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.
- package/dist/api/config.d.ts +0 -3
- package/dist/api/config.d.ts.map +1 -1
- package/dist/api/config.js +1 -1
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +14 -4
- package/dist/bin/cli.js +2 -58
- package/dist/syncer/api-parser.d.ts.map +1 -1
- package/dist/syncer/api-parser.js +3 -2
- package/dist/syncer/syncer.d.ts +2 -1
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +17 -18
- package/dist/types/types.d.ts +1 -1
- package/dist/ui/ai-api.d.ts +1 -0
- package/dist/ui/ai-api.d.ts.map +1 -0
- package/dist/ui/ai-api.js +50 -0
- package/dist/ui/ai-client.d.ts +1 -0
- package/dist/ui/ai-client.d.ts.map +1 -0
- package/dist/ui/ai-client.js +438 -0
- package/dist/ui/api.d.ts +3 -0
- package/dist/ui/api.d.ts.map +1 -0
- package/dist/ui/api.js +680 -0
- package/dist/ui-web/assets/brand-icons-Cu_C0hZ4.svg +1008 -0
- package/dist/ui-web/assets/brand-icons-F3SPCeH1.woff +0 -0
- package/dist/ui-web/assets/brand-icons-XL9sxUpA.woff2 +0 -0
- package/dist/ui-web/assets/brand-icons-sqJ2Pg7a.eot +0 -0
- package/dist/ui-web/assets/brand-icons-ubhWoxly.ttf +0 -0
- package/dist/ui-web/assets/flags-DOLqOU7Y.png +0 -0
- package/dist/ui-web/assets/icons-BOCtAERH.woff +0 -0
- package/dist/ui-web/assets/icons-CHzK1VD9.eot +0 -0
- package/dist/ui-web/assets/icons-D29ZQHHw.ttf +0 -0
- package/dist/ui-web/assets/icons-Du6TOHnR.woff2 +0 -0
- package/dist/ui-web/assets/icons-RwhydX30.svg +1518 -0
- package/dist/ui-web/assets/index-CpaB9P6g.css +1 -0
- package/dist/ui-web/assets/index-J9MCfjCd.js +95 -0
- package/dist/ui-web/assets/outline-icons-BfdLr8tr.svg +366 -0
- package/dist/ui-web/assets/outline-icons-DD8jm0uy.ttf +0 -0
- package/dist/ui-web/assets/outline-icons-DInHoiqI.woff2 +0 -0
- package/dist/ui-web/assets/outline-icons-LX8adJ4n.eot +0 -0
- package/dist/ui-web/assets/outline-icons-aQ88nltS.woff +0 -0
- package/dist/ui-web/assets/provider-utils_false-BKJD46kk.js +1 -0
- package/dist/ui-web/assets/provider-utils_false-Bu5lmX18.js +1 -0
- package/dist/ui-web/index.html +13 -0
- package/dist/ui-web/vite.svg +1 -0
- package/dist/vector/embedding.d.ts.map +1 -1
- package/dist/vector/embedding.js +7 -7
- package/package.json +7 -5
- package/src/api/config.ts +0 -3
- package/src/api/sonamu.ts +17 -4
- package/src/bin/cli.ts +1 -67
- package/src/syncer/api-parser.ts +2 -1
- package/src/syncer/syncer.ts +20 -21
- package/src/ui/ai-api.ts +60 -0
- package/src/ui/ai-client.ts +499 -0
- package/src/ui/api.ts +786 -0
- package/src/vector/embedding.ts +8 -6
package/dist/vector/embedding.js
CHANGED
|
@@ -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.
|
|
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": "
|
|
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
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
|
-
|
|
362
|
-
|
|
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"
|
|
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
|
-
}
|
package/src/syncer/api-parser.ts
CHANGED
|
@@ -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에 현재 메소드 타입 정보 확장
|
package/src/syncer/syncer.ts
CHANGED
|
@@ -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
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
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.
|
|
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
|
*/
|
package/src/ui/ai-api.ts
ADDED
|
@@ -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
|
+
// }
|