[#339] Backend preparation for 2FA support.

This commit is contained in:
Fabio Manganiello 2024-07-25 00:47:04 +02:00
parent 2cbb005c67
commit 8ec1ca8543
Signed by untrusted user: blacklight
GPG key ID: D90FBA7F76362774
6 changed files with 353 additions and 59 deletions

View file

@ -0,0 +1,207 @@
from typing import List, Optional
from flask import Blueprint, jsonify, request
from platypush.backend.http.app import template_folder
from platypush.backend.http.app.utils import UserAuthStatus, authenticate
from platypush.backend.http.utils import HttpUtils
from platypush.exceptions.user import (
InvalidCredentialsException,
InvalidOtpCodeException,
UserException,
)
from platypush.config import Config
from platypush.context import get_plugin
from platypush.user import UserManager
otp = Blueprint('otp', __name__, template_folder=template_folder)
# Declare routes list
__routes__ = [
otp,
]
def _get_otp_and_qrcode():
otp = get_plugin('otp') # pylint: disable=redefined-outer-name
qrcode = get_plugin('qrcode')
assert (
otp and qrcode
), 'The otp and/or qrcode plugins are not available in your installation'
return otp, qrcode
def _get_username():
user = HttpUtils.current_user()
if not user:
raise InvalidCredentialsException('Invalid user session')
return str(user.username)
def _get_otp_uri_and_qrcode(username: str, otp_secret: Optional[str] = None):
if not otp_secret:
return None, None
otp, qrcode = _get_otp_and_qrcode() # pylint: disable=redefined-outer-name
otp_uri = (
otp.provision_time_otp(
name=username,
secret=otp_secret,
issuer=f'platypush@{Config.get_device_id()}',
).output
if otp_secret
else None
)
otp_qrcode = (
qrcode.generate(content=otp_uri, format='png').output.get('data')
if otp_uri
else None
)
return otp_uri, otp_qrcode
def _verify_code(code: str, otp_secret: str) -> bool:
otp, _ = _get_otp_and_qrcode() # pylint: disable=redefined-outer-name
return otp.verify_time_otp(otp=code, secret=otp_secret).output
def _dump_response(
username: str,
otp_secret: Optional[str] = None,
backup_codes: Optional[List[str]] = None,
):
otp_uri, otp_qrcode = _get_otp_uri_and_qrcode(username, otp_secret)
return jsonify(
{
'username': username,
'otp_secret': otp_secret,
'otp_uri': otp_uri,
'qrcode': otp_qrcode,
'backup_codes': backup_codes or [],
}
)
def _get_otp():
username = _get_username()
user_manager = UserManager()
otp_secret = user_manager.get_otp_secret(username)
backup_codes = user_manager.get_user_backup_codes(username) if otp_secret else []
return _dump_response(
username=username,
otp_secret=otp_secret,
backup_codes=[str(c.code) for c in backup_codes],
)
def _authenticate_user(username: str, password: Optional[str]):
assert password, 'The password field is required when setting up OTP'
user, auth_status = UserManager().authenticate_user( # type: ignore
username, password, skip_2fa=True, with_status=True
)
if not user:
raise InvalidCredentialsException(auth_status.value[2])
def _post_otp():
body = request.json
assert body, 'Invalid request body'
username = _get_username()
dry_run = body.get('dry_run', False)
otp_secret = body.get('otp_secret')
if not dry_run:
_authenticate_user(username, body.get('password'))
if otp_secret:
code = body.get('code')
assert code, 'The code field is required when setting up OTP'
if not _verify_code(code, otp_secret):
raise InvalidOtpCodeException()
user_manager = UserManager()
user_otp, backup_codes = user_manager.enable_otp(
username=username,
otp_secret=otp_secret,
dry_run=dry_run,
)
return _dump_response(
username=username,
otp_secret=str(user_otp.otp_secret),
backup_codes=backup_codes,
)
def _delete_otp():
body = request.json
assert body, 'Invalid request body'
username = _get_username()
_authenticate_user(username, body.get('password'))
user_manager = UserManager()
user_manager.disable_otp(username)
return jsonify({'status': 'ok'})
@otp.route('/otp/config', methods=['GET', 'POST', 'DELETE'])
@authenticate()
def otp_route():
"""
:return: The user's current MFA/OTP configuration:
.. code-block:: json
{
"username": "testuser",
"otp_secret": "JBSA6ZUZ5DPEK7YV",
"otp_uri": "otpauth://totp/testuser?secret=JBSA6ZUZ5DPEK7YV&issuer=platypush@localhost",
"qrcode": "",
"backup_codes": [
"1A2B3C4D5E",
"6F7G8H9I0J",
"KLMNOPQRST",
"UVWXYZ1234",
"567890ABCD",
"EFGHIJKLMN",
"OPQRSTUVWX",
"YZ12345678",
"90ABCDEF12",
"34567890AB"
]
}
"""
try:
if request.method.lower() == 'get':
return _get_otp()
if request.method.lower() == 'post':
return _post_otp()
if request.method.lower() == 'delete':
return _delete_otp()
return jsonify({'error': 'Method not allowed'}), 405
except AssertionError as e:
return jsonify({'error': str(e)}), 400
except InvalidCredentialsException:
return UserAuthStatus.INVALID_CREDENTIALS.to_response()
except InvalidOtpCodeException:
return UserAuthStatus.INVALID_OTP_CODE.to_response()
except UserException as e:
return jsonify({'error': e.__class__.__name__, 'message': str(e)}), 401
except Exception as e:
HttpUtils.log.error(f'Error while processing OTP request: {e}', exc_info=True)
return jsonify({'error': str(e)}), 500
# vim:sw=4:ts=4:et:

View file

@ -7,6 +7,7 @@ class UserException(PlatypushException):
""" """
Base class for all user exceptions. Base class for all user exceptions.
""" """
def __init__(self, *args, user: Optional[Union[str, int]] = None, **kwargs): def __init__(self, *args, user: Optional[Union[str, int]] = None, **kwargs):
super().__init__(*args, **kwargs) super().__init__(*args, **kwargs)
self.user = user self.user = user
@ -16,6 +17,7 @@ class AuthenticationException(UserException):
""" """
Authentication error exception. Authentication error exception.
""" """
def __init__(self, error='Unauthorized', *args, **kwargs): def __init__(self, error='Unauthorized', *args, **kwargs):
super().__init__(error, *args, **kwargs) super().__init__(error, *args, **kwargs)
@ -24,6 +26,7 @@ class InvalidTokenException(AuthenticationException):
""" """
Exception raised in case of wrong user token. Exception raised in case of wrong user token.
""" """
def __init__(self, error='Invalid user token', *args, **kwargs): def __init__(self, error='Invalid user token', *args, **kwargs):
super().__init__(error, *args, **kwargs) super().__init__(error, *args, **kwargs)
@ -32,6 +35,7 @@ class InvalidCredentialsException(AuthenticationException):
""" """
Exception raised in case of wrong user token. Exception raised in case of wrong user token.
""" """
def __init__(self, error='Invalid credentials', *args, **kwargs): def __init__(self, error='Invalid credentials', *args, **kwargs):
super().__init__(error, *args, **kwargs) super().__init__(error, *args, **kwargs)
@ -40,5 +44,26 @@ class InvalidJWTTokenException(InvalidTokenException):
""" """
Exception raised in case of wrong/invalid JWT token. Exception raised in case of wrong/invalid JWT token.
""" """
def __init__(self, error='Invalid JWT token', *args, **kwargs): def __init__(self, error='Invalid JWT token', *args, **kwargs):
super().__init__(error, *args, **kwargs) super().__init__(error, *args, **kwargs)
class InvalidOtpCodeException(AuthenticationException):
"""
Exception raised in case of wrong OTP code.
"""
def __init__(self, error='Invalid OTP code', *args, **kwargs):
super().__init__(error, *args, **kwargs)
class OtpRecordAlreadyExistsException(UserException):
"""
Exception raised in case of an OTP record already existing for a user.
"""
def __init__(
self, *args, error='An OTP record already exists for this user', **kwargs
):
super().__init__(*args, error, **kwargs)

View file

@ -3,7 +3,6 @@ from typing import Optional
import pyotp import pyotp
from platypush import Response
from platypush.config import Config from platypush.config import Config
from platypush.plugins import Plugin, action from platypush.plugins import Plugin, action
@ -19,7 +18,7 @@ class OtpPlugin(Plugin):
secret: Optional[str] = None, secret: Optional[str] = None,
secret_path: Optional[str] = None, secret_path: Optional[str] = None,
provisioning_name: Optional[str] = None, provisioning_name: Optional[str] = None,
issuer_name: Optional[str] = None, issuer: Optional[str] = None,
**kwargs **kwargs
): ):
""" """
@ -29,17 +28,17 @@ class OtpPlugin(Plugin):
generated. generated.
:param provisioning_name: If you want to use the Google Authenticator, you can specify the default :param provisioning_name: If you want to use the Google Authenticator, you can specify the default
email address to associate to your OTPs for the provisioning process here. email address to associate to your OTPs for the provisioning process here.
:param issuer_name: If you want to use the Google Authenticator, you can specify the default :param issuer: If you want to use the Google Authenticator, you can specify the default
issuer name to display on your OTPs here. issuer name to display on your OTPs here.
""" """
super().__init__(**kwargs) super().__init__(**kwargs)
if not secret_path: if not secret_path:
secret_path = os.path.join(Config.get('workdir'), 'otp', 'secret') secret_path = os.path.join(Config.get_workdir(), 'otp', 'secret')
self.secret_path = secret_path self.secret_path = secret_path
self.secret = secret self.secret = secret
self.provisioning_name = provisioning_name self.provisioning_name = provisioning_name
self.issuer_name = issuer_name self.issuer = issuer
def _get_secret_from_path(self, secret_path: str) -> str: def _get_secret_from_path(self, secret_path: str) -> str:
if not os.path.isfile(secret_path): if not os.path.isfile(secret_path):
@ -75,7 +74,16 @@ class OtpPlugin(Plugin):
return pyotp.HOTP(self._get_secret(secret, secret_path)) return pyotp.HOTP(self._get_secret(secret, secret_path))
@action @action
def refresh_secret(self, secret_path: Optional[str] = None) -> Response: def generate_secret(self) -> str:
"""
Generate a new secret token for key generation.
:return: The new secret token.
"""
return pyotp.random_base32()
@action
def refresh_secret(self, secret_path: Optional[str] = None) -> str:
""" """
Refresh the secret token for key generation given a secret path. Refresh the secret token for key generation given a secret path.
@ -162,7 +170,7 @@ class OtpPlugin(Plugin):
def provision_time_otp( def provision_time_otp(
self, self,
name: Optional[str] = None, name: Optional[str] = None,
issuer_name: Optional[str] = None, issuer: Optional[str] = None,
secret: Optional[str] = None, secret: Optional[str] = None,
secret_path: Optional[str] = None, secret_path: Optional[str] = None,
) -> str: ) -> str:
@ -171,23 +179,23 @@ class OtpPlugin(Plugin):
:param name: Name or e-mail address associated to the account used by the Google Authenticator. :param name: Name or e-mail address associated to the account used by the Google Authenticator.
If None is specified then the value will be read from the configured ``provisioning_name``. If None is specified then the value will be read from the configured ``provisioning_name``.
:param issuer_name: Name of the issuer of the OTP (default: default configured ``issuer_name`` or None). :param issuer: Name of the issuer of the OTP (default: default configured ``issuer`` or None).
:param secret: Secret token to be used (overrides configured ``secret``). :param secret: Secret token to be used (overrides configured ``secret``).
:param secret_path: File containing the secret to be used (overrides configured ``secret_path``). :param secret_path: File containing the secret to be used (overrides configured ``secret_path``).
:return: Generated provisioning URI. :return: Generated provisioning URI.
""" """
name = name or self.provisioning_name name = name or self.provisioning_name
issuer_name = issuer_name or self.issuer_name issuer = issuer or self.issuer
assert name, 'No account name or default provisioning address provided' assert name, 'No account name or default provisioning address provided'
_otp = self._get_topt(secret, secret_path) _otp = self._get_topt(secret, secret_path)
return _otp.provisioning_uri(name, issuer_name=issuer_name) return _otp.provisioning_uri(name, issuer_name=issuer)
@action @action
def provision_counter_otp( def provision_counter_otp(
self, self,
name: Optional[str] = None, name: Optional[str] = None,
issuer_name: Optional[str] = None, issuer: Optional[str] = None,
initial_count=0, initial_count=0,
secret: Optional[str] = None, secret: Optional[str] = None,
secret_path: Optional[str] = None, secret_path: Optional[str] = None,
@ -197,19 +205,19 @@ class OtpPlugin(Plugin):
:param name: Name or e-mail address associated to the account used by the Google Authenticator. :param name: Name or e-mail address associated to the account used by the Google Authenticator.
If None is specified then the value will be read from the configured ``provisioning_name``. If None is specified then the value will be read from the configured ``provisioning_name``.
:param issuer_name: Name of the issuer of the OTP (default: default configured ``issuer_name`` or None). :param issuer: Name of the issuer of the OTP (default: default configured ``issuer`` or None).
:param initial_count: Initial value for the counter (default: 0). :param initial_count: Initial value for the counter (default: 0).
:param secret: Secret token to be used (overrides configured ``secret``). :param secret: Secret token to be used (overrides configured ``secret``).
:param secret_path: File containing the secret to be used (overrides configured ``secret_path``). :param secret_path: File containing the secret to be used (overrides configured ``secret_path``).
:return: Generated provisioning URI. :return: Generated provisioning URI.
""" """
name = name or self.provisioning_name name = name or self.provisioning_name
issuer_name = issuer_name or self.issuer_name issuer = issuer or self.issuer
assert name, 'No account name or default provisioning address provided' assert name, 'No account name or default provisioning address provided'
_otp = self._get_hopt(secret, secret_path) _otp = self._get_hopt(secret, secret_path)
return _otp.provisioning_uri( return _otp.provisioning_uri(
name, issuer_name=issuer_name, initial_count=initial_count name, issuer_name=issuer, initial_count=initial_count
) )

View file

@ -79,7 +79,7 @@ class UserPlugin(Plugin):
:param username: Username. :param username: Username.
:param password: Password. :param password: Password.
:param code: Optional 2FA code, if 2FA is enabled for the user. :param code: Optional 2FA code, if 2FA is enabled for the user.
:param return_error_details: If True then return the error details in :param return_details: If True then return the error details in
case of authentication failure. case of authentication failure.
:return: If ``return_details`` is False (default), the action returns :return: If ``return_details`` is False (default), the action returns
True if the provided credentials are valid, False otherwise. True if the provided credentials are valid, False otherwise.
@ -96,7 +96,7 @@ class UserPlugin(Plugin):
""" """
response = self.user_manager.authenticate_user( response = self.user_manager.authenticate_user(
username, password, code=code, return_error=return_details username, password, code=code, with_status=return_details
) )
if return_details: if return_details:

View file

@ -17,6 +17,7 @@ from platypush.context import get_plugin
from platypush.exceptions.user import ( from platypush.exceptions.user import (
InvalidJWTTokenException, InvalidJWTTokenException,
InvalidCredentialsException, InvalidCredentialsException,
OtpRecordAlreadyExistsException,
) )
from platypush.utils import get_or_generate_stored_rsa_key_pair, utcnow from platypush.utils import get_or_generate_stored_rsa_key_pair, utcnow
@ -63,7 +64,7 @@ class UserManager:
""" """
Get or generate the OTP RSA key pair. Get or generate the OTP RSA key pair.
""" """
return get_or_generate_stored_rsa_key_pair(cls._otp_keyfile, size=4096) return get_or_generate_stored_rsa_key_pair(cls._otp_keyfile, size=2048)
@staticmethod @staticmethod
def _encrypt(data: Union[str, bytes, dict, list, tuple], key: rsa.PublicKey) -> str: def _encrypt(data: Union[str, bytes, dict, list, tuple], key: rsa.PublicKey) -> str:
@ -151,10 +152,17 @@ class UserManager:
session.commit() session.commit()
return True return True
def authenticate_user(self, username, password, code=None, return_error=False): def authenticate_user(
self, username, password, code=None, skip_2fa=False, with_status=False
):
with self._get_session() as session: with self._get_session() as session:
return self._authenticate_user( return self._authenticate_user(
session, username, password, code=code, with_status=return_error session,
username,
password,
code=code,
skip_2fa=skip_2fa,
with_status=with_status,
) )
def authenticate_user_session(self, session_token, with_status=False): def authenticate_user_session(self, session_token, with_status=False):
@ -268,33 +276,45 @@ class UserManager:
) )
def create_otp_secret( def create_otp_secret(
self, username: str, expires_at: Optional[datetime.datetime] = None self,
username: str,
expires_at: Optional[datetime.datetime] = None,
otp_secret: Optional[str] = None,
dry_run: bool = False,
): ):
pubkey, _ = self._get_or_generate_otp_rsa_key_pair()
# Generate a new OTP secret and encrypt it with the OTP RSA key pair # Generate a new OTP secret and encrypt it with the OTP RSA key pair
otp_secret = "".join( otp_secret = otp_secret or "".join(
random.choice("ABCDEFGHIJKLMNOPQRSTUVWXYZ234567") for _ in range(32) random.choice("ABCDEFGHIJKLMNOPQRSTUVWXYZ234567") for _ in range(32)
) )
encrypted_secret = self._encrypt(otp_secret, pubkey)
with self._get_session(locked=True) as session: with self._get_session(locked=True) as session:
user = self._get_user(session, username) user = self._get_user(session, username)
assert user, f'No such user: {username}' if not user:
raise InvalidCredentialsException()
# Create a new OTP secret # Create a new OTP secret
user_otp = UserOtp( user_otp = UserOtp(
user_id=user.user_id, user_id=user.user_id,
otp_secret=encrypted_secret, otp_secret=otp_secret,
created_at=utcnow(), created_at=utcnow(),
expires_at=expires_at, expires_at=expires_at,
) )
# Remove any existing OTP secret and replace it with the new one if not dry_run:
session.query(UserOtp).filter_by(user_id=user.user_id).delete() # Store a copy of the OTP secret encrypted with the RSA public key
session.add(user_otp) pubkey, _ = self._get_or_generate_otp_rsa_key_pair()
session.commit() encrypted_secret = self._encrypt(otp_secret, pubkey)
encrypted_otp = UserOtp(
user_id=user_otp.user_id,
otp_secret=encrypted_secret,
created_at=user_otp.created_at,
expires_at=user_otp.expires_at,
)
# Remove any existing OTP secret and replace it with the new one
session.query(UserOtp).filter_by(user_id=user.user_id).delete()
session.add(encrypted_otp)
session.commit()
return user_otp return user_otp
@ -445,6 +465,7 @@ class UserManager:
username: str, username: str,
password: str, password: str,
code: Optional[str] = None, code: Optional[str] = None,
skip_2fa: bool = False,
with_status: bool = False, with_status: bool = False,
) -> Union[Optional['User'], Tuple[Optional['User'], 'AuthenticationStatus']]: ) -> Union[Optional['User'], Tuple[Optional['User'], 'AuthenticationStatus']]:
""" """
@ -478,7 +499,7 @@ class UserManager:
# The user doesn't have 2FA enabled and the password is correct: # The user doesn't have 2FA enabled and the password is correct:
# authentication successful # authentication successful
if not otp_secret: if skip_2fa or not otp_secret:
return user if not with_status else (user, AuthenticationStatus.OK) return user if not with_status else (user, AuthenticationStatus.OK)
# The user has 2FA enabled but the code is missing # The user has 2FA enabled but the code is missing
@ -501,23 +522,27 @@ class UserManager:
return user if not with_status else (user, AuthenticationStatus.OK) return user if not with_status else (user, AuthenticationStatus.OK)
def refresh_user_backup_codes(self, username: str): def refresh_user_backup_codes(self, username: str) -> List[str]:
""" """
Refresh the backup codes for a user with 2FA enabled. Refresh the backup codes for a user with 2FA enabled.
""" """
backup_codes = [
"".join(
random.choice("ABCDEFGHIJKLMNOPQRSTUVWXYZ234567") for _ in range(16)
)
for _ in range(10)
]
with self._get_session(locked=True) as session: with self._get_session(locked=True) as session:
user = self._get_user(session, username) user = self._get_user(session, username)
if not user: if not user:
return False return []
session.query(UserBackupCode).filter_by(user_id=user.user_id).delete() session.query(UserBackupCode).filter_by(user_id=user.user_id).delete()
pub_key, _ = self._get_or_generate_otp_rsa_key_pair() pub_key, _ = self._get_or_generate_otp_rsa_key_pair()
stored_codes = []
for _ in range(10): for backup_code in backup_codes:
backup_code = "".join(
random.choice("ABCDEFGHIJKLMNOPQRSTUVWXYZ234567") for _ in range(16)
)
user_backup_code = UserBackupCode( user_backup_code = UserBackupCode(
user_id=user.user_id, user_id=user.user_id,
code=self._encrypt(backup_code, pub_key), code=self._encrypt(backup_code, pub_key),
@ -526,9 +551,10 @@ class UserManager:
) )
session.add(user_backup_code) session.add(user_backup_code)
stored_codes.append(backup_code)
session.commit() session.commit()
return True return stored_codes
def get_user_backup_codes(self, username: str) -> List['UserBackupCode']: def get_user_backup_codes(self, username: str) -> List['UserBackupCode']:
with self._get_session() as session: with self._get_session() as session:
@ -537,12 +563,17 @@ class UserManager:
return [] return []
_, priv_key = self._get_or_generate_otp_rsa_key_pair() _, priv_key = self._get_or_generate_otp_rsa_key_pair()
codes = session.query(UserBackupCode).filter_by(user_id=user.user_id).all() return [
UserBackupCode(
for code in codes: user_id=code.user_id,
code.code = self._decrypt(code.code, priv_key) code=self._decrypt(code.code, priv_key),
created_at=code.created_at,
return codes expires_at=code.expires_at,
)
for code in session.query(UserBackupCode)
.filter_by(user_id=user.user_id)
.all()
]
def validate_backup_code(self, username: str, code: str) -> bool: def validate_backup_code(self, username: str, code: str) -> bool:
with self._get_session() as session: with self._get_session() as session:
@ -583,7 +614,7 @@ class UserManager:
return otp.verify_time_otp(otp=code, secret=otp_secret) return otp.verify_time_otp(otp=code, secret=otp_secret)
def disable_mfa(self, username: str): def disable_otp(self, username: str):
with self._get_session(locked=True) as session: with self._get_session(locked=True) as session:
user = self._get_user(session, username) user = self._get_user(session, username)
if not user: if not user:
@ -594,15 +625,30 @@ class UserManager:
session.commit() session.commit()
return True return True
def enable_mfa(self, username: str): def enable_otp(
self,
username: str,
dry_run: bool = False,
otp_secret: Optional[str] = None,
):
with self._get_session() as session: with self._get_session() as session:
user = self._get_user(session, username) user = self._get_user(session, username)
if not user: if not user:
return False raise InvalidCredentialsException()
self.create_otp_secret(username) user_otp = session.query(UserOtp).filter_by(user_id=user.user_id).first()
self.refresh_user_backup_codes(username) if user_otp:
return True raise OtpRecordAlreadyExistsException()
user_otp = self.create_otp_secret(
username, otp_secret=otp_secret, dry_run=dry_run
)
backup_codes = (
self.refresh_user_backup_codes(username) if not dry_run else []
)
return user_otp, backup_codes
# vim:sw=4:ts=4:et: # vim:sw=4:ts=4:et:

View file

@ -1,6 +1,13 @@
import enum import enum
from sqlalchemy import Column, Integer, String, DateTime, ForeignKey from sqlalchemy import (
Column,
DateTime,
ForeignKey,
Integer,
String,
UniqueConstraint,
)
from platypush.common.db import Base from platypush.common.db import Base
@ -28,9 +35,8 @@ class User(Base):
"""Models the User table""" """Models the User table"""
__tablename__ = 'user' __tablename__ = 'user'
__table_args__ = {'sqlite_autoincrement': True}
user_id = Column(Integer, primary_key=True) user_id = Column(Integer, primary_key=True, autoincrement=True)
username = Column(String, unique=True, nullable=False) username = Column(String, unique=True, nullable=False)
password = Column(String) password = Column(String)
password_salt = Column(String) password_salt = Column(String)
@ -42,9 +48,8 @@ class UserSession(Base):
"""Models the UserSession table""" """Models the UserSession table"""
__tablename__ = 'user_session' __tablename__ = 'user_session'
__table_args__ = {'sqlite_autoincrement': True}
session_id = Column(Integer, primary_key=True) session_id = Column(Integer, primary_key=True, autoincrement=True)
session_token = Column(String, unique=True, nullable=False) session_token = Column(String, unique=True, nullable=False)
csrf_token = Column(String, unique=True) csrf_token = Column(String, unique=True)
user_id = Column(Integer, ForeignKey('user.user_id'), nullable=False) user_id = Column(Integer, ForeignKey('user.user_id'), nullable=False)
@ -73,10 +78,13 @@ class UserBackupCode(Base):
__tablename__ = 'user_backup_code' __tablename__ = 'user_backup_code'
user_id = Column(Integer, ForeignKey('user.user_id'), primary_key=True) id = Column(Integer, primary_key=True, autoincrement=True)
code = Column(String, nullable=False, unique=True) user_id = Column(Integer, ForeignKey('user.user_id'))
code = Column(String, nullable=False)
created_at = Column(DateTime) created_at = Column(DateTime)
expires_at = Column(DateTime) expires_at = Column(DateTime)
UniqueConstraint(user_id, code)
# vim:sw=4:ts=4:et: # vim:sw=4:ts=4:et: