sonamu 0.8.21 → 0.8.23
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 +1 -1
- package/dist/api/config.d.ts.map +1 -1
- package/dist/api/config.js +1 -1
- package/dist/api/sonamu.d.ts +1 -1
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +14 -19
- package/dist/cone/cone-generator.js +29 -16
- package/dist/database/base-model.js +2 -2
- package/dist/database/db.d.ts +1 -1
- package/dist/database/db.d.ts.map +1 -1
- package/dist/database/db.js +1 -1
- package/dist/syncer/syncer.js +2 -2
- package/dist/testing/dev-test-routes.d.ts.map +1 -1
- package/dist/testing/dev-test-routes.js +4 -1
- package/dist/testing/global-setup.d.ts.map +1 -1
- package/dist/testing/global-setup.js +1 -4
- package/dist/ui/api.d.ts.map +1 -1
- package/dist/ui/api.js +11 -11
- package/dist/ui-web/assets/index-BrQKU3j9.css +1 -0
- package/dist/ui-web/assets/{index-DP968oXY.js → index-DIasN3Gt.js} +47 -47
- package/dist/ui-web/index.html +2 -2
- package/package.json +3 -3
- package/src/api/config.ts +1 -1
- package/src/api/sonamu.ts +15 -20
- package/src/cone/cone-generator.ts +39 -19
- package/src/database/base-model.ts +1 -1
- package/src/database/db.ts +1 -1
- package/src/skills/AGENTS.md +11 -1
- package/src/skills/sonamu/SKILL.md +42 -149
- package/src/skills/sonamu/cdd.md +172 -517
- package/src/skills/sonamu/cone.md +3 -4
- package/src/skills/sonamu/fixture-cli.md +1 -1
- package/src/skills/sonamu/workflow.md +72 -80
- package/src/syncer/syncer.ts +1 -1
- package/src/testing/dev-test-routes.ts +3 -0
- package/src/testing/global-setup.ts +0 -4
- package/src/ui/api.ts +6 -15
- package/dist/ui-web/assets/index-B7gc0Ygb.css +0 -1
- package/src/skills/project/business-logic.md +0 -270
- package/src/skills/project/requirements.md +0 -160
|
@@ -43,28 +43,41 @@ import * as path from "node:path";
|
|
|
43
43
|
return apiKey;
|
|
44
44
|
}
|
|
45
45
|
/**
|
|
46
|
-
*
|
|
46
|
+
* 도메인별 {domain}.contract.md와 architecture.md를 읽어 컨텍스트로 반환합니다.
|
|
47
47
|
*
|
|
48
|
-
*
|
|
49
|
-
*
|
|
48
|
+
* - contract/{domain}/{domain}.contract.md: 도메인 규칙과 결정 근거 (주 참조 대상)
|
|
49
|
+
* - .claude/skills/project/architecture.md: 엔티티 설계 구조 (보조 참조)
|
|
50
|
+
*
|
|
51
|
+
* cone 생성 시 LLM에게 전달하여 도메인 맥락에 맞는 메타데이터를 생성하도록 합니다.
|
|
50
52
|
*/ function readProjectSkills() {
|
|
51
53
|
try {
|
|
52
54
|
const { Sonamu } = require("../api");
|
|
53
55
|
const projectRoot = Sonamu.appRootPath;
|
|
54
|
-
const skillsDir = path.join(projectRoot, ".claude", "skills", "project");
|
|
55
|
-
if (!fs.existsSync(skillsDir)) {
|
|
56
|
-
return "";
|
|
57
|
-
}
|
|
58
|
-
const files = fs.readdirSync(skillsDir).filter((f)=>f.endsWith(".md")).sort();
|
|
59
|
-
if (files.length === 0) {
|
|
60
|
-
return "";
|
|
61
|
-
}
|
|
62
56
|
const contents = [];
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
57
|
+
// contract/**/*.contract.md 수집
|
|
58
|
+
const contractDir = path.join(projectRoot, "contract");
|
|
59
|
+
if (fs.existsSync(contractDir)) {
|
|
60
|
+
const domains = fs.readdirSync(contractDir, {
|
|
61
|
+
withFileTypes: true
|
|
62
|
+
}).filter((d)=>d.isDirectory()).map((d)=>d.name);
|
|
63
|
+
for (const domain of domains){
|
|
64
|
+
const domainDir = path.join(contractDir, domain);
|
|
65
|
+
const contractFiles = fs.readdirSync(domainDir).filter((f)=>f.endsWith(".contract.md"));
|
|
66
|
+
for (const file of contractFiles){
|
|
67
|
+
const filePath = path.join(domainDir, file);
|
|
68
|
+
const content = fs.readFileSync(filePath, "utf-8").trim();
|
|
69
|
+
if (content) {
|
|
70
|
+
contents.push(`--- contract/${domain}/${file} ---\n${content}`);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
// .claude/skills/project/architecture.md 보조 참조
|
|
76
|
+
const architecturePath = path.join(projectRoot, ".claude", "skills", "project", "architecture.md");
|
|
77
|
+
if (fs.existsSync(architecturePath)) {
|
|
78
|
+
const content = fs.readFileSync(architecturePath, "utf-8").trim();
|
|
66
79
|
if (content) {
|
|
67
|
-
contents.push(`---
|
|
80
|
+
contents.push(`--- architecture.md ---\n${content}`);
|
|
68
81
|
}
|
|
69
82
|
}
|
|
70
83
|
return contents.join("\n\n");
|
|
@@ -325,4 +338,4 @@ IMPORTANT: Return pure JSON only. Do NOT wrap in markdown code blocks.`;
|
|
|
325
338
|
return result;
|
|
326
339
|
}
|
|
327
340
|
|
|
328
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/cone/cone-generator.ts"],"sourcesContent":["import * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport type { Cone, EntityJson } from \"../types/types\";\n\n/**\n * Cone 생성 컨텍스트\n *\n * Entity 정보와 생성 옵션을 담고 있습니다.\n */\nexport type ConeGenerationContext = {\n  entity: EntityJson;\n  locale?: \"ko\" | \"en\" | \"ja\";\n  existingCones?: Record<string, Cone>;\n  /** true인 경우 note가 없는 cone만 생성 */\n  onlyEmpty?: boolean;\n};\n\n/**\n * Cone 생성 결과\n *\n * Entity, Props, Subsets, Enums의 cone 메타데이터를 담고 있습니다.\n */\nexport type ConeGenerationResult = {\n  entityCone?: Cone;\n  propCones: Record<string, Cone>;\n  subsetCones: Record<string, Cone>;\n  enumCones: Record<string, Cone>;\n  tokensUsed: number;\n};\n\n/**\n * LLM을 사용하여 Entity의 cone 메타데이터를 생성합니다.\n *\n * @param context - Entity 정보와 생성 옵션\n * @returns 생성된 cone 메타데이터\n */\nexport async function generateCones(context: ConeGenerationContext): Promise<ConeGenerationResult> {\n  const apiKey = getApiKey();\n  const prompt = buildPrompt(context);\n  const { text: responseText, tokensUsed } = await callAnthropicAPI(prompt, apiKey);\n  const result = parseConeResponse(responseText);\n  result.tokensUsed = tokensUsed;\n\n  if (context.existingCones) {\n    if (context.onlyEmpty) {\n      return mergeOnlyEmpty(result, context.existingCones);\n    }\n    return mergeWithExisting(result, context.existingCones);\n  }\n\n  return result;\n}\n\n/**\n * API 키를 가져옵니다.\n *\n * Sonamu.secret 또는 환경변수에서 가져옵니다.\n */\nfunction getApiKey(): string {\n  // Sonamu.secret은 런타임에 로드되므로 동적으로 import\n  let apiKey: string | undefined;\n\n  try {\n    // Sonamu가 초기화되어 있는 경우\n    const { Sonamu } = require(\"../api\");\n    apiKey = Sonamu.secret?.anthropic_api_key;\n  } catch {\n    // Sonamu가 초기화되지 않은 경우 (테스트 등)\n    apiKey = undefined;\n  }\n\n  if (!apiKey) {\n    apiKey = process.env.ANTHROPIC_API_KEY;\n  }\n\n  if (!apiKey) {\n    throw new Error(\n      \"ANTHROPIC_API_KEY not found. \" +\n        \"Set ANTHROPIC_API_KEY environment variable or add it to sonamu.secret.ts\",\n    );\n  }\n\n  return apiKey;\n}\n\n/**\n * 프로젝트 루트의 .claude/skills/project/*.md 파일들을 읽어 컨텍스트로 반환합니다.\n *\n * requirements.md 등 프로젝트의 비즈니스 요구사항과 도메인 지식을 담은 파일들을\n * cone 생성 시 LLM에게 전달하여 현실적인 메타데이터를 생성하도록 합니다.\n */\nfunction readProjectSkills(): string {\n  try {\n    const { Sonamu } = require(\"../api\");\n    const projectRoot = Sonamu.appRootPath;\n    const skillsDir = path.join(projectRoot, \".claude\", \"skills\", \"project\");\n\n    if (!fs.existsSync(skillsDir)) {\n      return \"\";\n    }\n\n    const files = fs\n      .readdirSync(skillsDir)\n      .filter((f: string) => f.endsWith(\".md\"))\n      .sort();\n    if (files.length === 0) {\n      return \"\";\n    }\n\n    const contents: string[] = [];\n    for (const file of files) {\n      const filePath = path.join(skillsDir, file);\n      const content = fs.readFileSync(filePath, \"utf-8\").trim();\n      if (content) {\n        contents.push(`--- ${file} ---\\n${content}`);\n      }\n    }\n\n    return contents.join(\"\\n\\n\");\n  } catch {\n    // Sonamu 미초기화 또는 파일 접근 오류 시 빈 문자열 반환\n    return \"\";\n  }\n}\n\n/**\n * LLM 프롬프트를 생성합니다.\n *\n * ai-client.ts 패턴을 참고하여 명확한 지시사항과 출력 형식을 제공합니다.\n */\nfunction buildPrompt(context: ConeGenerationContext): string {\n  const locale = context.locale || \"ko\";\n  const localeDesc = {\n    ko: \"Korean\",\n    en: \"English\",\n    ja: \"Japanese\",\n  }[locale];\n\n  const projectContext = readProjectSkills();\n  const projectSection = projectContext\n    ? `\\nPROJECT CONTEXT (business requirements and domain knowledge):\\n${projectContext}\\n\\nUse the above project context to understand the business domain, entity purposes, field meanings, and relationships. Generate cone metadata that reflects this project's actual requirements, not generic assumptions.\\n`\n    : \"\";\n\n  return `You are a Sonamu framework expert. Generate cone metadata for database entity fixture generation.\n\nCRITICAL PRIORITY RULE:\nThe \"note\" field is the PRIMARY source for fixture data generation. When --use-llm is enabled, the fixture generator reads cone.note and asks LLM to generate contextually appropriate data BEFORE falling back to fixtureGenerator.\nTherefore, cone.note must always contain rich, domain-specific descriptions with concrete examples and value ranges.\nfixtureGenerator is only a FALLBACK for when LLM is unavailable (no API key). Write it as a best-effort approximation, but never rely on it as the primary generation method.\n${projectSection}\nENTITY STRUCTURE:\n${JSON.stringify(context.entity, null, 2)}\n\nLOCALE: ${locale} (${localeDesc})\n\nINSTRUCTIONS:\n1. Entity cone metadata:\n   - note: Describe what this entity represents, its purpose, relationships, business context, and overall guidance for generating test data. Combine all relevant information into one coherent description.\n   - tags: Relevant categorization tags\n\n2. For each prop, generate appropriate cone metadata:\n   - note (MOST IMPORTANT): Describe what this field represents in business terms, and provide detailed guidance for realistic test data generation. Include concrete examples, value ranges, formatting rules, and domain constraints. This is the primary input LLM uses to generate fixture data.\n   - fixtureGenerator: faker.js expression as FALLBACK only (see rule 9 for exceptions). For free-text fields where faker cannot produce domain-appropriate content (description, summary, note, reason, title, etc.), prefer using faker.helpers.arrayElement([...]) with 5-10 domain-specific example values rather than faker.lorem.*.\n\n3. Field type → faker.js mapping:\n   - email → faker.internet.email()\n   - phone → faker.phone.number()\n   - name/username → faker.person.fullName() (with locale)\n   - birth_date → faker.date.birthdate({ min: 18, max: 65, mode: 'age' })\n   - salary → faker.number.int({ min: 30_000_000, max: 150_000_000 }) for ko locale\n   - company_name → faker.company.name()\n   - address → faker.location.streetAddress()\n\n4. Relation fields (BelongsToOne, OneToOne with hasJoinColumn):\n   - Always add dataSource: { strategy: \"recent\", config: { limit: 3-5 } }\n   - note: Explain what this relation represents and that it references existing data\n\n5. Subsets cone metadata (IMPORTANT - generate for ALL subsets):\n   - note: Describe what this subset represents, what fields it includes, and when to use it\n\n6. Enums cone metadata (IMPORTANT - generate for ALL enums):\n   - note: Describe what this enum represents. If any prop uses this enum type, include the same guidance from that prop's note.\n   - For each enum value, provide note explaining what that specific value means\n\n7. Korean field names (locale=ko):\n   - Infer meaning and generate appropriate faker\n   - \"이름\" → faker.person.fullName()\n   - \"생년월일\" → faker.date.birthdate()\n   - \"주소\" → faker.location.streetAddress()\n\n8. Locale-specific values:\n   - ko: Korean names, addresses, phone numbers (010-XXXX-XXXX format)\n   - en: English names, US addresses\n   - ja: Japanese names, addresses\n\n9. Correlated fields (IMPORTANT - do NOT use fixtureGenerator for these):\n   If multiple props are semantically related and must be consistent with each other (e.g. name + name_en, name + name_ja, title + title_en), do NOT set fixtureGenerator on any of them.\n   Instead, set only note with a clear description that explains the relationship.\n   Example: if name is a Korean full name like \"김민수\", then name_en must be its romanized form \"Kim Minsu\".\n   The fixture generator will pass all such props together to LLM in a single call to ensure consistency.\n   Detection rule: if a prop name matches another prop name with a locale suffix (_en, _ko, _ja, _cn) or vice versa, treat them as correlated.\n\n10. String PK — sequence vs UUID:\n   - DB sequence id: If a prop named \"id\" has type \"string\" and uses a DB sequence (indicated by dbDefault containing \"nextval\"), set fixtureStrategy: \"sequence\" and do NOT set fixtureGenerator. note should mention sequential number stored as string.\n   - better-auth entity id: Account, Session, Verification 엔티티의 id는 better-auth가 crypto.randomUUID()로 생성하는 UUID다. fixtureStrategy: \"sequence\"를 절대 사용하지 말고, fixtureGenerator: \"faker.string.uuid()\"를 사용한다.\n\n11. fixtureCompanions (IMPORTANT - never generate or modify):\n   - fixtureCompanions is user-declared metadata that triggers automatic companion fixture creation when a parent fixture is generated.\n   - Do NOT generate or suggest fixtureCompanions for any prop. Only users declare this intentionally.\n   - If a prop's existing cone already contains fixtureCompanions, preserve it exactly as-is in the propCones output. Do not remove, alter, or omit it.\n   - Example: if User entity's \"id\" prop cone has fixtureCompanions, include it unchanged in propCones[\"id\"].\n\n${\n  context.existingCones\n    ? `\nEXISTING CONES (preserve these if present):\n${JSON.stringify(context.existingCones, null, 2)}\n`\n    : \"\"\n}\n\nOUTPUT FORMAT:\nReturn ONLY valid JSON (no markdown, no code blocks). Use this exact structure:\n{\n  \"entityCone\": {\n    \"note\": \"Description of the entity, its purpose, and guidance for fixture generation\",\n    \"tags\": [\"optional\", \"tags\"]\n  },\n  \"propCones\": {\n    \"prop_name\": {\n      \"note\": \"Description of this field and guidance for realistic test data generation\",\n      \"fixtureGenerator\": \"faker.xxx.yyy()\",\n      \"dataSource\": { \"strategy\": \"recent\", \"config\": { \"limit\": 5 } }\n    }\n  },\n  \"subsetCones\": {\n    \"A\": {\n      \"note\": \"Description of subset A, what fields it includes, and when to use it\"\n    }\n  },\n  \"enumCones\": {\n    \"EnumName\": {\n      \"note\": \"Description of the enum and guidance for generating values\",\n      \"values\": {\n        \"VALUE_KEY\": {\n          \"note\": \"${localeDesc} description of this enum value\"\n        }\n      }\n    }\n  }\n}\n\nIMPORTANT: Return pure JSON only. Do NOT wrap in markdown code blocks.`;\n}\n\n/**\n * Anthropic API를 호출하여 LLM 응답을 받습니다.\n *\n * @param prompt - 생성할 프롬프트\n * @param apiKey - Anthropic API 키\n * @returns LLM 응답 텍스트 및 토큰 사용량\n */\nasync function callAnthropicAPI(\n  prompt: string,\n  apiKey: string,\n): Promise<{ text: string; tokensUsed: number }> {\n  try {\n    // @ai-sdk/anthropic과 ai 패키지는 optional dependency이므로 동적 import\n    const { createAnthropic } = await import(\"@ai-sdk/anthropic\");\n    const { generateText } = await import(\"ai\");\n\n    const anthropic = createAnthropic({\n      apiKey,\n    });\n\n    const { text, usage } = await generateText({\n      model: anthropic(\"claude-sonnet-4-5\"),\n      prompt,\n    });\n\n    const tokensUsed = usage?.totalTokens || 0;\n    if (usage) {\n      console.log(`[Cone Generator] Tokens used: ${tokensUsed}`);\n    }\n\n    return { text, tokensUsed };\n  } catch (error: unknown) {\n    if (error && typeof error === \"object\" && \"statusCode\" in error) {\n      const statusCode = (error as { statusCode: number }).statusCode;\n      if (statusCode === 429) {\n        throw new Error(\"Rate limit exceeded. Please try again later.\");\n      }\n    }\n\n    const message = error instanceof Error ? error.message : \"Unknown error\";\n    throw new Error(`LLM API failed: ${message}`);\n  }\n}\n\n/**\n * LLM 응답을 파싱하여 ConeGenerationResult로 변환합니다.\n *\n * Markdown 코드 블록이 포함되어 있으면 제거합니다.\n */\nfunction parseConeResponse(text: string): ConeGenerationResult {\n  let jsonText = text.trim();\n  jsonText = jsonText.replace(/^```json\\s*/i, \"\");\n  jsonText = jsonText.replace(/```\\s*$/, \"\");\n  jsonText = jsonText.trim();\n\n  try {\n    const parsed = JSON.parse(jsonText);\n\n    if (!parsed.propCones || typeof parsed.propCones !== \"object\") {\n      throw new Error(\"Invalid response: propCones is required and must be an object\");\n    }\n\n    return {\n      entityCone: parsed.entityCone,\n      propCones: parsed.propCones,\n      subsetCones: parsed.subsetCones || {},\n      enumCones: parsed.enumCones || {},\n      tokensUsed: 0,\n    };\n  } catch (error) {\n    const message = error instanceof Error ? error.message : \"Unknown error\";\n    throw new Error(\n      `Failed to parse LLM response: ${message}\\n\\n` +\n        `Original response:\\n${text}\\n\\n` +\n        `Cleaned JSON:\\n${jsonText}`,\n    );\n  }\n}\n\n/**\n * 생성된 cone을 기존 cone과 병합합니다.\n *\n * 기존 cone이 있으면 보존하고, 없는 경우에만 생성된 cone을 사용합니다.\n */\nfunction mergeWithExisting(\n  generated: ConeGenerationResult,\n  existing: Record<string, Cone>,\n): ConeGenerationResult {\n  const result = { ...generated };\n\n  const entityKey = `entity:${generated.entityCone ? \"present\" : \"missing\"}`;\n  if (existing[entityKey]) {\n    result.entityCone = existing[entityKey];\n  }\n\n  for (const propName of Object.keys(generated.propCones)) {\n    const key = `prop:${propName}`;\n    if (existing[key]) {\n      result.propCones[propName] = existing[key];\n    }\n  }\n\n  for (const enumId of Object.keys(generated.enumCones)) {\n    const key = `enum:${enumId}`;\n    if (existing[key]) {\n      result.enumCones[enumId] = existing[key];\n    }\n  }\n\n  for (const subsetKey of Object.keys(generated.subsetCones)) {\n    const key = `subset:${subsetKey}`;\n    if (existing[key]) {\n      result.subsetCones[subsetKey] = existing[key];\n    }\n  }\n\n  return result;\n}\n\n/**\n * note가 없는 cone만 생성하고 나머지는 보존합니다.\n *\n * 기존 cone에 note가 있으면 보존하고, 없으면 새로 생성된 cone을 사용합니다.\n */\nfunction mergeOnlyEmpty(\n  generated: ConeGenerationResult,\n  existing: Record<string, Cone>,\n): ConeGenerationResult {\n  const result = { ...generated };\n\n  // Entity cone: scale이 있으면 보존\n  const entityKey = `entity:${generated.entityCone ? \"present\" : \"missing\"}`;\n  if (existing[entityKey]?.note) {\n    result.entityCone = existing[entityKey];\n  }\n\n  // Prop cones: scale이 있으면 보존\n  for (const propName of Object.keys(generated.propCones)) {\n    const key = `prop:${propName}`;\n    if (existing[key]?.note) {\n      result.propCones[propName] = existing[key];\n    }\n  }\n\n  // Enum cones: scale이 있으면 보존\n  for (const enumId of Object.keys(generated.enumCones)) {\n    const key = `enum:${enumId}`;\n    if (existing[key]?.note) {\n      result.enumCones[enumId] = existing[key];\n    }\n  }\n\n  // Subset cones: scale이 있으면 보존\n  for (const subsetKey of Object.keys(generated.subsetCones)) {\n    const key = `subset:${subsetKey}`;\n    if (existing[key]?.note) {\n      result.subsetCones[subsetKey] = existing[key];\n    }\n  }\n\n  return result;\n}\n"],"names":["fs","path","generateCones","context","apiKey","getApiKey","prompt","buildPrompt","text","responseText","tokensUsed","callAnthropicAPI","result","parseConeResponse","existingCones","onlyEmpty","mergeOnlyEmpty","mergeWithExisting","Sonamu","require","secret","anthropic_api_key","undefined","process","env","ANTHROPIC_API_KEY","Error","readProjectSkills","projectRoot","appRootPath","skillsDir","join","existsSync","files","readdirSync","filter","f","endsWith","sort","length","contents","file","filePath","content","readFileSync","trim","push","locale","localeDesc","ko","en","ja","projectContext","projectSection","JSON","stringify","entity","createAnthropic","generateText","anthropic","usage","model","totalTokens","console","log","error","statusCode","message","jsonText","replace","parsed","parse","propCones","entityCone","subsetCones","enumCones","generated","existing","entityKey","propName","Object","keys","key","enumId","subsetKey","note"],"mappings":"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"}
|
|
341
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/cone/cone-generator.ts"],"sourcesContent":["import * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport type { Cone, EntityJson } from \"../types/types\";\n\n/**\n * Cone 생성 컨텍스트\n *\n * Entity 정보와 생성 옵션을 담고 있습니다.\n */\nexport type ConeGenerationContext = {\n  entity: EntityJson;\n  locale?: \"ko\" | \"en\" | \"ja\";\n  existingCones?: Record<string, Cone>;\n  /** true인 경우 note가 없는 cone만 생성 */\n  onlyEmpty?: boolean;\n};\n\n/**\n * Cone 생성 결과\n *\n * Entity, Props, Subsets, Enums의 cone 메타데이터를 담고 있습니다.\n */\nexport type ConeGenerationResult = {\n  entityCone?: Cone;\n  propCones: Record<string, Cone>;\n  subsetCones: Record<string, Cone>;\n  enumCones: Record<string, Cone>;\n  tokensUsed: number;\n};\n\n/**\n * LLM을 사용하여 Entity의 cone 메타데이터를 생성합니다.\n *\n * @param context - Entity 정보와 생성 옵션\n * @returns 생성된 cone 메타데이터\n */\nexport async function generateCones(context: ConeGenerationContext): Promise<ConeGenerationResult> {\n  const apiKey = getApiKey();\n  const prompt = buildPrompt(context);\n  const { text: responseText, tokensUsed } = await callAnthropicAPI(prompt, apiKey);\n  const result = parseConeResponse(responseText);\n  result.tokensUsed = tokensUsed;\n\n  if (context.existingCones) {\n    if (context.onlyEmpty) {\n      return mergeOnlyEmpty(result, context.existingCones);\n    }\n    return mergeWithExisting(result, context.existingCones);\n  }\n\n  return result;\n}\n\n/**\n * API 키를 가져옵니다.\n *\n * Sonamu.secret 또는 환경변수에서 가져옵니다.\n */\nfunction getApiKey(): string {\n  // Sonamu.secret은 런타임에 로드되므로 동적으로 import\n  let apiKey: string | undefined;\n\n  try {\n    // Sonamu가 초기화되어 있는 경우\n    const { Sonamu } = require(\"../api\");\n    apiKey = Sonamu.secret?.anthropic_api_key;\n  } catch {\n    // Sonamu가 초기화되지 않은 경우 (테스트 등)\n    apiKey = undefined;\n  }\n\n  if (!apiKey) {\n    apiKey = process.env.ANTHROPIC_API_KEY;\n  }\n\n  if (!apiKey) {\n    throw new Error(\n      \"ANTHROPIC_API_KEY not found. \" +\n        \"Set ANTHROPIC_API_KEY environment variable or add it to sonamu.secret.ts\",\n    );\n  }\n\n  return apiKey;\n}\n\n/**\n * 도메인별 {domain}.contract.md와 architecture.md를 읽어 컨텍스트로 반환합니다.\n *\n * - contract/{domain}/{domain}.contract.md: 도메인 규칙과 결정 근거 (주 참조 대상)\n * - .claude/skills/project/architecture.md: 엔티티 설계 구조 (보조 참조)\n *\n * cone 생성 시 LLM에게 전달하여 도메인 맥락에 맞는 메타데이터를 생성하도록 합니다.\n */\nfunction readProjectSkills(): string {\n  try {\n    const { Sonamu } = require(\"../api\");\n    const projectRoot = Sonamu.appRootPath;\n    const contents: string[] = [];\n\n    // contract/**/*.contract.md 수집\n    const contractDir = path.join(projectRoot, \"contract\");\n    if (fs.existsSync(contractDir)) {\n      const domains = fs\n        .readdirSync(contractDir, { withFileTypes: true })\n        .filter((d) => d.isDirectory())\n        .map((d) => d.name);\n\n      for (const domain of domains) {\n        const domainDir = path.join(contractDir, domain);\n        const contractFiles = fs\n          .readdirSync(domainDir)\n          .filter((f: string) => f.endsWith(\".contract.md\"));\n\n        for (const file of contractFiles) {\n          const filePath = path.join(domainDir, file);\n          const content = fs.readFileSync(filePath, \"utf-8\").trim();\n          if (content) {\n            contents.push(`--- contract/${domain}/${file} ---\\n${content}`);\n          }\n        }\n      }\n    }\n\n    // .claude/skills/project/architecture.md 보조 참조\n    const architecturePath = path.join(\n      projectRoot,\n      \".claude\",\n      \"skills\",\n      \"project\",\n      \"architecture.md\",\n    );\n    if (fs.existsSync(architecturePath)) {\n      const content = fs.readFileSync(architecturePath, \"utf-8\").trim();\n      if (content) {\n        contents.push(`--- architecture.md ---\\n${content}`);\n      }\n    }\n\n    return contents.join(\"\\n\\n\");\n  } catch {\n    // Sonamu 미초기화 또는 파일 접근 오류 시 빈 문자열 반환\n    return \"\";\n  }\n}\n\n/**\n * LLM 프롬프트를 생성합니다.\n *\n * ai-client.ts 패턴을 참고하여 명확한 지시사항과 출력 형식을 제공합니다.\n */\nfunction buildPrompt(context: ConeGenerationContext): string {\n  const locale = context.locale || \"ko\";\n  const localeDesc = {\n    ko: \"Korean\",\n    en: \"English\",\n    ja: \"Japanese\",\n  }[locale];\n\n  const projectContext = readProjectSkills();\n  const projectSection = projectContext\n    ? `\\nPROJECT CONTEXT (business requirements and domain knowledge):\\n${projectContext}\\n\\nUse the above project context to understand the business domain, entity purposes, field meanings, and relationships. Generate cone metadata that reflects this project's actual requirements, not generic assumptions.\\n`\n    : \"\";\n\n  return `You are a Sonamu framework expert. Generate cone metadata for database entity fixture generation.\n\nCRITICAL PRIORITY RULE:\nThe \"note\" field is the PRIMARY source for fixture data generation. When --use-llm is enabled, the fixture generator reads cone.note and asks LLM to generate contextually appropriate data BEFORE falling back to fixtureGenerator.\nTherefore, cone.note must always contain rich, domain-specific descriptions with concrete examples and value ranges.\nfixtureGenerator is only a FALLBACK for when LLM is unavailable (no API key). Write it as a best-effort approximation, but never rely on it as the primary generation method.\n${projectSection}\nENTITY STRUCTURE:\n${JSON.stringify(context.entity, null, 2)}\n\nLOCALE: ${locale} (${localeDesc})\n\nINSTRUCTIONS:\n1. Entity cone metadata:\n   - note: Describe what this entity represents, its purpose, relationships, business context, and overall guidance for generating test data. Combine all relevant information into one coherent description.\n   - tags: Relevant categorization tags\n\n2. For each prop, generate appropriate cone metadata:\n   - note (MOST IMPORTANT): Describe what this field represents in business terms, and provide detailed guidance for realistic test data generation. Include concrete examples, value ranges, formatting rules, and domain constraints. This is the primary input LLM uses to generate fixture data.\n   - fixtureGenerator: faker.js expression as FALLBACK only (see rule 9 for exceptions). For free-text fields where faker cannot produce domain-appropriate content (description, summary, note, reason, title, etc.), prefer using faker.helpers.arrayElement([...]) with 5-10 domain-specific example values rather than faker.lorem.*.\n\n3. Field type → faker.js mapping:\n   - email → faker.internet.email()\n   - phone → faker.phone.number()\n   - name/username → faker.person.fullName() (with locale)\n   - birth_date → faker.date.birthdate({ min: 18, max: 65, mode: 'age' })\n   - salary → faker.number.int({ min: 30_000_000, max: 150_000_000 }) for ko locale\n   - company_name → faker.company.name()\n   - address → faker.location.streetAddress()\n\n4. Relation fields (BelongsToOne, OneToOne with hasJoinColumn):\n   - Always add dataSource: { strategy: \"recent\", config: { limit: 3-5 } }\n   - note: Explain what this relation represents and that it references existing data\n\n5. Subsets cone metadata (IMPORTANT - generate for ALL subsets):\n   - note: Describe what this subset represents, what fields it includes, and when to use it\n\n6. Enums cone metadata (IMPORTANT - generate for ALL enums):\n   - note: Describe what this enum represents. If any prop uses this enum type, include the same guidance from that prop's note.\n   - For each enum value, provide note explaining what that specific value means\n\n7. Korean field names (locale=ko):\n   - Infer meaning and generate appropriate faker\n   - \"이름\" → faker.person.fullName()\n   - \"생년월일\" → faker.date.birthdate()\n   - \"주소\" → faker.location.streetAddress()\n\n8. Locale-specific values:\n   - ko: Korean names, addresses, phone numbers (010-XXXX-XXXX format)\n   - en: English names, US addresses\n   - ja: Japanese names, addresses\n\n9. Correlated fields (IMPORTANT - do NOT use fixtureGenerator for these):\n   If multiple props are semantically related and must be consistent with each other (e.g. name + name_en, name + name_ja, title + title_en), do NOT set fixtureGenerator on any of them.\n   Instead, set only note with a clear description that explains the relationship.\n   Example: if name is a Korean full name like \"김민수\", then name_en must be its romanized form \"Kim Minsu\".\n   The fixture generator will pass all such props together to LLM in a single call to ensure consistency.\n   Detection rule: if a prop name matches another prop name with a locale suffix (_en, _ko, _ja, _cn) or vice versa, treat them as correlated.\n\n10. String PK — sequence vs UUID:\n   - DB sequence id: If a prop named \"id\" has type \"string\" and uses a DB sequence (indicated by dbDefault containing \"nextval\"), set fixtureStrategy: \"sequence\" and do NOT set fixtureGenerator. note should mention sequential number stored as string.\n   - better-auth entity id: Account, Session, Verification 엔티티의 id는 better-auth가 crypto.randomUUID()로 생성하는 UUID다. fixtureStrategy: \"sequence\"를 절대 사용하지 말고, fixtureGenerator: \"faker.string.uuid()\"를 사용한다.\n\n11. fixtureCompanions (IMPORTANT - never generate or modify):\n   - fixtureCompanions is user-declared metadata that triggers automatic companion fixture creation when a parent fixture is generated.\n   - Do NOT generate or suggest fixtureCompanions for any prop. Only users declare this intentionally.\n   - If a prop's existing cone already contains fixtureCompanions, preserve it exactly as-is in the propCones output. Do not remove, alter, or omit it.\n   - Example: if User entity's \"id\" prop cone has fixtureCompanions, include it unchanged in propCones[\"id\"].\n\n${\n  context.existingCones\n    ? `\nEXISTING CONES (preserve these if present):\n${JSON.stringify(context.existingCones, null, 2)}\n`\n    : \"\"\n}\n\nOUTPUT FORMAT:\nReturn ONLY valid JSON (no markdown, no code blocks). Use this exact structure:\n{\n  \"entityCone\": {\n    \"note\": \"Description of the entity, its purpose, and guidance for fixture generation\",\n    \"tags\": [\"optional\", \"tags\"]\n  },\n  \"propCones\": {\n    \"prop_name\": {\n      \"note\": \"Description of this field and guidance for realistic test data generation\",\n      \"fixtureGenerator\": \"faker.xxx.yyy()\",\n      \"dataSource\": { \"strategy\": \"recent\", \"config\": { \"limit\": 5 } }\n    }\n  },\n  \"subsetCones\": {\n    \"A\": {\n      \"note\": \"Description of subset A, what fields it includes, and when to use it\"\n    }\n  },\n  \"enumCones\": {\n    \"EnumName\": {\n      \"note\": \"Description of the enum and guidance for generating values\",\n      \"values\": {\n        \"VALUE_KEY\": {\n          \"note\": \"${localeDesc} description of this enum value\"\n        }\n      }\n    }\n  }\n}\n\nIMPORTANT: Return pure JSON only. Do NOT wrap in markdown code blocks.`;\n}\n\n/**\n * Anthropic API를 호출하여 LLM 응답을 받습니다.\n *\n * @param prompt - 생성할 프롬프트\n * @param apiKey - Anthropic API 키\n * @returns LLM 응답 텍스트 및 토큰 사용량\n */\nasync function callAnthropicAPI(\n  prompt: string,\n  apiKey: string,\n): Promise<{ text: string; tokensUsed: number }> {\n  try {\n    // @ai-sdk/anthropic과 ai 패키지는 optional dependency이므로 동적 import\n    const { createAnthropic } = await import(\"@ai-sdk/anthropic\");\n    const { generateText } = await import(\"ai\");\n\n    const anthropic = createAnthropic({\n      apiKey,\n    });\n\n    const { text, usage } = await generateText({\n      model: anthropic(\"claude-sonnet-4-5\"),\n      prompt,\n    });\n\n    const tokensUsed = usage?.totalTokens || 0;\n    if (usage) {\n      console.log(`[Cone Generator] Tokens used: ${tokensUsed}`);\n    }\n\n    return { text, tokensUsed };\n  } catch (error: unknown) {\n    if (error && typeof error === \"object\" && \"statusCode\" in error) {\n      const statusCode = (error as { statusCode: number }).statusCode;\n      if (statusCode === 429) {\n        throw new Error(\"Rate limit exceeded. Please try again later.\");\n      }\n    }\n\n    const message = error instanceof Error ? error.message : \"Unknown error\";\n    throw new Error(`LLM API failed: ${message}`);\n  }\n}\n\n/**\n * LLM 응답을 파싱하여 ConeGenerationResult로 변환합니다.\n *\n * Markdown 코드 블록이 포함되어 있으면 제거합니다.\n */\nfunction parseConeResponse(text: string): ConeGenerationResult {\n  let jsonText = text.trim();\n  jsonText = jsonText.replace(/^```json\\s*/i, \"\");\n  jsonText = jsonText.replace(/```\\s*$/, \"\");\n  jsonText = jsonText.trim();\n\n  try {\n    const parsed = JSON.parse(jsonText);\n\n    if (!parsed.propCones || typeof parsed.propCones !== \"object\") {\n      throw new Error(\"Invalid response: propCones is required and must be an object\");\n    }\n\n    return {\n      entityCone: parsed.entityCone,\n      propCones: parsed.propCones,\n      subsetCones: parsed.subsetCones || {},\n      enumCones: parsed.enumCones || {},\n      tokensUsed: 0,\n    };\n  } catch (error) {\n    const message = error instanceof Error ? error.message : \"Unknown error\";\n    throw new Error(\n      `Failed to parse LLM response: ${message}\\n\\n` +\n        `Original response:\\n${text}\\n\\n` +\n        `Cleaned JSON:\\n${jsonText}`,\n    );\n  }\n}\n\n/**\n * 생성된 cone을 기존 cone과 병합합니다.\n *\n * 기존 cone이 있으면 보존하고, 없는 경우에만 생성된 cone을 사용합니다.\n */\nfunction mergeWithExisting(\n  generated: ConeGenerationResult,\n  existing: Record<string, Cone>,\n): ConeGenerationResult {\n  const result = { ...generated };\n\n  const entityKey = `entity:${generated.entityCone ? \"present\" : \"missing\"}`;\n  if (existing[entityKey]) {\n    result.entityCone = existing[entityKey];\n  }\n\n  for (const propName of Object.keys(generated.propCones)) {\n    const key = `prop:${propName}`;\n    if (existing[key]) {\n      result.propCones[propName] = existing[key];\n    }\n  }\n\n  for (const enumId of Object.keys(generated.enumCones)) {\n    const key = `enum:${enumId}`;\n    if (existing[key]) {\n      result.enumCones[enumId] = existing[key];\n    }\n  }\n\n  for (const subsetKey of Object.keys(generated.subsetCones)) {\n    const key = `subset:${subsetKey}`;\n    if (existing[key]) {\n      result.subsetCones[subsetKey] = existing[key];\n    }\n  }\n\n  return result;\n}\n\n/**\n * note가 없는 cone만 생성하고 나머지는 보존합니다.\n *\n * 기존 cone에 note가 있으면 보존하고, 없으면 새로 생성된 cone을 사용합니다.\n */\nfunction mergeOnlyEmpty(\n  generated: ConeGenerationResult,\n  existing: Record<string, Cone>,\n): ConeGenerationResult {\n  const result = { ...generated };\n\n  // Entity cone: scale이 있으면 보존\n  const entityKey = `entity:${generated.entityCone ? \"present\" : \"missing\"}`;\n  if (existing[entityKey]?.note) {\n    result.entityCone = existing[entityKey];\n  }\n\n  // Prop cones: scale이 있으면 보존\n  for (const propName of Object.keys(generated.propCones)) {\n    const key = `prop:${propName}`;\n    if (existing[key]?.note) {\n      result.propCones[propName] = existing[key];\n    }\n  }\n\n  // Enum cones: scale이 있으면 보존\n  for (const enumId of Object.keys(generated.enumCones)) {\n    const key = `enum:${enumId}`;\n    if (existing[key]?.note) {\n      result.enumCones[enumId] = existing[key];\n    }\n  }\n\n  // Subset cones: scale이 있으면 보존\n  for (const subsetKey of Object.keys(generated.subsetCones)) {\n    const key = `subset:${subsetKey}`;\n    if (existing[key]?.note) {\n      result.subsetCones[subsetKey] = existing[key];\n    }\n  }\n\n  return result;\n}\n"],"names":["fs","path","generateCones","context","apiKey","getApiKey","prompt","buildPrompt","text","responseText","tokensUsed","callAnthropicAPI","result","parseConeResponse","existingCones","onlyEmpty","mergeOnlyEmpty","mergeWithExisting","Sonamu","require","secret","anthropic_api_key","undefined","process","env","ANTHROPIC_API_KEY","Error","readProjectSkills","projectRoot","appRootPath","contents","contractDir","join","existsSync","domains","readdirSync","withFileTypes","filter","d","isDirectory","map","name","domain","domainDir","contractFiles","f","endsWith","file","filePath","content","readFileSync","trim","push","architecturePath","locale","localeDesc","ko","en","ja","projectContext","projectSection","JSON","stringify","entity","createAnthropic","generateText","anthropic","usage","model","totalTokens","console","log","error","statusCode","message","jsonText","replace","parsed","parse","propCones","entityCone","subsetCones","enumCones","generated","existing","entityKey","propName","Object","keys","key","enumId","subsetKey","note"],"mappings":"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"}
|
|
@@ -273,7 +273,7 @@ import { UpsertBuilder } from "./upsert-builder.js";
|
|
|
273
273
|
const { default: SqlParser } = await import("node-sql-parser");
|
|
274
274
|
const parser = new SqlParser.Parser();
|
|
275
275
|
const parsedQuery = parser.astify(countPuri.toQuery(), {
|
|
276
|
-
database: Sonamu.config.database
|
|
276
|
+
database: Sonamu.config.database.database
|
|
277
277
|
});
|
|
278
278
|
const leftJoinTables = getJoinTables(parsedQuery, [
|
|
279
279
|
"LEFT JOIN"
|
|
@@ -390,4 +390,4 @@ import { UpsertBuilder } from "./upsert-builder.js";
|
|
|
390
390
|
}
|
|
391
391
|
export const BaseModel = new BaseModelClass();
|
|
392
392
|
|
|
393
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/base-model.ts"],"sourcesContent":["/** biome-ignore-all lint/suspicious/noExplicitAny: Puri의 타입은 개별 모델에서 확정되므로 BaseModel에서는 any를 허용함 */\nimport { getLogger, type Logger } from \"@logtape/logtape\";\nimport type { Knex } from \"knex\";\nimport { cloneDeep, group, isObject, omit, set } from \"radashi\";\nimport { type ListResult, normalizeFilterQuery, validateSonamuFilters } from \"..\";\nimport { Sonamu } from \"../api\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport type { FilterOperator, FilterQuery } from \"../filter/types\";\nimport { convertDomainToCategory } from \"../logger/category\";\nimport type { DatabaseSchemaExtend, SonamuQueryMode } from \"../types/types\";\nimport { getJoinTables, getTableNamesFromWhere } from \"../utils/sql-parser\";\nimport { chunk } from \"../utils/utils\";\nimport type { EnhancerMap, ResolveSubsetIntersection } from \"./base-model.types\";\nimport type { DBPreset } from \"./db\";\nimport { DB } from \"./db\";\nimport { Puri } from \"./puri\";\nimport type { UnionExtractedTTables } from \"./puri.types\";\nimport type { InferAllSubsets, PuriLoaderQueries, PuriSubsetFn } from \"./puri-subset.types\";\nimport { PuriWrapper } from \"./puri-wrapper\";\nimport { UpsertBuilder } from \"./upsert-builder\";\n\ntype UnknownDBRecord = Record<string, unknown>;\n\n/**\n * 모든 Model 클래스의 기본 클래스\n *\n * @template TSubsetKey - 서브셋 키 유니온 (예: \"A\" | \"P\" | \"SS\")\n * @template TSubsetMapping - 서브셋별 최종 결과 타입 매핑\n * @template TSubsetQueries - 서브셋 쿼리 함수 객체\n * @template TLoaderQueries - 서브셋별 로더 쿼리 배열 객체\n */\nexport class BaseModelClass<\n  TSubsetKey extends string = never,\n  TSubsetMapping extends Record<string, any> = never,\n  TSubsetQueries extends Record<TSubsetKey, PuriSubsetFn> = never,\n  TLoaderQueries extends PuriLoaderQueries<TSubsetKey> = never,\n> {\n  protected readonly logger: Logger;\n\n  constructor(\n    public readonly modelName: string = this.constructor.name,\n    protected subsetQueries?: TSubsetQueries,\n    protected loaderQueries?: TLoaderQueries,\n  ) {\n    this.logger = getLogger(convertDomainToCategory(this.modelName, \"model\"));\n  }\n\n  getDB(which: DBPreset): Knex {\n    return DB.getDB(which);\n  }\n\n  getPuri(which: DBPreset): PuriWrapper {\n    // 트랜잭션 컨텍스트에서 트랜잭션 획득\n    const trx = DB.getTransactionContext().getTransaction(which);\n    if (trx) {\n      return trx;\n    }\n\n    // 트랜잭션이 없으면 새로운 PuriWrapper 반환\n    const db = this.getDB(which);\n    return new PuriWrapper(db, new UpsertBuilder());\n  }\n\n  async destroy() {\n    return DB.destroy();\n  }\n\n  async getInsertedIds(\n    wdb: Knex,\n    rows: UnknownDBRecord[],\n    tableName: string,\n    unqKeyFields: string[],\n    chunkSize: number = 500,\n  ) {\n    if (!wdb) {\n      wdb = this.getDB(\"w\");\n    }\n\n    let unqKeys: string[];\n    let whereInField: string | Knex.Raw;\n    let selectField: string;\n\n    if (unqKeyFields.length > 1) {\n      whereInField = wdb.raw(`CONCAT_WS('_', '${unqKeyFields.join(\",\")}')`);\n      selectField = `${whereInField} as tmpUid`;\n      unqKeys = rows.map((row) => unqKeyFields.map((field) => row[field]).join(\"_\"));\n    } else {\n      whereInField = unqKeyFields[0];\n      selectField = unqKeyFields[0];\n      unqKeys = rows.map((row) => row[unqKeyFields[0]] as string);\n    }\n\n    let resultIds: number[] = [];\n    for (const items of chunk(unqKeys, chunkSize)) {\n      const dbRows = await wdb(tableName)\n        .select(\"id\", wdb.raw(selectField))\n        .whereIn(whereInField as string, items);\n      resultIds = resultIds.concat(\n        dbRows.map((dbRow: UnknownDBRecord) => parseInt(String(dbRow.id))),\n      );\n    }\n\n    return resultIds;\n  }\n\n  /**\n   * 특정 서브셋에 대한 쿼리 빌더 획득\n   *\n   * @returns qb - 쿼리 빌더 (조건 추가용)\n   * @returns onSubset - 특정 서브셋 전용 타입이 필요할 때 사용\n   */\n  getSubsetQueries<T extends TSubsetKey>(subset: T) {\n    if (!this.subsetQueries) {\n      throw new Error(\"subsetQueries is not defined\");\n    }\n\n    const puriWrapper = new PuriWrapper(this.getDB(\"r\"), new UpsertBuilder());\n    const qb = this.subsetQueries[subset]?.(puriWrapper);\n\n    // NonAllowedAsSingleTable: 단일 테이블 컬럼 접근 방지용 마커\n    type QBTables = UnionExtractedTTables<TSubsetKey, TSubsetQueries> & {\n      NonAllowedAsSingleTable: { __fulltext__: true };\n    };\n\n    return {\n      qb: qb as unknown as Puri<DatabaseSchemaExtend, QBTables, {}>,\n      onSubset: ((_subset: TSubsetKey | readonly TSubsetKey[]) => qb) as {\n        // 단일 키\n        <S extends TSubsetKey>(subset: S): ReturnType<TSubsetQueries[S]>;\n        // 키 배열 -> 교집합 반환\n        <Arr extends readonly TSubsetKey[]>(\n          subsets: [...Arr],\n        ): ResolveSubsetIntersection<Arr, TSubsetQueries>;\n      },\n    };\n  }\n\n  /**\n   * Enhancer 객체 생성 헬퍼\n   * 타입 검증 및 추론을 도와줌\n   */\n  createEnhancers<T extends TSubsetKey>(\n    enhancers: EnhancerMap<\n      T,\n      InferAllSubsets<TSubsetQueries, TLoaderQueries>,\n      TSubsetMapping,\n      TSubsetQueries\n    >,\n  ) {\n    return enhancers;\n  }\n\n  /**\n   * 서브셋 쿼리 실행\n   *\n   * 1. Sonamu 필터 적용 (타입 변환 포함)\n   * 2. 쿼리 실행 (pagination 적용)\n   * 3. 로더 실행 (1:N, N:M 관계 데이터 로딩)\n   * 4. Hydrate (flat → 중첩 객체)\n   * 5. Enhancer 적용 (virtual 필드 계산)\n   */\n  async executeSubsetQuery<\n    T extends TSubsetKey,\n    TComputedResults extends InferAllSubsets<TSubsetQueries, TLoaderQueries>,\n    LP extends {\n      num?: number;\n      page?: number;\n      queryMode?: SonamuQueryMode;\n      sonamuFilter?: Record<string, unknown>;\n    },\n  >(\n    params: {\n      subset: T;\n      qb: Puri<any, any, any>;\n      params: {\n        num: number;\n        page: number;\n        queryMode?: SonamuQueryMode;\n        sonamuFilter?: Record<string, unknown>;\n      };\n      debug?: boolean;\n      optimizeCountQuery?: boolean;\n    } & EnhancerParam<TSubsetKey, TComputedResults, TSubsetMapping, TSubsetQueries>,\n  ): Promise<ListResult<LP, TSubsetMapping[T]>> {\n    const { subset, qb, params: queryParams, debug = false, optimizeCountQuery = false } = params;\n\n    if (!this.loaderQueries) {\n      throw new Error(\"loaderQueries is not defined\");\n    }\n\n    // Sonamu Filter 적용\n    if (queryParams.sonamuFilter) {\n      const normalizedFilter = normalizeFilterQuery(queryParams.sonamuFilter);\n      this.applySonamuFilters(qb, normalizedFilter);\n    }\n\n    const { num, page } = queryParams;\n\n    // COUNT 쿼리 실행 (queryMode: list일 때는 0 리턴)\n    const total = await this.executeCountQuery(qb, queryParams, debug, optimizeCountQuery);\n\n    if (queryParams?.queryMode === \"count\") {\n      return { total } as ListResult<LP, TSubsetMapping[T]>;\n    }\n\n    // LIST 쿼리 실행\n    const computedRows = await this.executeListQuery(subset, qb, queryParams, num, page, debug);\n\n    // Enhancer 적용\n    const enhancer = (params as any).enhancers?.[subset];\n    const enhancedRows = (await Promise.all(\n      computedRows.map((row) => enhancer?.(row) ?? row),\n    )) as TSubsetMapping[T][];\n\n    // Internal 필드 제거\n    const entity = EntityManager.get(this.modelName);\n    const internalFields = entity.subsetsInternal[subset] ?? [];\n    const rows =\n      internalFields.length > 0\n        ? enhancedRows.map((row) => this.omitInternalFields(row, internalFields))\n        : enhancedRows;\n\n    if (queryParams.queryMode === \"list\") {\n      // 리스트만 리턴\n      return { rows } as ListResult<LP, TSubsetMapping[T]>;\n    } else {\n      // 둘다 리턴\n      return { rows, total } as ListResult<LP, TSubsetMapping[T]>;\n    }\n  }\n\n  /**\n   * 객체에서 internal 필드 제거\n   * 중첩 필드(예: \"user.email\") 및 배열(예: \"employees.salary\")도 처리\n   */\n  omitInternalFields<T extends object>(row: T, fields: string[]): T {\n    const result = cloneDeep(row);\n    for (const field of fields) {\n      this.deleteField(result, field.split(\".\"));\n    }\n    return result;\n  }\n\n  /**\n   * FilterQuery를 Puri QueryBuilder에 적용\n   *\n   * @param qb Puri QueryBuilder 인스턴스\n   * @param filters FilterQuery 객체\n   */\n  protected applySonamuFilters<TEntity = Record<string, unknown>>(\n    qb: Puri<any, any, any>,\n    filters?: FilterQuery<TEntity>,\n  ): void {\n    if (!filters) return;\n\n    const entity = EntityManager.get(this.modelName);\n\n    // 1. 필터 검증 (Entity 기반)\n    validateSonamuFilters(filters, entity);\n\n    // 2. 검증된 필터 적용\n    const puri = qb as any;\n\n    for (const [field, condition] of Object.entries(filters)) {\n      if (condition === undefined || condition === null) continue;\n\n      // 테이블명.필드명 형식으로 변환\n      const fullField = entity.getFullFieldName(field);\n\n      // 직접 값 (eq와 동일)\n      if (typeof condition !== \"object\" || Array.isArray(condition)) {\n        puri.where(fullField, condition);\n        continue;\n      }\n\n      // 연산자 객체\n      for (const [operator, value] of Object.entries(condition)) {\n        this.applyOperator(qb, fullField, operator as FilterOperator, value);\n      }\n    }\n  }\n\n  /**\n   * 단일 연산자를 QueryBuilder에 적용\n   */\n  private applyOperator(\n    qb: Puri<any, any, any>,\n    field: string,\n    operator: FilterOperator,\n    value: unknown,\n  ): void {\n    const puri = qb as any;\n\n    switch (operator) {\n      case \"eq\":\n        puri.where(field, value);\n        break;\n\n      case \"ne\":\n        puri.where(field, \"!=\", value);\n        break;\n\n      case \"gt\":\n        puri.where(field, \">\", value);\n        break;\n\n      case \"gte\":\n        puri.where(field, \">=\", value);\n        break;\n\n      case \"lt\":\n        puri.where(field, \"<\", value);\n        break;\n\n      case \"lte\":\n        puri.where(field, \"<=\", value);\n        break;\n\n      case \"in\":\n        puri.whereIn(field, value);\n        break;\n\n      case \"notIn\":\n        puri.whereNotIn(field, value);\n        break;\n\n      case \"contains\":\n        puri.where(field, \"like\", `%${value}%`);\n        break;\n\n      case \"startsWith\":\n        puri.where(field, \"like\", `${value}%`);\n        break;\n\n      case \"endsWith\":\n        puri.where(field, \"like\", `%${value}`);\n        break;\n\n      case \"isNull\":\n        puri.where(field, null);\n        break;\n\n      case \"isNotNull\":\n        puri.where(field, \"!=\", null);\n        break;\n\n      case \"before\":\n        puri.where(field, \"<\", value);\n        break;\n\n      case \"after\":\n        puri.where(field, \">\", value);\n        break;\n\n      case \"between\": {\n        if (Array.isArray(value) && value.length === 2) {\n          const [min, max] = value;\n          puri.where(field, \">=\", min).where(field, \"<=\", max);\n        }\n        break;\n      }\n\n      default:\n        console.warn(`Unsupported operator: ${operator}`);\n    }\n  }\n\n  /**\n   * 중첩 필드 삭제 (배열 내 객체도 처리)\n   */\n  deleteField(obj: any, parts: string[]): void {\n    if (!obj || typeof obj !== \"object\") {\n      return;\n    }\n\n    if (parts.length === 1) {\n      if (Array.isArray(obj)) {\n        obj.forEach((item) => {\n          if (item && typeof item === \"object\") {\n            delete item[parts[0]];\n          }\n        });\n      } else {\n        delete obj[parts[0]];\n      }\n      return;\n    }\n\n    const [first, ...rest] = parts;\n    const next = obj[first];\n\n    if (Array.isArray(next)) {\n      next.map((item) => this.deleteField(item, rest));\n    } else if (next && typeof next === \"object\") {\n      this.deleteField(next, rest);\n    }\n  }\n\n  /**\n   * COUNT 쿼리 실행 (내부 메서드)\n   */\n  private async executeCountQuery(\n    qb: Puri<any, any, any>,\n    params: { queryMode?: \"list\" | \"count\" | \"both\" },\n    debug: boolean,\n    optimizeCountQuery: boolean,\n  ): Promise<number> {\n    if (params.queryMode === \"list\") {\n      return 0;\n    }\n\n    const countPuri = qb.clone().clear(\"order\").clear(\"limit\").clear(\"offset\");\n\n    if (optimizeCountQuery) {\n      const { default: SqlParser } = await import(\"node-sql-parser\");\n      const parser = new SqlParser.Parser();\n      const parsedQuery = parser.astify(countPuri.toQuery(), {\n        database: Sonamu.config.database?.database,\n      });\n\n      const leftJoinTables = getJoinTables(parsedQuery, [\"LEFT JOIN\"]);\n      const whereTables = getTableNamesFromWhere(parsedQuery);\n\n      const tablesToRemove = leftJoinTables.filter((j) => !whereTables.includes(j));\n      tablesToRemove.forEach((table) => {\n        countPuri.clearJoin(table);\n      });\n    }\n\n    // COUNT(*)로 전체 레코드 수를 계산\n    // TODO: qb의 DISTINCT가 있는 경우 처리해야 함\n    const countResult: { total?: number } = await countPuri\n      .clear(\"select\")\n      .select({ total: Puri.rawNumber(`COUNT(*)::integer`) })\n      .first();\n\n    if (debug) {\n      countPuri.debug();\n    }\n\n    return countResult?.total ?? 0;\n  }\n\n  /**\n   * LIST 쿼리 실행 (내부 메서드)\n   */\n  private async executeListQuery<T extends TSubsetKey>(\n    subset: T,\n    qb: Puri<any, any, any>,\n    params: { queryMode?: \"list\" | \"count\" | \"both\" },\n    num: number,\n    page: number,\n    debug: boolean,\n  ): Promise<any[]> {\n    if (params.queryMode === \"count\") {\n      return [];\n    }\n\n    const limitedQb = (() => {\n      if (num === 0) {\n        return qb;\n      } else {\n        return qb.limit(num).offset(num * (page - 1));\n      }\n    })();\n    let unloadedRows = (await limitedQb) as any[];\n\n    if (debug) {\n      qb.debug();\n    }\n\n    // 로더 처리\n    const loaders = (this.loaderQueries as any)[subset];\n    if (loaders && Array.isArray(loaders)) {\n      unloadedRows = await this.processLoaders(unloadedRows, loaders, debug);\n    }\n\n    return this.hydrate(unloadedRows);\n  }\n\n  /**\n   * 재귀적 로더 처리\n   */\n  private async processLoaders(rows: any[], loaders: any[], debug: boolean): Promise<any[]> {\n    for (const resolveLoader of loaders) {\n      const { as, refId, qb: resolveLoaderQbFn, loaders: nestedLoaders } = resolveLoader;\n\n      const resolveLoaderQb = resolveLoaderQbFn(\n        new PuriWrapper(this.getDB(\"r\"), new UpsertBuilder()),\n        rows.map((row) => row[refId]),\n      );\n\n      if (debug) {\n        resolveLoaderQb.debug();\n      }\n\n      let loadedRows = (await resolveLoaderQb) as any[];\n\n      // 중첩 loaders가 있으면 재귀 처리\n      if (nestedLoaders && nestedLoaders.length > 0) {\n        loadedRows = await this.processLoaders(loadedRows, nestedLoaders, debug);\n      }\n\n      const subRowGroups = group(loadedRows, (row) => row.refId);\n\n      rows = rows.map((row) => {\n        row[as] = (subRowGroups[row[refId]] ?? []).map((r) => omit(r, [\"refId\"]));\n        return row;\n      });\n    }\n\n    return rows;\n  }\n\n  /**\n   * Flat 레코드를 중첩 객체로 변환\n   *\n   * - `user__name` → `{ user: { name } }`\n   * - nullable relation의 경우 id 필드가 null이면 객체 자체를 null로\n   */\n  hydrate<T extends UnknownDBRecord>(rows: T[]): T[] {\n    return rows.map((row: T) => {\n      // nullable relation 처리: 그룹의 id 필드가 null이면 객체 전체를 null로\n      const nestedKeys = Object.keys(row).filter((key) => key.includes(\"__\"));\n      const groups = Object.groupBy(nestedKeys, (key) => key.split(\"__\")[0]);\n\n      // id 필드가 null인 그룹 찾기 (예: parent__id가 null이면 parent 그룹 전체가 null)\n      const nullKeys = Object.entries(groups)\n        .filter(([groupKey, fields]) => {\n          if (!fields || fields.length === 0) return false;\n\n          // 그룹의 id 필드 찾기 (예: \"parent__id\")\n          const idField = `${groupKey}__id`;\n          if (idField in row) {\n            // id 필드가 null이면 객체 전체가 null\n            return row[idField] === null;\n          }\n\n          // id 필드가 없으면 기존 로직: 모든 필드가 null인지 확인\n          return fields.every(\n            (field) =>\n              row[field] === null || (Array.isArray(row[field]) && row[field].length === 0),\n          );\n        })\n        .map(([key]) => key);\n\n      const hydrated = Object.keys(row).reduce((r, field) => {\n        if (!field.includes(\"__\")) {\n          // 일반 필드: 배열 내 객체면 재귀 hydrate\n          if (Array.isArray(row[field]) && isObject(row[field][0])) {\n            r[field] = this.hydrate(row[field]);\n          } else {\n            r[field] = row[field];\n          }\n          return r;\n        }\n\n        // 중첩 필드 처리: user__name → user[name]\n        const parts = field.split(\"__\");\n        const objPath =\n          parts[0] +\n          parts\n            .slice(1)\n            .map((part) => `[${part}]`)\n            .join(\"\");\n\n        r = set(\n          r,\n          objPath,\n          row[field] && Array.isArray(row[field]) && isObject(row[field][0])\n            ? this.hydrate(row[field])\n            : row[field],\n        );\n\n        return r;\n      }, {} as UnknownDBRecord);\n\n      // null relation 처리\n      nullKeys.forEach((nullKey) => {\n        hydrated[nullKey] = null;\n      });\n\n      return hydrated;\n    }) as T[];\n  }\n}\n\n/**\n * Enhancer 파라미터 조건부 타입\n * RequiredEnhancerKeys가 없으면 enhancers 선택적, 있으면 필수\n */\ntype EnhancerParam<\n  TSubsetKey extends string,\n  TComputedResults extends Record<TSubsetKey, any>,\n  TSubsetMapping extends Record<TSubsetKey, any>,\n  TSubsetQueries extends Record<TSubsetKey, PuriSubsetFn>,\n> = [RequiredEnhancerKeys<TSubsetKey, TComputedResults, TSubsetMapping>] extends [never]\n  ? { enhancers?: EnhancerMap<TSubsetKey, TComputedResults, TSubsetMapping, TSubsetQueries> }\n  : { enhancers: EnhancerMap<TSubsetKey, TComputedResults, TSubsetMapping, TSubsetQueries> };\n\ntype RequiredEnhancerKeys<\n  TSubsetKey extends string,\n  TComputedResults extends Record<TSubsetKey, any>,\n  TSubsetMapping extends Record<TSubsetKey, any>,\n> = {\n  [K in TSubsetKey]: TComputedResults[K] extends TSubsetMapping[K] ? never : K;\n}[TSubsetKey];\n\nexport const BaseModel = new BaseModelClass();\n"],"names":["getLogger","cloneDeep","group","isObject","omit","set","normalizeFilterQuery","validateSonamuFilters","Sonamu","EntityManager","convertDomainToCategory","getJoinTables","getTableNamesFromWhere","chunk","DB","Puri","PuriWrapper","UpsertBuilder","BaseModelClass","logger","modelName","name","subsetQueries","loaderQueries","getDB","which","getPuri","trx","getTransactionContext","getTransaction","db","destroy","getInsertedIds","wdb","rows","tableName","unqKeyFields","chunkSize","unqKeys","whereInField","selectField","length","raw","join","map","row","field","resultIds","items","dbRows","select","whereIn","concat","dbRow","parseInt","String","id","getSubsetQueries","subset","Error","puriWrapper","qb","onSubset","_subset","createEnhancers","enhancers","executeSubsetQuery","params","queryParams","debug","optimizeCountQuery","sonamuFilter","normalizedFilter","applySonamuFilters","num","page","total","executeCountQuery","queryMode","computedRows","executeListQuery","enhancer","enhancedRows","Promise","all","entity","get","internalFields","subsetsInternal","omitInternalFields","fields","result","deleteField","split","filters","puri","condition","Object","entries","undefined","fullField","getFullFieldName","Array","isArray","where","operator","value","applyOperator","whereNotIn","min","max","console","warn","obj","parts","forEach","item","first","rest","next","countPuri","clone","clear","default","SqlParser","parser","Parser","parsedQuery","astify","toQuery","database","config","leftJoinTables","whereTables","tablesToRemove","filter","j","includes","table","clearJoin","countResult","rawNumber","limitedQb","limit","offset","unloadedRows","loaders","processLoaders","hydrate","resolveLoader","as","refId","resolveLoaderQbFn","nestedLoaders","resolveLoaderQb","loadedRows","subRowGroups","r","nestedKeys","keys","key","groups","groupBy","nullKeys","groupKey","idField","every","hydrated","reduce","objPath","slice","part","nullKey","BaseModel"],"mappings":"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"}
|
|
393
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/base-model.ts"],"sourcesContent":["/** biome-ignore-all lint/suspicious/noExplicitAny: Puri의 타입은 개별 모델에서 확정되므로 BaseModel에서는 any를 허용함 */\nimport { getLogger, type Logger } from \"@logtape/logtape\";\nimport type { Knex } from \"knex\";\nimport { cloneDeep, group, isObject, omit, set } from \"radashi\";\nimport { type ListResult, normalizeFilterQuery, validateSonamuFilters } from \"..\";\nimport { Sonamu } from \"../api\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport type { FilterOperator, FilterQuery } from \"../filter/types\";\nimport { convertDomainToCategory } from \"../logger/category\";\nimport type { DatabaseSchemaExtend, SonamuQueryMode } from \"../types/types\";\nimport { getJoinTables, getTableNamesFromWhere } from \"../utils/sql-parser\";\nimport { chunk } from \"../utils/utils\";\nimport type { EnhancerMap, ResolveSubsetIntersection } from \"./base-model.types\";\nimport type { DBPreset } from \"./db\";\nimport { DB } from \"./db\";\nimport { Puri } from \"./puri\";\nimport type { UnionExtractedTTables } from \"./puri.types\";\nimport type { InferAllSubsets, PuriLoaderQueries, PuriSubsetFn } from \"./puri-subset.types\";\nimport { PuriWrapper } from \"./puri-wrapper\";\nimport { UpsertBuilder } from \"./upsert-builder\";\n\ntype UnknownDBRecord = Record<string, unknown>;\n\n/**\n * 모든 Model 클래스의 기본 클래스\n *\n * @template TSubsetKey - 서브셋 키 유니온 (예: \"A\" | \"P\" | \"SS\")\n * @template TSubsetMapping - 서브셋별 최종 결과 타입 매핑\n * @template TSubsetQueries - 서브셋 쿼리 함수 객체\n * @template TLoaderQueries - 서브셋별 로더 쿼리 배열 객체\n */\nexport class BaseModelClass<\n  TSubsetKey extends string = never,\n  TSubsetMapping extends Record<string, any> = never,\n  TSubsetQueries extends Record<TSubsetKey, PuriSubsetFn> = never,\n  TLoaderQueries extends PuriLoaderQueries<TSubsetKey> = never,\n> {\n  protected readonly logger: Logger;\n\n  constructor(\n    public readonly modelName: string = this.constructor.name,\n    protected subsetQueries?: TSubsetQueries,\n    protected loaderQueries?: TLoaderQueries,\n  ) {\n    this.logger = getLogger(convertDomainToCategory(this.modelName, \"model\"));\n  }\n\n  getDB(which: DBPreset): Knex {\n    return DB.getDB(which);\n  }\n\n  getPuri(which: DBPreset): PuriWrapper {\n    // 트랜잭션 컨텍스트에서 트랜잭션 획득\n    const trx = DB.getTransactionContext().getTransaction(which);\n    if (trx) {\n      return trx;\n    }\n\n    // 트랜잭션이 없으면 새로운 PuriWrapper 반환\n    const db = this.getDB(which);\n    return new PuriWrapper(db, new UpsertBuilder());\n  }\n\n  async destroy() {\n    return DB.destroy();\n  }\n\n  async getInsertedIds(\n    wdb: Knex,\n    rows: UnknownDBRecord[],\n    tableName: string,\n    unqKeyFields: string[],\n    chunkSize: number = 500,\n  ) {\n    if (!wdb) {\n      wdb = this.getDB(\"w\");\n    }\n\n    let unqKeys: string[];\n    let whereInField: string | Knex.Raw;\n    let selectField: string;\n\n    if (unqKeyFields.length > 1) {\n      whereInField = wdb.raw(`CONCAT_WS('_', '${unqKeyFields.join(\",\")}')`);\n      selectField = `${whereInField} as tmpUid`;\n      unqKeys = rows.map((row) => unqKeyFields.map((field) => row[field]).join(\"_\"));\n    } else {\n      whereInField = unqKeyFields[0];\n      selectField = unqKeyFields[0];\n      unqKeys = rows.map((row) => row[unqKeyFields[0]] as string);\n    }\n\n    let resultIds: number[] = [];\n    for (const items of chunk(unqKeys, chunkSize)) {\n      const dbRows = await wdb(tableName)\n        .select(\"id\", wdb.raw(selectField))\n        .whereIn(whereInField as string, items);\n      resultIds = resultIds.concat(\n        dbRows.map((dbRow: UnknownDBRecord) => parseInt(String(dbRow.id))),\n      );\n    }\n\n    return resultIds;\n  }\n\n  /**\n   * 특정 서브셋에 대한 쿼리 빌더 획득\n   *\n   * @returns qb - 쿼리 빌더 (조건 추가용)\n   * @returns onSubset - 특정 서브셋 전용 타입이 필요할 때 사용\n   */\n  getSubsetQueries<T extends TSubsetKey>(subset: T) {\n    if (!this.subsetQueries) {\n      throw new Error(\"subsetQueries is not defined\");\n    }\n\n    const puriWrapper = new PuriWrapper(this.getDB(\"r\"), new UpsertBuilder());\n    const qb = this.subsetQueries[subset]?.(puriWrapper);\n\n    // NonAllowedAsSingleTable: 단일 테이블 컬럼 접근 방지용 마커\n    type QBTables = UnionExtractedTTables<TSubsetKey, TSubsetQueries> & {\n      NonAllowedAsSingleTable: { __fulltext__: true };\n    };\n\n    return {\n      qb: qb as unknown as Puri<DatabaseSchemaExtend, QBTables, {}>,\n      onSubset: ((_subset: TSubsetKey | readonly TSubsetKey[]) => qb) as {\n        // 단일 키\n        <S extends TSubsetKey>(subset: S): ReturnType<TSubsetQueries[S]>;\n        // 키 배열 -> 교집합 반환\n        <Arr extends readonly TSubsetKey[]>(\n          subsets: [...Arr],\n        ): ResolveSubsetIntersection<Arr, TSubsetQueries>;\n      },\n    };\n  }\n\n  /**\n   * Enhancer 객체 생성 헬퍼\n   * 타입 검증 및 추론을 도와줌\n   */\n  createEnhancers<T extends TSubsetKey>(\n    enhancers: EnhancerMap<\n      T,\n      InferAllSubsets<TSubsetQueries, TLoaderQueries>,\n      TSubsetMapping,\n      TSubsetQueries\n    >,\n  ) {\n    return enhancers;\n  }\n\n  /**\n   * 서브셋 쿼리 실행\n   *\n   * 1. Sonamu 필터 적용 (타입 변환 포함)\n   * 2. 쿼리 실행 (pagination 적용)\n   * 3. 로더 실행 (1:N, N:M 관계 데이터 로딩)\n   * 4. Hydrate (flat → 중첩 객체)\n   * 5. Enhancer 적용 (virtual 필드 계산)\n   */\n  async executeSubsetQuery<\n    T extends TSubsetKey,\n    TComputedResults extends InferAllSubsets<TSubsetQueries, TLoaderQueries>,\n    LP extends {\n      num?: number;\n      page?: number;\n      queryMode?: SonamuQueryMode;\n      sonamuFilter?: Record<string, unknown>;\n    },\n  >(\n    params: {\n      subset: T;\n      qb: Puri<any, any, any>;\n      params: {\n        num: number;\n        page: number;\n        queryMode?: SonamuQueryMode;\n        sonamuFilter?: Record<string, unknown>;\n      };\n      debug?: boolean;\n      optimizeCountQuery?: boolean;\n    } & EnhancerParam<TSubsetKey, TComputedResults, TSubsetMapping, TSubsetQueries>,\n  ): Promise<ListResult<LP, TSubsetMapping[T]>> {\n    const { subset, qb, params: queryParams, debug = false, optimizeCountQuery = false } = params;\n\n    if (!this.loaderQueries) {\n      throw new Error(\"loaderQueries is not defined\");\n    }\n\n    // Sonamu Filter 적용\n    if (queryParams.sonamuFilter) {\n      const normalizedFilter = normalizeFilterQuery(queryParams.sonamuFilter);\n      this.applySonamuFilters(qb, normalizedFilter);\n    }\n\n    const { num, page } = queryParams;\n\n    // COUNT 쿼리 실행 (queryMode: list일 때는 0 리턴)\n    const total = await this.executeCountQuery(qb, queryParams, debug, optimizeCountQuery);\n\n    if (queryParams?.queryMode === \"count\") {\n      return { total } as ListResult<LP, TSubsetMapping[T]>;\n    }\n\n    // LIST 쿼리 실행\n    const computedRows = await this.executeListQuery(subset, qb, queryParams, num, page, debug);\n\n    // Enhancer 적용\n    const enhancer = (params as any).enhancers?.[subset];\n    const enhancedRows = (await Promise.all(\n      computedRows.map((row) => enhancer?.(row) ?? row),\n    )) as TSubsetMapping[T][];\n\n    // Internal 필드 제거\n    const entity = EntityManager.get(this.modelName);\n    const internalFields = entity.subsetsInternal[subset] ?? [];\n    const rows =\n      internalFields.length > 0\n        ? enhancedRows.map((row) => this.omitInternalFields(row, internalFields))\n        : enhancedRows;\n\n    if (queryParams.queryMode === \"list\") {\n      // 리스트만 리턴\n      return { rows } as ListResult<LP, TSubsetMapping[T]>;\n    } else {\n      // 둘다 리턴\n      return { rows, total } as ListResult<LP, TSubsetMapping[T]>;\n    }\n  }\n\n  /**\n   * 객체에서 internal 필드 제거\n   * 중첩 필드(예: \"user.email\") 및 배열(예: \"employees.salary\")도 처리\n   */\n  omitInternalFields<T extends object>(row: T, fields: string[]): T {\n    const result = cloneDeep(row);\n    for (const field of fields) {\n      this.deleteField(result, field.split(\".\"));\n    }\n    return result;\n  }\n\n  /**\n   * FilterQuery를 Puri QueryBuilder에 적용\n   *\n   * @param qb Puri QueryBuilder 인스턴스\n   * @param filters FilterQuery 객체\n   */\n  protected applySonamuFilters<TEntity = Record<string, unknown>>(\n    qb: Puri<any, any, any>,\n    filters?: FilterQuery<TEntity>,\n  ): void {\n    if (!filters) return;\n\n    const entity = EntityManager.get(this.modelName);\n\n    // 1. 필터 검증 (Entity 기반)\n    validateSonamuFilters(filters, entity);\n\n    // 2. 검증된 필터 적용\n    const puri = qb as any;\n\n    for (const [field, condition] of Object.entries(filters)) {\n      if (condition === undefined || condition === null) continue;\n\n      // 테이블명.필드명 형식으로 변환\n      const fullField = entity.getFullFieldName(field);\n\n      // 직접 값 (eq와 동일)\n      if (typeof condition !== \"object\" || Array.isArray(condition)) {\n        puri.where(fullField, condition);\n        continue;\n      }\n\n      // 연산자 객체\n      for (const [operator, value] of Object.entries(condition)) {\n        this.applyOperator(qb, fullField, operator as FilterOperator, value);\n      }\n    }\n  }\n\n  /**\n   * 단일 연산자를 QueryBuilder에 적용\n   */\n  private applyOperator(\n    qb: Puri<any, any, any>,\n    field: string,\n    operator: FilterOperator,\n    value: unknown,\n  ): void {\n    const puri = qb as any;\n\n    switch (operator) {\n      case \"eq\":\n        puri.where(field, value);\n        break;\n\n      case \"ne\":\n        puri.where(field, \"!=\", value);\n        break;\n\n      case \"gt\":\n        puri.where(field, \">\", value);\n        break;\n\n      case \"gte\":\n        puri.where(field, \">=\", value);\n        break;\n\n      case \"lt\":\n        puri.where(field, \"<\", value);\n        break;\n\n      case \"lte\":\n        puri.where(field, \"<=\", value);\n        break;\n\n      case \"in\":\n        puri.whereIn(field, value);\n        break;\n\n      case \"notIn\":\n        puri.whereNotIn(field, value);\n        break;\n\n      case \"contains\":\n        puri.where(field, \"like\", `%${value}%`);\n        break;\n\n      case \"startsWith\":\n        puri.where(field, \"like\", `${value}%`);\n        break;\n\n      case \"endsWith\":\n        puri.where(field, \"like\", `%${value}`);\n        break;\n\n      case \"isNull\":\n        puri.where(field, null);\n        break;\n\n      case \"isNotNull\":\n        puri.where(field, \"!=\", null);\n        break;\n\n      case \"before\":\n        puri.where(field, \"<\", value);\n        break;\n\n      case \"after\":\n        puri.where(field, \">\", value);\n        break;\n\n      case \"between\": {\n        if (Array.isArray(value) && value.length === 2) {\n          const [min, max] = value;\n          puri.where(field, \">=\", min).where(field, \"<=\", max);\n        }\n        break;\n      }\n\n      default:\n        console.warn(`Unsupported operator: ${operator}`);\n    }\n  }\n\n  /**\n   * 중첩 필드 삭제 (배열 내 객체도 처리)\n   */\n  deleteField(obj: any, parts: string[]): void {\n    if (!obj || typeof obj !== \"object\") {\n      return;\n    }\n\n    if (parts.length === 1) {\n      if (Array.isArray(obj)) {\n        obj.forEach((item) => {\n          if (item && typeof item === \"object\") {\n            delete item[parts[0]];\n          }\n        });\n      } else {\n        delete obj[parts[0]];\n      }\n      return;\n    }\n\n    const [first, ...rest] = parts;\n    const next = obj[first];\n\n    if (Array.isArray(next)) {\n      next.map((item) => this.deleteField(item, rest));\n    } else if (next && typeof next === \"object\") {\n      this.deleteField(next, rest);\n    }\n  }\n\n  /**\n   * COUNT 쿼리 실행 (내부 메서드)\n   */\n  private async executeCountQuery(\n    qb: Puri<any, any, any>,\n    params: { queryMode?: \"list\" | \"count\" | \"both\" },\n    debug: boolean,\n    optimizeCountQuery: boolean,\n  ): Promise<number> {\n    if (params.queryMode === \"list\") {\n      return 0;\n    }\n\n    const countPuri = qb.clone().clear(\"order\").clear(\"limit\").clear(\"offset\");\n\n    if (optimizeCountQuery) {\n      const { default: SqlParser } = await import(\"node-sql-parser\");\n      const parser = new SqlParser.Parser();\n      const parsedQuery = parser.astify(countPuri.toQuery(), {\n        database: Sonamu.config.database.database,\n      });\n\n      const leftJoinTables = getJoinTables(parsedQuery, [\"LEFT JOIN\"]);\n      const whereTables = getTableNamesFromWhere(parsedQuery);\n\n      const tablesToRemove = leftJoinTables.filter((j) => !whereTables.includes(j));\n      tablesToRemove.forEach((table) => {\n        countPuri.clearJoin(table);\n      });\n    }\n\n    // COUNT(*)로 전체 레코드 수를 계산\n    // TODO: qb의 DISTINCT가 있는 경우 처리해야 함\n    const countResult: { total?: number } = await countPuri\n      .clear(\"select\")\n      .select({ total: Puri.rawNumber(`COUNT(*)::integer`) })\n      .first();\n\n    if (debug) {\n      countPuri.debug();\n    }\n\n    return countResult?.total ?? 0;\n  }\n\n  /**\n   * LIST 쿼리 실행 (내부 메서드)\n   */\n  private async executeListQuery<T extends TSubsetKey>(\n    subset: T,\n    qb: Puri<any, any, any>,\n    params: { queryMode?: \"list\" | \"count\" | \"both\" },\n    num: number,\n    page: number,\n    debug: boolean,\n  ): Promise<any[]> {\n    if (params.queryMode === \"count\") {\n      return [];\n    }\n\n    const limitedQb = (() => {\n      if (num === 0) {\n        return qb;\n      } else {\n        return qb.limit(num).offset(num * (page - 1));\n      }\n    })();\n    let unloadedRows = (await limitedQb) as any[];\n\n    if (debug) {\n      qb.debug();\n    }\n\n    // 로더 처리\n    const loaders = (this.loaderQueries as any)[subset];\n    if (loaders && Array.isArray(loaders)) {\n      unloadedRows = await this.processLoaders(unloadedRows, loaders, debug);\n    }\n\n    return this.hydrate(unloadedRows);\n  }\n\n  /**\n   * 재귀적 로더 처리\n   */\n  private async processLoaders(rows: any[], loaders: any[], debug: boolean): Promise<any[]> {\n    for (const resolveLoader of loaders) {\n      const { as, refId, qb: resolveLoaderQbFn, loaders: nestedLoaders } = resolveLoader;\n\n      const resolveLoaderQb = resolveLoaderQbFn(\n        new PuriWrapper(this.getDB(\"r\"), new UpsertBuilder()),\n        rows.map((row) => row[refId]),\n      );\n\n      if (debug) {\n        resolveLoaderQb.debug();\n      }\n\n      let loadedRows = (await resolveLoaderQb) as any[];\n\n      // 중첩 loaders가 있으면 재귀 처리\n      if (nestedLoaders && nestedLoaders.length > 0) {\n        loadedRows = await this.processLoaders(loadedRows, nestedLoaders, debug);\n      }\n\n      const subRowGroups = group(loadedRows, (row) => row.refId);\n\n      rows = rows.map((row) => {\n        row[as] = (subRowGroups[row[refId]] ?? []).map((r) => omit(r, [\"refId\"]));\n        return row;\n      });\n    }\n\n    return rows;\n  }\n\n  /**\n   * Flat 레코드를 중첩 객체로 변환\n   *\n   * - `user__name` → `{ user: { name } }`\n   * - nullable relation의 경우 id 필드가 null이면 객체 자체를 null로\n   */\n  hydrate<T extends UnknownDBRecord>(rows: T[]): T[] {\n    return rows.map((row: T) => {\n      // nullable relation 처리: 그룹의 id 필드가 null이면 객체 전체를 null로\n      const nestedKeys = Object.keys(row).filter((key) => key.includes(\"__\"));\n      const groups = Object.groupBy(nestedKeys, (key) => key.split(\"__\")[0]);\n\n      // id 필드가 null인 그룹 찾기 (예: parent__id가 null이면 parent 그룹 전체가 null)\n      const nullKeys = Object.entries(groups)\n        .filter(([groupKey, fields]) => {\n          if (!fields || fields.length === 0) return false;\n\n          // 그룹의 id 필드 찾기 (예: \"parent__id\")\n          const idField = `${groupKey}__id`;\n          if (idField in row) {\n            // id 필드가 null이면 객체 전체가 null\n            return row[idField] === null;\n          }\n\n          // id 필드가 없으면 기존 로직: 모든 필드가 null인지 확인\n          return fields.every(\n            (field) =>\n              row[field] === null || (Array.isArray(row[field]) && row[field].length === 0),\n          );\n        })\n        .map(([key]) => key);\n\n      const hydrated = Object.keys(row).reduce((r, field) => {\n        if (!field.includes(\"__\")) {\n          // 일반 필드: 배열 내 객체면 재귀 hydrate\n          if (Array.isArray(row[field]) && isObject(row[field][0])) {\n            r[field] = this.hydrate(row[field]);\n          } else {\n            r[field] = row[field];\n          }\n          return r;\n        }\n\n        // 중첩 필드 처리: user__name → user[name]\n        const parts = field.split(\"__\");\n        const objPath =\n          parts[0] +\n          parts\n            .slice(1)\n            .map((part) => `[${part}]`)\n            .join(\"\");\n\n        r = set(\n          r,\n          objPath,\n          row[field] && Array.isArray(row[field]) && isObject(row[field][0])\n            ? this.hydrate(row[field])\n            : row[field],\n        );\n\n        return r;\n      }, {} as UnknownDBRecord);\n\n      // null relation 처리\n      nullKeys.forEach((nullKey) => {\n        hydrated[nullKey] = null;\n      });\n\n      return hydrated;\n    }) as T[];\n  }\n}\n\n/**\n * Enhancer 파라미터 조건부 타입\n * RequiredEnhancerKeys가 없으면 enhancers 선택적, 있으면 필수\n */\ntype EnhancerParam<\n  TSubsetKey extends string,\n  TComputedResults extends Record<TSubsetKey, any>,\n  TSubsetMapping extends Record<TSubsetKey, any>,\n  TSubsetQueries extends Record<TSubsetKey, PuriSubsetFn>,\n> = [RequiredEnhancerKeys<TSubsetKey, TComputedResults, TSubsetMapping>] extends [never]\n  ? { enhancers?: EnhancerMap<TSubsetKey, TComputedResults, TSubsetMapping, TSubsetQueries> }\n  : { enhancers: EnhancerMap<TSubsetKey, TComputedResults, TSubsetMapping, TSubsetQueries> };\n\ntype RequiredEnhancerKeys<\n  TSubsetKey extends string,\n  TComputedResults extends Record<TSubsetKey, any>,\n  TSubsetMapping extends Record<TSubsetKey, any>,\n> = {\n  [K in TSubsetKey]: TComputedResults[K] extends TSubsetMapping[K] ? never : K;\n}[TSubsetKey];\n\nexport const BaseModel = new BaseModelClass();\n"],"names":["getLogger","cloneDeep","group","isObject","omit","set","normalizeFilterQuery","validateSonamuFilters","Sonamu","EntityManager","convertDomainToCategory","getJoinTables","getTableNamesFromWhere","chunk","DB","Puri","PuriWrapper","UpsertBuilder","BaseModelClass","logger","modelName","name","subsetQueries","loaderQueries","getDB","which","getPuri","trx","getTransactionContext","getTransaction","db","destroy","getInsertedIds","wdb","rows","tableName","unqKeyFields","chunkSize","unqKeys","whereInField","selectField","length","raw","join","map","row","field","resultIds","items","dbRows","select","whereIn","concat","dbRow","parseInt","String","id","getSubsetQueries","subset","Error","puriWrapper","qb","onSubset","_subset","createEnhancers","enhancers","executeSubsetQuery","params","queryParams","debug","optimizeCountQuery","sonamuFilter","normalizedFilter","applySonamuFilters","num","page","total","executeCountQuery","queryMode","computedRows","executeListQuery","enhancer","enhancedRows","Promise","all","entity","get","internalFields","subsetsInternal","omitInternalFields","fields","result","deleteField","split","filters","puri","condition","Object","entries","undefined","fullField","getFullFieldName","Array","isArray","where","operator","value","applyOperator","whereNotIn","min","max","console","warn","obj","parts","forEach","item","first","rest","next","countPuri","clone","clear","default","SqlParser","parser","Parser","parsedQuery","astify","toQuery","database","config","leftJoinTables","whereTables","tablesToRemove","filter","j","includes","table","clearJoin","countResult","rawNumber","limitedQb","limit","offset","unloadedRows","loaders","processLoaders","hydrate","resolveLoader","as","refId","resolveLoaderQbFn","nestedLoaders","resolveLoaderQb","loadedRows","subRowGroups","r","nestedKeys","keys","key","groups","groupBy","nullKeys","groupKey","idField","every","hydrated","reduce","objPath","slice","part","nullKey","BaseModel"],"mappings":"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"}
|
package/dist/database/db.d.ts
CHANGED
|
@@ -26,7 +26,7 @@ export declare class DBClass {
|
|
|
26
26
|
private getWorkerDB;
|
|
27
27
|
getDBConfig(which: DBPreset): Knex.Config;
|
|
28
28
|
destroy(): Promise<void>;
|
|
29
|
-
generateDBConfig(config:
|
|
29
|
+
generateDBConfig(config: SonamuConfig["database"]): SonamuDBConfig;
|
|
30
30
|
testTransaction: Knex.Transaction | null;
|
|
31
31
|
createTestTransaction(): Promise<Knex.Transaction>;
|
|
32
32
|
clearTestTransaction(): Promise<void>;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"db.d.ts","sourceRoot":"","sources":["../../src/database/db.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAChD,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAGjC,OAAO,KAAK,EAAkB,YAAY,EAAE,MAAM,eAAe,CAAC;AAElE,OAAO,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAC;AAU3D,MAAM,MAAM,QAAQ,GAAG,GAAG,GAAG,GAAG,CAAC;AAEjC,MAAM,MAAM,cAAc,GAAG;IAC3B,kBAAkB,EAAE,IAAI,CAAC,MAAM,CAAC;IAChC,iBAAiB,EAAE,IAAI,CAAC,MAAM,CAAC;IAC/B,iBAAiB,EAAE,IAAI,CAAC,MAAM,CAAC;IAC/B,gBAAgB,EAAE,IAAI,CAAC,MAAM,CAAC;IAC9B,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC;IACrB,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC;CACnB,CAAC;AAEF,qBAAa,OAAO;IAClB,OAAO,CAAC,GAAG,CAAC,CAAO;IACnB,OAAO,CAAC,GAAG,CAAC,CAAO;IACnB,OAAO,CAAC,SAAS,CAAgC;IAE1C,kBAAkB,wCAA+C;IAEjE,kBAAkB,CAAC,CAAC,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;IAI7D,qBAAqB,IAAI,kBAAkB;IAIlD,KAAK,CAAC,KAAK,EAAE,QAAQ,GAAG,IAAI;IAsC5B;;;OAGG;IACH,OAAO,CAAC,WAAW;IA+BnB,WAAW,CAAC,KAAK,EAAE,QAAQ,GAAG,IAAI,CAAC,MAAM;IA2BnC,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;IAgBvB,gBAAgB,CAAC,MAAM,EAAE,
|
|
1
|
+
{"version":3,"file":"db.d.ts","sourceRoot":"","sources":["../../src/database/db.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAChD,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAGjC,OAAO,KAAK,EAAkB,YAAY,EAAE,MAAM,eAAe,CAAC;AAElE,OAAO,EAAE,kBAAkB,EAAE,MAAM,uBAAuB,CAAC;AAU3D,MAAM,MAAM,QAAQ,GAAG,GAAG,GAAG,GAAG,CAAC;AAEjC,MAAM,MAAM,cAAc,GAAG;IAC3B,kBAAkB,EAAE,IAAI,CAAC,MAAM,CAAC;IAChC,iBAAiB,EAAE,IAAI,CAAC,MAAM,CAAC;IAC/B,iBAAiB,EAAE,IAAI,CAAC,MAAM,CAAC;IAC/B,gBAAgB,EAAE,IAAI,CAAC,MAAM,CAAC;IAC9B,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC;IACrB,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC;CACnB,CAAC;AAEF,qBAAa,OAAO;IAClB,OAAO,CAAC,GAAG,CAAC,CAAO;IACnB,OAAO,CAAC,GAAG,CAAC,CAAO;IACnB,OAAO,CAAC,SAAS,CAAgC;IAE1C,kBAAkB,wCAA+C;IAEjE,kBAAkB,CAAC,CAAC,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC;IAI7D,qBAAqB,IAAI,kBAAkB;IAIlD,KAAK,CAAC,KAAK,EAAE,QAAQ,GAAG,IAAI;IAsC5B;;;OAGG;IACH,OAAO,CAAC,WAAW;IA+BnB,WAAW,CAAC,KAAK,EAAE,QAAQ,GAAG,IAAI,CAAC,MAAM;IA2BnC,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;IAgBvB,gBAAgB,CAAC,MAAM,EAAE,YAAY,CAAC,UAAU,CAAC,GAAG,cAAc;IAsDlE,eAAe,EAAE,IAAI,CAAC,WAAW,GAAG,IAAI,CAAQ;IACjD,qBAAqB,IAAI,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC;IAKlD,oBAAoB,IAAI,OAAO,CAAC,IAAI,CAAC;IAIrC,iBAAiB,IAAI,OAAO,CAAC,IAAI,CAAC;CAIzC;AACD,eAAO,MAAM,EAAE,SAAgB,CAAC"}
|
package/dist/database/db.js
CHANGED
|
@@ -173,4 +173,4 @@ export class DBClass {
|
|
|
173
173
|
}
|
|
174
174
|
export const DB = new DBClass();
|
|
175
175
|
|
|
176
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/db.ts"],"sourcesContent":["import assert from \"assert\";\nimport { AsyncLocalStorage } from \"async_hooks\";\nimport type { Knex } from \"knex\";\nimport { assign } from \"radashi\";\nimport { Sonamu } from \"../api\";\nimport type { DatabaseConfig, SonamuConfig } from \"../api/config\";\nimport { createKnexInstance } from \"./knex\";\nimport { TransactionContext } from \"./transaction-context\";\n\n/**\n * 여러 설정 객체를 순차적으로 deep merge합니다.\n * undefined/null인 인자는 무시됩니다.\n */\nfunction mergeConfigs<T extends object>(...configs: (Partial<T> | undefined | null)[]): T {\n  return configs.reduce<T>((acc, config) => (config ? assign(acc, config as T) : acc), {} as T);\n}\n\nexport type DBPreset = \"w\" | \"r\";\n\nexport type SonamuDBConfig = {\n  development_master: Knex.Config;\n  development_slave: Knex.Config;\n  production_master: Knex.Config;\n  production_slave: Knex.Config;\n  fixture: Knex.Config;\n  test: Knex.Config;\n};\n\nexport class DBClass {\n  private wdb?: Knex;\n  private rdb?: Knex;\n  private workerDBs: Map<number, Knex> = new Map();\n\n  public transactionStorage = new AsyncLocalStorage<TransactionContext>();\n\n  public runWithTransaction<T>(callback: () => Promise<T>): Promise<T> {\n    return this.transactionStorage.run(new TransactionContext(), callback);\n  }\n\n  public getTransactionContext(): TransactionContext {\n    return this.transactionStorage.getStore() ?? new TransactionContext();\n  }\n\n  getDB(which: DBPreset): Knex {\n    const dbConfig = Sonamu.dbConfig;\n\n    // 테스트 트랜잭션 격리\n    if (process.env.NODE_ENV === \"test\") {\n      // 병렬 테스트 모드: worker별 DB 사용\n      if (process.env.SONAMU_WORKER_DB === \"true\") {\n        return this.getWorkerDB(dbConfig);\n      }\n\n      // 기존 단일 테스트 로직\n      if (this.testTransaction) {\n        return this.testTransaction;\n      } else if (this.wdb) {\n        return this.wdb;\n      } else {\n        this.wdb = createKnexInstance({\n          ...dbConfig.test,\n          // 단일 풀\n          pool: {\n            min: 1,\n            max: 1,\n          },\n        });\n        return this.wdb;\n      }\n    }\n\n    const instanceName = which === \"w\" ? \"wdb\" : \"rdb\";\n\n    if (!this[instanceName]) {\n      const config = this.getDBConfig(which);\n      this[instanceName] = createKnexInstance(config);\n    }\n\n    return this[instanceName];\n  }\n\n  /**\n   * 병렬 테스트에서 worker별 DB 인스턴스를 반환합니다.\n   * VITEST_POOL_ID 환경변수로 worker를 식별하여 해당 DB에 연결합니다.\n   */\n  private getWorkerDB(dbConfig: SonamuDBConfig): Knex {\n    // 트랜잭션이 있으면 트랜잭션 반환\n    if (this.testTransaction) {\n      return this.testTransaction;\n    }\n\n    const workerId = parseInt(process.env.VITEST_POOL_ID ?? \"1\", 10);\n\n    // Worker별 DB 인스턴스 캐싱\n    if (!this.workerDBs.has(workerId)) {\n      const baseTestConfig = dbConfig.test;\n      const connection = baseTestConfig.connection as { database: string };\n      const workerDbName = `${connection.database}_${workerId}`;\n\n      const workerConfig = {\n        ...baseTestConfig,\n        connection: {\n          ...connection,\n          database: workerDbName,\n        },\n        pool: { min: 1, max: 1 },\n      };\n\n      this.workerDBs.set(workerId, createKnexInstance(workerConfig));\n    }\n\n    const db = this.workerDBs.get(workerId);\n    assert(db, `Worker DB ${workerId} not found`);\n    return db;\n  }\n\n  getDBConfig(which: DBPreset): Knex.Config {\n    const dbConfig = Sonamu.dbConfig;\n    if (process.env.NODE_ENV === \"test\") {\n      return {\n        ...dbConfig.test,\n        // 단일 풀\n        pool: {\n          min: 1,\n          max: 1,\n        },\n      };\n    }\n    switch (process.env.NODE_ENV ?? \"development\") {\n      case \"development\":\n      case \"staging\":\n        return which === \"w\"\n          ? dbConfig.development_master\n          : (dbConfig.development_slave ?? dbConfig.development_master);\n      case \"production\":\n        return which === \"w\"\n          ? dbConfig.production_master\n          : (dbConfig.production_slave ?? dbConfig.production_master);\n      default:\n        throw new Error(`현재 ENV ${process.env.NODE_ENV}에는 설정 가능한 DB설정이 없습니다.`);\n    }\n  }\n\n  async destroy(): Promise<void> {\n    if (this.wdb !== undefined) {\n      await this.wdb.destroy();\n      this.wdb = undefined;\n    }\n    if (this.rdb !== undefined) {\n      await this.rdb.destroy();\n      this.rdb = undefined;\n    }\n    // 병렬 테스트용 worker DB들도 정리\n    for (const db of this.workerDBs.values()) {\n      await db.destroy();\n    }\n    this.workerDBs.clear();\n  }\n\n  public generateDBConfig(config: NonNullable<SonamuConfig[\"database\"]>): SonamuDBConfig {\n    const defaultKnexConfig: Partial<DatabaseConfig> = assign(\n      {\n        client: \"postgresql\",\n        pool: {\n          min: 1,\n          max: 5,\n        },\n        migrations: {\n          directory: \"./src/migrations\",\n        },\n        connection: {\n          database: config.name,\n          ...config.defaultOptions?.connection,\n        },\n      },\n      config.defaultOptions,\n    );\n\n    // biome-ignore format: 설정 구조 가독성을 위해 여러 줄로 유지\n    return {\n      // 여기에 나열한 순서대로 Sonamu UI의 DB Migration 탭에 표시됩니다.\n      test: mergeConfigs(\n        defaultKnexConfig, \n        { connection: { database: `${config.name}_test` } },\n        config.environments?.test\n      ),\n      fixture: mergeConfigs(\n        defaultKnexConfig, \n        { connection: { database: `${config.name}_fixture` } },\n        config.environments?.fixture,\n      ),\n      development_master: mergeConfigs(\n        defaultKnexConfig, \n        config.environments?.development\n      ),\n      development_slave: mergeConfigs(\n        defaultKnexConfig,\n        config.environments?.development,\n        config.environments?.development_slave,\n      ),\n      production_master: mergeConfigs(\n        defaultKnexConfig, \n        config.environments?.production\n      ),\n      production_slave: mergeConfigs(\n        defaultKnexConfig,\n        config.environments?.production,\n        config.environments?.production_slave,\n      ),\n    };\n  }\n\n  // Test 환경에서 트랜잭션 사용\n  public testTransaction: Knex.Transaction | null = null;\n  async createTestTransaction(): Promise<Knex.Transaction> {\n    const db = this.getDB(\"w\");\n    this.testTransaction = await db.transaction();\n    return this.testTransaction;\n  }\n  async clearTestTransaction(): Promise<void> {\n    await this.testTransaction?.rollback();\n    this.testTransaction = null;\n  }\n  async getTestConnection(): Promise<Knex> {\n    const db = this.getDB(\"w\");\n    return db;\n  }\n}\nexport const DB = new DBClass();\n"],"names":["assert","AsyncLocalStorage","assign","Sonamu","createKnexInstance","TransactionContext","mergeConfigs","configs","reduce","acc","config","DBClass","wdb","rdb","workerDBs","Map","transactionStorage","runWithTransaction","callback","run","getTransactionContext","getStore","getDB","which","dbConfig","process","env","NODE_ENV","SONAMU_WORKER_DB","getWorkerDB","testTransaction","test","pool","min","max","instanceName","getDBConfig","workerId","parseInt","VITEST_POOL_ID","has","baseTestConfig","connection","workerDbName","database","workerConfig","set","db","get","development_master","development_slave","production_master","production_slave","Error","destroy","undefined","values","clear","generateDBConfig","defaultKnexConfig","client","migrations","directory","name","defaultOptions","environments","fixture","development","production","createTestTransaction","transaction","clearTestTransaction","rollback","getTestConnection","DB"],"mappings":"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"}
|
|
176
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/db.ts"],"sourcesContent":["import assert from \"assert\";\nimport { AsyncLocalStorage } from \"async_hooks\";\nimport type { Knex } from \"knex\";\nimport { assign } from \"radashi\";\nimport { Sonamu } from \"../api\";\nimport type { DatabaseConfig, SonamuConfig } from \"../api/config\";\nimport { createKnexInstance } from \"./knex\";\nimport { TransactionContext } from \"./transaction-context\";\n\n/**\n * 여러 설정 객체를 순차적으로 deep merge합니다.\n * undefined/null인 인자는 무시됩니다.\n */\nfunction mergeConfigs<T extends object>(...configs: (Partial<T> | undefined | null)[]): T {\n  return configs.reduce<T>((acc, config) => (config ? assign(acc, config as T) : acc), {} as T);\n}\n\nexport type DBPreset = \"w\" | \"r\";\n\nexport type SonamuDBConfig = {\n  development_master: Knex.Config;\n  development_slave: Knex.Config;\n  production_master: Knex.Config;\n  production_slave: Knex.Config;\n  fixture: Knex.Config;\n  test: Knex.Config;\n};\n\nexport class DBClass {\n  private wdb?: Knex;\n  private rdb?: Knex;\n  private workerDBs: Map<number, Knex> = new Map();\n\n  public transactionStorage = new AsyncLocalStorage<TransactionContext>();\n\n  public runWithTransaction<T>(callback: () => Promise<T>): Promise<T> {\n    return this.transactionStorage.run(new TransactionContext(), callback);\n  }\n\n  public getTransactionContext(): TransactionContext {\n    return this.transactionStorage.getStore() ?? new TransactionContext();\n  }\n\n  getDB(which: DBPreset): Knex {\n    const dbConfig = Sonamu.dbConfig;\n\n    // 테스트 트랜잭션 격리\n    if (process.env.NODE_ENV === \"test\") {\n      // 병렬 테스트 모드: worker별 DB 사용\n      if (process.env.SONAMU_WORKER_DB === \"true\") {\n        return this.getWorkerDB(dbConfig);\n      }\n\n      // 기존 단일 테스트 로직\n      if (this.testTransaction) {\n        return this.testTransaction;\n      } else if (this.wdb) {\n        return this.wdb;\n      } else {\n        this.wdb = createKnexInstance({\n          ...dbConfig.test,\n          // 단일 풀\n          pool: {\n            min: 1,\n            max: 1,\n          },\n        });\n        return this.wdb;\n      }\n    }\n\n    const instanceName = which === \"w\" ? \"wdb\" : \"rdb\";\n\n    if (!this[instanceName]) {\n      const config = this.getDBConfig(which);\n      this[instanceName] = createKnexInstance(config);\n    }\n\n    return this[instanceName];\n  }\n\n  /**\n   * 병렬 테스트에서 worker별 DB 인스턴스를 반환합니다.\n   * VITEST_POOL_ID 환경변수로 worker를 식별하여 해당 DB에 연결합니다.\n   */\n  private getWorkerDB(dbConfig: SonamuDBConfig): Knex {\n    // 트랜잭션이 있으면 트랜잭션 반환\n    if (this.testTransaction) {\n      return this.testTransaction;\n    }\n\n    const workerId = parseInt(process.env.VITEST_POOL_ID ?? \"1\", 10);\n\n    // Worker별 DB 인스턴스 캐싱\n    if (!this.workerDBs.has(workerId)) {\n      const baseTestConfig = dbConfig.test;\n      const connection = baseTestConfig.connection as { database: string };\n      const workerDbName = `${connection.database}_${workerId}`;\n\n      const workerConfig = {\n        ...baseTestConfig,\n        connection: {\n          ...connection,\n          database: workerDbName,\n        },\n        pool: { min: 1, max: 1 },\n      };\n\n      this.workerDBs.set(workerId, createKnexInstance(workerConfig));\n    }\n\n    const db = this.workerDBs.get(workerId);\n    assert(db, `Worker DB ${workerId} not found`);\n    return db;\n  }\n\n  getDBConfig(which: DBPreset): Knex.Config {\n    const dbConfig = Sonamu.dbConfig;\n    if (process.env.NODE_ENV === \"test\") {\n      return {\n        ...dbConfig.test,\n        // 단일 풀\n        pool: {\n          min: 1,\n          max: 1,\n        },\n      };\n    }\n    switch (process.env.NODE_ENV ?? \"development\") {\n      case \"development\":\n      case \"staging\":\n        return which === \"w\"\n          ? dbConfig.development_master\n          : (dbConfig.development_slave ?? dbConfig.development_master);\n      case \"production\":\n        return which === \"w\"\n          ? dbConfig.production_master\n          : (dbConfig.production_slave ?? dbConfig.production_master);\n      default:\n        throw new Error(`현재 ENV ${process.env.NODE_ENV}에는 설정 가능한 DB설정이 없습니다.`);\n    }\n  }\n\n  async destroy(): Promise<void> {\n    if (this.wdb !== undefined) {\n      await this.wdb.destroy();\n      this.wdb = undefined;\n    }\n    if (this.rdb !== undefined) {\n      await this.rdb.destroy();\n      this.rdb = undefined;\n    }\n    // 병렬 테스트용 worker DB들도 정리\n    for (const db of this.workerDBs.values()) {\n      await db.destroy();\n    }\n    this.workerDBs.clear();\n  }\n\n  public generateDBConfig(config: SonamuConfig[\"database\"]): SonamuDBConfig {\n    const defaultKnexConfig: Partial<DatabaseConfig> = assign(\n      {\n        client: \"postgresql\",\n        pool: {\n          min: 1,\n          max: 5,\n        },\n        migrations: {\n          directory: \"./src/migrations\",\n        },\n        connection: {\n          database: config.name,\n          ...config.defaultOptions?.connection,\n        },\n      },\n      config.defaultOptions,\n    );\n\n    // biome-ignore format: 설정 구조 가독성을 위해 여러 줄로 유지\n    return {\n      // 여기에 나열한 순서대로 Sonamu UI의 DB Migration 탭에 표시됩니다.\n      test: mergeConfigs(\n        defaultKnexConfig, \n        { connection: { database: `${config.name}_test` } },\n        config.environments?.test\n      ),\n      fixture: mergeConfigs(\n        defaultKnexConfig, \n        { connection: { database: `${config.name}_fixture` } },\n        config.environments?.fixture,\n      ),\n      development_master: mergeConfigs(\n        defaultKnexConfig, \n        config.environments?.development\n      ),\n      development_slave: mergeConfigs(\n        defaultKnexConfig,\n        config.environments?.development,\n        config.environments?.development_slave,\n      ),\n      production_master: mergeConfigs(\n        defaultKnexConfig, \n        config.environments?.production\n      ),\n      production_slave: mergeConfigs(\n        defaultKnexConfig,\n        config.environments?.production,\n        config.environments?.production_slave,\n      ),\n    };\n  }\n\n  // Test 환경에서 트랜잭션 사용\n  public testTransaction: Knex.Transaction | null = null;\n  async createTestTransaction(): Promise<Knex.Transaction> {\n    const db = this.getDB(\"w\");\n    this.testTransaction = await db.transaction();\n    return this.testTransaction;\n  }\n  async clearTestTransaction(): Promise<void> {\n    await this.testTransaction?.rollback();\n    this.testTransaction = null;\n  }\n  async getTestConnection(): Promise<Knex> {\n    const db = this.getDB(\"w\");\n    return db;\n  }\n}\nexport const DB = new DBClass();\n"],"names":["assert","AsyncLocalStorage","assign","Sonamu","createKnexInstance","TransactionContext","mergeConfigs","configs","reduce","acc","config","DBClass","wdb","rdb","workerDBs","Map","transactionStorage","runWithTransaction","callback","run","getTransactionContext","getStore","getDB","which","dbConfig","process","env","NODE_ENV","SONAMU_WORKER_DB","getWorkerDB","testTransaction","test","pool","min","max","instanceName","getDBConfig","workerId","parseInt","VITEST_POOL_ID","has","baseTestConfig","connection","workerDbName","database","workerConfig","set","db","get","development_master","development_slave","production_master","production_slave","Error","destroy","undefined","values","clear","generateDBConfig","defaultKnexConfig","client","migrations","directory","name","defaultOptions","environments","fixture","development","production","createTestTransaction","transaction","clearTestTransaction","rollback","getTestConnection","DB"],"mappings":"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"}
|