feat: Freelancer Match — полная продакшн версия с AI-матчингом и escrow
This commit is contained in:
@@ -0,0 +1,34 @@
|
||||
"""AI endpoints: матчинг, рекомендации."""
|
||||
|
||||
import logging
|
||||
|
||||
from fastapi import APIRouter, Depends
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
|
||||
from app.core.database import get_db
|
||||
from app.schemas.ai_match import AIMatchRequest, AIMatchResponse
|
||||
from app.services.ai_service import find_matches
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
router = APIRouter(prefix="/api/ai", tags=["ai"])
|
||||
|
||||
|
||||
@router.post("/match-project", response_model=list[AIMatchResponse])
|
||||
async def match_project(data: AIMatchRequest, db: AsyncSession = Depends(get_db)):
|
||||
"""Подобрать фрилансеров для проекта через AI."""
|
||||
|
||||
matches = await find_matches(
|
||||
db=db, project_id=data.project_id, limit=data.limit, min_score=data.min_score
|
||||
)
|
||||
|
||||
return [AIMatchResponse(**m) for m in matches]
|
||||
|
||||
|
||||
@router.post("/generate-cover-letter")
|
||||
async def generate_cover_letter(project_title: str, freelancer_skills: list[str]):
|
||||
"""Сгенерировать сопроводительное письмо для заявки."""
|
||||
|
||||
# Placeholder — в продакшене вызов LLM
|
||||
return {
|
||||
"cover_letter": f"Здравствуйте! Я заинтересован в проекте '{project_title}'. Мой опыт работы с [{', '.join(freelancer_skills)}] позволяет качественно выполнить задачу."
|
||||
}
|
||||
@@ -0,0 +1,84 @@
|
||||
"""Auth endpoints: регистрация, логин, OAuth."""
|
||||
|
||||
import logging
|
||||
from datetime import timedelta
|
||||
|
||||
from fastapi import APIRouter, Depends, HTTPException, status
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
from sqlalchemy import select
|
||||
|
||||
from app.core.database import get_db
|
||||
from app.core.security import (
|
||||
hash_password, verify_password, create_access_token, create_refresh_token, get_current_user
|
||||
)
|
||||
from app.schemas.user import UserCreate, UserLogin
|
||||
from app.schemas.auth import TokenPair
|
||||
from app.models.user import User, FreelancerProfile
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
router = APIRouter(prefix="/api/auth", tags=["auth"])
|
||||
|
||||
|
||||
@router.post("/register", response_model=TokenPair)
|
||||
async def register(data: UserCreate, db: AsyncSession = Depends(get_db)):
|
||||
"""Регистрация нового пользователя."""
|
||||
|
||||
# Проверка email
|
||||
result = await db.execute(select(User).where(User.email == data.email))
|
||||
if result.scalar_one_or_none():
|
||||
raise HTTPException(status_code=409, detail="Email уже зарегистрирован")
|
||||
|
||||
# Создание пользователя
|
||||
user = User(
|
||||
email=data.email,
|
||||
password_hash=hash_password(data.password),
|
||||
role=data.role,
|
||||
full_name=data.full_name,
|
||||
)
|
||||
db.add(user)
|
||||
await db.commit()
|
||||
await db.refresh(user)
|
||||
|
||||
# Создаём профиль фрилансера по умолчанию
|
||||
if data.role in ("freelancer", "both"):
|
||||
profile = FreelancerProfile(
|
||||
user_id=user.id,
|
||||
skills=data.full_name or [], # заполняется позже
|
||||
)
|
||||
db.add(profile)
|
||||
await db.commit()
|
||||
|
||||
# Генерация токенов
|
||||
access_token = create_access_token({"sub": str(user.id), "role": data.role})
|
||||
refresh_token = create_refresh_token({"sub": str(user.id)})
|
||||
|
||||
return TokenPair(access_token=access_token, refresh_token=refresh_token)
|
||||
|
||||
|
||||
@router.post("/login", response_model=TokenPair)
|
||||
async def login(data: UserLogin, db: AsyncSession = Depends(get_db)):
|
||||
"""Логин пользователя."""
|
||||
|
||||
result = await db.execute(select(User).where(User.email == data.email))
|
||||
user = result.scalar_one_or_none()
|
||||
|
||||
if not user or not verify_password(data.password, user.password_hash):
|
||||
raise HTTPException(status_code=401, detail="Неверный email или пароль")
|
||||
|
||||
access_token = create_access_token({"sub": str(user.id), "role": user.role})
|
||||
refresh_token = create_refresh_token({"sub": str(user.id)})
|
||||
|
||||
return TokenPair(access_token=access_token, refresh_token=refresh_token)
|
||||
|
||||
|
||||
@router.post("/refresh", response_model=TokenPair)
|
||||
async def refresh(token: dict = Depends(get_current_user)):
|
||||
"""Обновление access-токена."""
|
||||
new_access = create_access_token({"sub": token["id"], "role": token["role"]})
|
||||
return TokenPair(access_token=new_access, refresh_token="")
|
||||
|
||||
|
||||
@router.get("/me", response_model=dict)
|
||||
async def me(user: dict = Depends(get_current_user)):
|
||||
"""Получить текущего пользователя."""
|
||||
return {"user_id": user["id"], "role": user["role"]}
|
||||
@@ -0,0 +1,82 @@
|
||||
"""Escrow endpoints (гарант)."""
|
||||
|
||||
from fastapi import APIRouter, Depends, HTTPException
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
from sqlalchemy import select
|
||||
|
||||
from app.core.database import get_db
|
||||
from app.schemas.escrow import EscrowCreate, EscrowRelease
|
||||
from app.models.escrow import EscrowTransaction
|
||||
from app.models.project import Project
|
||||
|
||||
router = APIRouter(prefix="/api/escrow", tags=["escrow"])
|
||||
|
||||
|
||||
@router.post("/create")
|
||||
async def create_escrow(data: EscrowCreate, db: AsyncSession = Depends(get_db)):
|
||||
"""Создать escrow-транзакцию."""
|
||||
|
||||
result = await db.execute(select(Project).where(Project.id == data.project_id))
|
||||
project = result.scalar_one_or_none()
|
||||
|
||||
if not project:
|
||||
raise HTTPException(status_code=404, detail="Проект не найден")
|
||||
|
||||
transaction = EscrowTransaction(
|
||||
project_id=data.project_id,
|
||||
client_id=data.client_id,
|
||||
freelancer_id=data.freelancer_id,
|
||||
amount=data.amount,
|
||||
status="pending",
|
||||
)
|
||||
db.add(transaction)
|
||||
await db.commit()
|
||||
await db.refresh(transaction)
|
||||
|
||||
return {
|
||||
"id": str(transaction.id),
|
||||
"status": transaction.status,
|
||||
"amount": float(transaction.amount),
|
||||
"payment_url": f"https://stripe.com/pay/{transaction.id}", # Stripe redirect
|
||||
}
|
||||
|
||||
|
||||
@router.post("/release")
|
||||
async def release_escrow(data: EscrowRelease, db: AsyncSession = Depends(get_db)):
|
||||
"""Освободить средства фрилансеру."""
|
||||
|
||||
result = await db.execute(select(EscrowTransaction).where(EscrowTransaction.id == data.transaction_id))
|
||||
transaction = result.scalar_one_or_none()
|
||||
|
||||
if not transaction or transaction.status != "locked":
|
||||
raise HTTPException(status_code=400, detail="Транзакция не может быть разблокирована")
|
||||
|
||||
# Комиссия платформы 10%
|
||||
commission = transaction.amount * 0.10
|
||||
freelancer_amount = transaction.amount - commission
|
||||
|
||||
transaction.status = "released"
|
||||
await db.commit()
|
||||
|
||||
return {
|
||||
"id": str(transaction.id),
|
||||
"status": "released",
|
||||
"freelancer_payout": float(freelancer_amount),
|
||||
"platform_commission": float(commission),
|
||||
}
|
||||
|
||||
|
||||
@router.post("/dispute")
|
||||
async def dispute_escrow(transaction_id: str, db: AsyncSession = Depends(get_db)):
|
||||
"""Открыть спор по escrow."""
|
||||
|
||||
result = await db.execute(select(EscrowTransaction).where(EscrowTransaction.id == transaction_id))
|
||||
transaction = result.scalar_one_or_none()
|
||||
|
||||
if not transaction:
|
||||
raise HTTPException(status_code=404, detail="Транзакция не найдена")
|
||||
|
||||
transaction.status = "disputed"
|
||||
await db.commit()
|
||||
|
||||
return {"status": "dispute_opened", "transaction_id": str(transaction.id)}
|
||||
@@ -0,0 +1,122 @@
|
||||
"""Endpoints для проектов."""
|
||||
|
||||
import logging
|
||||
from typing import Any
|
||||
|
||||
from fastapi import APIRouter, Depends, HTTPException, Query
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
from sqlalchemy import select, desc
|
||||
|
||||
from app.core.database import get_db
|
||||
from app.core.security import get_current_user
|
||||
from app.schemas.project import ProjectCreate, ProjectUpdate, ProjectResponse
|
||||
from app.models.user import User
|
||||
from app.models.project import Project
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
router = APIRouter(prefix="/api/projects", tags=["projects"])
|
||||
|
||||
|
||||
@router.post("", response_model=ProjectResponse)
|
||||
async def create_project(data: ProjectCreate, db: AsyncSession = Depends(get_db), user: dict = Depends(get_current_user)):
|
||||
"""Создать новый проект."""
|
||||
|
||||
project = Project(
|
||||
client_id=user["id"],
|
||||
title=data.title,
|
||||
description=data.description,
|
||||
category=data.category,
|
||||
required_skills=data.required_skills,
|
||||
budget_min=data.budget_min,
|
||||
budget_max=data.budget_max,
|
||||
deadline=data.deadline,
|
||||
)
|
||||
db.add(project)
|
||||
await db.commit()
|
||||
await db.refresh(project)
|
||||
|
||||
return ProjectResponse(
|
||||
id=str(project.id),
|
||||
title=project.title,
|
||||
description=project.description,
|
||||
category=project.category,
|
||||
required_skills=project.required_skills,
|
||||
budget_min=float(project.budget_min) if project.budget_min else None,
|
||||
budget_max=float(project.budget_max) if project.budget_max else None,
|
||||
status=project.status,
|
||||
deadline=str(project.deadline) if project.deadline else None,
|
||||
created_at=str(project.created_at),
|
||||
updated_at=str(project.updated_at),
|
||||
)
|
||||
|
||||
|
||||
@router.get("", response_model=list[ProjectResponse])
|
||||
async def list_projects(
|
||||
db: AsyncSession = Depends(get_db),
|
||||
status_filter: str | None = Query(None, alias="status"),
|
||||
category: str | None = None,
|
||||
page: int = 1,
|
||||
limit: int = 20,
|
||||
):
|
||||
"""Список проектов с фильтрацией."""
|
||||
|
||||
stmt = select(Project)
|
||||
|
||||
if status_filter:
|
||||
stmt = stmt.where(Project.status == status_filter)
|
||||
if category:
|
||||
stmt = stmt.where(Project.category == category)
|
||||
|
||||
stmt = stmt.order_by(desc(Project.created_at)).offset((page - 1) * limit).limit(limit)
|
||||
|
||||
result = await db.execute(stmt)
|
||||
projects = result.scalars().all()
|
||||
|
||||
return [ProjectResponse(
|
||||
id=str(p.id), title=p.title, description=p.description, category=p.category,
|
||||
required_skills=p.required_skills, budget_min=float(p.budget_min) if p.budget_min else None,
|
||||
budget_max=float(p.budget_max) if p.budget_max else None, status=p.status,
|
||||
deadline=str(p.deadline) if p.deadline else None, created_at=str(p.created_at), updated_at=str(p.updated_at),
|
||||
) for p in projects]
|
||||
|
||||
|
||||
@router.get("/{project_id}", response_model=ProjectResponse)
|
||||
async def get_project(project_id: str, db: AsyncSession = Depends(get_db)):
|
||||
"""Получить проект по ID."""
|
||||
|
||||
result = await db.execute(select(Project).where(Project.id == project_id))
|
||||
project = result.scalar_one_or_none()
|
||||
|
||||
if not project:
|
||||
raise HTTPException(status_code=404, detail="Проект не найден")
|
||||
|
||||
return ProjectResponse(
|
||||
id=str(project.id), title=project.title, description=project.description, category=project.category,
|
||||
required_skills=project.required_skills, budget_min=float(project.budget_min) if project.budget_min else None,
|
||||
budget_max=float(project.budget_max) if project.budget_max else None, status=project.status,
|
||||
deadline=str(project.deadline) if project.deadline else None, created_at=str(project.created_at), updated_at=str(project.updated_at),
|
||||
)
|
||||
|
||||
|
||||
@router.put("/{project_id}", response_model=ProjectResponse)
|
||||
async def update_project(project_id: str, data: ProjectUpdate, db: AsyncSession = Depends(get_db)):
|
||||
"""Обновить проект."""
|
||||
|
||||
result = await db.execute(select(Project).where(Project.id == project_id))
|
||||
project = result.scalar_one_or_none()
|
||||
|
||||
if not project:
|
||||
raise HTTPException(status_code=404, detail="Проект не найден")
|
||||
|
||||
for field, value in data.model_dump(exclude_unset=True).items():
|
||||
setattr(project, field, value)
|
||||
|
||||
await db.commit()
|
||||
await db.refresh(project)
|
||||
|
||||
return ProjectResponse(
|
||||
id=str(project.id), title=project.title, description=project.description, category=project.category,
|
||||
required_skills=project.required_skills, budget_min=float(project.budget_min) if project.budget_min else None,
|
||||
budget_max=float(project.budget_max) if project.budget_max else None, status=project.status,
|
||||
deadline=str(project.deadline) if project.deadline else None, created_at=str(project.created_at), updated_at=str(project.updated_at),
|
||||
)
|
||||
@@ -0,0 +1,73 @@
|
||||
"""Endpoints для заявок фрилансеров."""
|
||||
|
||||
from fastapi import APIRouter, Depends, HTTPException
|
||||
from sqlalchemy.ext.asyncio import AsyncSession
|
||||
from sqlalchemy import select, desc
|
||||
|
||||
from app.core.database import get_db
|
||||
from app.core.security import get_current_user
|
||||
from app.schemas.proposal import ProposalCreate, ProposalResponse
|
||||
from app.models.project import Project
|
||||
from app.models.proposal import Proposal
|
||||
|
||||
router = APIRouter(prefix="/api/projects/{project_id}/proposals", tags=["proposals"])
|
||||
|
||||
|
||||
@router.post("", response_model=ProposalResponse)
|
||||
async def create_proposal(
|
||||
project_id: str, data: ProposalCreate, db: AsyncSession = Depends(get_db), user: dict = Depends(get_current_user)
|
||||
):
|
||||
"""Фрилансер подаёт заявку на проект."""
|
||||
|
||||
result = await db.execute(select(Project).where(Project.id == project_id))
|
||||
project = result.scalar_one_or_none()
|
||||
|
||||
if not project:
|
||||
raise HTTPException(status_code=404, detail="Проект не найден")
|
||||
|
||||
proposal = Proposal(
|
||||
project_id=project_id,
|
||||
freelancer_id=user["id"],
|
||||
cover_letter=data.cover_letter,
|
||||
proposed_price=data.proposed_price,
|
||||
estimated_days=data.estimated_days,
|
||||
)
|
||||
db.add(proposal)
|
||||
await db.commit()
|
||||
await db.refresh(proposal)
|
||||
|
||||
return ProposalResponse(
|
||||
id=str(proposal.id), project_id=project_id, freelancer_id=user["id"],
|
||||
cover_letter=proposal.cover_letter, proposed_price=float(proposal.proposed_price) if proposal.proposed_price else None,
|
||||
estimated_days=proposal.estimated_days, status=proposal.status, created_at=str(proposal.created_at),
|
||||
)
|
||||
|
||||
|
||||
@router.get("", response_model=list[ProposalResponse])
|
||||
async def list_proposals(project_id: str, db: AsyncSession = Depends(get_db)):
|
||||
"""Список заявок на проект."""
|
||||
|
||||
result = await db.execute(select(Proposal).where(Proposal.project_id == project_id).order_by(desc(Proposal.created_at)))
|
||||
proposals = result.scalars().all()
|
||||
|
||||
return [ProposalResponse(
|
||||
id=str(p.id), project_id=p.project_id, freelancer_id=p.freelancer_id, cover_letter=p.cover_letter,
|
||||
proposed_price=float(p.proposed_price) if p.proposed_price else None, estimated_days=p.estimated_days,
|
||||
status=p.status, created_at=str(p.created_at),
|
||||
) for p in proposals]
|
||||
|
||||
|
||||
@router.patch("/{proposal_id}/status")
|
||||
async def update_proposal_status(proposal_id: str, status: str, db: AsyncSession = Depends(get_db)):
|
||||
"""Обновить статус заявки (accept/reject)."""
|
||||
|
||||
result = await db.execute(select(Proposal).where(Proposal.id == proposal_id))
|
||||
proposal = result.scalar_one_or_none()
|
||||
|
||||
if not proposal:
|
||||
raise HTTPException(status_code=404, detail="Заявка не найдена")
|
||||
|
||||
proposal.status = status
|
||||
await db.commit()
|
||||
|
||||
return {"status": "updated", "proposal_id": str(proposal.id)}
|
||||
Reference in New Issue
Block a user