[DB] - supabase/migrations/002_project_milestones.sql 추가 quotes.user_id 컬럼 + project_milestones 테이블 생성 SQL [API] - GET /api/projects — 로그인 사용자의 프로젝트+마일스톤 조회 - POST /api/projects/link — 견적서 토큰으로 계정에 프로젝트 연결 - GET/POST /api/admin/milestones — 관리자 마일스톤 목록/기본 7단계 초기화 - PATCH/DELETE /api/admin/milestones/[id] — 관리자 단계별 상태·메모 업데이트 [UI — 마이페이지] - '프로젝트 현황' 탭 신규 추가 (Tab type 확장) - 진행률 바, 단계별 타임라인, 개발자 메모 표시 - 견적서 코드 입력 → 계정 연결 폼 [UI — 관리자 견적서 편집] - '진행 단계' 탭 추가: 기본 7단계 초기화, 단계별 status/메모 편집 [마케팅 카피] - page.tsx PROMISES 4번째 추가: "진행 현황 마이페이지 실시간 확인" - freelance 보증 카드 5번째 추가: 실시간 진행 현황 (그리드 2×5) - services/website trust badge 5번째 추가 Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
46 lines
1.7 KiB
TypeScript
46 lines
1.7 KiB
TypeScript
import { NextResponse } from 'next/server';
|
|
import { createAdminClient } from '@/lib/supabase/admin';
|
|
import { createClient } from '@/lib/supabase/server';
|
|
|
|
export const runtime = 'nodejs';
|
|
|
|
export async function POST(request: Request) {
|
|
const supabase = await createClient();
|
|
const { data: { user } } = await supabase.auth.getUser();
|
|
if (!user) return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
|
|
|
const body = await request.json();
|
|
const token = (body.token as string | undefined)?.trim();
|
|
if (!token) return NextResponse.json({ error: '견적서 코드를 입력해주세요' }, { status: 400 });
|
|
|
|
const admin = createAdminClient();
|
|
|
|
const { data: quote, error } = await admin
|
|
.from('quotes')
|
|
.select('id, status, user_id, client_email')
|
|
.eq('public_token', token)
|
|
.single();
|
|
|
|
if (error || !quote) {
|
|
return NextResponse.json({ error: '견적서를 찾을 수 없습니다. 코드를 다시 확인해주세요.' }, { status: 404 });
|
|
}
|
|
if (quote.status === 'draft') {
|
|
return NextResponse.json({ error: '아직 발송되지 않은 견적서입니다.' }, { status: 400 });
|
|
}
|
|
if (quote.user_id && quote.user_id !== user.id) {
|
|
return NextResponse.json({ error: '이미 다른 계정에 연결된 견적서입니다.' }, { status: 400 });
|
|
}
|
|
if (quote.user_id === user.id) {
|
|
return NextResponse.json({ success: true, quoteId: quote.id, alreadyLinked: true });
|
|
}
|
|
|
|
const { error: updateErr } = await admin
|
|
.from('quotes')
|
|
.update({ user_id: user.id, updated_at: new Date().toISOString() })
|
|
.eq('id', quote.id);
|
|
|
|
if (updateErr) return NextResponse.json({ error: updateErr.message }, { status: 500 });
|
|
|
|
return NextResponse.json({ success: true, quoteId: quote.id });
|
|
}
|