diff --git a/docs/superpowers/plans/2026-06-02-insta-cardnews-upgrade.md b/docs/superpowers/plans/2026-06-02-insta-cardnews-upgrade.md new file mode 100644 index 0000000..9fcacf0 --- /dev/null +++ b/docs/superpowers/plans/2026-06-02-insta-cardnews-upgrade.md @@ -0,0 +1,408 @@ +# 인스타 카드뉴스 품질 고도화 + 업로드 친화 패키지 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:** 인스타 카드를 모던 미니멀 디자인 시스템으로 격상하고(렌더 견고화로 known-issue 해결), 완성 패키지를 zip으로 받아 인스타에 쉽게 업로드(반자동)할 수 있게 한다. + +**Architecture:** 디자인 시스템 Jinja 템플릿(페이지 타입별 레이아웃)을 web-ai insta-render 워커(authoritative)와 insta-lab(참조 복사본)에 작성. 워커 `card_renderer.py`에 `document.fonts.ready` 대기 + PNG 검증 추가. card_writer 프롬프트에 글자수 가이드. insta-lab에 zip 패키지 API + web-ui 다운로드 버튼. Graph API 미사용(반자동). + +**Tech Stack:** Jinja2 + HTML/CSS, Playwright(Chromium), FastAPI, pytest / React+Vite(web-ui). + +**Spec:** `docs/superpowers/specs/2026-06-02-insta-cardnews-upgrade-design.md` + +**⚠️ 3 repo 작업** (커밋·배포 경로 다름): +- `web-backend/insta-lab` — git push → Gitea webhook 자동배포 (NAS) +- `web-ai/services/insta-render` — **별도 repo(ai-trade.git), Windows 머신 구동** — 워커가 실제 렌더하는 authoritative 템플릿 위치 +- `web-ui` — **별도 repo**, `npm run release:nas` 수동 배포 + +--- + +## 검증된 컨텍스트 +- 워커 렌더: `web-ai/services/insta-render/card_renderer.py` — `_build_pages(slate)`가 10 spec 생성(cover page_no=1 / body page_no=2~9 / cta page_no=10, 각 `page_type`/`headline`/`body`/`accent_color`/`cta`/`page_no`/`total_pages`). `CARD_TEMPLATE_DIR`(기본 `/app/templates`)에서 `{theme}/card.html.j2` 로드 → `page.goto(file://, networkidle)` → `screenshot(full_page=False)` @viewport 1080×1350. +- 워커 템플릿 실제 위치: `web-ai/services/insta-render/templates/default/card.html.j2` (현재 insta-lab과 동일한 55줄 기본형). **이게 렌더에 쓰이는 authoritative 파일.** +- 카피: `insta-lab/app/card_writer.py` `DEFAULT_PROMPT`(DB `slate_writer` 오버라이드 가능). 산출: cover_copy{headline,body,accent_color}/body_copies[8]{headline,body}/cta_copy{headline,body,cta}/suggested_caption/hashtags[]. +- 슬레이트 PNG: 워커가 `INSTA_MEDIA_ROOT/{slate_id}/{page_no:02d}.png` 저장. NAS에서 `card_assets` 테이블 + `db.list_card_assets(slate_id)`(page_index + 파일경로)로 추적. `GET /api/insta/slates/{id}/assets/{page}`가 단일 PNG 서빙(파일경로 읽어 반환). +- 슬레이트 데이터: `db.get_card_slate(slate_id)` + `db.list_card_assets(slate_id)`. `GET /api/insta/slates/{id}`가 slate + assets 반환. + +--- + +# Phase 1 — 모던 미니멀 디자인 시스템 템플릿 (web-ai authoritative + insta-lab 복사본) + +## Task 1.1: 디자인 시스템 card.html.j2 작성 + +**Files:** +- Modify: `web-ai/services/insta-render/templates/default/card.html.j2` (**렌더 authoritative**) +- Modify: `web-backend/insta-lab/app/templates/default/card.html.j2` (참조 복사본 — 동일 내용 유지) + +> 두 파일을 **동일 내용**으로 작성한다. 워커가 web-ai 쪽을 렌더하지만 insta-lab 복사본도 일관성 위해 갱신. + +- [ ] **Step 1: 디자인 시스템 템플릿 작성** — 아래 전체 내용으로 두 파일을 교체: + +```html + + + + + + + +
+
+ + {% if page_type == 'cover' %} + {{ category_label|default(headline[:0]) }}{{ '오늘의 이슈' if not category_label }} +
+

{{ headline }}

+

{{ body }}

+
+ {% elif page_type == 'cta' %} +
+

{{ headline }}

+

{{ body }}

+ {% if cta %}
{{ cta }}
{% endif %} +
+ {% else %} + {{ '%02d'|format(page_no - 1) }} +
+

{{ headline }}

+

{{ body }}

+
+ {% endif %} + + +
+ + +``` +> 디자인 노트: 페이지 타입별 분기(cover 대형 헤드라인+서브+배지 / body 좌상단 인덱스 `01~08`(page_no-1)+헤드라인+본문+진행 점 / cta 요약+CTA pill). `-webkit-line-clamp`로 오버플로우 2차 방어(글자수 가이드가 1차). `accent_color`는 기존 데이터. `brand_handle`은 미설정 시 빈칸(추후 핸들 주입 가능). Pretendard CDN(@import) — Phase 2의 fonts.ready 대기와 짝. + +- [ ] **Step 2: 렌더 스모크 확인 (web-ai)** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-ai/services/insta-render && python -c "from jinja2 import Environment, FileSystemLoader; e=Environment(loader=FileSystemLoader('templates')); t=e.get_template('default/card.html.j2'); [print(pt, len(t.render(page_type=pt, page_no=n, total_pages=10, headline='테스트 헤드라인', body='본문 테스트입니다.', accent_color='#0F62FE', cta='팔로우')) > 0) for pt,n in [('cover',1),('body',3),('cta',10)]]"` + Expected: `True` 3줄 (3 페이지 타입 모두 렌더 예외 없음). + +- [ ] **Step 3: Commit (2 repo 각각)** +```bash +# web-ai repo +cd /c/Users/jaeoh/Desktop/workspace/web-ai && git add services/insta-render/templates/default/card.html.j2 && git commit -m "feat(insta-render): 모던 미니멀 디자인 시스템 템플릿" +# insta-lab repo (참조 복사본) +cd /c/Users/jaeoh/Desktop/workspace/web-backend && git add insta-lab/app/templates/default/card.html.j2 && git commit -m "feat(insta-lab): default 템플릿 디자인 시스템 동기화(참조용)" +``` +> 커밋 메시지 trailer 각각에 `Co-Authored-By: Claude Opus 4.8 (1M context) ` 추가. + +--- + +# Phase 2 — 렌더 견고화 (web-ai 워커, known-issue 해결) + +## Task 2.1: fonts.ready 대기 + PNG 비어있음 검증 + +**Files:** +- Modify: `web-ai/services/insta-render/card_renderer.py` (`_render_slate_locked`) +- Test: `web-ai/services/insta-render/tests/test_worker.py` (또는 기존 테스트 파일에 추가) + +- [ ] **Step 1: 실패 테스트** — `tests/test_worker.py`에 추가 (실제 Chromium 렌더 + 검증). 워커 테스트 관례 확인 후 맞출 것; pytest-asyncio 사용 가정: +```python +import os +import pytest +from card_renderer import render_slate, init_browser, shutdown_browser + +@pytest.mark.asyncio +async def test_render_produces_nonempty_1080x1350(tmp_path, monkeypatch): + monkeypatch.setattr("card_renderer.INSTA_MEDIA_ROOT", str(tmp_path)) + await init_browser() + try: + slate = { + "cover_copy": {"headline": "헤드라인", "body": "서브", "accent_color": "#0F62FE"}, + "body_copies": [{"headline": f"포인트{i}", "body": "본문"} for i in range(8)], + "cta_copy": {"headline": "요약", "body": "마무리", "cta": "팔로우"}, + } + paths = await render_slate(slate, slate_id=99999) + assert len(paths) == 10 + for p in paths: + assert os.path.getsize(p) > 1000 # 비어있지 않음 + finally: + await shutdown_browser() +``` + +- [ ] **Step 2: 실패/현황 확인** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-ai/services/insta-render && python -m pytest tests/test_worker.py::test_render_produces_nonempty_1080x1350 -v` + Expected: 현재 코드로도 통과할 수 있으나(렌더 자체는 동작), 폰트/검증 보강 전이므로 FAIL이 아니면 다음 Step에서 검증 로직 추가가 의미를 갖도록 진행. (Playwright/Chromium 미설치 환경이면 `playwright install chromium` 필요 — 안 되면 DONE_WITH_CONCERNS로 보고) + +- [ ] **Step 3: card_renderer 보강** — `_render_slate_locked`의 페이지 루프에서 `page.goto` 직후·`screenshot` 직전에 폰트 대기 추가, screenshot 후 비어있음 검증: +```python + try: + await page.goto(f"file://{html_path}", wait_until="networkidle") + await page.evaluate("document.fonts.ready") # 웹폰트 로딩 완료까지 대기 + out_path = os.path.join(out_dir, f"{spec['page_no']:02d}.png") + await page.screenshot(path=out_path, full_page=False, omit_background=False) + if os.path.getsize(out_path) < 1000: # 빈/깨진 PNG 방어 + raise RuntimeError(f"rendered PNG too small: {out_path}") + paths.append(out_path) + finally: + ... +``` + +- [ ] **Step 4: 통과 확인** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-ai/services/insta-render && python -m pytest tests/test_worker.py -v` Expected: PASS + +- [ ] **Step 5: Commit (web-ai repo)** +```bash +cd /c/Users/jaeoh/Desktop/workspace/web-ai && git add services/insta-render/card_renderer.py services/insta-render/tests/test_worker.py && git commit -m "fix(insta-render): fonts.ready 대기 + PNG 비어있음 검증 (렌더 known-issue 해결)" +``` + +--- + +# Phase 3 — 카피 글자수 가이드 (insta-lab) + +## Task 3.1: card_writer 프롬프트에 글자수 상한 추가 + +**Files:** +- Modify: `web-backend/insta-lab/app/card_writer.py` (`DEFAULT_PROMPT`) +- Test: `web-backend/insta-lab/app/test_card_writer_prompt.py` (NEW) + +- [ ] **Step 1: 실패 테스트** + +`insta-lab/app/test_card_writer_prompt.py`: +```python +from app import card_writer + +def test_default_prompt_has_length_guidance(): + p = card_writer.DEFAULT_PROMPT + # 글자수 가이드가 프롬프트에 포함됐는지 + assert "22자" in p and "120자" in p + # 포맷 placeholder는 유지 + assert "{category}" in p and "{keyword}" in p and "{articles}" in p +``` + +- [ ] **Step 2: 실패 확인** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-backend/insta-lab && python -m pytest app/test_card_writer_prompt.py -v` Expected: FAIL + +- [ ] **Step 3: DEFAULT_PROMPT에 가이드 추가** — `DEFAULT_PROMPT` 문자열의 JSON 스키마 안내 뒤(닫는 `}}` 다음)에 글자수 가이드 문단 추가: +```python +DEFAULT_PROMPT = """너는 인스타그램 카드 뉴스 카피라이터다. +카테고리: {category} +키워드: {keyword} +참고 기사: +{articles} + +10페이지 인스타 카드용 카피를 다음 JSON 한 객체로만 출력해라 (코드펜스 금지): +{{ + "cover_copy": {{"headline": "<훅 한 줄>", "body": "<서브카피 1~2줄>", "accent_color": "#hex"}}, + "body_copies": [ + {{"headline": "<포인트 헤드라인>", "body": "<2~4문장 본문>"}}, + ... (총 8개) + ], + "cta_copy": {{"headline": "<요약 한 줄>", "body": "<마무리 1~2줄>", "cta": "팔로우/저장 등"}}, + "suggested_caption": "<인스타 캡션 본문>", + "hashtags": ["#태그1", "#태그2", ...] +}} + +[글자수 제약 — 카드 디자인 박스에 맞게 반드시 준수] +- cover_copy.headline: 22자 이내 +- body_copies[].headline: 26자 이내 +- body_copies[].body: 120자 이내 (2~4문장) +- cta_copy.headline: 22자 이내 +초과하면 잘리므로 간결하고 임팩트 있게 작성한다. +""" +``` + +- [ ] **Step 4: 통과 확인** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-backend/insta-lab && python -m pytest app/test_card_writer_prompt.py -v` Expected: PASS + +- [ ] **Step 5: Commit (insta-lab)** +```bash +cd /c/Users/jaeoh/Desktop/workspace/web-backend && git add insta-lab/app/card_writer.py insta-lab/app/test_card_writer_prompt.py && git commit -m "feat(insta-lab): card_writer 프롬프트에 글자수 가이드(오버플로우 예방)" +``` +> 주의: 운영 DB에 `slate_writer` prompt_template 오버라이드가 있으면 DEFAULT_PROMPT 대신 그게 쓰임 → 배포 후 필요 시 `PUT /api/insta/templates/prompts/slate_writer`로 동일 가이드 반영(plan §검증에서 안내). + +--- + +# Phase 4 — zip 패키지 다운로드 API (insta-lab) + +## Task 4.1: GET /api/insta/slates/{id}/package + +**Files:** +- Modify: `web-backend/insta-lab/app/main.py` (엔드포인트 추가) +- Test: `web-backend/insta-lab/app/test_package_api.py` (NEW) + +- [ ] **Step 1: (확인됨) asset 스키마** — `card_assets(slate_id, page_index, file_path, file_hash)`. `db.list_card_assets(slate_id)` → 각 row에 `file_path`·`page_index`. `db.add_card_asset(slate_id, page_index, file_path, file_hash="")`. `db.add_card_slate(row: dict)`. 기존 `/assets/{page}`는 `FileResponse(match["file_path"], media_type="image/png")`. zip 엔드포인트는 동일하게 `a["file_path"]`를 읽는다. + +- [ ] **Step 2: 실패 테스트** + +`insta-lab/app/test_package_api.py`: +```python +import io, os, tempfile, zipfile, sys +from fastapi.testclient import TestClient + +def _client(monkeypatch): + sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))) + from app import config, db + tmp = tempfile.mkdtemp() + monkeypatch.setattr(config, "INSTA_DATA_PATH", tmp, raising=False) + monkeypatch.setattr(db, "DB_PATH", os.path.join(tmp, "insta.db"), raising=False) + db.init_db() + from app.main import app + return TestClient(app), db, tmp + +def test_package_zip_contains_pngs_and_caption(monkeypatch): + client, db, tmp = _client(monkeypatch) + # 슬레이트 + 2개 asset(실제 PNG 파일) 시드 + sid = db.add_card_slate({"keyword":"k","category":"economy","status":"rendered", + "cover_copy":{"headline":"h"}, "body_copies":[{"headline":"b","body":"x"}]*8, + "cta_copy":{}, "suggested_caption":"캡션입니다", "hashtags":["#a","#b"]}) + cards_dir = os.path.join(tmp, "insta_cards", str(sid)); os.makedirs(cards_dir, exist_ok=True) + for pg in (1,2): + fp = os.path.join(cards_dir, f"{pg:02d}.png") + with open(fp, "wb") as f: f.write(b"\x89PNG\r\n" + b"0"*2000) + db.add_card_asset(slate_id=sid, page_index=pg, file_path=fp) + r = client.get(f"/api/insta/slates/{sid}/package") + assert r.status_code == 200 + assert r.headers["content-type"] == "application/zip" + z = zipfile.ZipFile(io.BytesIO(r.content)) + names = z.namelist() + assert any(n.endswith(".png") for n in names) + assert "caption.txt" in names + cap = z.read("caption.txt").decode("utf-8") + assert "캡션입니다" in cap and "#a" in cap +``` +> `db.add_card_slate`/`add_card_asset`/`list_card_assets`의 실제 시그니처·컬럼명은 db.py 확인 후 맞출 것. asset 경로 컬럼이 `path`가 아니면 테스트·구현 모두 조정. + +- [ ] **Step 3: 실패 확인** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-backend/insta-lab && python -m pytest app/test_package_api.py -v` Expected: FAIL (404) + +- [ ] **Step 4: 엔드포인트 구현** — `insta-lab/app/main.py`에 추가 (`/assets/{page}` 엔드포인트 근처, 동일한 asset 파일경로 접근 방식 사용. `import io, zipfile`은 상단에 추가): +```python +@app.get("/api/insta/slates/{slate_id}/package") +def download_package(slate_id: int): + slate = db.get_card_slate(slate_id) + if not slate: + raise HTTPException(404, "slate not found") + assets = sorted(db.list_card_assets(slate_id), key=lambda a: a["page_index"]) + if not assets: + raise HTTPException(409, "아직 렌더된 카드가 없습니다") + buf = io.BytesIO() + with zipfile.ZipFile(buf, "w", zipfile.ZIP_DEFLATED) as z: + for a in assets: + fp = a["file_path"] + if os.path.exists(fp): + z.write(fp, arcname=f"{a['page_index']:02d}.png") + caption = (slate.get("suggested_caption") or "").strip() + tags = slate.get("hashtags") or [] + if isinstance(tags, str): + import json as _json + try: tags = _json.loads(tags) + except Exception: tags = [] + caption_full = caption + ("\n\n" + " ".join(tags) if tags else "") + z.writestr("caption.txt", caption_full) + buf.seek(0) + from fastapi.responses import StreamingResponse + return StreamingResponse(buf, media_type="application/zip", headers={ + "Content-Disposition": f'attachment; filename="insta_slate_{slate_id}.zip"'}) +``` +> `HTTPException`/`os`는 main.py에 이미 import됨. `slate.get("hashtags")`가 JSON 문자열일 수 있어 방어 파싱. + +- [ ] **Step 5: 통과 확인** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-backend/insta-lab && python -m pytest app/test_package_api.py -v` Expected: PASS + +- [ ] **Step 6: Commit (insta-lab)** +```bash +cd /c/Users/jaeoh/Desktop/workspace/web-backend && git add insta-lab/app/main.py insta-lab/app/test_package_api.py && git commit -m "feat(insta-lab): 슬레이트 zip 패키지 다운로드 API (10 PNG + caption.txt)" +``` + +--- + +# Phase 5 — web-ui 패키지 다운로드 버튼 (별도 repo: web-ui) + +## Task 5.1: 슬레이트 상세에 다운로드 버튼 + +**Files:** +- Modify: `web-ui/src/api.js` (헬퍼) +- Modify: insta 카드 페이지 (`web-ui/src/pages/insta/InstaCards.jsx` 또는 슬레이트 상세 컴포넌트) + +- [ ] **Step 1: 구조 확인** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-ui && git checkout -b feat/insta-package-download && grep -rln "insta\|슬레이트\|slate" src/pages/insta/ src/api.js 2>/dev/null | head` 로 슬레이트 상세 UI + apiGet 패턴 확인. + +- [ ] **Step 2: api.js 헬퍼 + 다운로드** — `src/api.js`에 패키지 URL 헬퍼 추가(파일 다운로드는 새 탭/anchor로): +```javascript +export const instaPackageUrl = (slateId) => `/api/insta/slates/${slateId}/package`; +``` +슬레이트 상세 컴포넌트에 버튼 추가 (기존 버튼 스타일 맞춤): +```jsx + + 📦 패키지 다운로드 (10장 + 캡션) + +``` +> import에 `instaPackageUrl` 추가. 실제 슬레이트 객체의 id 필드명·버튼 클래스는 Step 1 확인 결과에 맞출 것. + +- [ ] **Step 3: 빌드 확인** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-ui && npm run build` Expected: exit 0 + +- [ ] **Step 4: Commit (web-ui repo)** +```bash +cd /c/Users/jaeoh/Desktop/workspace/web-ui && git add src/ && git commit -m "feat: 인스타 슬레이트 패키지 다운로드 버튼" +``` + +--- + +# Phase 6 — 통합 검증 + +## Task 6.1: 회귀 + 배포 안내 + +- [ ] **Step 1: insta-lab 테스트** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-backend/insta-lab && python -m pytest app/ -q` (Playwright 의존 테스트는 web-ai에만 있음). 신규 통과 + 회귀 없음. (`_shared` import로 main 로드 시 PYTHONPATH 필요하면 test에 sys.path.insert 적용 — Phase 4 test가 이미 처리) +- [ ] **Step 2: web-ai 테스트** — Run: `cd /c/Users/jaeoh/Desktop/workspace/web-ai/services/insta-render && python -m pytest -q` (Chromium 필요; 미설치 시 `playwright install chromium`). +- [ ] **Step 3: 배포 안내** — 3 repo 각각 push/배포: + - insta-lab: `git push origin main` → webhook 자동배포(NAS). + - web-ai: Windows 머신에서 워커 repo pull + 재시작 (insta-render 서비스). **신규 템플릿이 워커 CARD_TEMPLATE_DIR에 반영돼야 효과 발생.** + - web-ui: `npm run release:nas`. + - 배포 후 슬레이트 1건 생성 → 카드 PNG 육안 확인(디자인 시스템 적용·폰트 정상) → `/package` zip 다운로드 확인. DB `slate_writer` 오버라이드 존재 시 글자수 가이드 반영. + +--- + +## Self-Review 체크리스트 결과 +- **Spec 커버리지**: 디자인 시스템 템플릿(Task 1.1) / 렌더 견고화 fonts.ready+검증(2.1) / 카피 글자수 가이드(3.1) / zip 패키지(4.1) / web-ui 버튼(5.1) / 검증(6.1). known-issue(폰트·오버플로우)=2.1+템플릿 clamp. 모두 매핑. +- **Placeholder**: 모든 코드 step에 실제 코드. db asset 컬럼명·web-ui 슬레이트 필드·워커 테스트 관례는 "Step에서 확인 후 맞춤" 명시(코드베이스 의존, 합리적). brand_handle 기본 빈칸(미설정 허용). +- **타입 일관성**: 템플릿이 쓰는 spec 키(page_type/page_no/total_pages/headline/body/accent_color/cta)가 워커 `_build_pages` 산출과 일치. zip 엔드포인트가 쓰는 `list_card_assets`/`get_card_slate`/`suggested_caption`/`hashtags`는 기존 db/슬레이트 스키마와 일치(Step 1에서 asset 경로 컬럼명만 확인). +- **3 repo 경로**: 각 Task에 repo별 cd + 커밋 분리 명시.