sonamu 0.7.12 → 0.7.13

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 (55) hide show
  1. package/dist/api/config.d.ts +0 -3
  2. package/dist/api/config.d.ts.map +1 -1
  3. package/dist/api/config.js +1 -1
  4. package/dist/api/sonamu.d.ts.map +1 -1
  5. package/dist/api/sonamu.js +14 -4
  6. package/dist/bin/cli.js +2 -58
  7. package/dist/syncer/api-parser.d.ts.map +1 -1
  8. package/dist/syncer/api-parser.js +3 -2
  9. package/dist/syncer/syncer.d.ts +2 -1
  10. package/dist/syncer/syncer.d.ts.map +1 -1
  11. package/dist/syncer/syncer.js +17 -18
  12. package/dist/types/types.d.ts +1 -1
  13. package/dist/ui/ai-api.d.ts +1 -0
  14. package/dist/ui/ai-api.d.ts.map +1 -0
  15. package/dist/ui/ai-api.js +50 -0
  16. package/dist/ui/ai-client.d.ts +1 -0
  17. package/dist/ui/ai-client.d.ts.map +1 -0
  18. package/dist/ui/ai-client.js +438 -0
  19. package/dist/ui/api.d.ts +3 -0
  20. package/dist/ui/api.d.ts.map +1 -0
  21. package/dist/ui/api.js +680 -0
  22. package/dist/ui-web/assets/brand-icons-Cu_C0hZ4.svg +1008 -0
  23. package/dist/ui-web/assets/brand-icons-F3SPCeH1.woff +0 -0
  24. package/dist/ui-web/assets/brand-icons-XL9sxUpA.woff2 +0 -0
  25. package/dist/ui-web/assets/brand-icons-sqJ2Pg7a.eot +0 -0
  26. package/dist/ui-web/assets/brand-icons-ubhWoxly.ttf +0 -0
  27. package/dist/ui-web/assets/flags-DOLqOU7Y.png +0 -0
  28. package/dist/ui-web/assets/icons-BOCtAERH.woff +0 -0
  29. package/dist/ui-web/assets/icons-CHzK1VD9.eot +0 -0
  30. package/dist/ui-web/assets/icons-D29ZQHHw.ttf +0 -0
  31. package/dist/ui-web/assets/icons-Du6TOHnR.woff2 +0 -0
  32. package/dist/ui-web/assets/icons-RwhydX30.svg +1518 -0
  33. package/dist/ui-web/assets/index-CpaB9P6g.css +1 -0
  34. package/dist/ui-web/assets/index-J9MCfjCd.js +95 -0
  35. package/dist/ui-web/assets/outline-icons-BfdLr8tr.svg +366 -0
  36. package/dist/ui-web/assets/outline-icons-DD8jm0uy.ttf +0 -0
  37. package/dist/ui-web/assets/outline-icons-DInHoiqI.woff2 +0 -0
  38. package/dist/ui-web/assets/outline-icons-LX8adJ4n.eot +0 -0
  39. package/dist/ui-web/assets/outline-icons-aQ88nltS.woff +0 -0
  40. package/dist/ui-web/assets/provider-utils_false-BKJD46kk.js +1 -0
  41. package/dist/ui-web/assets/provider-utils_false-Bu5lmX18.js +1 -0
  42. package/dist/ui-web/index.html +13 -0
  43. package/dist/ui-web/vite.svg +1 -0
  44. package/dist/vector/embedding.d.ts.map +1 -1
  45. package/dist/vector/embedding.js +7 -7
  46. package/package.json +7 -5
  47. package/src/api/config.ts +0 -3
  48. package/src/api/sonamu.ts +17 -4
  49. package/src/bin/cli.ts +1 -67
  50. package/src/syncer/api-parser.ts +2 -1
  51. package/src/syncer/syncer.ts +20 -21
  52. package/src/ui/ai-api.ts +60 -0
  53. package/src/ui/ai-client.ts +499 -0
  54. package/src/ui/api.ts +786 -0
  55. package/src/vector/embedding.ts +8 -6
@@ -1,6 +1,4 @@
1
- import { createOpenAI } from "@ai-sdk/openai";
2
1
  import { embedMany } from "ai";
3
- import { VoyageAIClient } from "voyageai";
4
2
  import { Sonamu } from "../api/sonamu.js";
5
3
  import { DEFAULT_VECTOR_CONFIG } from "./config.js";
6
4
  /**
@@ -34,7 +32,8 @@ import { DEFAULT_VECTOR_CONFIG } from "./config.js";
34
32
  }
35
33
  /**
36
34
  * Voyage AI 클라이언트 초기화
37
- */ getVoyageClient() {
35
+ */ async getVoyageClient() {
36
+ const { VoyageAIClient } = await import("voyageai");
38
37
  const apiKey = Sonamu.secrets?.voyage_api_key ?? process.env.VOYAGE_API_KEY;
39
38
  if (!apiKey) {
40
39
  throw new Error("VOYAGE_API_KEY가 설정되지 않았습니다. 환경변수를 확인하세요.");
@@ -45,7 +44,8 @@ import { DEFAULT_VECTOR_CONFIG } from "./config.js";
45
44
  }
46
45
  /**
47
46
  * OpenAI provider 생성
48
- */ getOpenAIProvider() {
47
+ */ async getOpenAIProvider() {
48
+ const { createOpenAI } = await import("@ai-sdk/openai");
49
49
  const apiKey = Sonamu.secrets?.openai_api_key ?? process.env.OPENAI_API_KEY;
50
50
  if (!apiKey) {
51
51
  throw new Error("OPENAI_API_KEY가 설정되지 않았습니다. 환경변수를 확인하세요.");
@@ -85,7 +85,7 @@ import { DEFAULT_VECTOR_CONFIG } from "./config.js";
85
85
  /**
86
86
  * Voyage AI 임베딩
87
87
  */ async embedVoyage(texts, inputType) {
88
- const client = this.getVoyageClient();
88
+ const client = await this.getVoyageClient();
89
89
  const voyageConfig = this.config.voyage;
90
90
  const response = await client.embed({
91
91
  input: texts,
@@ -104,7 +104,7 @@ import { DEFAULT_VECTOR_CONFIG } from "./config.js";
104
104
  /**
105
105
  * OpenAI 임베딩
106
106
  */ async embedOpenAI(texts) {
107
- const openai = this.getOpenAIProvider();
107
+ const openai = await this.getOpenAIProvider();
108
108
  const openaiConfig = this.config.openai;
109
109
  const model = openai.embeddingModel(openaiConfig.model);
110
110
  const { embeddings, usage } = await embedMany({
@@ -125,4 +125,4 @@ import { DEFAULT_VECTOR_CONFIG } from "./config.js";
125
125
  }
126
126
  export const Embedding = new EmbeddingClass();
127
127
 
128
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/vector/embedding.ts"],"sourcesContent":["import { createOpenAI, type OpenAIProvider } from \"@ai-sdk/openai\";\nimport { type EmbeddingModel, embedMany } from \"ai\";\nimport { VoyageAIClient } from \"voyageai\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { DEFAULT_VECTOR_CONFIG } from \"./config\";\nimport type {\n  EmbeddingProvider,\n  EmbeddingResult,\n  ProgressCallback,\n  VectorConfig,\n  VectorInputType,\n} from \"./types\";\n\n/**\n * 임베딩 클라이언트\n * Voyage AI와 OpenAI 임베딩을 SDK 방식으로 통합 지원\n */\nexport class EmbeddingClass {\n  private config: VectorConfig;\n\n  constructor(config: Partial<VectorConfig> = {}) {\n    this.config = {\n      voyage: { ...DEFAULT_VECTOR_CONFIG.voyage, ...config.voyage },\n      openai: { ...DEFAULT_VECTOR_CONFIG.openai, ...config.openai },\n      chunking: { ...DEFAULT_VECTOR_CONFIG.chunking, ...config.chunking },\n      search: { ...DEFAULT_VECTOR_CONFIG.search, ...config.search },\n      pgvector: { ...DEFAULT_VECTOR_CONFIG.pgvector, ...config.pgvector },\n    };\n  }\n\n  /**\n   * Voyage AI 클라이언트 초기화\n   */\n  private getVoyageClient(): VoyageAIClient {\n    const apiKey = Sonamu.secrets?.voyage_api_key ?? process.env.VOYAGE_API_KEY;\n    if (!apiKey) {\n      throw new Error(\"VOYAGE_API_KEY가 설정되지 않았습니다. 환경변수를 확인하세요.\");\n    }\n    return new VoyageAIClient({ apiKey });\n  }\n\n  /**\n   * OpenAI provider 생성\n   */\n  private getOpenAIProvider(): OpenAIProvider {\n    const apiKey = Sonamu.secrets?.openai_api_key ?? process.env.OPENAI_API_KEY;\n    if (!apiKey) {\n      throw new Error(\"OPENAI_API_KEY가 설정되지 않았습니다. 환경변수를 확인하세요.\");\n    }\n    return createOpenAI({ apiKey });\n  }\n\n  /**\n   * 텍스트 임베딩 생성\n   * @param texts - 임베딩할 텍스트 배열 (batchSize이상 시 자동 분할)\n   * @param provider - 'voyage' | 'openai'\n   * @param inputType - 'document' | 'query' (Voyage AI만 해당)\n   * @param onProgress - 진행률 콜백\n   */\n  async embed(\n    texts: string[],\n    provider: EmbeddingProvider,\n    inputType: VectorInputType = \"document\",\n    onProgress?: ProgressCallback,\n  ): Promise<EmbeddingResult[]> {\n    const maxBatchSize =\n      provider === \"voyage\" ? this.config.voyage.batchSize : this.config.openai.batchSize;\n\n    // batchSize이하면 바로 호출\n    if (texts.length <= maxBatchSize) {\n      return provider === \"voyage\"\n        ? await this.embedVoyage(texts, inputType)\n        : await this.embedOpenAI(texts);\n    }\n\n    // batchSize이상이면 자동으로 나눠서 처리\n    const batches = Array.from({ length: Math.ceil(texts.length / maxBatchSize) }, (_, i) =>\n      texts.slice(i * maxBatchSize, (i + 1) * maxBatchSize),\n    );\n\n    const results = await Promise.all(\n      batches.map((batch) =>\n        provider === \"voyage\" ? this.embedVoyage(batch, inputType) : this.embedOpenAI(batch),\n      ),\n    );\n\n    onProgress?.(texts.length, texts.length);\n    return results.flat();\n  }\n\n  /**\n   * 단일 텍스트 임베딩 (편의 메서드)\n   */\n  async embedOne(\n    text: string,\n    provider: EmbeddingProvider,\n    inputType: VectorInputType = \"document\",\n  ): Promise<EmbeddingResult> {\n    const results = await this.embed([text], provider, inputType);\n    return results[0];\n  }\n\n  /**\n   * Voyage AI 임베딩\n   */\n  private async embedVoyage(\n    texts: string[],\n    inputType: VectorInputType,\n  ): Promise<EmbeddingResult[]> {\n    const client = this.getVoyageClient();\n    const voyageConfig = this.config.voyage;\n\n    const response = await client.embed({\n      input: texts,\n      model: voyageConfig.model,\n      inputType: inputType,\n    });\n    if (!response.data) {\n      throw new Error(\"Voyage API: 응답 데이터가 없습니다.\");\n    }\n\n    return response.data.map((item) => ({\n      embedding: item.embedding ?? [],\n      model: voyageConfig.model,\n      tokenCount: response.usage?.totalTokens ?? 0,\n    }));\n  }\n\n  /**\n   * OpenAI 임베딩\n   */\n  private async embedOpenAI(texts: string[]): Promise<EmbeddingResult[]> {\n    const openai = this.getOpenAIProvider();\n    const openaiConfig = this.config.openai;\n    const model = openai.embeddingModel(openaiConfig.model);\n\n    const { embeddings, usage } = await embedMany({\n      model: model as EmbeddingModel,\n      values: texts,\n    });\n\n    return embeddings.map((embedding) => ({\n      embedding,\n      model: openaiConfig.model,\n      tokenCount: usage?.tokens ?? 0,\n    }));\n  }\n\n  /**\n   * 임베딩 provider의 차원 수 반환\n   */\n  getDimensions(provider: EmbeddingProvider): number {\n    return provider === \"voyage\" ? this.config.voyage.dimensions : this.config.openai.dimensions;\n  }\n}\nexport const Embedding = new EmbeddingClass();\n"],"names":["createOpenAI","embedMany","VoyageAIClient","Sonamu","DEFAULT_VECTOR_CONFIG","EmbeddingClass","config","voyage","openai","chunking","search","pgvector","getVoyageClient","apiKey","secrets","voyage_api_key","process","env","VOYAGE_API_KEY","Error","getOpenAIProvider","openai_api_key","OPENAI_API_KEY","embed","texts","provider","inputType","onProgress","maxBatchSize","batchSize","length","embedVoyage","embedOpenAI","batches","Array","from","Math","ceil","_","i","slice","results","Promise","all","map","batch","flat","embedOne","text","client","voyageConfig","response","input","model","data","item","embedding","tokenCount","usage","totalTokens","openaiConfig","embeddingModel","embeddings","values","tokens","getDimensions","dimensions","Embedding"],"mappings":"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"}
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/vector/embedding.ts"],"sourcesContent":["import type { OpenAIProvider } from \"@ai-sdk/openai\";\nimport { type EmbeddingModel, embedMany } from \"ai\";\nimport type { VoyageAIClient } from \"voyageai\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { DEFAULT_VECTOR_CONFIG } from \"./config\";\nimport type {\n  EmbeddingProvider,\n  EmbeddingResult,\n  ProgressCallback,\n  VectorConfig,\n  VectorInputType,\n} from \"./types\";\n\n/**\n * 임베딩 클라이언트\n * Voyage AI와 OpenAI 임베딩을 SDK 방식으로 통합 지원\n */\nexport class EmbeddingClass {\n  private config: VectorConfig;\n\n  constructor(config: Partial<VectorConfig> = {}) {\n    this.config = {\n      voyage: { ...DEFAULT_VECTOR_CONFIG.voyage, ...config.voyage },\n      openai: { ...DEFAULT_VECTOR_CONFIG.openai, ...config.openai },\n      chunking: { ...DEFAULT_VECTOR_CONFIG.chunking, ...config.chunking },\n      search: { ...DEFAULT_VECTOR_CONFIG.search, ...config.search },\n      pgvector: { ...DEFAULT_VECTOR_CONFIG.pgvector, ...config.pgvector },\n    };\n  }\n\n  /**\n   * Voyage AI 클라이언트 초기화\n   */\n  private async getVoyageClient(): Promise<VoyageAIClient> {\n    const { VoyageAIClient } = await import(\"voyageai\");\n    const apiKey = Sonamu.secrets?.voyage_api_key ?? process.env.VOYAGE_API_KEY;\n    if (!apiKey) {\n      throw new Error(\"VOYAGE_API_KEY가 설정되지 않았습니다. 환경변수를 확인하세요.\");\n    }\n    return new VoyageAIClient({ apiKey });\n  }\n\n  /**\n   * OpenAI provider 생성\n   */\n  private async getOpenAIProvider(): Promise<OpenAIProvider> {\n    const { createOpenAI } = await import(\"@ai-sdk/openai\");\n    const apiKey = Sonamu.secrets?.openai_api_key ?? process.env.OPENAI_API_KEY;\n    if (!apiKey) {\n      throw new Error(\"OPENAI_API_KEY가 설정되지 않았습니다. 환경변수를 확인하세요.\");\n    }\n    return createOpenAI({ apiKey });\n  }\n\n  /**\n   * 텍스트 임베딩 생성\n   * @param texts - 임베딩할 텍스트 배열 (batchSize이상 시 자동 분할)\n   * @param provider - 'voyage' | 'openai'\n   * @param inputType - 'document' | 'query' (Voyage AI만 해당)\n   * @param onProgress - 진행률 콜백\n   */\n  async embed(\n    texts: string[],\n    provider: EmbeddingProvider,\n    inputType: VectorInputType = \"document\",\n    onProgress?: ProgressCallback,\n  ): Promise<EmbeddingResult[]> {\n    const maxBatchSize =\n      provider === \"voyage\" ? this.config.voyage.batchSize : this.config.openai.batchSize;\n\n    // batchSize이하면 바로 호출\n    if (texts.length <= maxBatchSize) {\n      return provider === \"voyage\"\n        ? await this.embedVoyage(texts, inputType)\n        : await this.embedOpenAI(texts);\n    }\n\n    // batchSize이상이면 자동으로 나눠서 처리\n    const batches = Array.from({ length: Math.ceil(texts.length / maxBatchSize) }, (_, i) =>\n      texts.slice(i * maxBatchSize, (i + 1) * maxBatchSize),\n    );\n\n    const results = await Promise.all(\n      batches.map((batch) =>\n        provider === \"voyage\" ? this.embedVoyage(batch, inputType) : this.embedOpenAI(batch),\n      ),\n    );\n\n    onProgress?.(texts.length, texts.length);\n    return results.flat();\n  }\n\n  /**\n   * 단일 텍스트 임베딩 (편의 메서드)\n   */\n  async embedOne(\n    text: string,\n    provider: EmbeddingProvider,\n    inputType: VectorInputType = \"document\",\n  ): Promise<EmbeddingResult> {\n    const results = await this.embed([text], provider, inputType);\n    return results[0];\n  }\n\n  /**\n   * Voyage AI 임베딩\n   */\n  private async embedVoyage(\n    texts: string[],\n    inputType: VectorInputType,\n  ): Promise<EmbeddingResult[]> {\n    const client = await this.getVoyageClient();\n    const voyageConfig = this.config.voyage;\n\n    const response = await client.embed({\n      input: texts,\n      model: voyageConfig.model,\n      inputType: inputType,\n    });\n    if (!response.data) {\n      throw new Error(\"Voyage API: 응답 데이터가 없습니다.\");\n    }\n\n    return response.data.map((item) => ({\n      embedding: item.embedding ?? [],\n      model: voyageConfig.model,\n      tokenCount: response.usage?.totalTokens ?? 0,\n    }));\n  }\n\n  /**\n   * OpenAI 임베딩\n   */\n  private async embedOpenAI(texts: string[]): Promise<EmbeddingResult[]> {\n    const openai = await this.getOpenAIProvider();\n    const openaiConfig = this.config.openai;\n    const model = openai.embeddingModel(openaiConfig.model);\n\n    const { embeddings, usage } = await embedMany({\n      model: model as EmbeddingModel,\n      values: texts,\n    });\n\n    return embeddings.map((embedding) => ({\n      embedding,\n      model: openaiConfig.model,\n      tokenCount: usage?.tokens ?? 0,\n    }));\n  }\n\n  /**\n   * 임베딩 provider의 차원 수 반환\n   */\n  getDimensions(provider: EmbeddingProvider): number {\n    return provider === \"voyage\" ? this.config.voyage.dimensions : this.config.openai.dimensions;\n  }\n}\nexport const Embedding = new EmbeddingClass();\n"],"names":["embedMany","Sonamu","DEFAULT_VECTOR_CONFIG","EmbeddingClass","config","voyage","openai","chunking","search","pgvector","getVoyageClient","VoyageAIClient","apiKey","secrets","voyage_api_key","process","env","VOYAGE_API_KEY","Error","getOpenAIProvider","createOpenAI","openai_api_key","OPENAI_API_KEY","embed","texts","provider","inputType","onProgress","maxBatchSize","batchSize","length","embedVoyage","embedOpenAI","batches","Array","from","Math","ceil","_","i","slice","results","Promise","all","map","batch","flat","embedOne","text","client","voyageConfig","response","input","model","data","item","embedding","tokenCount","usage","totalTokens","openaiConfig","embeddingModel","embeddings","values","tokens","getDimensions","dimensions","Embedding"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "sonamu",
3
- "version": "0.7.12",
3
+ "version": "0.7.13",
4
4
  "description": "Sonamu — TypeScript Fullstack API Framework",
5
5
  "keywords": [
6
6
  "typescript",
@@ -68,6 +68,7 @@
68
68
  "knex": "^3.1.0",
69
69
  "mime-types": "^3.0.1",
70
70
  "minimatch": "^10.0.3",
71
+ "node-cron": "^4.2.1",
71
72
  "node-sql-parser": "^5.2.0",
72
73
  "pg": "^8.16.3",
73
74
  "prompts": "^2.4.2",
@@ -76,10 +77,9 @@
76
77
  "tsicli": "^1.0.5",
77
78
  "vitest": "^4.0.10",
78
79
  "zod": "^4.1.12",
79
- "node-cron": "^4.2.1",
80
80
  "@sonamu-kit/hmr-hook": "^0.4.1",
81
- "@sonamu-kit/ts-loader": "^2.1.3",
82
81
  "@sonamu-kit/tasks": "^0.0.1",
82
+ "@sonamu-kit/ts-loader": "^2.1.3",
83
83
  "@sonamu-kit/hmr-runner": "^0.1.1"
84
84
  },
85
85
  "devDependencies": {
@@ -104,8 +104,8 @@
104
104
  "ai": "6.0.0-beta.138",
105
105
  "fastify": "^4.23.2",
106
106
  "knex": "^3.1.0",
107
- "typescript": "^5.9.3",
108
107
  "pgvector": "^0.2.1",
108
+ "typescript": "^5.9.3",
109
109
  "voyageai": "^0.0.8"
110
110
  },
111
111
  "peerDependenciesMeta": {
@@ -130,7 +130,9 @@
130
130
  },
131
131
  "scripts": {
132
132
  "dev": "nodemon exec",
133
- "build": "rm -rf dist && swc src -d dist --strip-leading-paths && tsc --emitDeclarationOnly",
133
+ "build": "run-s 'build:sonamu' 'build:ui-web'",
134
+ "build:sonamu": "rm -rf dist && swc src -d dist --strip-leading-paths && tsc --emitDeclarationOnly",
135
+ "build:ui-web": "cd ui-web && pnpm install && pnpm build",
134
136
  "test:type": "vitest --typecheck test-d.ts"
135
137
  }
136
138
  }
package/src/api/config.ts CHANGED
@@ -30,9 +30,6 @@ export type SonamuConfig = {
30
30
  sync: {
31
31
  targets: string[]; // "web", "app" 등
32
32
  };
33
- ui?: {
34
- port: number;
35
- };
36
33
 
37
34
  database: {
38
35
  // 데이터베이스
package/src/api/sonamu.ts CHANGED
@@ -231,8 +231,6 @@ class SonamuClass {
231
231
  await this.syncer.sync();
232
232
 
233
233
  await this.startWatcher();
234
-
235
- this.syncer.syncUI();
236
234
  }
237
235
 
238
236
  this.isInitialized = true;
@@ -346,10 +344,19 @@ class SonamuClass {
346
344
  },
347
345
  );
348
346
 
347
+ // Sonamu UI API
348
+ const { sonamuUIApiPlugin } = await import("../ui/api");
349
+ server.register(sonamuUIApiPlugin);
350
+
349
351
  // API 라우팅 (로컬HMR 상태와 구분)
350
352
  const { isLocal } = await import("../utils/controller");
351
353
  if (isLocal()) {
352
354
  server.all("*", async (request, reply) => {
355
+ // Sonamu UI
356
+ if (request.url.startsWith("/sonamu-ui")) {
357
+ return;
358
+ }
359
+
353
360
  const found = this.syncer.apis.find(
354
361
  (api) =>
355
362
  this.config.api.route.prefix + api.path === request.url.split("?")[0] &&
@@ -358,8 +365,14 @@ class SonamuClass {
358
365
  if (found) {
359
366
  return this.getApiHandler(found, config)(request, reply);
360
367
  }
361
- const { NotFoundException } = await import("../exceptions/so-exceptions");
362
- throw new NotFoundException("존재하지 않는 API 접근입니다.");
368
+
369
+ if (request.url.startsWith("/api/")) {
370
+ const { NotFoundException } = await import("../exceptions/so-exceptions");
371
+ throw new NotFoundException(`존재하지 않는 API 접근입니다. ${request.url}`);
372
+ }
373
+
374
+ // 일반 파일 접근시 별도의 에러 출력하지 않음
375
+ return;
363
376
  });
364
377
  } else {
365
378
  for (const api of this.syncer.apis) {
package/src/bin/cli.ts CHANGED
@@ -22,7 +22,7 @@ import { BUILD_DIR, SWC_BUILD_COMMAND, TSC_TYPE_CHECK_COMMAND } from "./build-co
22
22
  let migrator: Migrator;
23
23
 
24
24
  async function bootstrap() {
25
- const notToInit = ["dev", "build", "start", "ui"].includes(process.argv[2] ?? "");
25
+ const notToInit = ["dev", "build", "start"].includes(process.argv[2] ?? "");
26
26
  if (!notToInit) {
27
27
  await Sonamu.init(false, false);
28
28
  }
@@ -58,7 +58,6 @@ async function bootstrap() {
58
58
  ["scaffold", "model_test", "#entityId"],
59
59
  ["scaffold", "view_list", "#entityId"],
60
60
  ["scaffold", "view_form", "#entityId"],
61
- ["ui"],
62
61
  ["sync"],
63
62
  ["dev"],
64
63
  ["build"],
@@ -74,7 +73,6 @@ async function bootstrap() {
74
73
  stub_entity,
75
74
  scaffold_model,
76
75
  scaffold_model_test,
77
- ui,
78
76
  // scaffold_view_list,
79
77
  // scaffold_view_form,
80
78
  sync,
@@ -452,67 +450,3 @@ async function scaffold_model_test(entityId: string) {
452
450
  entityId,
453
451
  });
454
452
  }
455
-
456
- async function ui() {
457
- try {
458
- const apiRootPath = findApiRootPath();
459
-
460
- // 사용자 프로젝트의 패키지들 중에서 @sonamu-kit/ui를 찾습니다.
461
- // 이를 위해서 createRequire를 사용하여 프로젝트 경로 기준으로 resolve합니다.
462
- const projectRequire = createRequire(path.join(apiRootPath, "package.json"));
463
- const uiPackagePath = projectRequire.resolve("@sonamu-kit/ui"); // 없으면 여기서 터져요(MODULE_NOT_FOUND)
464
- const uiNodePath = path.join(path.dirname(uiPackagePath), "run-ui.js");
465
-
466
- if (!(await exists(uiNodePath))) {
467
- console.log(
468
- chalk.red(`UI runner script not found at ${uiNodePath}. Please rebuild @sonamu-kit/ui.`),
469
- );
470
- return;
471
- }
472
-
473
- // UI를 별도 프로세스로 실행 (hmr-hook 활성화)
474
- const uiProcess = spawn(
475
- process.execPath,
476
- [
477
- "--import",
478
- "sonamu/ts-loader-register",
479
- "--import",
480
- "sonamu/hmr-hook-register",
481
- "--enable-source-maps",
482
- "--no-warnings",
483
- uiNodePath,
484
- ],
485
- {
486
- stdio: "inherit",
487
- env: {
488
- ...process.env,
489
- HOT: "yes",
490
- API_ROOT_PATH: apiRootPath, // UI는 얘만 알면 돼요! 나머지는 얘가 떠서 알아서 할 것임 ㅎ
491
- },
492
- },
493
- );
494
-
495
- // 종료 처리
496
- const cleanup = () => {
497
- console.log(chalk.yellow("\n\n👋 Shutting down UI server..."));
498
- uiProcess.kill("SIGTERM");
499
- process.exit(0);
500
- };
501
-
502
- process.on("SIGINT", cleanup);
503
- process.on("SIGTERM", cleanup);
504
-
505
- uiProcess.on("exit", (code) => {
506
- if (code !== 0) {
507
- console.error(chalk.red(`❌ UI server exited with code ${code}`));
508
- process.exit(code || 1);
509
- }
510
- });
511
- } catch (e: unknown) {
512
- if (e instanceof Error && e.message.includes("isn't declared")) {
513
- console.log(`You need to install ${chalk.blue(`@sonamu-kit/ui`)} first.`);
514
- return;
515
- }
516
- throw e;
517
- }
518
- }
@@ -98,7 +98,8 @@ export async function readApisFromFile(filePath: AbsolutePath): Promise<Extended
98
98
  // const p = path.join(tmpdir(), "sonamu-syncer-error.json");
99
99
  // writeFileSync(p, JSON.stringify(registeredApis, null, 2));
100
100
  // execSync(`open ${p}`);
101
- throw new Error(`현재 파일에 사전 등록된 API가 없습니다. ${filePath}`);
101
+ // throw new Error(`현재 파일에 사전 등록된 API가 없습니다. ${filePath}`);
102
+ return [];
102
103
  }
103
104
 
104
105
  // 등록된 API에 현재 메소드 타입 정보 확장
@@ -1,6 +1,7 @@
1
1
  import { hot } from "@sonamu-kit/hmr-hook";
2
2
  import assert from "assert";
3
3
  import chalk from "chalk";
4
+ import { EventEmitter } from "events";
4
5
  import { mkdir, readFile, writeFile } from "fs/promises";
5
6
  import inflection from "inflection";
6
7
  import { minimatch } from "minimatch";
@@ -45,6 +46,7 @@ export class Syncer {
45
46
  models: LoadedModels = {};
46
47
  workflows: Map<string, WorkflowMetadata[]> = new Map();
47
48
  isSyncing: boolean = false;
49
+ eventEmitter: EventEmitter = new EventEmitter();
48
50
 
49
51
  /**
50
52
  * 체크섬이 변경된 부분에 대해 싱크를 진행합니다.
@@ -100,17 +102,24 @@ export class Syncer {
100
102
  console.log(chalk.bold(`🔄 Invalidated:`));
101
103
 
102
104
  for (const invalidatedPath of invalidatedPaths) {
103
- // 만약 model.ts 파일이 변경(invalidate)되었다? 그러면 registeredApis 중에서 이 모델에 해당하는 api들은 지워줘요.
104
- // registeredApis는 통으로 날려버릴 없습니다. registeredApis 올라오는 친구들은 초기 로드시 또는 HMR시에만 등록되기 때문입니다.
105
- // 따라서 model.ts 파일의 변경으로 다음번 새로운 eval이 예상되는 시점에서만, 모델에서 나온 registeredApis들을 지워줄 수 있습니다.
106
- const removedApis = this.removeInvalidatedRegisteredApis(invalidatedPath);
107
- if (removedApis.length > 0) {
108
- console.log(
109
- chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`),
110
- chalk.gray(`(with ${removedApis.length} APIs)`),
105
+ try {
106
+ // 만약 model.ts 파일이 변경(invalidate)되었다? 그러면 registeredApis 중에서 모델에 해당하는 api들은 지워줘요.
107
+ // registeredApis는 통으로 날려버릴 없습니다. registeredApis에 올라오는 친구들은 초기 로드시 또는 HMR시에만 등록되기 때문입니다.
108
+ // 따라서 model.ts 파일의 변경으로 다음번 새로운 eval이 예상되는 이 시점에서만, 이 모델에서 나온 registeredApis들을 지워줄 수 있습니다.
109
+ const removedApis = this.removeInvalidatedRegisteredApis(invalidatedPath);
110
+ if (removedApis.length > 0) {
111
+ console.log(
112
+ chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`),
113
+ chalk.gray(`(with ${removedApis.length} APIs)`),
114
+ );
115
+ } else {
116
+ console.log(chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`));
117
+ }
118
+ } catch (e) {
119
+ console.error(e);
120
+ console.error(
121
+ chalk.red(`Failed to remove invalidated registered APIs for ${invalidatedPath}`),
111
122
  );
112
- } else {
113
- console.log(chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`));
114
123
  }
115
124
  }
116
125
  }
@@ -132,7 +141,7 @@ export class Syncer {
132
141
  await this.autoloadApis();
133
142
  await this.autoloadWorkflows();
134
143
 
135
- this.syncUI();
144
+ this.eventEmitter.emit("onHMRCompleted");
136
145
  }
137
146
 
138
147
  removeInvalidatedRegisteredApis(
@@ -551,16 +560,6 @@ export class Syncer {
551
560
  );
552
561
  }
553
562
 
554
- syncUI() {
555
- const uiPort = Sonamu.config.ui?.port ?? 57000;
556
-
557
- if (!isTest()) {
558
- fetch(`http://127.0.0.1:${uiPort}/api/reload`, {
559
- method: "GET",
560
- }).catch((e) => console.log(chalk.dim(`Failed to reload Sonamu UI: ${e.message}`)));
561
- }
562
- }
563
-
564
563
  /**
565
564
  * 하위호환용 프록시 메소드입니다.
566
565
  */
@@ -0,0 +1,60 @@
1
+ // import { convertToModelMessages, type UIMessage } from "ai";
2
+ // import type { FastifyInstance } from "fastify";
3
+ // import { BadRequestException, type FixtureRecord } from "sonamu";
4
+ // import { aiClient } from "./ai-client";
5
+
6
+ // export async function setAiApi(server: FastifyInstance) {
7
+ // await aiClient.init();
8
+
9
+ // server.post("/api/ai/fixture/chat", async (request, reply) => {
10
+ // const { messages, fixtureRecords } = request.body as {
11
+ // messages: UIMessage[];
12
+ // fixtureRecords?: FixtureRecord[];
13
+ // };
14
+
15
+ // if (!fixtureRecords || fixtureRecords.length === 0) {
16
+ // throw new BadRequestException("픽스쳐 레코드가 없습니다. 픽스쳐 조회 후 시도하세요.");
17
+ // }
18
+
19
+ // const result = aiClient.handleFixture(convertToModelMessages(messages), fixtureRecords);
20
+ // const response = result.toUIMessageStreamResponse();
21
+
22
+ // reply.raw.writeHead(response.status, Object.fromEntries(response.headers.entries()));
23
+
24
+ // if (response.body) {
25
+ // const reader = response.body.getReader();
26
+ // while (true) {
27
+ // const { done, value } = await reader.read();
28
+ // if (done) break;
29
+ // reply.raw.write(value);
30
+ // }
31
+ // }
32
+
33
+ // reply.raw.end();
34
+ // return reply;
35
+ // });
36
+
37
+ // // Entity/Enum 생성용 AI Chat Stream
38
+ // server.post("/api/ai/entity/chat", async (request, reply) => {
39
+ // const { messages } = request.body as {
40
+ // messages: UIMessage[];
41
+ // };
42
+
43
+ // const result = aiClient.handleEntity(convertToModelMessages(messages));
44
+ // const response = result.toUIMessageStreamResponse();
45
+
46
+ // reply.raw.writeHead(response.status, Object.fromEntries(response.headers.entries()));
47
+
48
+ // if (response.body) {
49
+ // const reader = response.body.getReader();
50
+ // while (true) {
51
+ // const { done, value } = await reader.read();
52
+ // if (done) break;
53
+ // reply.raw.write(value);
54
+ // }
55
+ // }
56
+
57
+ // reply.raw.end();
58
+ // return reply;
59
+ // });
60
+ // }