feat: добавить отзывы, milestone-платежи, портфолио, skill-tests и верификацию (фичи Upwork/Fiverr)
Фичи конкурентов внедрены: - Reviews API + UI — система отзывов с рейтингом 1-5 звёзд - Milestones (Upwork-style) — разделение escrow на этапы с submit/approve - Portfolio — портфолио фрилансера с превью работ и технологиями - Skill Tests (Upwork-style) — сертификация навыков с тестами - Verification Badges — верификация email/phone/id/bank Модели: Milestone, PortfolioItem, SkillTest/SkillTestResult, Verification
This commit is contained in:
@@ -5,14 +5,20 @@ from app.models.project import Project
|
||||
from app.models.proposal import Proposal
|
||||
from app.models.ai_match import AIMatch
|
||||
from app.models.escrow import EscrowTransaction
|
||||
from app.models.milestone import Milestone
|
||||
from app.models.work_session import WorkSession
|
||||
from app.models.review import Review
|
||||
from app.models.message import Message
|
||||
from app.models.notification import Notification
|
||||
from app.models.portfolio import PortfolioItem
|
||||
from app.models.skill_test import SkillTest, SkillTestResult
|
||||
from app.models.verification import Verification
|
||||
|
||||
__all__ = [
|
||||
"User", "FreelancerProfile", "ClientProfile",
|
||||
"Project", "Proposal", "AIMatch",
|
||||
"EscrowTransaction", "WorkSession",
|
||||
"EscrowTransaction", "Milestone", "WorkSession",
|
||||
"Review", "Message", "Notification",
|
||||
"PortfolioItem", "SkillTest", "SkillTestResult",
|
||||
"Verification",
|
||||
]
|
||||
|
||||
@@ -0,0 +1,26 @@
|
||||
"""Модель Milestone-платежей (Upwork-style)."""
|
||||
|
||||
import uuid
|
||||
from datetime import datetime, timezone
|
||||
|
||||
from sqlalchemy import Column, DateTime, Enum, Float, ForeignKey, func
|
||||
from sqlalchemy.dialects.postgresql import UUID as PG_UUID
|
||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||
|
||||
from app.core.database import Base
|
||||
|
||||
|
||||
class Milestone(Base):
|
||||
__tablename__ = "milestones"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
project_id: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), ForeignKey("projects.id"), nullable=False)
|
||||
escrow_transaction_id: Mapped[uuid.UUID | None] = mapped_column(PG_UUID(as_uuid=True), ForeignKey("escrow_transactions.id"))
|
||||
title: Mapped[str] = mapped_column(String(255), nullable=False)
|
||||
description: Mapped[str | None] = mapped_column(Text)
|
||||
amount: Mapped[float] = mapped_column(Float(precision=10, scale=2), nullable=False)
|
||||
status: Mapped[str] = mapped_column(Enum("pending", "funded", "in_progress", "submitted", "approved", "disputed"), default="pending")
|
||||
due_date: Mapped[datetime | None] = mapped_column(DateTime(timezone=True))
|
||||
submitted_at: Mapped[datetime | None] = mapped_column(DateTime(timezone=True))
|
||||
approved_at: Mapped[datetime | None] = mapped_column(DateTime(timezone=True))
|
||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now())
|
||||
@@ -0,0 +1,23 @@
|
||||
"""Модель портфолио фрилансера."""
|
||||
|
||||
import uuid
|
||||
from datetime import datetime, timezone
|
||||
|
||||
from sqlalchemy import Column, DateTime, ForeignKey, Integer, String, Text, func
|
||||
from sqlalchemy.dialects.postgresql import UUID as PG_UUID
|
||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||
|
||||
from app.core.database import Base
|
||||
|
||||
|
||||
class PortfolioItem(Base):
|
||||
__tablename__ = "portfolio_items"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
freelancer_id: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), ForeignKey("users.id"), nullable=False)
|
||||
title: Mapped[str] = mapped_column(String(255), nullable=False)
|
||||
description: Mapped[str | None] = mapped_column(Text)
|
||||
image_url: Mapped[str | None] = mapped_column(Text) # URL превью работы
|
||||
live_url: Mapped[str | None] = mapped_column(Text) # Ссылка на работу
|
||||
technologies: Mapped[list] = mapped_column("technologies", postgresql.JSONB, default=list)
|
||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now())
|
||||
@@ -0,0 +1,33 @@
|
||||
"""Модель Skill Tests (сертификация навыков как на Upwork)."""
|
||||
|
||||
import uuid
|
||||
from datetime import datetime, timezone
|
||||
|
||||
from sqlalchemy import Column, DateTime, Enum, Float, ForeignKey, Integer, String, Text, func
|
||||
from sqlalchemy.dialects.postgresql import UUID as PG_UUID
|
||||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||||
|
||||
from app.core.database import Base
|
||||
|
||||
|
||||
class SkillTest(Base):
|
||||
__tablename__ = "skill_tests"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
name: Mapped[str] = mapped_column(String(255), nullable=False) # например "Python Basics"
|
||||
category: Mapped[str] = mapped_column(String(100)) # programming, design, etc.
|
||||
questions_count: Mapped[int] = mapped_column(Integer, default=40)
|
||||
passing_score: Mapped[float] = mapped_column(Float(precision=5, scale=2), default=70.0)
|
||||
|
||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now())
|
||||
|
||||
|
||||
class SkillTestResult(Base):
|
||||
__tablename__ = "skill_test_results"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
user_id: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), ForeignKey("users.id"), nullable=False)
|
||||
skill_test_id: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), ForeignKey("skill_tests.id"), nullable=False)
|
||||
score: Mapped[float] = mapped_column(Float(precision=5, scale=2))
|
||||
passed: Mapped[bool] = mapped_column(default=False)
|
||||
completed_at: Mapped[datetime | None] = mapped_column(DateTime(timezone=True))
|
||||
@@ -0,0 +1,27 @@
|
||||
"""Модель верификации профиля (Verified Badges)."""
|
||||
|
||||
import uuid
|
||||
from datetime import datetime, timezone
|
||||
|
||||
from sqlalchemy import Column, DateTime, Boolean, String, func
|
||||
from sqlalchemy.dialects.postgresql import UUID as PG_UUID
|
||||
from sqlalchemy.orm import Mapped, mapped_column
|
||||
|
||||
from app.core.database import Base
|
||||
|
||||
|
||||
class Verification(Base):
|
||||
__tablename__ = "verifications"
|
||||
|
||||
id: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), primary_key=True, default=uuid.uuid4)
|
||||
user_id: Mapped[uuid.UUID] = mapped_column(PG_UUID(as_uuid=True), nullable=False, unique=True)
|
||||
|
||||
# Типы верификации
|
||||
is_email_verified: Mapped[bool] = mapped_column(default=False)
|
||||
is_phone_verified: Mapped[bool] = mapped_column(default=False)
|
||||
is_id_verified: Mapped[bool] = mapped_column(default=False) # ID document
|
||||
is_bank_verified: Mapped[bool] = mapped_column(default=False) # Bank account
|
||||
|
||||
verified_at: Mapped[datetime | None] = mapped_column(DateTime(timezone=True))
|
||||
|
||||
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now())
|
||||
Reference in New Issue
Block a user