Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com> Claude-Session: https://claude.ai/code/session_01AAtcmKKtqDUe4NyVgy1aLQ
24 KiB
Phase 1 외주 코어 Implementation Plan
For agentic workers: REQUIRED SUB-SKILL: Use superpowers:subagent-driven-development (recommended) or superpowers:executing-plans to implement this plan task-by-task. Steps use checkbox (
- [ ]) syntax for tracking.
Goal: 발주서 표면화(mypage·admin) + /showcase 제작 사례 허브 + admin 광고 관리 + packs 페이지 정리로 외주 코어를 정비한다.
Architecture: 8개 태스크, 서로 파일 비중첩. 신규 데이터 단일 소스는 lib/showcase-samples.ts(데모 메타)와 ad_channels 테이블(광고 채널). mypage는 기존 의뢰 카드를 유지한 채 상단에 발주·진행 섹션을 추가(정보 손실 없음).
Tech Stack: Next.js 16 (App Router, TS), Tailwind v4 (--jsm-* 토큰), Supabase, vitest
Spec: docs/superpowers/specs/2026-07-02-phase1-outsourcing-core-design.md
Global Constraints
- 디자인 가드레일: gradient / blur / 보라(violet/purple) / 이모지 금지,
--jsm-*토큰만 (신규 공개 페이지) - 카피 가드레일: "대기업 N년차" 류 자격 어필 금지 — "실서비스 직접 운영" 실증 서술만
- next.config.ts redirects() 수정 금지
/api/admin/packs·/api/admin/packs/upload-url은 삭제 금지 (products·mypage가 공유)app/work/website/samples/**데모 8종 수정 금지 (링크만 연결)- 기존 supabase/migrations/ 파일 삭제·수정 금지, 신규만 추가
- 커밋은 스코프 파일만 스테이징 —
git add -A·git commit -a금지 - 각 Task 종료 시
npm test전체 통과 +npm run build성공 후 커밋 - 커밋 트레일러:
Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
확인된 기존 계약 (구현 시 그대로 사용)
GET /api/projects→{ projects: [{ id, title, status, total, created_at, milestones: [{ quote_id, step_number, title, status: 'pending'|'in_progress'|'completed', ... }] }] }— quotes.status ∈ sent/accepted/in_progress/completed/delivered 필터POST /api/projects/link— body{ token: string }→ 200{ success: true, quoteId, alreadyLinked? }/ 4xx·5xx{ error: string }- admin API 인증 패턴:
cookies()→admin_token→verifyAdminTokenNode(token)실패 시 401 →createAdminClient()(참고:app/api/admin/services/route.ts) - mypage requests 탭:
contact_requests기반 카드 리스트(변수명orders), 탭 keyrequests— 라벨만 변경, 기존 카드 유지
Task 1: showcase 데이터 모듈 (TDD)
Files:
- Create:
lib/showcase-samples.ts - Test:
lib/__tests__/showcase-samples.test.ts
Interfaces:
-
Produces:
SHOWCASE_SAMPLES: ShowcaseSample[],type ShowcaseSample = { slug: string; title: string; description: string; tags: string[] }— Task 2가 import -
Step 1: 실패 테스트 작성
lib/__tests__/showcase-samples.test.ts:
import { describe, it, expect } from 'vitest';
import { SHOWCASE_SAMPLES } from '../showcase-samples';
const EXPECTED_SLUGS = [
'bakery', 'corporate', 'dashboard', 'game',
'interior', 'portfolio', 'reading', 'shopping',
];
describe('SHOWCASE_SAMPLES', () => {
it('데모 8종의 slug가 정확히 존재한다', () => {
expect(SHOWCASE_SAMPLES.map((s) => s.slug).sort()).toEqual([...EXPECTED_SLUGS].sort());
});
it('모든 항목에 title/description/tags가 채워져 있다', () => {
for (const s of SHOWCASE_SAMPLES) {
expect(s.title.length).toBeGreaterThan(0);
expect(s.description.length).toBeGreaterThan(0);
expect(s.tags.length).toBeGreaterThan(0);
}
});
it('demo 경로는 /work/website/samples/[slug] 형식이다', () => {
for (const s of SHOWCASE_SAMPLES) {
expect(`/work/website/samples/${s.slug}`).toMatch(/^\/work\/website\/samples\/[a-z]+$/);
}
});
});
-
Step 2: 실패 확인 — Run:
npx vitest run lib/__tests__/showcase-samples.test.ts/ Expected: FAIL (모듈 없음) -
Step 3: 구현
lib/showcase-samples.ts:
/** /showcase 제작 사례 허브의 데모 카드 단일 소스. 데모 실체는 app/work/website/samples/[slug]. */
export type ShowcaseSample = {
slug: string;
title: string;
description: string;
tags: string[];
};
export const SHOWCASE_SAMPLES: ShowcaseSample[] = [
{ slug: 'corporate', title: '기업 홈페이지', description: 'IT 기업 소개·서비스·문의까지 담은 반응형 공식 홈페이지.', tags: ['기업', '반응형', 'SEO'] },
{ slug: 'shopping', title: '쇼핑몰', description: '상품 목록·상세·장바구니 흐름을 갖춘 커머스 데모.', tags: ['커머스', '결제 흐름'] },
{ slug: 'dashboard', title: 'SaaS 대시보드', description: '지표 카드·차트·테이블로 구성한 관리자 대시보드.', tags: ['대시보드', '차트'] },
{ slug: 'bakery', title: '베이커리 브랜드', description: '메뉴·매장·브랜드 스토리를 담은 로컬 비즈니스 사이트.', tags: ['브랜드', '로컬'] },
{ slug: 'interior', title: '인테리어 포트폴리오', description: '시공 사례 중심의 갤러리형 인테리어 회사 사이트.', tags: ['갤러리', '포트폴리오'] },
{ slug: 'portfolio', title: '디자이너 포트폴리오', description: '작업물·경력·연락처를 담은 개인 포트폴리오.', tags: ['개인', '포트폴리오'] },
{ slug: 'game', title: '게임 프로모션', description: '출시 게임을 소개하는 인터랙티브 프로모션 페이지.', tags: ['프로모션', '랜딩'] },
{ slug: 'reading', title: '도서 콘텐츠', description: '책 소개·리뷰 중심의 콘텐츠 페이지.', tags: ['콘텐츠', '블로그'] },
];
- Step 4: 통과 확인 — Run:
npx vitest run lib/__tests__/showcase-samples.test.ts/ Expected: 3 tests PASS - Step 5: 커밋 —
git add lib/showcase-samples.ts lib/__tests__/showcase-samples.test.ts && git commit -m "feat(phase1): showcase 데모 메타 단일 소스 + 무결성 테스트"
Task 2: /showcase 페이지 + TopNav + robots
Files:
- Create:
app/showcase/page.tsx - Modify:
app/components/TopNav.tsx:9-12(NAV_LINKS에 항목 추가) - Modify:
app/robots.ts:9(죽은 경로 3개 제거)
Interfaces:
-
Consumes: Task 1의
SHOWCASE_SAMPLES,ShowcaseSample -
Produces: 공개 라우트
/showcase -
Step 1: /showcase 페이지 구현
app/showcase/page.tsx — 서버 컴포넌트. 구조(스타일은 기존 app/products/page.tsx의 카드·섹션 패턴을 Read 후 동일 관용구로):
import type { Metadata } from 'next';
import Link from 'next/link';
import { SHOWCASE_SAMPLES } from '@/lib/showcase-samples';
export const metadata: Metadata = {
title: '제작 사례 | 쟁승메이드',
description: '직접 설계·개발한 웹사이트 데모와 실서비스 운영 사례.',
};
// 실운영 서비스(개인 NAS 실서비스 — 외부 링크 없음, 실증 서술만)
const LIVE_SERVICES = [
{ title: '로또 분석 랩', desc: '회차 수집·통계 분석·리포트 자동 생성까지 무인 운영' },
{ title: '주식 자동매매 대시보드', desc: '시세 수집·스크리너·자동 주문을 하나의 콘솔로 운영' },
{ title: 'AI 미디어 파이프라인', desc: '음악·영상·이미지 생성 워커를 큐 기반으로 상시 가동' },
{ title: '여행 사진 갤러리', desc: '수천 장 사진의 지역 분류·썸네일·지도 탐색 자동화' },
];
export default function ShowcasePage() {
return (
<div>
{/* Hero: h1 "제작 사례" + 부제 "실서비스를 직접 만들고 운영하며 검증한 방식 그대로 만듭니다." */}
{/* 섹션 1: 웹사이트 데모 — SHOWCASE_SAMPLES.map 카드 그리드(md:grid-cols-2 lg:grid-cols-4)
카드: title, description, tags(작은 pill), "데모 보기" 링크
<Link href={`/work/website/samples/${s.slug}`} target="_blank" rel="noopener noreferrer"> */}
{/* 섹션 2: 실서비스 운영 — LIVE_SERVICES 카드(링크 없음) + 하단 캡션
"위 서비스들은 개인 인프라에서 상시 운영 중인 실제 서비스입니다." */}
{/* CTA: "이런 걸 만들어 드립니다" → /outsourcing#contact 버튼(--jsm-accent) */}
</div>
);
}
주석 블록은 실제 JSX로 구현한다(플레이스홀더로 남기지 말 것). 색상은 var(--jsm-*) 토큰/기존 Tailwind 클래스 관용구만.
- Step 2: TopNav 링크 추가
app/components/TopNav.tsx의 NAV_LINKS 배열:
const NAV_LINKS = [
{ href: '/outsourcing', label: '외주 개발' },
{ href: '/products', label: '소프트웨어' },
{ href: '/showcase', label: '제작 사례' },
];
- Step 3: robots.ts 정리
disallow: ['/admin/', '/api/', '/mypage/', '/portfolio/'],
(죽은 경로 /payment/·/freelance·/services/website 제거. /showcase는 allow '/'에 포함되므로 추가 불필요)
- Step 4: 검증 —
npm test && npm run build/ Expected: PASS + 빌드 라우트에/showcase등장. 가드레일 grep:grep -nE "gradient|violet|purple|blur" app/showcase/page.tsx→ 0건 - Step 5: 커밋 —
git add app/showcase/page.tsx app/components/TopNav.tsx app/robots.ts && git commit -m "feat(phase1): /showcase 제작 사례 허브 + TopNav 제작 사례 + robots 죽은 경로 정리"
Task 3: mypage 발주·진행 섹션
Files:
- Modify:
app/mypage/page.tsx(탭 라벨 263행, requests 탭 렌더 518행~)
Interfaces:
-
Consumes:
GET /api/projects,POST /api/projects/link(위 "확인된 기존 계약") -
Produces: 없음 (말단 UI)
-
Step 1: 상태·타입·페치 추가
app/mypage/page.tsx에 (기존 state들 옆):
type ProjectMilestone = { quote_id: string; step_number: number; title: string; status: 'pending' | 'in_progress' | 'completed' };
type Project = { id: string; title: string; status: string; total: number; created_at: string; milestones: ProjectMilestone[] };
const QUOTE_STATUS_LABELS: Record<string, string> = {
sent: '견적 발송', accepted: '발주 확정', in_progress: '진행중', completed: '완료', delivered: '납품 완료',
};
const [projects, setProjects] = useState<Project[]>([]);
const [linkCode, setLinkCode] = useState('');
const [linkMsg, setLinkMsg] = useState<string | null>(null);
const [linking, setLinking] = useState(false);
const loadProjects = useCallback(async () => {
try {
const res = await fetch('/api/projects');
if (!res.ok) return;
const d = await res.json();
setProjects(d.projects ?? []);
} catch { /* 미로그인/네트워크 — 무시 */ }
}, []);
// 기존 초기 로드 useEffect에 loadProjects() 추가
const handleLink = async () => {
if (!linkCode.trim() || linking) return;
setLinking(true); setLinkMsg(null);
try {
const res = await fetch('/api/projects/link', {
method: 'POST', headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ token: linkCode.trim() }),
});
const d = await res.json();
if (!res.ok) { setLinkMsg(d.error ?? '연결에 실패했습니다.'); return; }
setLinkMsg(d.alreadyLinked ? '이미 연결된 견적서입니다.' : '견적서가 연결되었습니다.');
setLinkCode('');
await loadProjects();
} catch { setLinkMsg('연결에 실패했습니다. 다시 시도해주세요.'); }
finally { setLinking(false); }
};
-
Step 2: 탭 라벨 변경 — 263행:
label: '내 의뢰'→label: '발주·진행'(keyrequests유지) -
Step 3: requests 탭 상단에 발주·진행 섹션 추가
{tab === 'requests' && ( 블록 최상단(기존 의뢰 카드 리스트는 그대로 아래 유지):
{/* 발주·진행 (quotes 기반 — 견적 수락 시 발주서로 전환) */}
<section className="mb-8">
<div className="flex items-center justify-between mb-3">
<h2 className="text-base font-bold">발주·진행</h2>
{/* 견적코드 연결: 접이식 — input + 연결 버튼 + linkMsg 표시 */}
</div>
{projects.length === 0 ? (
<p className="text-sm text-slate-500">진행 중인 발주가 없습니다. 견적서 코드를 입력해 연결하거나 새로 의뢰해 보세요.</p>
) : (
<div className="space-y-3">
{projects.map((p) => (
<div key={p.id} className="rounded-lg border border-[var(--jsm-line)] bg-[var(--jsm-surface)] p-4">
{/* 헤더: p.title + 상태 뱃지(QUOTE_STATUS_LABELS[p.status] ?? p.status)
+ accepted/in_progress/completed/delivered면 "발주서" 뱃지 병기 */}
{/* 총액: p.total.toLocaleString('ko-KR') + '원' */}
{/* 마일스톤 타임라인: p.milestones step_number 순.
completed=accent 채움, in_progress=accent 테두리, pending=회색.
각 스텝: 번호 원 + title */}
</div>
))}
</div>
)}
</section>
주석 블록은 실제 JSX로 구현. 견적코드 폼: <input value={linkCode} ...> + <button onClick={handleLink} disabled={linking}>연결</button> + {linkMsg && <p ...>{linkMsg}</p>}. 스타일은 파일 내 기존 폼·뱃지 관용구를 따른다.
- Step 4: 검증 —
npm test && npm run buildPASS. 수동 확인 항목(보고서에 기재): 로그인 후 /mypage?tab=requests에서 발주 섹션·견적코드 폼 렌더 - Step 5: 커밋 —
git add app/mypage/page.tsx && git commit -m "feat(phase1): mypage 발주·진행 섹션 — projects API 배선 + 견적코드 연결"
Task 4: admin/quotes 발주 뱃지 + 상태 확장
Files:
- Modify:
app/admin/quotes/page.tsx:12,19(status 타입·STATUS 맵)
Interfaces:
-
Consumes: quotes.status 값 집합 sent/accepted/in_progress/completed/delivered (+draft/rejected)
-
Produces: 없음
-
Step 1: 타입·STATUS 맵 확장
12행 타입에 'in_progress' | 'completed' | 'delivered' 추가. 19행 STATUS 맵(기존 draft/sent/accepted/rejected 스타일 관용구 유지)에:
in_progress: { label: '진행중 · 발주', ... },
completed: { label: '완료 · 발주', ... },
delivered: { label: '납품 완료 · 발주', ... },
그리고 기존 accepted 라벨을 '수락 · 발주'로 변경. 색상 값은 파일 내 기존 STATUS 항목의 색 체계에서 선택(초록 계열=진행/완료, 기존 관용구 확인 후).
- Step 2: 검증 —
npm test && npm run buildPASS - Step 3: 커밋 —
git add app/admin/quotes/page.tsx && git commit -m "feat(phase1): admin 견적 리스트 발주 뱃지 + 진행 상태 라벨 확장"
Task 5: ad_channels 마이그레이션 + CRUD API
Files:
- Create:
supabase/migrations/2026-07-02-phase1-ad-channels.sql - Create:
app/api/admin/ad-channels/route.ts - Create:
app/api/admin/ad-channels/[id]/route.ts
Interfaces:
-
Produces:
GET /api/admin/ad-channels→{ channels: AdChannel[] },POSTbody{ name, url?, memo? }→{ channel },PATCH /api/admin/ad-channels/[id]body{ name?, url?, status?, memo? }→{ success: true },DELETE→{ success: true }.AdChannel = { id: string; name: string; url: string | null; status: 'active'|'paused'; memo: string | null; created_at: string; updated_at: string }— Task 6이 소비 -
Step 1: 마이그레이션 파일 — 스펙 §WS3의 SQL 그대로:
CREATE TABLE IF NOT EXISTS ad_channels (
id uuid PRIMARY KEY DEFAULT gen_random_uuid(),
name text NOT NULL,
url text,
status text NOT NULL DEFAULT 'active' CHECK (status IN ('active','paused')),
memo text,
created_at timestamptz NOT NULL DEFAULT now(),
updated_at timestamptz NOT NULL DEFAULT now()
);
ALTER TABLE ad_channels ENABLE ROW LEVEL SECURITY;
-- service_role(관리자 API)만 접근 — 별도 policy 없음(기본 거부)
- Step 2: 목록/생성 API
app/api/admin/ad-channels/route.ts (인증 패턴은 app/api/admin/services/route.ts를 Read 후 동일하게):
import { NextResponse } from 'next/server';
import { cookies } from 'next/headers';
import { createAdminClient } from '@/lib/supabase/admin';
import { verifyAdminTokenNode } from '@/lib/admin-auth';
export const runtime = 'nodejs';
async function requireAdmin() {
const cookieStore = await cookies();
const token = cookieStore.get('admin_token')?.value;
if (!token || !verifyAdminTokenNode(token)) return null;
return createAdminClient();
}
export async function GET() {
const supabase = await requireAdmin();
if (!supabase) return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
const { data, error } = await supabase.from('ad_channels').select('*').order('created_at', { ascending: false });
if (error) return NextResponse.json({ error: error.message }, { status: 500 });
return NextResponse.json({ channels: data ?? [] });
}
export async function POST(request: Request) {
const supabase = await requireAdmin();
if (!supabase) return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
const body = await request.json();
const name = (body.name as string | undefined)?.trim();
if (!name) return NextResponse.json({ error: '채널명을 입력해주세요.' }, { status: 400 });
const { data, error } = await supabase
.from('ad_channels')
.insert({ name, url: body.url?.trim() || null, memo: body.memo?.trim() || null })
.select().single();
if (error) return NextResponse.json({ error: error.message }, { status: 500 });
return NextResponse.json({ channel: data });
}
- Step 3: 수정/삭제 API
app/api/admin/ad-channels/[id]/route.ts:
import { NextResponse } from 'next/server';
import { cookies } from 'next/headers';
import { createAdminClient } from '@/lib/supabase/admin';
import { verifyAdminTokenNode } from '@/lib/admin-auth';
export const runtime = 'nodejs';
async function requireAdmin() {
const cookieStore = await cookies();
const token = cookieStore.get('admin_token')?.value;
if (!token || !verifyAdminTokenNode(token)) return null;
return createAdminClient();
}
export async function PATCH(request: Request, { params }: { params: Promise<{ id: string }> }) {
const supabase = await requireAdmin();
if (!supabase) return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
const { id } = await params;
const body = await request.json();
const patch: Record<string, unknown> = { updated_at: new Date().toISOString() };
if (typeof body.name === 'string' && body.name.trim()) patch.name = body.name.trim();
if ('url' in body) patch.url = body.url?.trim() || null;
if ('memo' in body) patch.memo = body.memo?.trim() || null;
if (body.status === 'active' || body.status === 'paused') patch.status = body.status;
const { error } = await supabase.from('ad_channels').update(patch).eq('id', id);
if (error) return NextResponse.json({ error: error.message }, { status: 500 });
return NextResponse.json({ success: true });
}
export async function DELETE(_request: Request, { params }: { params: Promise<{ id: string }> }) {
const supabase = await requireAdmin();
if (!supabase) return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
const { id } = await params;
const { error } = await supabase.from('ad_channels').delete().eq('id', id);
if (error) return NextResponse.json({ error: error.message }, { status: 500 });
return NextResponse.json({ success: true });
}
(주의: Next 16에서 route context의 params는 Promise — 기존 app/api/admin/quotes/[id]/route.ts의 시그니처를 Read 후 동일 관용구로 맞출 것)
- Step 4: 검증 —
npm test && npm run buildPASS (라우트 2개 빌드 등장) - Step 5: 커밋 —
git add supabase/migrations/2026-07-02-phase1-ad-channels.sql app/api/admin/ad-channels && git commit -m "feat(phase1): ad_channels 테이블 + admin CRUD API"
Task 6: admin 광고 관리 페이지 재편 + 사이드바
Files:
- Modify:
app/admin/marketing/page.tsx(2탭 재구성 — 기존 에셋 UI는 탭 안으로 이동) - Modify:
app/admin/components/AdminSidebar.tsx:90-91(마케팅 에셋→광고 관리)
Interfaces:
-
Consumes: Task 5의 ad-channels API 계약 (
AdChannel타입 포함) -
Produces: 없음
-
Step 1: 페이지 2탭 재구성
app/admin/marketing/page.tsx: 파일 상단에 type AdminTab = 'channels' | 'assets' state 추가, 페이지 타이틀 광고 관리. 기존 에셋 렌더 전체(통계 카드·그리드·모달)를 {tab === 'assets' && (...)}로 감싸고, channels 탭(기본값)에 채널 CRUD UI 구현:
// channels 탭 구성:
// - 상단: 신규 채널 추가 폼 (name 필수, url, memo) → POST /api/admin/ad-channels
// - 테이블: 채널명 | URL(외부 링크, 없으면 '-') | 상태 토글(active↔paused, PATCH) | 메모(인라인 편집 또는 표시) | 등록일 | 삭제 버튼(confirm 후 DELETE)
// - fetch는 useEffect 초기 1회 GET + 각 뮤테이션 후 재조회
// - 에러는 상단 배너 텍스트로 표시
주석은 실제 구현으로. 스타일은 기존 admin 페이지들(orders/products)의 테이블·버튼 관용구를 Read 후 동일하게.
- Step 2: 사이드바 라벨 —
label: '마케팅 에셋'→label: '광고 관리'(href 유지) - Step 3: 검증 —
npm test && npm run buildPASS - Step 4: 커밋 —
git add app/admin/marketing/page.tsx app/admin/components/AdminSidebar.tsx && git commit -m "feat(phase1): admin 광고 관리 — 채널·캠페인 CRUD 탭 + 에셋 탭 재편"
Task 7: admin/packs 페이지 제거
Files:
- Delete:
app/admin/packs/page.tsx(디렉토리째) - Modify:
app/admin/components/AdminSidebar.tsx:80-81(팩 자료메뉴 항목 객체 제거)
Interfaces:
-
Consumes: 없음
-
Produces: 없음.
/api/admin/packs*는 유지(products·mypage 공유 — Global Constraints) -
Step 1: 삭제 —
git rm -r app/admin/packs+ AdminSidebar에서 href/admin/packs객체(svg 포함) 제거 -
Step 2: 잔존 참조 확인 —
grep -rn "admin/packs" app lib→ 허용되는 매치:app/admin/products/page.tsx·app/mypage의/api/admin/packs(API 호출)뿐. 페이지 라우트/admin/packshref 참조 0건 -
Step 3: 검증 —
npm test && npm run buildPASS -
Step 4: 커밋 —
git add -u app/admin && git commit -m "chore(phase1): admin/packs 레거시 페이지 제거 (API는 products·mypage 공유로 유지)"
Task 8: CLAUDE.md 갱신 + 이메일 경로 점검 + 최종 검증
Files:
- Modify:
CLAUDE.md(IA 표·admin 서술)
Interfaces:
-
Consumes: Task 1~7 완료 상태
-
Produces: 문서 정합 + 검증 보고
-
Step 1: CLAUDE.md 갱신
-
핵심 IA 표에
| /showcase | 제작 사례 — 웹 데모 8종 + 실서비스 운영 사례 |추가 -
/mypage행:4탭: 프로필 / 발주·진행(발주서·마일스톤·견적코드 연결) / 내 제품(다운로드) / 주문 내역으로 갱신 -
admin 서술:
packs제거,marketing→광고 관리(채널 CRUD + 에셋)반영 -
파일 구조 트리에
showcase/page.tsx,api/admin/ad-channels/추가 -
Step 2: 이메일 경로 점검 (변경 없음, 검증만)
lib/request-emails.ts·lib/order-emails.ts의 export 함수들이 각각app/api/contact/route.ts·app/api/admin/quotes/[id]/send/route.ts·app/api/quote/[token]/route.ts·app/api/orders/route.ts·app/api/admin/orders/route.ts에서 여전히 import·호출되는지 grep으로 확인하고 결과를 보고서에 기재 (Phase 0 삭제가 메일 경로를 건드리지 않았음을 실증) -
Step 3: 최종 검증
npm test # showcase-samples 테스트 포함 전체 PASS
npm run build # /showcase 라우트 존재, /admin/packs 라우트 소멸
grep -nE "gradient|violet|purple|blur" app/showcase/page.tsx # 0건
-
Step 4: 커밋 —
git add CLAUDE.md && git commit -m "docs(phase1): CLAUDE.md — showcase·발주 탭·광고 관리·packs 정리 반영" -
Step 5: CEO 안내 (보고)
-
2026-07-02-phase1-ad-channels.sql을 클라우드 Supabase + NAS self-host 양쪽 적용 -
수동 확인 2종: /mypage 발주·진행 탭(견적코드 연결), /admin/marketing 채널 CRUD