nexus/.planning/phases/43-documents-branding/43-02-PLAN.md

13 KiB

phase plan type wave depends_on files_modified autonomous requirements must_haves
43-documents-branding 02 execute 2
43-01
server/src/services/renderers/brand-renderer.ts
server/src/services/content-job-runner.ts
server/src/__tests__/brand-renderer.test.ts
true
BRAND-01
BRAND-02
BRAND-03
BRAND-04
BRAND-05
BRAND-06
truths artifacts key_links
renderBrandKit returns a brand-kit-bundle with all required fields populated
Logo SVG is generated via LLM and cleaned with SVGO
Avatar PNGs are rasterized at 5 sizes: 512, 256, 128, 64, 32
Social images are generated for twitter-profile, twitter-banner, linkedin-profile, linkedin-banner, instagram-profile
Email signature and letterhead are generated as HTML strings
Brand guidelines PDF is generated via Playwright
All assets are packaged into a ZIP via archiver with correct folder structure
content-job-runner dispatches brand-kit jobType to brand-renderer
path provides exports
server/src/services/renderers/brand-renderer.ts Brand kit orchestration renderer
renderBrandKit
path provides min_lines
server/src/__tests__/brand-renderer.test.ts Unit tests for brand renderer 60
from to via pattern
server/src/services/content-job-runner.ts server/src/services/renderers/brand-renderer.ts dynamic import in renderContent switch case "brand-kit"
from to via pattern
server/src/services/renderers/brand-renderer.ts archiver import archiver for ZIP packaging archiver
from to via pattern
server/src/services/renderers/brand-renderer.ts server/src/services/renderers/diagram-renderer.js resolveBrowserPath for Playwright PDF resolveBrowserPath
Brand identity kit renderer that orchestrates logo, avatars, social images, templates, guidelines PDF, and ZIP packaging.

Purpose: Enable full brand identity generation from a single conversation — produces logo, avatars, social platform images, email signature, letterhead, guidelines PDF, and a downloadable ZIP of everything. Output: Working brand-renderer.ts with job-runner wiring and tests.

<execution_context> @$HOME/.claude/get-shit-done/workflows/execute-plan.md @$HOME/.claude/get-shit-done/templates/summary.md </execution_context>

@.planning/PROJECT.md @.planning/ROADMAP.md @.planning/STATE.md @.planning/phases/43-documents-branding/43-RESEARCH.md @.planning/phases/43-documents-branding/43-01-SUMMARY.md

@server/src/services/renderers/types.ts @server/src/services/content-job-runner.ts @server/src/services/renderers/diagram-renderer.ts @server/src/services/renderers/icon-renderer.ts @server/src/services/puter-inference.ts

From server/src/services/renderers/types.ts (after Plan 01):

export interface BrandKitBundle {
  type: "brand-kit-bundle";
  spec: { name: string; tagline: string; primaryColor: string; secondaryColor: string; fontStyle: string; industry: string; };
  logoSvgBase64: string;
  avatarPngs: Record<string, string>;      // "512"|"256"|"128"|"64"|"32" -> base64
  socialImages: Record<string, string>;    // "twitter-profile"|"twitter-banner"|"linkedin-profile"|"linkedin-banner"|"instagram-profile" -> base64
  signatureHtml: string;
  letterheadHtml: string;
  guidelinesPdfBase64: string;
  zipBase64: string;
}
export interface RenderResult { filename: string; contentType: string; buffer: Buffer; }

From server/src/services/renderers/diagram-renderer.ts:

export function resolveBrowserPath(): string;

From server/src/services/renderers/icon-renderer.ts:

export function validateAndCleanSvg(raw: string): { svg: string; warnings: string[] };

From server/src/services/puter-inference.ts:

export async function puterChatComplete(messages: ChatMessage[], model?: string): Promise<string>;

From server/src/services/renderers/wallpaper-renderer.ts:

export const PLATFORM_DIMENSIONS: Record<string, { width: number; height: number }>;
// Includes twitter-profile (400x400), twitter-banner (1500x500), linkedin-profile (400x400), linkedin-banner (1584x396), instagram-1080 (1080x1080)
Task 1: Create brand-renderer with orchestrated sub-renders and ZIP packaging server/src/services/renderers/brand-renderer.ts, server/src/__tests__/brand-renderer.test.ts - server/src/services/renderers/types.ts (BrandKitBundle type from Plan 01) - server/src/services/renderers/icon-renderer.ts (validateAndCleanSvg for SVG cleanup, LLM SVG prompt pattern) - server/src/services/renderers/wallpaper-renderer.ts (PLATFORM_DIMENSIONS export, sharp usage for SVG-to-PNG, stripMarkdownFences) - server/src/services/renderers/diagram-renderer.ts (resolveBrowserPath, Playwright launch + page.pdf pattern) - server/src/services/puter-inference.ts (puterChatComplete) - server/src/__tests__/diagram-renderer.test.ts (mock patterns) - Test 1: renderBrandKit returns brand-kit-bundle with all required top-level fields present (spec, logoSvgBase64, avatarPngs, socialImages, signatureHtml, letterheadHtml, guidelinesPdfBase64, zipBase64) - Test 2: spec contains name, tagline, primaryColor, secondaryColor, fontStyle, industry - Test 3: avatarPngs has keys "512", "256", "128", "64", "32" — all non-empty base64 strings - Test 4: socialImages has keys twitter-profile, twitter-banner, linkedin-profile, linkedin-banner, instagram-profile — all non-empty - Test 5: signatureHtml and letterheadHtml are non-empty strings - Test 6: guidelinesPdfBase64 is a non-empty string - Test 7: zipBase64 decodes to a buffer starting with PK (0x50, 0x4B — ZIP magic bytes) Create server/src/services/renderers/brand-renderer.ts with these internal functions:
**BrandSpec interface** (local, not exported — matches BrandKitBundle.spec shape):
```typescript
interface BrandSpec {
  name: string; tagline: string; primaryColor: string; secondaryColor: string;
  fontStyle: string; industry: string; logoDescription: string;
}
```

**extractBrandSpec(prompt)** — LLM call with system prompt instructing JSON output with the BrandSpec fields. Parse JSON from response (strip markdown fences first). If parsing fails, provide sensible defaults.

**generateLogoSvg(spec)** — LLM call asking for a simple, clean SVG logo mark based on spec.logoDescription, spec.primaryColor, spec.secondaryColor. System prompt: "Output ONLY valid SVG. No text explanations. Simple geometric shapes. viewBox 0 0 512 512. Use only the provided colors." Clean output with validateAndCleanSvg from icon-renderer.

**rasterizeAvatars(logoSvg)** — Use sharp to resize the SVG to [512, 256, 128, 64, 32]px PNG. Return Record<string, string> mapping size to base64. Use `sharp(Buffer.from(logoSvg)).resize(size, size).png().toBuffer()` for each size.

**generateSocialImages(spec, logoSvg)** — For each of 5 platforms (twitter-profile 400x400, twitter-banner 1500x500, linkedin-profile 400x400, linkedin-banner 1584x396, instagram-profile 1080x1080): create an SVG template with brand-colored background + centered logo, rasterize with sharp to the target dimensions. Return Record<string, string>.

**generateTemplates(spec)** — Two LLM calls:
1. Email signature HTML — system prompt: "Generate a professional HTML email signature with name, tagline, colors. Inline CSS only. No external resources. Keep under 200 lines."
2. Letterhead HTML — system prompt: "Generate a professional HTML letterhead template with header, footer, placeholder body. Inline CSS only. No external resources."
Strip markdown fences from both.

**generateGuidelinesPdf(spec, logoSvg)** — LLM generates a brand guidelines HTML page (system prompt: "Generate a brand guidelines document as HTML. Include: brand name, tagline, color palette with hex codes, typography guidance, logo usage rules, spacing guidelines. Inline CSS only. No external resources. Use web-safe fonts."). Render to PDF via Playwright (same pattern as pdf-renderer: resolveBrowserPath, chromium.launch, page.setContent, page.pdf A4). Return Buffer.

**buildZip(assets)** — Use archiver v7 to create an in-memory ZIP buffer with folder structure:
```
brand-kit/logo/logo.svg
brand-kit/logo/logo-512.png, logo-256.png, logo-128.png, logo-64.png, logo-32.png
brand-kit/social/twitter-profile.png, twitter-banner.png, linkedin-profile.png, linkedin-banner.png, instagram-profile.png
brand-kit/templates/email-signature.html, letterhead.html
brand-kit/guidelines.pdf
```
Use the streaming pattern from RESEARCH.md Pattern 3: Writable sink, archive.pipe(sink), archive.append(buffer, { name }), archive.finalize(), resolve on sink finish.

**renderBrandKit(input)** — Main exported function. Extract `prompt` from input. Call extractBrandSpec, generateLogoSvg, rasterizeAvatars, generateSocialImages, generateTemplates, generateGuidelinesPdf, buildZip in sequence. Assemble BrandKitBundle. Return RenderResult.

CRITICAL constraints:
- Open Playwright browser ONCE for the entire brand kit job (guidelines PDF), not per sub-step
- Use try/finally to close browser
- Social image SVGs are simple templates (colored rect + embedded logo), NOT LLM-generated — keeps them fast and predictable
- All base64 encoding via Buffer.from(x).toString("base64")

Create server/src/__tests__/brand-renderer.test.ts:
- vi.mock("playwright-core") — same pattern as pdf-renderer tests
- vi.mock("../services/puter-inference.js") — mock puterChatComplete to return different responses based on system prompt content (JSON for spec, SVG for logo, HTML for templates/guidelines)
- vi.mock("../services/renderers/diagram-renderer.js") — resolveBrowserPath returns "/fake/chromium"
- vi.mock("../services/renderers/icon-renderer.js") — validateAndCleanSvg returns { svg: input, warnings: [] }
- vi.mock("sharp") — mock sharp() chain: resize().png().toBuffer() returns a small Buffer
- vi.mock("archiver") — mock archiver() to return an object with append, pipe, finalize, on("error") that triggers the sink finish event
- Write tests per behavior section
- grep -q "renderBrandKit" server/src/services/renderers/brand-renderer.ts - grep -q "archiver" server/src/services/renderers/brand-renderer.ts - grep -q "resolveBrowserPath" server/src/services/renderers/brand-renderer.ts - grep -q "validateAndCleanSvg" server/src/services/renderers/brand-renderer.ts - grep -q "brand-kit-bundle" server/src/services/renderers/brand-renderer.ts - grep -q "zipBase64" server/src/services/renderers/brand-renderer.ts pnpm --filter @paperclipai/server test --run src/__tests__/brand-renderer.test.ts Brand renderer produces complete BrandKitBundle with logo SVG, 5 avatar sizes, 5 social images, email signature HTML, letterhead HTML, guidelines PDF, and ZIP package. All tests pass. Task 2: Wire brand-kit jobType into content-job-runner server/src/services/content-job-runner.ts - server/src/services/content-job-runner.ts (current switch with pdf-document case from Plan 01) Add a new case to the `renderContent()` switch in content-job-runner.ts, after the "pdf-document" case and before the `default` case:
```typescript
case "brand-kit": {
  const { renderBrandKit } = await import("./renderers/brand-renderer.js");
  return renderBrandKit(input);
}
```

Follow exact pattern of existing cases (dynamic import with .js extension, destructured named export).
- grep -q '"brand-kit"' server/src/services/content-job-runner.ts - grep -q 'renderBrandKit' server/src/services/content-job-runner.ts pnpm --filter @paperclipai/server exec tsc --noEmit content-job-runner dispatches "brand-kit" jobType to renderBrandKit. TypeScript compiles cleanly. - `pnpm --filter @paperclipai/server test --run src/__tests__/brand-renderer.test.ts` passes - `pnpm --filter @paperclipai/server exec tsc --noEmit` compiles without errors - `grep -q "brand-kit" server/src/services/content-job-runner.ts` succeeds

<success_criteria>

  • Brand renderer orchestrates 7 sub-steps: spec extraction, logo SVG, avatar rasterization, social images, templates, guidelines PDF, ZIP
  • All BrandKitBundle fields populated with non-empty values
  • ZIP contains correct folder structure with all assets
  • Job runner dispatches brand-kit jobs correctly
  • All tests pass, TypeScript clean </success_criteria>
After completion, create `.planning/phases/43-documents-branding/43-02-SUMMARY.md`