- app/api/projects, link/route: Cookie + Bearer 토큰 이중 인증 지원 (E2E 테스트 대응) - app/mypage: 로또 기록 탭 제거, 구독 빈 상태 프롬프트 서비스로 변경 - scripts/test-flow.mjs: 견적서 발송→연결→마일스톤 진행 E2E 테스트 스크립트 - supabase/migrations/003: quotes RLS 비활성화 (관리자 서버 전용 접근) - marketing/kmong-images: 크몽 서비스 A 상세 이미지 5장 (HTML 스크린샷용) Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
67 lines
2.2 KiB
TypeScript
67 lines
2.2 KiB
TypeScript
import { NextResponse } from 'next/server';
|
|
import { createAdminClient } from '@/lib/supabase/admin';
|
|
import { createClient } from '@/lib/supabase/server';
|
|
import { createClient as createSupabaseClient } from '@supabase/supabase-js';
|
|
|
|
export const runtime = 'nodejs';
|
|
|
|
export async function GET(request: Request) {
|
|
// Cookie 기반 또는 Bearer 토큰 기반 인증 모두 지원
|
|
const authHeader = request.headers.get('authorization');
|
|
let user = null;
|
|
|
|
if (authHeader?.startsWith('Bearer ')) {
|
|
const token = authHeader.slice(7);
|
|
const client = createSupabaseClient(
|
|
process.env.NEXT_PUBLIC_SUPABASE_URL!,
|
|
process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY!,
|
|
);
|
|
const { data } = await client.auth.getUser(token);
|
|
user = data?.user ?? null;
|
|
} else {
|
|
const supabase = await createClient();
|
|
const { data } = await supabase.auth.getUser();
|
|
user = data?.user ?? null;
|
|
}
|
|
if (!user) return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
|
|
|
const admin = createAdminClient();
|
|
|
|
const { data: quotes, error: qErr } = await admin
|
|
.from('quotes')
|
|
.select('id, title, status, items, created_at')
|
|
.eq('user_id', user.id)
|
|
.in('status', ['sent', 'accepted', 'in_progress', 'completed', 'delivered'])
|
|
.order('created_at', { ascending: false });
|
|
|
|
if (qErr) return NextResponse.json({ error: qErr.message }, { status: 500 });
|
|
if (!quotes?.length) return NextResponse.json({ projects: [] });
|
|
|
|
const quoteIds = quotes.map((q) => q.id);
|
|
|
|
const { data: milestones } = await admin
|
|
.from('project_milestones')
|
|
.select('*')
|
|
.in('quote_id', quoteIds)
|
|
.order('step_number', { ascending: true });
|
|
|
|
const projects = quotes.map((q) => ({
|
|
id: q.id,
|
|
title: q.title,
|
|
status: q.status,
|
|
total: Array.isArray(q.items)
|
|
? q.items.reduce(
|
|
(s: number, i: { unitPrice?: number; quantity?: number }) =>
|
|
s + ((i.unitPrice ?? 0) * (i.quantity ?? 1)),
|
|
0
|
|
)
|
|
: 0,
|
|
created_at: q.created_at,
|
|
milestones: (milestones ?? [])
|
|
.filter((m) => m.quote_id === q.id)
|
|
.sort((a, b) => a.step_number - b.step_number),
|
|
}));
|
|
|
|
return NextResponse.json({ projects });
|
|
}
|