from flask import request from flask_restx import Resource, fields, reqparse from constants.languages import supported_language from controllers.console import api, console_ns from controllers.console.error import AlreadyActivateError from extensions.ext_database import db from libs.datetime_utils import naive_utc_now from libs.helper import StrLen, email, extract_remote_ip, timezone from models.account import AccountStatus from services.account_service import AccountService, RegisterService active_check_parser = reqparse.RequestParser() active_check_parser.add_argument( "workspace_id", type=str, required=False, nullable=True, location="args", help="Workspace ID" ) active_check_parser.add_argument( "email", type=email, required=False, nullable=True, location="args", help="Email address" ) active_check_parser.add_argument( "token", type=str, required=True, nullable=False, location="args", help="Activation token" ) @console_ns.route("/activate/check") class ActivateCheckApi(Resource): @api.doc("check_activation_token") @api.doc(description="Check if activation token is valid") @api.expect(active_check_parser) @api.response( 200, "Success", api.model( "ActivationCheckResponse", { "is_valid": fields.Boolean(description="Whether token is valid"), "data": fields.Raw(description="Activation data if valid"), }, ), ) def get(self): args = active_check_parser.parse_args() workspaceId = args["workspace_id"] reg_email = args["email"] token = args["token"] invitation = RegisterService.get_invitation_if_token_valid(workspaceId, reg_email, token) if invitation: data = invitation.get("data", {}) tenant = invitation.get("tenant", None) workspace_name = tenant.name if tenant else None workspace_id = tenant.id if tenant else None invitee_email = data.get("email") if data else None return { "is_valid": invitation is not None, "data": {"workspace_name": workspace_name, "workspace_id": workspace_id, "email": invitee_email}, } else: return {"is_valid": False} active_parser = reqparse.RequestParser() active_parser.add_argument("workspace_id", type=str, required=False, nullable=True, location="json") active_parser.add_argument("email", type=email, required=False, nullable=True, location="json") active_parser.add_argument("token", type=str, required=True, nullable=False, location="json") active_parser.add_argument("name", type=StrLen(30), required=True, nullable=False, location="json") active_parser.add_argument( "interface_language", type=supported_language, required=True, nullable=False, location="json" ) active_parser.add_argument("timezone", type=timezone, required=True, nullable=False, location="json") @console_ns.route("/activate") class ActivateApi(Resource): @api.doc("activate_account") @api.doc(description="Activate account with invitation token") @api.expect(active_parser) @api.response( 200, "Account activated successfully", api.model( "ActivationResponse", { "result": fields.String(description="Operation result"), "data": fields.Raw(description="Login token data"), }, ), ) @api.response(400, "Already activated or invalid token") def post(self): args = active_parser.parse_args() invitation = RegisterService.get_invitation_if_token_valid(args["workspace_id"], args["email"], args["token"]) if invitation is None: raise AlreadyActivateError() RegisterService.revoke_token(args["workspace_id"], args["email"], args["token"]) account = invitation["account"] account.name = args["name"] account.interface_language = args["interface_language"] account.timezone = args["timezone"] account.interface_theme = "light" account.status = AccountStatus.ACTIVE.value account.initialized_at = naive_utc_now() db.session.commit() token_pair = AccountService.login(account, ip_address=extract_remote_ip(request)) return {"result": "success", "data": token_pair.model_dump()}