Files
jaengseung-made/app/api/projects/link/route.ts
gahusb fe1e8ffcf0 fix: 외주 플랫폼 전환율 개선 + API 보안 정비 + 시크릿 노출 제거
[Backend API]
- contact/route: 문의 내역 contact_requests DB 저장 추가 (이메일+DB 병행)
- projects/route, link/route: 미사용 Bearer 토큰 인증 제거, Cookie 전용
- projects/route: DB 에러 메시지 클라이언트 노출 차단 (console.error로 전환)
- quote/[token]/route: valid_until 만료 검증 + expired 플래그 응답 추가

[Frontend UX]
- mypage: 로또 잔존 코드 완전 제거 (PLAN_LABELS, lotto_history 쿼리)
- mypage: 기본 탭 projects로 변경, 탭 순서 외주 고객 우선 재배치
- freelance: 포트폴리오 가격대 뱃지 추가, 각 항목 CTA 링크 추가
- freelance: 후기 섹션 하단 CTA 블록 추가

[견적서 페이지]
- quote/[token]/page: 만료 견적서 경고 배너 + 수락 버튼 숨김
- quote/layout: DashboardShell 없이 독립 렌더링

[보안]
- test-flow.mjs: 하드코딩 시크릿 → .env.test 환경변수 참조로 교체
- GitGuardian 3건 대응 (admin password, JWT, test password)

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-02 08:49:05 +09:00

49 lines
1.9 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) {
console.error('[Projects/Link] DB update error:', updateErr.message);
return NextResponse.json({ error: '견적서 연결에 실패했습니다. 다시 시도해주세요.' }, { status: 500 });
}
return NextResponse.json({ success: true, quoteId: quote.id });
}