sonamu 0.8.5 → 0.8.6
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.
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"dev-vitest-manager.d.ts","sourceRoot":"","sources":["../../src/testing/dev-vitest-manager.ts"],"names":[],"mappings":"AAUA,MAAM,MAAM,SAAS,GAAG;IACtB,EAAE,EAAE,OAAO,CAAC;IACZ,OAAO,EAAE;QACP,KAAK,EAAE,MAAM,CAAC;QACd,MAAM,EAAE,MAAM,CAAC;QACf,MAAM,EAAE,MAAM,CAAC;QACf,OAAO,EAAE,MAAM,CAAC;QAChB,UAAU,EAAE,MAAM,CAAC;KACpB,CAAC;IACF,MAAM,EAAE,UAAU,EAAE,CAAC;CACtB,CAAC;AAEF,MAAM,MAAM,UAAU,GAAG;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;CACf,CAAC;AAEF,MAAM,MAAM,aAAa,GAAG;IAC1B,KAAK,EAAE,OAAO,CAAC;IACf,OAAO,EAAE,OAAO,CAAC;IACjB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;CAC1B,CAAC;AAQF,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,SAAS,CAAuB;IACxC,OAAO,CAAC,KAAK,CAAoB;IACjC,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,MAAM,CAAS;IAEjB,KAAK,CAAC,gBAAgB,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA6B/C,GAAG,CAAC,IAAI,EAAE;QAAE,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;QAAC,OAAO,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,OAAO,CAAC,SAAS,CAAC;IAe3E,SAAS,IAAI,aAAa;IAQpB,QAAQ,IAAI,OAAO,CAAC,IAAI,CAAC;
|
|
1
|
+
{"version":3,"file":"dev-vitest-manager.d.ts","sourceRoot":"","sources":["../../src/testing/dev-vitest-manager.ts"],"names":[],"mappings":"AAUA,MAAM,MAAM,SAAS,GAAG;IACtB,EAAE,EAAE,OAAO,CAAC;IACZ,OAAO,EAAE;QACP,KAAK,EAAE,MAAM,CAAC;QACd,MAAM,EAAE,MAAM,CAAC;QACf,MAAM,EAAE,MAAM,CAAC;QACf,OAAO,EAAE,MAAM,CAAC;QAChB,UAAU,EAAE,MAAM,CAAC;KACpB,CAAC;IACF,MAAM,EAAE,UAAU,EAAE,CAAC;CACtB,CAAC;AAEF,MAAM,MAAM,UAAU,GAAG;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;CACf,CAAC;AAEF,MAAM,MAAM,aAAa,GAAG;IAC1B,KAAK,EAAE,OAAO,CAAC;IACf,OAAO,EAAE,OAAO,CAAC;IACjB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;CAC1B,CAAC;AAQF,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,SAAS,CAAuB;IACxC,OAAO,CAAC,KAAK,CAAoB;IACjC,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,MAAM,CAAS;IAEjB,KAAK,CAAC,gBAAgB,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA6B/C,GAAG,CAAC,IAAI,EAAE;QAAE,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;QAAC,OAAO,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,OAAO,CAAC,SAAS,CAAC;IAe3E,SAAS,IAAI,aAAa;IAQpB,QAAQ,IAAI,OAAO,CAAC,IAAI,CAAC;YAoBjB,YAAY;YA2BZ,UAAU;IAkCxB,OAAO,CAAC,cAAc;IA4BtB,OAAO,CAAC,iBAAiB;IA4BzB,OAAO,CAAC,iBAAiB;CAe1B"}
|
|
@@ -59,6 +59,13 @@ export class DevVitestManager {
|
|
|
59
59
|
return;
|
|
60
60
|
}
|
|
61
61
|
this.closed = true;
|
|
62
|
+
// 큐에 남은 작업들을 reject하여 호출자가 영구 대기하지 않도록 정리
|
|
63
|
+
while(this.queue.length > 0){
|
|
64
|
+
const entry = this.queue.shift();
|
|
65
|
+
if (entry) {
|
|
66
|
+
entry.reject(new Error("DevVitestManager is being shut down"));
|
|
67
|
+
}
|
|
68
|
+
}
|
|
62
69
|
if (this.vitest) {
|
|
63
70
|
await this.vitest.close();
|
|
64
71
|
this.vitest = null;
|
|
@@ -180,4 +187,4 @@ export class DevVitestManager {
|
|
|
180
187
|
}
|
|
181
188
|
}
|
|
182
189
|
|
|
183
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/testing/dev-vitest-manager.ts"],"sourcesContent":["import type { UserConfig as ViteUserConfig } from \"vite\";\nimport type {\n  CliOptions,\n  TestCase,\n  TestModule,\n  TestRunResult,\n  TestSpecification,\n  Vitest,\n} from \"vitest/node\";\n\nexport type RunResult = {\n  ok: boolean;\n  summary: {\n    total: number;\n    passed: number;\n    failed: number;\n    skipped: number;\n    durationMs: number;\n  };\n  failed: FailedTest[];\n};\n\nexport type FailedTest = {\n  name: string;\n  file: string;\n  error: string;\n};\n\nexport type ManagerStatus = {\n  ready: boolean;\n  running: boolean;\n  lastRunAt: string | null;\n};\n\ntype QueueEntry = {\n  task: () => Promise<RunResult>;\n  resolve: (result: RunResult) => void;\n  reject: (error: unknown) => void;\n};\n\nexport class DevVitestManager {\n  private vitest: Vitest | null = null;\n  private running = false;\n  private lastRunAt: string | null = null;\n  private queue: QueueEntry[] = [];\n  private processing = false;\n  private closed = false;\n\n  async start(vitestConfigPath?: string): Promise<void> {\n    // 이미 시작된 경우 중복 초기화를 방지\n    if (this.vitest) {\n      return;\n    }\n\n    const { createVitest } = await import(\"vitest/node\");\n\n    const viteOverrides: ViteUserConfig = {\n      server: { watch: null },\n    };\n\n    const cliOptions: CliOptions = {\n      watch: true,\n      standalone: true,\n      forceRerunTriggers: [],\n      config: vitestConfigPath,\n      env: {\n        NODE_ENV: \"test\",\n      },\n    };\n\n    this.vitest = await createVitest(\"test\", cliOptions, viteOverrides);\n    await this.vitest.init();\n\n    this.vitest.onFilterWatchedSpecification((_spec) => false);\n    this.closed = false;\n  }\n\n  async run(opts: { files?: string[]; pattern?: string }): Promise<RunResult> {\n    if (this.closed) {\n      throw new Error(\"DevVitestManager is already shut down\");\n    }\n    if (!this.vitest) {\n      throw new Error(\"DevVitestManager is not started\");\n    }\n\n    return new Promise<RunResult>((resolve, reject) => {\n      const task = () => this.executeRun(opts);\n      this.queue.push({ task, resolve, reject });\n      this.processQueue();\n    });\n  }\n\n  getStatus(): ManagerStatus {\n    return {\n      ready: this.vitest !== null && !this.closed,\n      running: this.running,\n      lastRunAt: this.lastRunAt,\n    };\n  }\n\n  async shutdown(): Promise<void> {\n    if (this.closed) {\n      return;\n    }\n    this.closed = true;\n\n    if (this.vitest) {\n      await this.vitest.close();\n      this.vitest = null;\n    }\n  }\n\n  private async processQueue(): Promise<void> {\n    if (this.processing) {\n      return;\n    }\n    this.processing = true;\n\n    try {\n      while (this.queue.length > 0) {\n        const entry = this.queue.shift();\n        if (!entry) break;\n        if (this.closed) {\n          entry.reject(new Error(\"DevVitestManager is already shut down\"));\n          continue;\n        }\n\n        try {\n          const result = await entry.task();\n          entry.resolve(result);\n        } catch (err) {\n          entry.reject(err);\n        }\n      }\n    } finally {\n      this.processing = false;\n    }\n  }\n\n  private async executeRun(opts: { files?: string[]; pattern?: string }): Promise<RunResult> {\n    const vitest = this.vitest;\n    if (!vitest) {\n      throw new Error(\"DevVitestManager is not started\");\n    }\n\n    this.running = true;\n    const startTime = Date.now();\n\n    if (opts.pattern) {\n      vitest.setGlobalTestNamePattern(opts.pattern);\n    }\n\n    try {\n      const specs: TestSpecification[] = opts.files\n        ? await vitest.globTestSpecifications(opts.files)\n        : await vitest.globTestSpecifications();\n\n      const allTestsRun = !opts.files || opts.files.length === 0;\n      const runResult: TestRunResult = await vitest.runTestSpecifications(specs, allTestsRun);\n\n      const durationMs = Date.now() - startTime;\n      this.lastRunAt = new Date().toISOString();\n\n      const specModuleIds = new Set(specs.map((s) => s.moduleId));\n      return this.collectResults(runResult, durationMs, specModuleIds);\n    } finally {\n      if (opts.pattern) {\n        vitest.resetGlobalTestNamePattern();\n      }\n      this.running = false;\n    }\n  }\n\n  private collectResults(\n    runResult: TestRunResult,\n    durationMs: number,\n    specModuleIds: Set<string>,\n  ): RunResult {\n    let total = 0;\n    let passed = 0;\n    let failed = 0;\n    let skipped = 0;\n    const failedTests: FailedTest[] = [];\n\n    for (const testModule of runResult.testModules) {\n      if (!specModuleIds.has(testModule.moduleId)) continue;\n      this.collectFromModule(testModule, failedTests, (counts) => {\n        total += counts.total;\n        passed += counts.passed;\n        failed += counts.failed;\n        skipped += counts.skipped;\n      });\n    }\n\n    return {\n      ok: failed === 0,\n      summary: { total, passed, failed, skipped, durationMs },\n      failed: failedTests,\n    };\n  }\n\n  private collectFromModule(\n    testModule: TestModule,\n    failedTests: FailedTest[],\n    addCounts: (counts: { total: number; passed: number; failed: number; skipped: number }) => void,\n  ): void {\n    let total = 0;\n    let passed = 0;\n    let failed = 0;\n    let skipped = 0;\n\n    for (const testCase of testModule.children.allTests()) {\n      total++;\n      const result = testCase.result();\n\n      if (result.state === \"passed\") {\n        passed++;\n      } else if (result.state === \"failed\") {\n        failed++;\n        failedTests.push(this.extractFailedTest(testCase, testModule));\n      } else {\n        // pending/skipped 상태는 skipped로 집계\n        skipped++;\n      }\n    }\n\n    addCounts({ total, passed, failed, skipped });\n  }\n\n  private extractFailedTest(testCase: TestCase, testModule: TestModule): FailedTest {\n    const result = testCase.result();\n    let errorMessage = \"Unknown error\";\n\n    if (result.state === \"failed\" && result.errors.length > 0) {\n      const firstError = result.errors[0];\n      errorMessage = firstError.message ?? String(firstError);\n    }\n\n    return {\n      name: testCase.fullName,\n      file: testModule.moduleId,\n      error: errorMessage,\n    };\n  }\n}\n"],"names":["DevVitestManager","vitest","running","lastRunAt","queue","processing","closed","start","vitestConfigPath","createVitest","viteOverrides","server","watch","cliOptions","standalone","forceRerunTriggers","config","env","NODE_ENV","init","onFilterWatchedSpecification","_spec","run","opts","Error","Promise","resolve","reject","task","executeRun","push","processQueue","getStatus","ready","shutdown","close","length","entry","shift","result","err","startTime","Date","now","pattern","setGlobalTestNamePattern","specs","files","globTestSpecifications","allTestsRun","runResult","runTestSpecifications","durationMs","toISOString","specModuleIds","Set","map","s","moduleId","collectResults","resetGlobalTestNamePattern","total","passed","failed","skipped","failedTests","testModule","testModules","has","collectFromModule","counts","ok","summary","addCounts","testCase","children","allTests","state","extractFailedTest","errorMessage","errors","firstError","message","String","name","fullName","file","error"],"mappings":"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"}
|
|
190
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/testing/dev-vitest-manager.ts"],"sourcesContent":["import type { UserConfig as ViteUserConfig } from \"vite\";\nimport type {\n  CliOptions,\n  TestCase,\n  TestModule,\n  TestRunResult,\n  TestSpecification,\n  Vitest,\n} from \"vitest/node\";\n\nexport type RunResult = {\n  ok: boolean;\n  summary: {\n    total: number;\n    passed: number;\n    failed: number;\n    skipped: number;\n    durationMs: number;\n  };\n  failed: FailedTest[];\n};\n\nexport type FailedTest = {\n  name: string;\n  file: string;\n  error: string;\n};\n\nexport type ManagerStatus = {\n  ready: boolean;\n  running: boolean;\n  lastRunAt: string | null;\n};\n\ntype QueueEntry = {\n  task: () => Promise<RunResult>;\n  resolve: (result: RunResult) => void;\n  reject: (error: unknown) => void;\n};\n\nexport class DevVitestManager {\n  private vitest: Vitest | null = null;\n  private running = false;\n  private lastRunAt: string | null = null;\n  private queue: QueueEntry[] = [];\n  private processing = false;\n  private closed = false;\n\n  async start(vitestConfigPath?: string): Promise<void> {\n    // 이미 시작된 경우 중복 초기화를 방지\n    if (this.vitest) {\n      return;\n    }\n\n    const { createVitest } = await import(\"vitest/node\");\n\n    const viteOverrides: ViteUserConfig = {\n      server: { watch: null },\n    };\n\n    const cliOptions: CliOptions = {\n      watch: true,\n      standalone: true,\n      forceRerunTriggers: [],\n      config: vitestConfigPath,\n      env: {\n        NODE_ENV: \"test\",\n      },\n    };\n\n    this.vitest = await createVitest(\"test\", cliOptions, viteOverrides);\n    await this.vitest.init();\n\n    this.vitest.onFilterWatchedSpecification((_spec) => false);\n    this.closed = false;\n  }\n\n  async run(opts: { files?: string[]; pattern?: string }): Promise<RunResult> {\n    if (this.closed) {\n      throw new Error(\"DevVitestManager is already shut down\");\n    }\n    if (!this.vitest) {\n      throw new Error(\"DevVitestManager is not started\");\n    }\n\n    return new Promise<RunResult>((resolve, reject) => {\n      const task = () => this.executeRun(opts);\n      this.queue.push({ task, resolve, reject });\n      this.processQueue();\n    });\n  }\n\n  getStatus(): ManagerStatus {\n    return {\n      ready: this.vitest !== null && !this.closed,\n      running: this.running,\n      lastRunAt: this.lastRunAt,\n    };\n  }\n\n  async shutdown(): Promise<void> {\n    if (this.closed) {\n      return;\n    }\n    this.closed = true;\n\n    // 큐에 남은 작업들을 reject하여 호출자가 영구 대기하지 않도록 정리\n    while (this.queue.length > 0) {\n      const entry = this.queue.shift();\n      if (entry) {\n        entry.reject(new Error(\"DevVitestManager is being shut down\"));\n      }\n    }\n\n    if (this.vitest) {\n      await this.vitest.close();\n      this.vitest = null;\n    }\n  }\n\n  private async processQueue(): Promise<void> {\n    if (this.processing) {\n      return;\n    }\n    this.processing = true;\n\n    try {\n      while (this.queue.length > 0) {\n        const entry = this.queue.shift();\n        if (!entry) break;\n        if (this.closed) {\n          entry.reject(new Error(\"DevVitestManager is already shut down\"));\n          continue;\n        }\n\n        try {\n          const result = await entry.task();\n          entry.resolve(result);\n        } catch (err) {\n          entry.reject(err);\n        }\n      }\n    } finally {\n      this.processing = false;\n    }\n  }\n\n  private async executeRun(opts: { files?: string[]; pattern?: string }): Promise<RunResult> {\n    const vitest = this.vitest;\n    if (!vitest) {\n      throw new Error(\"DevVitestManager is not started\");\n    }\n\n    this.running = true;\n    const startTime = Date.now();\n\n    if (opts.pattern) {\n      vitest.setGlobalTestNamePattern(opts.pattern);\n    }\n\n    try {\n      const specs: TestSpecification[] = opts.files\n        ? await vitest.globTestSpecifications(opts.files)\n        : await vitest.globTestSpecifications();\n\n      const allTestsRun = !opts.files || opts.files.length === 0;\n      const runResult: TestRunResult = await vitest.runTestSpecifications(specs, allTestsRun);\n\n      const durationMs = Date.now() - startTime;\n      this.lastRunAt = new Date().toISOString();\n\n      const specModuleIds = new Set(specs.map((s) => s.moduleId));\n      return this.collectResults(runResult, durationMs, specModuleIds);\n    } finally {\n      if (opts.pattern) {\n        vitest.resetGlobalTestNamePattern();\n      }\n      this.running = false;\n    }\n  }\n\n  private collectResults(\n    runResult: TestRunResult,\n    durationMs: number,\n    specModuleIds: Set<string>,\n  ): RunResult {\n    let total = 0;\n    let passed = 0;\n    let failed = 0;\n    let skipped = 0;\n    const failedTests: FailedTest[] = [];\n\n    for (const testModule of runResult.testModules) {\n      if (!specModuleIds.has(testModule.moduleId)) continue;\n      this.collectFromModule(testModule, failedTests, (counts) => {\n        total += counts.total;\n        passed += counts.passed;\n        failed += counts.failed;\n        skipped += counts.skipped;\n      });\n    }\n\n    return {\n      ok: failed === 0,\n      summary: { total, passed, failed, skipped, durationMs },\n      failed: failedTests,\n    };\n  }\n\n  private collectFromModule(\n    testModule: TestModule,\n    failedTests: FailedTest[],\n    addCounts: (counts: { total: number; passed: number; failed: number; skipped: number }) => void,\n  ): void {\n    let total = 0;\n    let passed = 0;\n    let failed = 0;\n    let skipped = 0;\n\n    for (const testCase of testModule.children.allTests()) {\n      total++;\n      const result = testCase.result();\n\n      if (result.state === \"passed\") {\n        passed++;\n      } else if (result.state === \"failed\") {\n        failed++;\n        failedTests.push(this.extractFailedTest(testCase, testModule));\n      } else {\n        // pending/skipped 상태는 skipped로 집계\n        skipped++;\n      }\n    }\n\n    addCounts({ total, passed, failed, skipped });\n  }\n\n  private extractFailedTest(testCase: TestCase, testModule: TestModule): FailedTest {\n    const result = testCase.result();\n    let errorMessage = \"Unknown error\";\n\n    if (result.state === \"failed\" && result.errors.length > 0) {\n      const firstError = result.errors[0];\n      errorMessage = firstError.message ?? String(firstError);\n    }\n\n    return {\n      name: testCase.fullName,\n      file: testModule.moduleId,\n      error: errorMessage,\n    };\n  }\n}\n"],"names":["DevVitestManager","vitest","running","lastRunAt","queue","processing","closed","start","vitestConfigPath","createVitest","viteOverrides","server","watch","cliOptions","standalone","forceRerunTriggers","config","env","NODE_ENV","init","onFilterWatchedSpecification","_spec","run","opts","Error","Promise","resolve","reject","task","executeRun","push","processQueue","getStatus","ready","shutdown","length","entry","shift","close","result","err","startTime","Date","now","pattern","setGlobalTestNamePattern","specs","files","globTestSpecifications","allTestsRun","runResult","runTestSpecifications","durationMs","toISOString","specModuleIds","Set","map","s","moduleId","collectResults","resetGlobalTestNamePattern","total","passed","failed","skipped","failedTests","testModule","testModules","has","collectFromModule","counts","ok","summary","addCounts","testCase","children","allTests","state","extractFailedTest","errorMessage","errors","firstError","message","String","name","fullName","file","error"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "sonamu",
|
|
3
|
-
"version": "0.8.
|
|
3
|
+
"version": "0.8.6",
|
|
4
4
|
"description": "Sonamu — TypeScript Fullstack API Framework",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"typescript",
|
|
@@ -119,10 +119,10 @@
|
|
|
119
119
|
"tsicli": "^1.0.5",
|
|
120
120
|
"vite": "7.3.0",
|
|
121
121
|
"vitest": "^4.0.10",
|
|
122
|
-
"@sonamu-kit/hmr-hook": "^0.4.1",
|
|
123
122
|
"@sonamu-kit/hmr-runner": "^0.1.1",
|
|
124
123
|
"@sonamu-kit/tasks": "^0.2.0",
|
|
125
|
-
"@sonamu-kit/ts-loader": "^2.1.3"
|
|
124
|
+
"@sonamu-kit/ts-loader": "^2.1.3",
|
|
125
|
+
"@sonamu-kit/hmr-hook": "^0.4.1"
|
|
126
126
|
},
|
|
127
127
|
"devDependencies": {
|
|
128
128
|
"@biomejs/biome": "^2.3.13",
|
|
@@ -25,65 +25,21 @@ Sonamu 프레임워크로 프로젝트를 개발하기 위한 Claude Code skill
|
|
|
25
25
|
|
|
26
26
|
## 개발 흐름
|
|
27
27
|
|
|
28
|
-
|
|
29
|
-
```bash
|
|
30
|
-
pnpm create sonamu [프로젝트명] --yes
|
|
31
|
-
```
|
|
28
|
+
**CRITICAL: 모든 개발은 반드시 `workflow.md`의 PHASE 0 → 7 순서를 따른다. 단계를 건너뛰거나 순서를 바꾸지 않는다.**
|
|
32
29
|
|
|
33
|
-
### 2. Sonamu 링크 설정 (선택)
|
|
34
|
-
**Skills 원본 동기화가 필요한 경우만** `pnpm-workspace.yaml`에 추가:
|
|
35
|
-
```yaml
|
|
36
|
-
overrides:
|
|
37
|
-
sonamu: link:../../sonamu/modules/sonamu
|
|
38
30
|
```
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
31
|
+
PHASE 0: 프로젝트 초기화 (프로젝트 생성 + auth generate)
|
|
32
|
+
PHASE 1: 엔티티 설계 (Auth 엔티티 확인 후 나머지 설계)
|
|
33
|
+
PHASE 2: 엔티티 생성
|
|
34
|
+
PHASE 3: 마이그레이션
|
|
35
|
+
PHASE 4: Cone 생성 (LLM 사용 여부 사용자에게 확인 후 진행)
|
|
36
|
+
PHASE 5: 스캐폴딩
|
|
37
|
+
PHASE 6: 테스트 작성 (완료 후 Fixture 생성 여부 확인, LLM 사용 여부도 확인)
|
|
38
|
+
PHASE 7: API 개발
|
|
39
|
+
PHASE 8: Frontend 개발
|
|
46
40
|
```
|
|
47
|
-
> 빌드 실패 시 → 4번 도커 먼저 실행 후 5번 dev 서버 올리고 재시도
|
|
48
|
-
|
|
49
|
-
### 4. Docker 실행
|
|
50
|
-
```bash
|
|
51
|
-
cd packages/api
|
|
52
|
-
pnpm docker:up
|
|
53
|
-
```
|
|
54
|
-
|
|
55
|
-
### 5. 개발 서버 실행
|
|
56
|
-
```bash
|
|
57
|
-
pnpm dev
|
|
58
|
-
```
|
|
59
|
-
> 3번 빌드가 실패했다면, dev 서버가 올라온 후 프로젝트 루트에서 `pnpm -r build` 재시도
|
|
60
|
-
|
|
61
|
-
### 6. Auth 엔티티 생성 (별도 터미널)
|
|
62
|
-
**dev 실행 중**에:
|
|
63
|
-
```bash
|
|
64
|
-
pnpm sonamu auth generate
|
|
65
|
-
```
|
|
66
|
-
> dev 모드에서 실행해야 types도 자동 생성됨
|
|
67
|
-
|
|
68
|
-
### 7. Subset 확인
|
|
69
|
-
Sonamu UI Entity 메뉴에서 subset 체크
|
|
70
|
-
|
|
71
|
-
### 8. Migration
|
|
72
|
-
```bash
|
|
73
|
-
pnpm sonamu migrate run
|
|
74
|
-
```
|
|
75
|
-
|
|
76
|
-
### 9. Scaffolding
|
|
77
|
-
Sonamu UI에서 Model/View Scaffolding 실행
|
|
78
|
-
|
|
79
|
-
### 10. API 단위테스트
|
|
80
|
-
```bash
|
|
81
|
-
pnpm test:watch
|
|
82
|
-
```
|
|
83
|
-
|
|
84
|
-
### 11. 프론트엔드 개발
|
|
85
41
|
|
|
86
|
-
**상세 내용:** `
|
|
42
|
+
**상세 내용:** `workflow.md` 참조
|
|
87
43
|
|
|
88
44
|
---
|
|
89
45
|
|
|
@@ -5,12 +5,79 @@ description: Sonamu 전체 개발 워크플로우. 엔티티 설계부터 Fronte
|
|
|
5
5
|
|
|
6
6
|
# Sonamu 전체 개발 워크플로우
|
|
7
7
|
|
|
8
|
-
사용자가 시스템 구축을 요청하면 다음
|
|
8
|
+
사용자가 시스템 구축을 요청하면 다음 9단계로 진행한다.
|
|
9
|
+
|
|
10
|
+
**CRITICAL: 이 워크플로우는 반드시 순서대로 진행한다. 단계를 건너뛰거나 순서를 바꾸지 않는다.**
|
|
9
11
|
|
|
10
12
|
**CRITICAL: 요구사항이 이미 제공된 경우에도 설계 및 비즈니스 로직은 반드시 사용자와 함께 확인한다.**
|
|
11
13
|
요구사항 명세는 출발점일 뿐이다. Entity 구조, 관계, 필드, 상태 전이, 권한 규칙 등은 항상 사용자에게 질문하고 승인을 받아야 한다. 과거에 비슷한 요구사항을 받은 적이 있더라도 이번 프로젝트의 설계는 새로 확인한다.
|
|
12
14
|
|
|
13
|
-
## 사용자 요청 → 완성까지
|
|
15
|
+
## 사용자 요청 → 완성까지 9단계
|
|
16
|
+
|
|
17
|
+
### PHASE 0: 프로젝트 초기화
|
|
18
|
+
|
|
19
|
+
**목표:** 프로젝트 생성 및 better-auth 기반 엔티티 준비
|
|
20
|
+
|
|
21
|
+
**참조 스킬:** project-init.md, create-sonamu.md, auth.md
|
|
22
|
+
|
|
23
|
+
**절차:**
|
|
24
|
+
|
|
25
|
+
1. **작업 경로 확인**
|
|
26
|
+
- 현재 디렉토리 또는 지정 디렉토리 확인
|
|
27
|
+
- `project-init.md` "0. 작업 경로 확인" 참조
|
|
28
|
+
|
|
29
|
+
2. **프로젝트 생성**
|
|
30
|
+
```bash
|
|
31
|
+
pnpm create sonamu [프로젝트명] --yes
|
|
32
|
+
```
|
|
33
|
+
기존 프로젝트면 경로 확인 후 진행.
|
|
34
|
+
|
|
35
|
+
3. **요구사항 문서화** (요구사항 제공 시)
|
|
36
|
+
- `.claude/skills/project/requirements.md` 작성
|
|
37
|
+
- `project-init.md` "요구사항 문서화" 참조
|
|
38
|
+
|
|
39
|
+
4. **의존성 설치 및 빌드**
|
|
40
|
+
```bash
|
|
41
|
+
pnpm install
|
|
42
|
+
pnpm -r build
|
|
43
|
+
```
|
|
44
|
+
> 빌드 실패 시 → 5번 Docker 먼저 실행 후 6번 dev 서버 올리고 재시도
|
|
45
|
+
|
|
46
|
+
5. **Docker 실행**
|
|
47
|
+
```bash
|
|
48
|
+
cd packages/api
|
|
49
|
+
pnpm docker:up
|
|
50
|
+
```
|
|
51
|
+
|
|
52
|
+
6. **개발 서버 실행**
|
|
53
|
+
```bash
|
|
54
|
+
pnpm dev
|
|
55
|
+
```
|
|
56
|
+
|
|
57
|
+
7. **Auth 엔티티 생성** (별도 터미널, dev 실행 중)
|
|
58
|
+
```bash
|
|
59
|
+
cd packages/api
|
|
60
|
+
pnpm sonamu auth generate
|
|
61
|
+
```
|
|
62
|
+
생성되는 엔티티: `User`, `Session`, `Account`, `Verification`
|
|
63
|
+
> dev 모드에서 실행해야 types 파일도 자동 생성됨
|
|
64
|
+
|
|
65
|
+
8. **생성된 Auth 엔티티 확인**
|
|
66
|
+
- Sonamu UI (`http://localhost:34900/sonamu-ui`) → Entity 메뉴에서 4개 엔티티 확인
|
|
67
|
+
- **User 엔티티 필드 파악** (id 타입, 기본 제공 필드 목록)
|
|
68
|
+
- 이후 PHASE 1 엔티티 설계 시 User와의 관계를 고려해야 하므로 반드시 확인
|
|
69
|
+
|
|
70
|
+
**완료 기준:**
|
|
71
|
+
|
|
72
|
+
- [ ] 프로젝트 생성 완료
|
|
73
|
+
- [ ] Docker 실행 중
|
|
74
|
+
- [ ] dev 서버 실행 중
|
|
75
|
+
- [ ] `pnpm sonamu auth generate` 완료
|
|
76
|
+
- [ ] User, Session, Account, Verification 엔티티 확인 완료
|
|
77
|
+
|
|
78
|
+
**다음 단계:** PHASE 1 엔티티 설계
|
|
79
|
+
|
|
80
|
+
---
|
|
14
81
|
|
|
15
82
|
### PHASE 1: 엔티티 설계
|
|
16
83
|
|
|
@@ -18,6 +85,9 @@ description: Sonamu 전체 개발 워크플로우. 엔티티 설계부터 Fronte
|
|
|
18
85
|
|
|
19
86
|
**참조 스킬:** entity-basic.md, entity-relations.md
|
|
20
87
|
|
|
88
|
+
**CRITICAL: PHASE 0에서 생성된 Auth 엔티티(User, Session, Account, Verification)를 확인한 후 설계를 시작한다.**
|
|
89
|
+
User 엔티티는 이미 생성되어 있으므로 중복 생성하지 않는다. 다른 엔티티에서 User를 참조하는 관계는 이미 존재하는 User 엔티티 구조에 맞춰 설계한다.
|
|
90
|
+
|
|
21
91
|
**절차:**
|
|
22
92
|
|
|
23
93
|
1. **비즈니스 플로우 작성**
|
|
@@ -218,11 +288,71 @@ pnpm dev # 이 상태로 유지하면서 작업
|
|
|
218
288
|
- [ ] Migration 오류 없이 완료
|
|
219
289
|
- [ ] DB에 테이블 생성 확인 (psql 또는 DB UI)
|
|
220
290
|
|
|
221
|
-
**다음 단계:** PHASE 4
|
|
291
|
+
**다음 단계:** PHASE 4 Cone 생성
|
|
292
|
+
|
|
293
|
+
---
|
|
294
|
+
|
|
295
|
+
### PHASE 4: Cone 생성
|
|
296
|
+
|
|
297
|
+
**목표:** Entity에 fixture 생성 가이드 메타데이터(cone) 부착
|
|
298
|
+
|
|
299
|
+
**참조 스킬:** fixture-cli.md (고급: cone 메타데이터 섹션)
|
|
300
|
+
|
|
301
|
+
**Cone이란?**
|
|
302
|
+
각 Entity/Prop/Subset/Enum에 `cone` 메타데이터를 추가하여 fixture 생성 시 더 현실적인 테스트 데이터를 만들 수 있게 하는 기능이다. `note`, `fixtureGenerator`, `fixtureDefault`, `dataSource` 등의 속성으로 구성된다.
|
|
303
|
+
|
|
304
|
+
**절차:**
|
|
305
|
+
|
|
306
|
+
1. **LLM 사용 여부 확인**
|
|
307
|
+
|
|
308
|
+
사용자에게 묻는다:
|
|
309
|
+
> "Cone 메타데이터를 생성할게요. 요구사항 명세(`requirements.md`)를 참고하여 LLM이 각 필드에 맞는 설명과 faker 생성 로직을 자동으로 작성할 수 있습니다.
|
|
310
|
+
> LLM(Claude API)을 호출하여 생성하갬습니까?
|
|
311
|
+
> 1. 예 — AI가 요구사항 기반 상세 cone 생성 (ANTHROPIC_API_KEY 필요, 소량 과금)
|
|
312
|
+
> 2. 아니오 — faker 타입 매핑 기반 템플릿 cone 자동 생성 (API 키 불필요)"
|
|
313
|
+
|
|
314
|
+
**선택 1 — LLM 사용:**
|
|
315
|
+
```bash
|
|
316
|
+
# 단일 Entity
|
|
317
|
+
pnpm sonamu cone gen [EntityId]
|
|
318
|
+
|
|
319
|
+
# 전체 Entity
|
|
320
|
+
pnpm sonamu cone gen --all
|
|
321
|
+
|
|
322
|
+
# 전체 재생성 (기존 덮어쓰기)
|
|
323
|
+
pnpm sonamu cone gen --all --regenerate
|
|
324
|
+
|
|
325
|
+
# 언어 지정 (기본: sonamu.config.ts의 i18n.defaultLocale)
|
|
326
|
+
pnpm sonamu cone gen --all --locale ko
|
|
327
|
+
```
|
|
328
|
+
> ANTHROPIC_API_KEY가 없으면 에러 발생 → `sonamu.secret.ts` 또는 환경변수 설정 필요
|
|
329
|
+
|
|
330
|
+
**선택 2 — 템플릿 cone 사용:**
|
|
331
|
+
```bash
|
|
332
|
+
# stub entity 시 이미 템플릿 cone이 생성되어 있음 (별도 작업 불필요)
|
|
333
|
+
# 필요 시 개별 Entity의 entity.json을 직접 편집
|
|
334
|
+
```
|
|
335
|
+
> stub entity 실행 시 faker 타입 매핑 기반 기본 cone이 자동 생성된다.
|
|
336
|
+
> 나중에 `pnpm sonamu cone gen [EntityId]`로 AI 업그레이드 가능.
|
|
337
|
+
|
|
338
|
+
2. **생성 결과 확인**
|
|
339
|
+
- 각 Entity의 `entity.json` → `cone` 필드 확인
|
|
340
|
+
- `note`: 필드 설명 및 fixture 생성 가이드
|
|
341
|
+
- `fixtureGenerator`: faker.js 표현식
|
|
342
|
+
- `dataSource`: BelongsToOne 관계 참조 전략
|
|
343
|
+
- 필요 시 직접 편집하여 커스터마이징
|
|
344
|
+
|
|
345
|
+
**완료 기준:**
|
|
346
|
+
|
|
347
|
+
- [ ] 모든 Entity의 `entity.json`에 cone 메타데이터 존재
|
|
348
|
+
- [ ] BelongsToOne 관계 필드에 `dataSource` 설정 확인
|
|
349
|
+
- [ ] LLM 사용 시 비용 확인 (콘솔 출력 참고)
|
|
350
|
+
|
|
351
|
+
**다음 단계:** PHASE 5 스캐폴딩
|
|
222
352
|
|
|
223
353
|
---
|
|
224
354
|
|
|
225
|
-
### PHASE
|
|
355
|
+
### PHASE 5: 스캐폴딩
|
|
226
356
|
|
|
227
357
|
**목표:** Model, API, Frontend 코드 자동 생성
|
|
228
358
|
|
|
@@ -299,11 +429,11 @@ pnpm dev # 이 상태로 유지하면서 작업
|
|
|
299
429
|
- [ ] Relation이 있는 경우 ko.ts 키 추가 완료
|
|
300
430
|
- [ ] types.ts nullable 필드 처리 완료
|
|
301
431
|
|
|
302
|
-
**다음 단계:** PHASE
|
|
432
|
+
**다음 단계:** PHASE 6 테스트 작성
|
|
303
433
|
|
|
304
434
|
---
|
|
305
435
|
|
|
306
|
-
### PHASE
|
|
436
|
+
### PHASE 6: 테스트 작성
|
|
307
437
|
|
|
308
438
|
**목표:** 업무 프로세스 기반 모듈 테스트 작성
|
|
309
439
|
|
|
@@ -406,11 +536,47 @@ pnpm dev # 이 상태로 유지하면서 작업
|
|
|
406
536
|
- [ ] 업무 시나리오 검증 완료
|
|
407
537
|
- [ ] 모든 그룹 커밋 완료
|
|
408
538
|
|
|
409
|
-
|
|
539
|
+
**[사용자 확인] Fixture 생성**
|
|
540
|
+
|
|
541
|
+
테스트 작성 완료 후 사용자에게 묻는다:
|
|
542
|
+
|
|
543
|
+
> "Fixture 데이터를 생성할게요.
|
|
544
|
+
> 테스트 DB에 현실적인 샘플 데이터를 체우면 확장성 테스트나 프론트엔드 개발에 도움이 됩니다.
|
|
545
|
+
> Fixture를 생성하시갌습니까?
|
|
546
|
+
> 1. 예 — Fixture 생성 진행
|
|
547
|
+
> 2. 아니오 — PHASE 7 API 개발로 바로 진행"
|
|
548
|
+
|
|
549
|
+
**예 선택 시 — Fixture 생성 방식 확인:**
|
|
550
|
+
|
|
551
|
+
> "Fixture 데이터를 어떻게 생성할까요?
|
|
552
|
+
> 1. LLM 사용 — `cone.note` 기반으로 Claude API가 현실적인 데이터 생성 (ANTHROPIC_API_KEY 필요)
|
|
553
|
+
> 2. 템플릿 사용 — faker.js 기반 자동 생성 (API 키 불필요)"
|
|
554
|
+
|
|
555
|
+
**LLM 사용 시:**
|
|
556
|
+
```bash
|
|
557
|
+
# 대화형 모드
|
|
558
|
+
pnpm sonamu fixture gen
|
|
559
|
+
|
|
560
|
+
# 또는 직접 지정
|
|
561
|
+
pnpm sonamu fixture gen --include User,Post,Comment --count 10 --use-llm
|
|
562
|
+
|
|
563
|
+
# 후 테스트 DB 동기화
|
|
564
|
+
pnpm sonamu fixture sync
|
|
565
|
+
```
|
|
566
|
+
|
|
567
|
+
**템플릿 사용 시:**
|
|
568
|
+
```bash
|
|
569
|
+
pnpm sonamu fixture gen --include User,Post,Comment --count 10
|
|
570
|
+
pnpm sonamu fixture sync
|
|
571
|
+
```
|
|
572
|
+
|
|
573
|
+
> fixture gen 세부 옵션 및 생성 전략은 `fixture-cli.md` 참조
|
|
574
|
+
|
|
575
|
+
**다음 단계:** PHASE 7 API 개발
|
|
410
576
|
|
|
411
577
|
---
|
|
412
578
|
|
|
413
|
-
### PHASE
|
|
579
|
+
### PHASE 7: API 개발
|
|
414
580
|
|
|
415
581
|
**목표:** 요구사항에 따른 비즈니스 로직 API 구현
|
|
416
582
|
|
|
@@ -464,11 +630,11 @@ pnpm dev # 이 상태로 유지하면서 작업
|
|
|
464
630
|
- [ ] API 테스트 통과
|
|
465
631
|
- [ ] Build 성공
|
|
466
632
|
|
|
467
|
-
**다음 단계:** PHASE
|
|
633
|
+
**다음 단계:** PHASE 8 Frontend 개발
|
|
468
634
|
|
|
469
635
|
---
|
|
470
636
|
|
|
471
|
-
### PHASE
|
|
637
|
+
### PHASE 8: Frontend 개발
|
|
472
638
|
|
|
473
639
|
**목표:** 화면에서 실제 동작 확인
|
|
474
640
|
|
|
@@ -547,20 +713,29 @@ pnpm dev # 이 상태로 유지하면서 작업
|
|
|
547
713
|
|
|
548
714
|
## 빠른 참조 테이블
|
|
549
715
|
|
|
550
|
-
| 단계
|
|
551
|
-
|
|
|
552
|
-
|
|
|
553
|
-
|
|
|
554
|
-
|
|
|
555
|
-
|
|
|
556
|
-
|
|
|
557
|
-
|
|
|
558
|
-
|
|
|
716
|
+
| 단계 | 예상 시간 | 핵심 명령어 | 핵심 스킬 |
|
|
717
|
+
| -------------------- | ----------- | ----------------------------- | ----------------------------- |
|
|
718
|
+
| **0. 프로젝트 초기화** | **5-10분** | **`create sonamu`, `auth generate`** | **project-init.md, auth.md** |
|
|
719
|
+
| 1. 엔티티 설계 | 5-10분 | (대화) | entity-basic.md |
|
|
720
|
+
| 2. 엔티티 생성 | 10-15분 | `stub entity`, `sync` | entity-basic.md |
|
|
721
|
+
| 3. 마이그레이션 | 5분 | `migration:latest` | migration.md |
|
|
722
|
+
| **4. Cone 생성** | **5-10분** | **`cone gen`** | **fixture-cli.md** |
|
|
723
|
+
| 5. 스캐폴딩 | 5-10분 | `scaffold`, `build` | scaffolding.md |
|
|
724
|
+
| 6. 테스트 | 30-60분 | `test`, `test:watch` | testing.md |
|
|
725
|
+
| **7. API 개발** | **1-3시간** | **@api 데코레이터** | **api.md, model.md** |
|
|
726
|
+
| **8. Frontend** | **2-5시간** | **Service, useTypeForm** | **frontend.md** |
|
|
559
727
|
|
|
560
728
|
---
|
|
561
729
|
|
|
562
730
|
## 각 단계의 완료 확인
|
|
563
731
|
|
|
732
|
+
### PHASE 0 완료 시
|
|
733
|
+
|
|
734
|
+
```
|
|
735
|
+
프로젝트 초기화 완료 (Auth 엔티티 생성 포함)
|
|
736
|
+
→ 다음: PHASE 1 엔티티 설계 (entity-basic.md)
|
|
737
|
+
```
|
|
738
|
+
|
|
564
739
|
### PHASE 1 완료 시
|
|
565
740
|
|
|
566
741
|
```
|
|
@@ -579,32 +754,40 @@ pnpm dev # 이 상태로 유지하면서 작업
|
|
|
579
754
|
|
|
580
755
|
```
|
|
581
756
|
마이그레이션 완료
|
|
582
|
-
→ 다음: PHASE 4
|
|
757
|
+
→ 다음: PHASE 4 Cone 생성 (fixture-cli.md)
|
|
583
758
|
```
|
|
584
759
|
|
|
585
760
|
### PHASE 4 완료 시
|
|
586
761
|
|
|
587
762
|
```
|
|
588
|
-
|
|
589
|
-
→ 다음: PHASE 5
|
|
763
|
+
Cone 생성 완료
|
|
764
|
+
→ 다음: PHASE 5 스캐폴딩 (scaffolding.md)
|
|
590
765
|
```
|
|
591
766
|
|
|
592
767
|
### PHASE 5 완료 시
|
|
593
768
|
|
|
594
769
|
```
|
|
595
|
-
|
|
596
|
-
→ 다음: PHASE 6
|
|
770
|
+
스ce90폴딩 완료
|
|
771
|
+
→ 다음: PHASE 6 테스트 작성 (testing.md)
|
|
597
772
|
```
|
|
598
773
|
|
|
599
774
|
### PHASE 6 완료 시
|
|
600
775
|
|
|
601
776
|
```
|
|
602
|
-
|
|
603
|
-
|
|
777
|
+
테스트 작성 완료
|
|
778
|
+
[사용자 확인] Fixture 생성 여부 확인 원스톱
|
|
779
|
+
→ 다음: PHASE 7 API 개발 (api.md)
|
|
604
780
|
```
|
|
605
781
|
|
|
606
782
|
### PHASE 7 완료 시
|
|
607
783
|
|
|
784
|
+
```
|
|
785
|
+
API 개발 완료
|
|
786
|
+
→ 다음: PHASE 8 Frontend 개발 (frontend.md)
|
|
787
|
+
```
|
|
788
|
+
|
|
789
|
+
### PHASE 8 완료 시
|
|
790
|
+
|
|
608
791
|
```
|
|
609
792
|
Frontend 개발 완료
|
|
610
793
|
전체 워크플로우 완료
|
|
@@ -104,6 +104,14 @@ export class DevVitestManager {
|
|
|
104
104
|
}
|
|
105
105
|
this.closed = true;
|
|
106
106
|
|
|
107
|
+
// 큐에 남은 작업들을 reject하여 호출자가 영구 대기하지 않도록 정리
|
|
108
|
+
while (this.queue.length > 0) {
|
|
109
|
+
const entry = this.queue.shift();
|
|
110
|
+
if (entry) {
|
|
111
|
+
entry.reject(new Error("DevVitestManager is being shut down"));
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
|
|
107
115
|
if (this.vitest) {
|
|
108
116
|
await this.vitest.close();
|
|
109
117
|
this.vitest = null;
|