Add member profile questions, admin tooling, legal pages, and fast tests
- Add configurable profile questions with conditional visibility, admin-only fields, user answers, and seeded onboarding/volunteer questions
- Add admin UI for managing profile questions and member profile answers
- Add volunteer level/profile data support across backend schemas, models, API, and migration
- Update dashboard/profile UI, super admin menu, membership service types, and related styling
- Add privacy policy, terms of service, cookie notice, and footer links
- Add frontend Vitest coverage for profile question logic
- Add backend pytest coverage for profile answer normalization and validation
- Update restart.sh to build, run frontend/backend unit tests, and restart only after tests pass
- Refresh README, quickstart, project structure, instructions, and Square docs to match current app features
- Protect feature flag reload behind super-admin access
- Restrict admin-triggered password resets so admins can only reset member accounts
- Replace email template HTML preview rendering with escaped text preview
- Update docs for feature flag reload access, password reset scope, and email template preview safety
-- test user questions are also made by AI and not very useful. but i didn't know what to put there so its good enough for a test
This commit is contained in:
@@ -37,6 +37,13 @@ from .schemas import (
|
||||
EventRSVPBase,
|
||||
EventRSVPUpdate,
|
||||
EventRSVPResponse,
|
||||
QuestionOption,
|
||||
ProfileQuestionCreate,
|
||||
ProfileQuestionUpdate,
|
||||
ProfileQuestionResponse,
|
||||
ProfileQuestionForUser,
|
||||
ProfileAnswerUpdate,
|
||||
ProfileAnswersUpdateRequest,
|
||||
)
|
||||
|
||||
__all__ = [
|
||||
@@ -78,4 +85,11 @@ __all__ = [
|
||||
"EventRSVPBase",
|
||||
"EventRSVPUpdate",
|
||||
"EventRSVPResponse",
|
||||
"QuestionOption",
|
||||
"ProfileQuestionCreate",
|
||||
"ProfileQuestionUpdate",
|
||||
"ProfileQuestionResponse",
|
||||
"ProfileQuestionForUser",
|
||||
"ProfileAnswerUpdate",
|
||||
"ProfileAnswersUpdateRequest",
|
||||
]
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
from pydantic import BaseModel, EmailStr, Field, ConfigDict
|
||||
from typing import Optional
|
||||
from typing import Optional, Literal, Any
|
||||
from datetime import datetime, date
|
||||
from ..models.models import UserRole, MembershipStatus, PaymentStatus, PaymentMethod
|
||||
|
||||
@@ -24,6 +24,7 @@ class UserUpdate(BaseModel):
|
||||
phone: Optional[str] = None
|
||||
address: Optional[str] = None
|
||||
role: Optional[UserRole] = None
|
||||
volunteer_level: Optional[str] = Field(None, max_length=50)
|
||||
|
||||
|
||||
class UserResponse(UserBase):
|
||||
@@ -31,6 +32,7 @@ class UserResponse(UserBase):
|
||||
|
||||
id: int
|
||||
role: UserRole
|
||||
volunteer_level: Optional[str] = None
|
||||
is_active: bool
|
||||
created_at: datetime
|
||||
last_login: Optional[datetime] = None
|
||||
@@ -285,3 +287,80 @@ class EventRSVPResponse(EventRSVPBase):
|
||||
attended: bool
|
||||
created_at: datetime
|
||||
updated_at: datetime
|
||||
|
||||
|
||||
# Profile Question Schemas
|
||||
ProfileQuestionInputType = Literal["text", "number", "boolean", "date", "select"]
|
||||
|
||||
|
||||
class QuestionOption(BaseModel):
|
||||
label: str = Field(..., min_length=1, max_length=100)
|
||||
value: str = Field(..., min_length=1, max_length=100)
|
||||
|
||||
|
||||
class ProfileQuestionBase(BaseModel):
|
||||
key: str = Field(..., min_length=2, max_length=100, pattern=r"^[a-z0-9_]+$")
|
||||
label: str = Field(..., min_length=2, max_length=255)
|
||||
help_text: Optional[str] = None
|
||||
input_type: ProfileQuestionInputType
|
||||
placeholder: Optional[str] = Field(None, max_length=255)
|
||||
options: Optional[list[QuestionOption]] = None
|
||||
is_required: bool = False
|
||||
is_active: bool = True
|
||||
admin_only_edit: bool = False
|
||||
display_order: int = 0
|
||||
depends_on_question_id: Optional[int] = None
|
||||
depends_on_value: Optional[str] = Field(None, max_length=255)
|
||||
|
||||
|
||||
class ProfileQuestionCreate(ProfileQuestionBase):
|
||||
pass
|
||||
|
||||
|
||||
class ProfileQuestionUpdate(BaseModel):
|
||||
key: Optional[str] = Field(None, min_length=2, max_length=100, pattern=r"^[a-z0-9_]+$")
|
||||
label: Optional[str] = Field(None, min_length=2, max_length=255)
|
||||
help_text: Optional[str] = None
|
||||
input_type: Optional[ProfileQuestionInputType] = None
|
||||
placeholder: Optional[str] = Field(None, max_length=255)
|
||||
options: Optional[list[QuestionOption]] = None
|
||||
is_required: Optional[bool] = None
|
||||
is_active: Optional[bool] = None
|
||||
admin_only_edit: Optional[bool] = None
|
||||
display_order: Optional[int] = None
|
||||
depends_on_question_id: Optional[int] = None
|
||||
depends_on_value: Optional[str] = Field(None, max_length=255)
|
||||
|
||||
|
||||
class ProfileQuestionResponse(BaseModel):
|
||||
model_config = ConfigDict(from_attributes=True)
|
||||
|
||||
id: int
|
||||
key: str
|
||||
label: str
|
||||
help_text: Optional[str] = None
|
||||
input_type: ProfileQuestionInputType
|
||||
placeholder: Optional[str] = None
|
||||
options: list[QuestionOption] = []
|
||||
is_required: bool
|
||||
is_active: bool
|
||||
admin_only_edit: bool
|
||||
display_order: int
|
||||
depends_on_question_id: Optional[int] = None
|
||||
depends_on_value: Optional[str] = None
|
||||
created_at: datetime
|
||||
updated_at: datetime
|
||||
|
||||
|
||||
class ProfileQuestionForUser(ProfileQuestionResponse):
|
||||
answer: Optional[Any] = None
|
||||
can_edit: bool = True
|
||||
|
||||
|
||||
class ProfileAnswerUpdate(BaseModel):
|
||||
question_id: int
|
||||
value: Optional[Any] = None
|
||||
|
||||
|
||||
class ProfileAnswersUpdateRequest(BaseModel):
|
||||
answers: list[ProfileAnswerUpdate]
|
||||
|
||||
Reference in New Issue
Block a user