@su-record/vibe 2.6.28 → 2.6.31

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (243) hide show
  1. package/CLAUDE.md +235 -202
  2. package/LICENSE +21 -21
  3. package/README.md +276 -267
  4. package/agents/architect-low.md +41 -41
  5. package/agents/architect-medium.md +59 -59
  6. package/agents/architect.md +80 -80
  7. package/agents/build-error-resolver.md +115 -115
  8. package/agents/compounder.md +261 -261
  9. package/agents/diagrammer.md +178 -178
  10. package/agents/docs/api-documenter.md +99 -99
  11. package/agents/docs/changelog-writer.md +93 -93
  12. package/agents/e2e-tester.md +266 -266
  13. package/agents/explorer-low.md +42 -42
  14. package/agents/explorer-medium.md +59 -59
  15. package/agents/explorer.md +48 -48
  16. package/agents/implementer-low.md +43 -43
  17. package/agents/implementer-medium.md +52 -52
  18. package/agents/implementer.md +54 -54
  19. package/agents/planning/requirements-analyst.md +84 -84
  20. package/agents/planning/ux-advisor.md +83 -83
  21. package/agents/qa/acceptance-tester.md +86 -86
  22. package/agents/qa/edge-case-finder.md +93 -93
  23. package/agents/refactor-cleaner.md +143 -143
  24. package/agents/research/best-practices-agent.md +199 -199
  25. package/agents/research/codebase-patterns-agent.md +157 -157
  26. package/agents/research/framework-docs-agent.md +188 -188
  27. package/agents/research/security-advisory-agent.md +213 -213
  28. package/agents/review/architecture-reviewer.md +107 -107
  29. package/agents/review/complexity-reviewer.md +116 -116
  30. package/agents/review/data-integrity-reviewer.md +88 -88
  31. package/agents/review/git-history-reviewer.md +103 -103
  32. package/agents/review/performance-reviewer.md +86 -86
  33. package/agents/review/python-reviewer.md +150 -150
  34. package/agents/review/rails-reviewer.md +139 -139
  35. package/agents/review/react-reviewer.md +144 -144
  36. package/agents/review/security-reviewer.md +80 -80
  37. package/agents/review/simplicity-reviewer.md +140 -140
  38. package/agents/review/test-coverage-reviewer.md +116 -116
  39. package/agents/review/typescript-reviewer.md +127 -127
  40. package/agents/searcher.md +54 -54
  41. package/agents/simplifier.md +120 -120
  42. package/agents/tester.md +49 -49
  43. package/agents/ui-previewer.md +268 -268
  44. package/commands/vibe.analyze.md +356 -356
  45. package/commands/vibe.reason.md +329 -329
  46. package/commands/vibe.review.md +423 -423
  47. package/commands/vibe.run.md +1423 -1313
  48. package/commands/vibe.spec.md +1054 -1054
  49. package/commands/vibe.spec.review.md +412 -412
  50. package/commands/vibe.trace.md +161 -161
  51. package/commands/vibe.utils.md +376 -376
  52. package/commands/vibe.verify.md +375 -375
  53. package/dist/cli/collaborator.js +52 -52
  54. package/dist/cli/detect.js +32 -32
  55. package/dist/cli/hud.js +20 -20
  56. package/dist/cli/index.js +112 -112
  57. package/dist/cli/llm.js +144 -144
  58. package/dist/cli/postinstall.js +858 -858
  59. package/dist/lib/DeepInit.js +24 -24
  60. package/dist/lib/IterationTracker.js +11 -11
  61. package/dist/lib/PythonParser.js +108 -108
  62. package/dist/lib/ReviewRace.js +96 -96
  63. package/dist/lib/SkillFrontmatter.js +28 -28
  64. package/dist/lib/SkillQualityGate.js +9 -9
  65. package/dist/lib/SkillRepository.js +159 -159
  66. package/dist/lib/UltraQA.js +77 -77
  67. package/dist/lib/gemini-api.d.ts +13 -0
  68. package/dist/lib/gemini-api.d.ts.map +1 -1
  69. package/dist/lib/gemini-api.js +63 -5
  70. package/dist/lib/gemini-api.js.map +1 -1
  71. package/dist/lib/gpt-api.js +4 -4
  72. package/dist/lib/memory/KnowledgeGraph.js +4 -4
  73. package/dist/lib/memory/MemorySearch.js +43 -43
  74. package/dist/lib/memory/MemoryStorage.js +130 -130
  75. package/dist/lib/memory/ObservationStore.js +28 -28
  76. package/dist/lib/memory/SessionRAGRetriever.js +7 -7
  77. package/dist/lib/memory/SessionRAGStore.js +216 -216
  78. package/dist/lib/memory/SessionSummarizer.js +9 -9
  79. package/dist/orchestrator/AgentManager.js +12 -12
  80. package/dist/orchestrator/MultiLlmResearch.js +8 -8
  81. package/dist/orchestrator/SmartRouter.js +11 -11
  82. package/dist/orchestrator/SwarmOrchestrator.test.js +16 -16
  83. package/dist/orchestrator/parallelResearch.js +24 -24
  84. package/dist/tools/convention/analyzeComplexity.test.js +115 -115
  85. package/dist/tools/convention/validateCodeQuality.test.js +104 -104
  86. package/dist/tools/spec/prdParser.test.js +171 -171
  87. package/dist/tools/spec/specGenerator.js +169 -169
  88. package/dist/tools/spec/traceabilityMatrix.js +64 -64
  89. package/dist/tools/spec/traceabilityMatrix.test.js +28 -28
  90. package/hooks/hooks.json +115 -115
  91. package/hooks/scripts/code-check.js +70 -70
  92. package/hooks/scripts/code-review.js +22 -22
  93. package/hooks/scripts/complexity.js +22 -22
  94. package/hooks/scripts/compound.js +23 -23
  95. package/hooks/scripts/context-save.js +53 -53
  96. package/hooks/scripts/gemini-ui-gen.js +281 -281
  97. package/hooks/scripts/generate-brand-assets.js +474 -474
  98. package/hooks/scripts/hud-multiline.js +262 -262
  99. package/hooks/scripts/hud-status.js +291 -291
  100. package/hooks/scripts/keyword-detector.js +214 -214
  101. package/hooks/scripts/llm-orchestrate.js +245 -171
  102. package/hooks/scripts/post-edit.js +97 -97
  103. package/hooks/scripts/post-tool-verify.js +210 -210
  104. package/hooks/scripts/pre-tool-guard.js +125 -125
  105. package/hooks/scripts/prompt-dispatcher.js +161 -161
  106. package/hooks/scripts/recall.js +22 -22
  107. package/hooks/scripts/session-start.js +30 -30
  108. package/hooks/scripts/skill-injector.js +191 -191
  109. package/hooks/scripts/utils.js +97 -97
  110. package/languages/csharp-unity.md +515 -515
  111. package/languages/gdscript-godot.md +470 -470
  112. package/languages/ruby-rails.md +489 -489
  113. package/languages/typescript-angular.md +433 -433
  114. package/languages/typescript-astro.md +416 -416
  115. package/languages/typescript-electron.md +406 -406
  116. package/languages/typescript-nestjs.md +524 -524
  117. package/languages/typescript-svelte.md +407 -407
  118. package/languages/typescript-tauri.md +365 -365
  119. package/package.json +84 -84
  120. package/skills/brand-assets.md +141 -141
  121. package/skills/commerce-patterns.md +361 -361
  122. package/skills/context7-usage.md +102 -102
  123. package/skills/e2e-commerce.md +304 -304
  124. package/skills/frontend-design.md +92 -92
  125. package/skills/git-worktree.md +181 -181
  126. package/skills/parallel-research.md +77 -77
  127. package/skills/priority-todos.md +239 -239
  128. package/skills/seo-checklist.md +244 -244
  129. package/skills/tool-fallback.md +190 -190
  130. package/skills/vibe-capabilities.md +161 -161
  131. package/vibe/constitution.md +227 -227
  132. package/vibe/rules/core/communication-guide.md +98 -98
  133. package/vibe/rules/core/development-philosophy.md +52 -52
  134. package/vibe/rules/core/quick-start.md +102 -102
  135. package/vibe/rules/quality/bdd-contract-testing.md +393 -393
  136. package/vibe/rules/quality/checklist.md +276 -276
  137. package/vibe/rules/quality/testing-strategy.md +440 -440
  138. package/vibe/rules/standards/anti-patterns.md +541 -541
  139. package/vibe/rules/standards/code-structure.md +291 -291
  140. package/vibe/rules/standards/complexity-metrics.md +313 -313
  141. package/vibe/rules/standards/naming-conventions.md +198 -198
  142. package/vibe/setup.sh +31 -31
  143. package/vibe/templates/constitution-template.md +252 -252
  144. package/vibe/templates/contract-backend-template.md +526 -526
  145. package/vibe/templates/contract-frontend-template.md +599 -599
  146. package/vibe/templates/feature-template.md +96 -96
  147. package/vibe/templates/spec-template.md +221 -221
  148. package/dist/cli/mcp.d.ts +0 -49
  149. package/dist/cli/mcp.d.ts.map +0 -1
  150. package/dist/cli/mcp.js +0 -169
  151. package/dist/cli/mcp.js.map +0 -1
  152. package/dist/lib/gemini-mcp.d.ts +0 -10
  153. package/dist/lib/gemini-mcp.d.ts.map +0 -1
  154. package/dist/lib/gemini-mcp.js +0 -353
  155. package/dist/lib/gemini-mcp.js.map +0 -1
  156. package/dist/lib/gpt-mcp.d.ts +0 -10
  157. package/dist/lib/gpt-mcp.d.ts.map +0 -1
  158. package/dist/lib/gpt-mcp.js +0 -352
  159. package/dist/lib/gpt-mcp.js.map +0 -1
  160. package/dist/tools/analytics/getUsageAnalytics.d.ts +0 -10
  161. package/dist/tools/analytics/getUsageAnalytics.d.ts.map +0 -1
  162. package/dist/tools/analytics/getUsageAnalytics.js +0 -246
  163. package/dist/tools/analytics/getUsageAnalytics.js.map +0 -1
  164. package/dist/tools/analytics/index.d.ts +0 -5
  165. package/dist/tools/analytics/index.d.ts.map +0 -1
  166. package/dist/tools/analytics/index.js +0 -5
  167. package/dist/tools/analytics/index.js.map +0 -1
  168. package/dist/tools/convention/getCodingGuide.d.ts +0 -7
  169. package/dist/tools/convention/getCodingGuide.d.ts.map +0 -1
  170. package/dist/tools/convention/getCodingGuide.js +0 -69
  171. package/dist/tools/convention/getCodingGuide.js.map +0 -1
  172. package/dist/tools/planning/analyzeRequirements.d.ts +0 -9
  173. package/dist/tools/planning/analyzeRequirements.d.ts.map +0 -1
  174. package/dist/tools/planning/analyzeRequirements.js +0 -171
  175. package/dist/tools/planning/analyzeRequirements.js.map +0 -1
  176. package/dist/tools/planning/createUserStories.d.ts +0 -9
  177. package/dist/tools/planning/createUserStories.d.ts.map +0 -1
  178. package/dist/tools/planning/createUserStories.js +0 -124
  179. package/dist/tools/planning/createUserStories.js.map +0 -1
  180. package/dist/tools/planning/featureRoadmap.d.ts +0 -10
  181. package/dist/tools/planning/featureRoadmap.d.ts.map +0 -1
  182. package/dist/tools/planning/featureRoadmap.js +0 -207
  183. package/dist/tools/planning/featureRoadmap.js.map +0 -1
  184. package/dist/tools/planning/generatePrd.d.ts +0 -11
  185. package/dist/tools/planning/generatePrd.d.ts.map +0 -1
  186. package/dist/tools/planning/generatePrd.js +0 -161
  187. package/dist/tools/planning/generatePrd.js.map +0 -1
  188. package/dist/tools/planning/index.d.ts +0 -8
  189. package/dist/tools/planning/index.d.ts.map +0 -1
  190. package/dist/tools/planning/index.js +0 -8
  191. package/dist/tools/planning/index.js.map +0 -1
  192. package/dist/tools/prompt/analyzePrompt.d.ts +0 -7
  193. package/dist/tools/prompt/analyzePrompt.d.ts.map +0 -1
  194. package/dist/tools/prompt/analyzePrompt.js +0 -150
  195. package/dist/tools/prompt/analyzePrompt.js.map +0 -1
  196. package/dist/tools/prompt/enhancePrompt.d.ts +0 -8
  197. package/dist/tools/prompt/enhancePrompt.d.ts.map +0 -1
  198. package/dist/tools/prompt/enhancePrompt.js +0 -110
  199. package/dist/tools/prompt/enhancePrompt.js.map +0 -1
  200. package/dist/tools/prompt/enhancePromptGemini.d.ts +0 -8
  201. package/dist/tools/prompt/enhancePromptGemini.d.ts.map +0 -1
  202. package/dist/tools/prompt/enhancePromptGemini.js +0 -332
  203. package/dist/tools/prompt/enhancePromptGemini.js.map +0 -1
  204. package/dist/tools/prompt/index.d.ts +0 -7
  205. package/dist/tools/prompt/index.d.ts.map +0 -1
  206. package/dist/tools/prompt/index.js +0 -7
  207. package/dist/tools/prompt/index.js.map +0 -1
  208. package/dist/tools/reasoning/applyReasoningFramework.d.ts +0 -8
  209. package/dist/tools/reasoning/applyReasoningFramework.d.ts.map +0 -1
  210. package/dist/tools/reasoning/applyReasoningFramework.js +0 -266
  211. package/dist/tools/reasoning/applyReasoningFramework.js.map +0 -1
  212. package/dist/tools/reasoning/index.d.ts +0 -5
  213. package/dist/tools/reasoning/index.d.ts.map +0 -1
  214. package/dist/tools/reasoning/index.js +0 -5
  215. package/dist/tools/reasoning/index.js.map +0 -1
  216. package/dist/tools/thinking/analyzeProblem.d.ts +0 -7
  217. package/dist/tools/thinking/analyzeProblem.d.ts.map +0 -1
  218. package/dist/tools/thinking/analyzeProblem.js +0 -55
  219. package/dist/tools/thinking/analyzeProblem.js.map +0 -1
  220. package/dist/tools/thinking/breakDownProblem.d.ts +0 -8
  221. package/dist/tools/thinking/breakDownProblem.d.ts.map +0 -1
  222. package/dist/tools/thinking/breakDownProblem.js +0 -145
  223. package/dist/tools/thinking/breakDownProblem.js.map +0 -1
  224. package/dist/tools/thinking/createThinkingChain.d.ts +0 -7
  225. package/dist/tools/thinking/createThinkingChain.d.ts.map +0 -1
  226. package/dist/tools/thinking/createThinkingChain.js +0 -44
  227. package/dist/tools/thinking/createThinkingChain.js.map +0 -1
  228. package/dist/tools/thinking/formatAsPlan.d.ts +0 -9
  229. package/dist/tools/thinking/formatAsPlan.d.ts.map +0 -1
  230. package/dist/tools/thinking/formatAsPlan.js +0 -78
  231. package/dist/tools/thinking/formatAsPlan.js.map +0 -1
  232. package/dist/tools/thinking/index.d.ts +0 -10
  233. package/dist/tools/thinking/index.d.ts.map +0 -1
  234. package/dist/tools/thinking/index.js +0 -10
  235. package/dist/tools/thinking/index.js.map +0 -1
  236. package/dist/tools/thinking/stepByStepAnalysis.d.ts +0 -8
  237. package/dist/tools/thinking/stepByStepAnalysis.d.ts.map +0 -1
  238. package/dist/tools/thinking/stepByStepAnalysis.js +0 -63
  239. package/dist/tools/thinking/stepByStepAnalysis.js.map +0 -1
  240. package/dist/tools/thinking/thinkAloudProcess.d.ts +0 -8
  241. package/dist/tools/thinking/thinkAloudProcess.d.ts.map +0 -1
  242. package/dist/tools/thinking/thinkAloudProcess.js +0 -80
  243. package/dist/tools/thinking/thinkAloudProcess.js.map +0 -1
@@ -1,474 +1,474 @@
1
- #!/usr/bin/env node
2
- /**
3
- * Brand Assets Generator
4
- * Gemini Image API를 사용하여 앱 아이콘/파비콘 자동 생성
5
- *
6
- * Usage:
7
- * node generate-brand-assets.js --name "AppName" --color "#2F6BFF" --style "modern" --output "./public"
8
- * node generate-brand-assets.js --spec ".claude/vibe/specs/feature.md" --output "./public"
9
- */
10
-
11
- import fs from 'fs';
12
- import path from 'path';
13
- import os from 'os';
14
- import https from 'https';
15
- import { execSync } from 'child_process';
16
-
17
- // ============================================
18
- // Configuration
19
- // ============================================
20
-
21
- const CONFIG_PATH = path.join(os.homedir(), '.config', 'vibe', 'gemini.json');
22
-
23
- const ICON_SIZES = {
24
- 'favicon-16x16.png': 16,
25
- 'favicon-32x32.png': 32,
26
- 'apple-touch-icon.png': 180,
27
- 'android-chrome-192x192.png': 192,
28
- 'android-chrome-512x512.png': 512,
29
- };
30
-
31
- // ============================================
32
- // Gemini API
33
- // ============================================
34
-
35
- function getGeminiApiKey() {
36
- // 1. Environment variable
37
- if (process.env.GEMINI_API_KEY) {
38
- return process.env.GEMINI_API_KEY;
39
- }
40
-
41
- // 2. Config file
42
- if (fs.existsSync(CONFIG_PATH)) {
43
- try {
44
- const config = JSON.parse(fs.readFileSync(CONFIG_PATH, 'utf8'));
45
- if (config.apiKey) return config.apiKey;
46
- } catch {
47
- // ignore
48
- }
49
- }
50
-
51
- return null;
52
- }
53
-
54
- async function generateImageWithGemini(prompt, apiKey) {
55
- // Nano Banana (Gemini 2.5 Flash Image) - fast image generation for icons/logos
56
- // For professional assets, use gemini-3-pro-image-preview (Nano Banana Pro)
57
- const url = `https://generativelanguage.googleapis.com/v1beta/models/gemini-2.5-flash-image:generateContent?key=${apiKey}`;
58
-
59
- const requestBody = {
60
- contents: [{
61
- parts: [{
62
- text: `Generate an app icon image. ${prompt}
63
-
64
- Requirements:
65
- - Square format (1:1 aspect ratio)
66
- - Simple, recognizable design
67
- - Works well at small sizes
68
- - No text or letters in the icon
69
- - Professional and modern look
70
- - Single focal element on solid or gradient background`
71
- }]
72
- }],
73
- generationConfig: {
74
- responseModalities: ['TEXT', 'IMAGE'],
75
- }
76
- };
77
-
78
- return new Promise((resolve, reject) => {
79
- const req = https.request(url, {
80
- method: 'POST',
81
- headers: {
82
- 'Content-Type': 'application/json',
83
- }
84
- }, (res) => {
85
- let data = '';
86
- res.on('data', chunk => data += chunk);
87
- res.on('end', () => {
88
- try {
89
- const result = JSON.parse(data);
90
- if (result.error) {
91
- reject(new Error(result.error.message));
92
- return;
93
- }
94
-
95
- // Find image part in response
96
- const parts = result.candidates?.[0]?.content?.parts || [];
97
- for (const part of parts) {
98
- if (part.inlineData?.mimeType?.startsWith('image/')) {
99
- resolve({
100
- data: Buffer.from(part.inlineData.data, 'base64'),
101
- mimeType: part.inlineData.mimeType
102
- });
103
- return;
104
- }
105
- }
106
-
107
- reject(new Error('No image in response'));
108
- } catch (e) {
109
- reject(e);
110
- }
111
- });
112
- });
113
-
114
- req.on('error', reject);
115
- req.write(JSON.stringify(requestBody));
116
- req.end();
117
- });
118
- }
119
-
120
- // ============================================
121
- // Image Processing
122
- // ============================================
123
-
124
- function resizeImage(inputPath, outputPath, size) {
125
- // Try sharp via npx, fallback to sips (macOS) or convert (ImageMagick)
126
- try {
127
- // Check if sharp is available
128
- execSync(`npx sharp-cli resize ${size} ${size} -i "${inputPath}" -o "${outputPath}"`, {
129
- stdio: 'pipe'
130
- });
131
- return true;
132
- } catch {
133
- // Fallback: macOS sips
134
- if (process.platform === 'darwin') {
135
- try {
136
- execSync(`sips -z ${size} ${size} "${inputPath}" --out "${outputPath}"`, {
137
- stdio: 'pipe'
138
- });
139
- return true;
140
- } catch {
141
- // ignore
142
- }
143
- }
144
-
145
- // Fallback: ImageMagick convert
146
- try {
147
- execSync(`convert "${inputPath}" -resize ${size}x${size} "${outputPath}"`, {
148
- stdio: 'pipe'
149
- });
150
- return true;
151
- } catch {
152
- // ignore
153
- }
154
- }
155
-
156
- return false;
157
- }
158
-
159
- function createFavicon(inputPath, outputPath) {
160
- // Create ICO file with multiple sizes
161
- try {
162
- // Try ImageMagick
163
- execSync(`convert "${inputPath}" -define icon:auto-resize=48,32,16 "${outputPath}"`, {
164
- stdio: 'pipe'
165
- });
166
- return true;
167
- } catch {
168
- // Fallback: just copy 32x32 as ico
169
- const png32 = outputPath.replace('favicon.ico', 'favicon-32x32.png');
170
- if (fs.existsSync(png32)) {
171
- fs.copyFileSync(png32, outputPath);
172
- return true;
173
- }
174
- }
175
- return false;
176
- }
177
-
178
- // ============================================
179
- // SPEC Parsing
180
- // ============================================
181
-
182
- function parseSpecForBrand(specPath) {
183
- if (!fs.existsSync(specPath)) {
184
- return null;
185
- }
186
-
187
- const content = fs.readFileSync(specPath, 'utf8');
188
- const brand = {
189
- name: null,
190
- color: null,
191
- style: null,
192
- concept: null
193
- };
194
-
195
- // Extract app name from title or context
196
- const titleMatch = content.match(/^#\s+(.+?)(?:\s+SPEC)?$/m);
197
- if (titleMatch) {
198
- brand.name = titleMatch[1].trim();
199
- }
200
-
201
- // Extract from context section
202
- const contextMatch = content.match(/<context>([\s\S]*?)<\/context>/i);
203
- if (contextMatch) {
204
- const context = contextMatch[1];
205
-
206
- // App Name
207
- const nameMatch = context.match(/App\s*Name[:\s]+([^\n]+)/i);
208
- if (nameMatch) brand.name = nameMatch[1].trim();
209
-
210
- // Primary Color
211
- const colorMatch = context.match(/Primary\s*Color[:\s]+(#[0-9A-Fa-f]{6}|#[0-9A-Fa-f]{3})/i);
212
- if (colorMatch) brand.color = colorMatch[1];
213
-
214
- // Style
215
- const styleMatch = context.match(/Style[:\s]+([^\n]+)/i);
216
- if (styleMatch) brand.style = styleMatch[1].trim();
217
-
218
- // Icon Concept
219
- const conceptMatch = context.match(/Icon\s*Concept[:\s]+([^\n]+)/i);
220
- if (conceptMatch) brand.concept = conceptMatch[1].trim();
221
- }
222
-
223
- // Fallback: extract from any brand-related content
224
- if (!brand.color) {
225
- const anyColor = content.match(/#([0-9A-Fa-f]{6}|[0-9A-Fa-f]{3})\b/);
226
- if (anyColor) brand.color = '#' + anyColor[1];
227
- }
228
-
229
- return brand;
230
- }
231
-
232
- // ============================================
233
- // Prompt Generation
234
- // ============================================
235
-
236
- function buildIconPrompt(brand) {
237
- const parts = [];
238
-
239
- if (brand.name) {
240
- parts.push(`App icon for "${brand.name}"`);
241
- } else {
242
- parts.push('Modern app icon');
243
- }
244
-
245
- if (brand.concept) {
246
- parts.push(`Concept: ${brand.concept}`);
247
- }
248
-
249
- if (brand.style) {
250
- parts.push(`Style: ${brand.style}`);
251
- } else {
252
- parts.push('Style: Modern, minimalist, professional');
253
- }
254
-
255
- if (brand.color) {
256
- parts.push(`Primary color: ${brand.color}`);
257
- }
258
-
259
- return parts.join('. ');
260
- }
261
-
262
- // ============================================
263
- // Fallback: Text Monogram
264
- // ============================================
265
-
266
- function generateFallbackIcon(name, color, outputPath) {
267
- // Generate simple SVG monogram
268
- const initial = (name || 'A')[0].toUpperCase();
269
- const bgColor = color || '#4A90D9';
270
-
271
- const svg = `<?xml version="1.0" encoding="UTF-8"?>
272
- <svg width="512" height="512" viewBox="0 0 512 512" xmlns="http://www.w3.org/2000/svg">
273
- <rect width="512" height="512" rx="64" fill="${bgColor}"/>
274
- <text x="256" y="320" font-family="Arial, sans-serif" font-size="280" font-weight="bold" fill="white" text-anchor="middle">${initial}</text>
275
- </svg>`;
276
-
277
- const svgPath = outputPath.replace(/\.(png|ico)$/, '.svg');
278
- fs.writeFileSync(svgPath, svg);
279
-
280
- // Convert SVG to PNG if possible
281
- try {
282
- execSync(`convert "${svgPath}" -resize 512x512 "${outputPath}"`, { stdio: 'pipe' });
283
- fs.unlinkSync(svgPath);
284
- return true;
285
- } catch {
286
- // Keep SVG as fallback
287
- console.log(` Created SVG fallback: ${svgPath}`);
288
- return false;
289
- }
290
- }
291
-
292
- // ============================================
293
- // Manifest Generation
294
- // ============================================
295
-
296
- function generateWebManifest(outputDir, name) {
297
- const manifest = {
298
- name: name || 'App',
299
- short_name: name || 'App',
300
- icons: [
301
- { src: '/android-chrome-192x192.png', sizes: '192x192', type: 'image/png' },
302
- { src: '/android-chrome-512x512.png', sizes: '512x512', type: 'image/png' }
303
- ],
304
- theme_color: '#ffffff',
305
- background_color: '#ffffff',
306
- display: 'standalone'
307
- };
308
-
309
- const manifestPath = path.join(outputDir, 'site.webmanifest');
310
- fs.writeFileSync(manifestPath, JSON.stringify(manifest, null, 2));
311
- return manifestPath;
312
- }
313
-
314
- // ============================================
315
- // Main
316
- // ============================================
317
-
318
- async function main() {
319
- const args = process.argv.slice(2);
320
-
321
- // Parse arguments
322
- let name = null;
323
- let color = null;
324
- let style = null;
325
- let specPath = null;
326
- let outputDir = './public';
327
- let force = false;
328
-
329
- for (let i = 0; i < args.length; i++) {
330
- switch (args[i]) {
331
- case '--name':
332
- name = args[++i];
333
- break;
334
- case '--color':
335
- color = args[++i];
336
- break;
337
- case '--style':
338
- style = args[++i];
339
- break;
340
- case '--spec':
341
- specPath = args[++i];
342
- break;
343
- case '--output':
344
- outputDir = args[++i];
345
- break;
346
- case '--force':
347
- case '--regenerate':
348
- force = true;
349
- break;
350
- case '--help':
351
- console.log(`
352
- Brand Assets Generator
353
-
354
- Usage:
355
- node generate-brand-assets.js [options]
356
-
357
- Options:
358
- --name <name> App name
359
- --color <hex> Primary color (e.g., #2F6BFF)
360
- --style <style> Design style keywords
361
- --spec <path> Path to SPEC file to extract brand info
362
- --output <dir> Output directory (default: ./public)
363
- --force Regenerate even if icons exist
364
-
365
- Examples:
366
- node generate-brand-assets.js --name "MyApp" --color "#FF5722" --output "./public"
367
- node generate-brand-assets.js --spec ".claude/vibe/specs/my-feature.md"
368
- `);
369
- process.exit(0);
370
- }
371
- }
372
-
373
- // Check if icons already exist
374
- const faviconPath = path.join(outputDir, 'favicon.ico');
375
- if (fs.existsSync(faviconPath) && !force) {
376
- console.log('Brand assets already exist. Use --force to regenerate.');
377
- process.exit(0);
378
- }
379
-
380
- // Parse SPEC if provided
381
- if (specPath) {
382
- const brand = parseSpecForBrand(specPath);
383
- if (brand) {
384
- name = name || brand.name;
385
- color = color || brand.color;
386
- style = style || brand.style;
387
- }
388
- }
389
-
390
- // Ensure output directory
391
- if (!fs.existsSync(outputDir)) {
392
- fs.mkdirSync(outputDir, { recursive: true });
393
- }
394
-
395
- // Check Gemini API key
396
- const apiKey = getGeminiApiKey();
397
-
398
- console.log('Generating brand assets...');
399
- console.log(` Name: ${name || '(not specified)'}`);
400
- console.log(` Color: ${color || '(not specified)'}`);
401
- console.log(` Style: ${style || '(not specified)'}`);
402
- console.log(` Output: ${outputDir}`);
403
- console.log(` Gemini: ${apiKey ? 'configured' : 'not configured'}`);
404
- console.log('');
405
-
406
- const masterIconPath = path.join(outputDir, 'icon-master.png');
407
- let generated = false;
408
-
409
- // Try Gemini Image API
410
- if (apiKey) {
411
- try {
412
- const prompt = buildIconPrompt({ name, color, style });
413
- console.log('Generating icon with Gemini...');
414
- console.log(` Prompt: ${prompt}`);
415
-
416
- const image = await generateImageWithGemini(prompt, apiKey);
417
- fs.writeFileSync(masterIconPath, image.data);
418
- console.log(' Master icon generated successfully');
419
- generated = true;
420
- } catch (err) {
421
- console.log(` Gemini generation failed: ${err.message}`);
422
- console.log(' Falling back to text monogram...');
423
- }
424
- }
425
-
426
- // Fallback to monogram
427
- if (!generated) {
428
- console.log('Generating fallback monogram icon...');
429
- generateFallbackIcon(name, color, masterIconPath);
430
- if (fs.existsSync(masterIconPath)) {
431
- generated = true;
432
- }
433
- }
434
-
435
- if (!generated) {
436
- console.error('Failed to generate icon');
437
- process.exit(1);
438
- }
439
-
440
- // Resize to all required sizes
441
- console.log('Resizing to platform sizes...');
442
- for (const [filename, size] of Object.entries(ICON_SIZES)) {
443
- const outPath = path.join(outputDir, filename);
444
- if (resizeImage(masterIconPath, outPath, size)) {
445
- console.log(` Created: ${filename}`);
446
- } else {
447
- console.log(` Warning: Could not create ${filename}`);
448
- }
449
- }
450
-
451
- // Create favicon.ico
452
- console.log('Creating favicon.ico...');
453
- if (createFavicon(masterIconPath, faviconPath)) {
454
- console.log(' Created: favicon.ico');
455
- }
456
-
457
- // Generate web manifest
458
- const manifestPath = generateWebManifest(outputDir, name);
459
- console.log(` Created: ${path.basename(manifestPath)}`);
460
-
461
- // Cleanup master icon
462
- if (fs.existsSync(masterIconPath)) {
463
- fs.unlinkSync(masterIconPath);
464
- }
465
-
466
- console.log('');
467
- console.log('Brand assets generation complete!');
468
- console.log(`Output directory: ${outputDir}`);
469
- }
470
-
471
- main().catch(err => {
472
- console.error('Error:', err.message);
473
- process.exit(1);
474
- });
1
+ #!/usr/bin/env node
2
+ /**
3
+ * Brand Assets Generator
4
+ * Gemini Image API를 사용하여 앱 아이콘/파비콘 자동 생성
5
+ *
6
+ * Usage:
7
+ * node generate-brand-assets.js --name "AppName" --color "#2F6BFF" --style "modern" --output "./public"
8
+ * node generate-brand-assets.js --spec ".claude/vibe/specs/feature.md" --output "./public"
9
+ */
10
+
11
+ import fs from 'fs';
12
+ import path from 'path';
13
+ import os from 'os';
14
+ import https from 'https';
15
+ import { execSync } from 'child_process';
16
+
17
+ // ============================================
18
+ // Configuration
19
+ // ============================================
20
+
21
+ const CONFIG_PATH = path.join(os.homedir(), '.config', 'vibe', 'gemini.json');
22
+
23
+ const ICON_SIZES = {
24
+ 'favicon-16x16.png': 16,
25
+ 'favicon-32x32.png': 32,
26
+ 'apple-touch-icon.png': 180,
27
+ 'android-chrome-192x192.png': 192,
28
+ 'android-chrome-512x512.png': 512,
29
+ };
30
+
31
+ // ============================================
32
+ // Gemini API
33
+ // ============================================
34
+
35
+ function getGeminiApiKey() {
36
+ // 1. Environment variable
37
+ if (process.env.GEMINI_API_KEY) {
38
+ return process.env.GEMINI_API_KEY;
39
+ }
40
+
41
+ // 2. Config file
42
+ if (fs.existsSync(CONFIG_PATH)) {
43
+ try {
44
+ const config = JSON.parse(fs.readFileSync(CONFIG_PATH, 'utf8'));
45
+ if (config.apiKey) return config.apiKey;
46
+ } catch {
47
+ // ignore
48
+ }
49
+ }
50
+
51
+ return null;
52
+ }
53
+
54
+ async function generateImageWithGemini(prompt, apiKey) {
55
+ // Nano Banana (Gemini 2.5 Flash Image) - fast image generation for icons/logos
56
+ // For professional assets, use gemini-3-pro-image-preview (Nano Banana Pro)
57
+ const url = `https://generativelanguage.googleapis.com/v1beta/models/gemini-2.5-flash-image:generateContent?key=${apiKey}`;
58
+
59
+ const requestBody = {
60
+ contents: [{
61
+ parts: [{
62
+ text: `Generate an app icon image. ${prompt}
63
+
64
+ Requirements:
65
+ - Square format (1:1 aspect ratio)
66
+ - Simple, recognizable design
67
+ - Works well at small sizes
68
+ - No text or letters in the icon
69
+ - Professional and modern look
70
+ - Single focal element on solid or gradient background`
71
+ }]
72
+ }],
73
+ generationConfig: {
74
+ responseModalities: ['TEXT', 'IMAGE'],
75
+ }
76
+ };
77
+
78
+ return new Promise((resolve, reject) => {
79
+ const req = https.request(url, {
80
+ method: 'POST',
81
+ headers: {
82
+ 'Content-Type': 'application/json',
83
+ }
84
+ }, (res) => {
85
+ let data = '';
86
+ res.on('data', chunk => data += chunk);
87
+ res.on('end', () => {
88
+ try {
89
+ const result = JSON.parse(data);
90
+ if (result.error) {
91
+ reject(new Error(result.error.message));
92
+ return;
93
+ }
94
+
95
+ // Find image part in response
96
+ const parts = result.candidates?.[0]?.content?.parts || [];
97
+ for (const part of parts) {
98
+ if (part.inlineData?.mimeType?.startsWith('image/')) {
99
+ resolve({
100
+ data: Buffer.from(part.inlineData.data, 'base64'),
101
+ mimeType: part.inlineData.mimeType
102
+ });
103
+ return;
104
+ }
105
+ }
106
+
107
+ reject(new Error('No image in response'));
108
+ } catch (e) {
109
+ reject(e);
110
+ }
111
+ });
112
+ });
113
+
114
+ req.on('error', reject);
115
+ req.write(JSON.stringify(requestBody));
116
+ req.end();
117
+ });
118
+ }
119
+
120
+ // ============================================
121
+ // Image Processing
122
+ // ============================================
123
+
124
+ function resizeImage(inputPath, outputPath, size) {
125
+ // Try sharp via npx, fallback to sips (macOS) or convert (ImageMagick)
126
+ try {
127
+ // Check if sharp is available
128
+ execSync(`npx sharp-cli resize ${size} ${size} -i "${inputPath}" -o "${outputPath}"`, {
129
+ stdio: 'pipe'
130
+ });
131
+ return true;
132
+ } catch {
133
+ // Fallback: macOS sips
134
+ if (process.platform === 'darwin') {
135
+ try {
136
+ execSync(`sips -z ${size} ${size} "${inputPath}" --out "${outputPath}"`, {
137
+ stdio: 'pipe'
138
+ });
139
+ return true;
140
+ } catch {
141
+ // ignore
142
+ }
143
+ }
144
+
145
+ // Fallback: ImageMagick convert
146
+ try {
147
+ execSync(`convert "${inputPath}" -resize ${size}x${size} "${outputPath}"`, {
148
+ stdio: 'pipe'
149
+ });
150
+ return true;
151
+ } catch {
152
+ // ignore
153
+ }
154
+ }
155
+
156
+ return false;
157
+ }
158
+
159
+ function createFavicon(inputPath, outputPath) {
160
+ // Create ICO file with multiple sizes
161
+ try {
162
+ // Try ImageMagick
163
+ execSync(`convert "${inputPath}" -define icon:auto-resize=48,32,16 "${outputPath}"`, {
164
+ stdio: 'pipe'
165
+ });
166
+ return true;
167
+ } catch {
168
+ // Fallback: just copy 32x32 as ico
169
+ const png32 = outputPath.replace('favicon.ico', 'favicon-32x32.png');
170
+ if (fs.existsSync(png32)) {
171
+ fs.copyFileSync(png32, outputPath);
172
+ return true;
173
+ }
174
+ }
175
+ return false;
176
+ }
177
+
178
+ // ============================================
179
+ // SPEC Parsing
180
+ // ============================================
181
+
182
+ function parseSpecForBrand(specPath) {
183
+ if (!fs.existsSync(specPath)) {
184
+ return null;
185
+ }
186
+
187
+ const content = fs.readFileSync(specPath, 'utf8');
188
+ const brand = {
189
+ name: null,
190
+ color: null,
191
+ style: null,
192
+ concept: null
193
+ };
194
+
195
+ // Extract app name from title or context
196
+ const titleMatch = content.match(/^#\s+(.+?)(?:\s+SPEC)?$/m);
197
+ if (titleMatch) {
198
+ brand.name = titleMatch[1].trim();
199
+ }
200
+
201
+ // Extract from context section
202
+ const contextMatch = content.match(/<context>([\s\S]*?)<\/context>/i);
203
+ if (contextMatch) {
204
+ const context = contextMatch[1];
205
+
206
+ // App Name
207
+ const nameMatch = context.match(/App\s*Name[:\s]+([^\n]+)/i);
208
+ if (nameMatch) brand.name = nameMatch[1].trim();
209
+
210
+ // Primary Color
211
+ const colorMatch = context.match(/Primary\s*Color[:\s]+(#[0-9A-Fa-f]{6}|#[0-9A-Fa-f]{3})/i);
212
+ if (colorMatch) brand.color = colorMatch[1];
213
+
214
+ // Style
215
+ const styleMatch = context.match(/Style[:\s]+([^\n]+)/i);
216
+ if (styleMatch) brand.style = styleMatch[1].trim();
217
+
218
+ // Icon Concept
219
+ const conceptMatch = context.match(/Icon\s*Concept[:\s]+([^\n]+)/i);
220
+ if (conceptMatch) brand.concept = conceptMatch[1].trim();
221
+ }
222
+
223
+ // Fallback: extract from any brand-related content
224
+ if (!brand.color) {
225
+ const anyColor = content.match(/#([0-9A-Fa-f]{6}|[0-9A-Fa-f]{3})\b/);
226
+ if (anyColor) brand.color = '#' + anyColor[1];
227
+ }
228
+
229
+ return brand;
230
+ }
231
+
232
+ // ============================================
233
+ // Prompt Generation
234
+ // ============================================
235
+
236
+ function buildIconPrompt(brand) {
237
+ const parts = [];
238
+
239
+ if (brand.name) {
240
+ parts.push(`App icon for "${brand.name}"`);
241
+ } else {
242
+ parts.push('Modern app icon');
243
+ }
244
+
245
+ if (brand.concept) {
246
+ parts.push(`Concept: ${brand.concept}`);
247
+ }
248
+
249
+ if (brand.style) {
250
+ parts.push(`Style: ${brand.style}`);
251
+ } else {
252
+ parts.push('Style: Modern, minimalist, professional');
253
+ }
254
+
255
+ if (brand.color) {
256
+ parts.push(`Primary color: ${brand.color}`);
257
+ }
258
+
259
+ return parts.join('. ');
260
+ }
261
+
262
+ // ============================================
263
+ // Fallback: Text Monogram
264
+ // ============================================
265
+
266
+ function generateFallbackIcon(name, color, outputPath) {
267
+ // Generate simple SVG monogram
268
+ const initial = (name || 'A')[0].toUpperCase();
269
+ const bgColor = color || '#4A90D9';
270
+
271
+ const svg = `<?xml version="1.0" encoding="UTF-8"?>
272
+ <svg width="512" height="512" viewBox="0 0 512 512" xmlns="http://www.w3.org/2000/svg">
273
+ <rect width="512" height="512" rx="64" fill="${bgColor}"/>
274
+ <text x="256" y="320" font-family="Arial, sans-serif" font-size="280" font-weight="bold" fill="white" text-anchor="middle">${initial}</text>
275
+ </svg>`;
276
+
277
+ const svgPath = outputPath.replace(/\.(png|ico)$/, '.svg');
278
+ fs.writeFileSync(svgPath, svg);
279
+
280
+ // Convert SVG to PNG if possible
281
+ try {
282
+ execSync(`convert "${svgPath}" -resize 512x512 "${outputPath}"`, { stdio: 'pipe' });
283
+ fs.unlinkSync(svgPath);
284
+ return true;
285
+ } catch {
286
+ // Keep SVG as fallback
287
+ console.log(` Created SVG fallback: ${svgPath}`);
288
+ return false;
289
+ }
290
+ }
291
+
292
+ // ============================================
293
+ // Manifest Generation
294
+ // ============================================
295
+
296
+ function generateWebManifest(outputDir, name) {
297
+ const manifest = {
298
+ name: name || 'App',
299
+ short_name: name || 'App',
300
+ icons: [
301
+ { src: '/android-chrome-192x192.png', sizes: '192x192', type: 'image/png' },
302
+ { src: '/android-chrome-512x512.png', sizes: '512x512', type: 'image/png' }
303
+ ],
304
+ theme_color: '#ffffff',
305
+ background_color: '#ffffff',
306
+ display: 'standalone'
307
+ };
308
+
309
+ const manifestPath = path.join(outputDir, 'site.webmanifest');
310
+ fs.writeFileSync(manifestPath, JSON.stringify(manifest, null, 2));
311
+ return manifestPath;
312
+ }
313
+
314
+ // ============================================
315
+ // Main
316
+ // ============================================
317
+
318
+ async function main() {
319
+ const args = process.argv.slice(2);
320
+
321
+ // Parse arguments
322
+ let name = null;
323
+ let color = null;
324
+ let style = null;
325
+ let specPath = null;
326
+ let outputDir = './public';
327
+ let force = false;
328
+
329
+ for (let i = 0; i < args.length; i++) {
330
+ switch (args[i]) {
331
+ case '--name':
332
+ name = args[++i];
333
+ break;
334
+ case '--color':
335
+ color = args[++i];
336
+ break;
337
+ case '--style':
338
+ style = args[++i];
339
+ break;
340
+ case '--spec':
341
+ specPath = args[++i];
342
+ break;
343
+ case '--output':
344
+ outputDir = args[++i];
345
+ break;
346
+ case '--force':
347
+ case '--regenerate':
348
+ force = true;
349
+ break;
350
+ case '--help':
351
+ console.log(`
352
+ Brand Assets Generator
353
+
354
+ Usage:
355
+ node generate-brand-assets.js [options]
356
+
357
+ Options:
358
+ --name <name> App name
359
+ --color <hex> Primary color (e.g., #2F6BFF)
360
+ --style <style> Design style keywords
361
+ --spec <path> Path to SPEC file to extract brand info
362
+ --output <dir> Output directory (default: ./public)
363
+ --force Regenerate even if icons exist
364
+
365
+ Examples:
366
+ node generate-brand-assets.js --name "MyApp" --color "#FF5722" --output "./public"
367
+ node generate-brand-assets.js --spec ".claude/vibe/specs/my-feature.md"
368
+ `);
369
+ process.exit(0);
370
+ }
371
+ }
372
+
373
+ // Check if icons already exist
374
+ const faviconPath = path.join(outputDir, 'favicon.ico');
375
+ if (fs.existsSync(faviconPath) && !force) {
376
+ console.log('Brand assets already exist. Use --force to regenerate.');
377
+ process.exit(0);
378
+ }
379
+
380
+ // Parse SPEC if provided
381
+ if (specPath) {
382
+ const brand = parseSpecForBrand(specPath);
383
+ if (brand) {
384
+ name = name || brand.name;
385
+ color = color || brand.color;
386
+ style = style || brand.style;
387
+ }
388
+ }
389
+
390
+ // Ensure output directory
391
+ if (!fs.existsSync(outputDir)) {
392
+ fs.mkdirSync(outputDir, { recursive: true });
393
+ }
394
+
395
+ // Check Gemini API key
396
+ const apiKey = getGeminiApiKey();
397
+
398
+ console.log('Generating brand assets...');
399
+ console.log(` Name: ${name || '(not specified)'}`);
400
+ console.log(` Color: ${color || '(not specified)'}`);
401
+ console.log(` Style: ${style || '(not specified)'}`);
402
+ console.log(` Output: ${outputDir}`);
403
+ console.log(` Gemini: ${apiKey ? 'configured' : 'not configured'}`);
404
+ console.log('');
405
+
406
+ const masterIconPath = path.join(outputDir, 'icon-master.png');
407
+ let generated = false;
408
+
409
+ // Try Gemini Image API
410
+ if (apiKey) {
411
+ try {
412
+ const prompt = buildIconPrompt({ name, color, style });
413
+ console.log('Generating icon with Gemini...');
414
+ console.log(` Prompt: ${prompt}`);
415
+
416
+ const image = await generateImageWithGemini(prompt, apiKey);
417
+ fs.writeFileSync(masterIconPath, image.data);
418
+ console.log(' Master icon generated successfully');
419
+ generated = true;
420
+ } catch (err) {
421
+ console.log(` Gemini generation failed: ${err.message}`);
422
+ console.log(' Falling back to text monogram...');
423
+ }
424
+ }
425
+
426
+ // Fallback to monogram
427
+ if (!generated) {
428
+ console.log('Generating fallback monogram icon...');
429
+ generateFallbackIcon(name, color, masterIconPath);
430
+ if (fs.existsSync(masterIconPath)) {
431
+ generated = true;
432
+ }
433
+ }
434
+
435
+ if (!generated) {
436
+ console.error('Failed to generate icon');
437
+ process.exit(1);
438
+ }
439
+
440
+ // Resize to all required sizes
441
+ console.log('Resizing to platform sizes...');
442
+ for (const [filename, size] of Object.entries(ICON_SIZES)) {
443
+ const outPath = path.join(outputDir, filename);
444
+ if (resizeImage(masterIconPath, outPath, size)) {
445
+ console.log(` Created: ${filename}`);
446
+ } else {
447
+ console.log(` Warning: Could not create ${filename}`);
448
+ }
449
+ }
450
+
451
+ // Create favicon.ico
452
+ console.log('Creating favicon.ico...');
453
+ if (createFavicon(masterIconPath, faviconPath)) {
454
+ console.log(' Created: favicon.ico');
455
+ }
456
+
457
+ // Generate web manifest
458
+ const manifestPath = generateWebManifest(outputDir, name);
459
+ console.log(` Created: ${path.basename(manifestPath)}`);
460
+
461
+ // Cleanup master icon
462
+ if (fs.existsSync(masterIconPath)) {
463
+ fs.unlinkSync(masterIconPath);
464
+ }
465
+
466
+ console.log('');
467
+ console.log('Brand assets generation complete!');
468
+ console.log(`Output directory: ${outputDir}`);
469
+ }
470
+
471
+ main().catch(err => {
472
+ console.error('Error:', err.message);
473
+ process.exit(1);
474
+ });