All checks were successful
Gitea Actions Demo / build-and-push (push) Successful in 23s
- Added `marked_for_deletion` and `marked_for_deletion_at` fields to User model (Python and TypeScript) with serialization updates - Created POST /api/user/mark-for-deletion endpoint with JWT auth, error handling, and SSE event trigger - Blocked login and password reset for marked users; added new error codes ACCOUNT_MARKED_FOR_DELETION and ALREADY_MARKED - Updated UserProfile.vue with "Delete My Account" button, confirmation modal (email input), loading state, success/error modals, and sign-out/redirect logic - Synced error codes and model fields between backend and frontend - Added and updated backend and frontend tests to cover all flows and edge cases - All Acceptance Criteria from the spec are complete and verified
66 lines
2.4 KiB
Python
66 lines
2.4 KiB
Python
from dataclasses import dataclass, field
|
|
from models.base import BaseModel
|
|
|
|
@dataclass
|
|
class User(BaseModel):
|
|
first_name: str
|
|
last_name: str
|
|
email: str
|
|
password: str # In production, this should be hashed
|
|
verified: bool = False
|
|
verify_token: str | None = None
|
|
verify_token_created: str | None = None
|
|
reset_token: str | None = None
|
|
reset_token_created: str | None = None
|
|
image_id: str | None = None
|
|
pin: str = ''
|
|
pin_setup_code: str = ''
|
|
pin_setup_code_created: str | None = None
|
|
marked_for_deletion: bool = False
|
|
marked_for_deletion_at: str | None = None
|
|
|
|
@classmethod
|
|
def from_dict(cls, d: dict):
|
|
return cls(
|
|
first_name=d.get('first_name'),
|
|
last_name=d.get('last_name'),
|
|
email=d.get('email'),
|
|
password=d.get('password'),
|
|
verified=d.get('verified', False),
|
|
verify_token=d.get('verify_token'),
|
|
verify_token_created=d.get('verify_token_created'),
|
|
reset_token=d.get('reset_token'),
|
|
reset_token_created=d.get('reset_token_created'),
|
|
image_id=d.get('image_id'),
|
|
pin=d.get('pin', ''),
|
|
pin_setup_code=d.get('pin_setup_code', ''),
|
|
pin_setup_code_created=d.get('pin_setup_code_created'),
|
|
marked_for_deletion=d.get('marked_for_deletion', False),
|
|
marked_for_deletion_at=d.get('marked_for_deletion_at'),
|
|
id=d.get('id'),
|
|
created_at=d.get('created_at'),
|
|
updated_at=d.get('updated_at')
|
|
|
|
)
|
|
|
|
def to_dict(self):
|
|
base = super().to_dict()
|
|
base.update({
|
|
'first_name': self.first_name,
|
|
'last_name': self.last_name,
|
|
'email': self.email,
|
|
'password': self.password,
|
|
'verified': self.verified,
|
|
'verify_token': self.verify_token,
|
|
'verify_token_created': self.verify_token_created,
|
|
'reset_token': self.reset_token,
|
|
'reset_token_created': self.reset_token_created,
|
|
'image_id': self.image_id,
|
|
'pin': self.pin,
|
|
'pin_setup_code': self.pin_setup_code,
|
|
'pin_setup_code_created': self.pin_setup_code_created,
|
|
'marked_for_deletion': self.marked_for_deletion,
|
|
'marked_for_deletion_at': self.marked_for_deletion_at
|
|
})
|
|
return base
|