sonamu 0.8.25 → 0.8.26

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"file":"knex-adapter.d.ts","sourceRoot":"","sources":["../../src/auth/knex-adapter.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAMrD,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAGjC,UAAU,YAAY;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,MAAM,EAAE,GAAG,MAAM,EAAE,GAAG,IAAI,GAAG,IAAI,CAAC;IACrE,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED;;;;;GAKG;AACH,eAAO,MAAM,iBAAiB,SAwGpB,SAAS,iBAAiB,uDAInC,CAAC;AAEF;;;;GAIG;AACH,wBAAgB,UAAU,CACxB,KAAK,EAAE,IAAI,CAAC,YAAY,EACxB,UAAU,EAAE,YAAY,EAAE,GACzB,IAAI,CAAC,YAAY,CAmBnB"}
1
+ {"version":3,"file":"knex-adapter.d.ts","sourceRoot":"","sources":["../../src/auth/knex-adapter.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAMrD,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAGjC,UAAU,YAAY;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,MAAM,EAAE,GAAG,MAAM,EAAE,GAAG,IAAI,GAAG,IAAI,CAAC;IACrE,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED;;;;;GAKG;AACH,eAAO,MAAM,iBAAiB,SAyGpB,SAAS,iBAAiB,uDAInC,CAAC;AAEF;;;;GAIG;AACH,wBAAgB,UAAU,CACxB,KAAK,EAAE,IAAI,CAAC,YAAY,EACxB,UAAU,EAAE,YAAY,EAAE,GACzB,IAAI,CAAC,YAAY,CAmBnB"}
@@ -8,7 +8,7 @@ import { DB } from "../database/db.js";
8
8
  */ export const sonamuKnexAdapter = ()=>{
9
9
  let lazyOptions = null;
10
10
  const createCustomAdapter = (getDb)=>{
11
- return ()=>({
11
+ return ({ getFieldName })=>({
12
12
  create: async ({ model, data })=>{
13
13
  const [row] = await getDb()(model).insert(data).returning("*");
14
14
  return row;
@@ -25,7 +25,11 @@ import { DB } from "../database/db.js";
25
25
  query = applyWhere(query, where);
26
26
  }
27
27
  if (sortBy) {
28
- query = query.orderBy(sortBy.field, sortBy.direction);
28
+ const dbField = getFieldName({
29
+ model,
30
+ field: sortBy.field
31
+ });
32
+ query = query.orderBy(dbField, sortBy.direction);
29
33
  }
30
34
  if (limit) {
31
35
  query = query.limit(limit);
@@ -156,4 +160,4 @@ function applyCondition(query, condition, method) {
156
160
  }
157
161
  }
158
162
 
159
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/auth/knex-adapter.ts"],"sourcesContent":["import type { BetterAuthOptions } from \"better-auth\";\nimport type {\n  AdapterFactoryCustomizeAdapterCreator,\n  DBTransactionAdapter,\n} from \"better-auth/adapters\";\nimport { createAdapterFactory } from \"better-auth/adapters\";\nimport type { Knex } from \"knex\";\nimport { DB } from \"../database/db\";\n\ninterface CleanedWhere {\n  field: string;\n  value: string | number | boolean | string[] | number[] | Date | null;\n  operator: string;\n  connector: string;\n}\n\n/**\n * better-auth용 Sonamu knex 어댑터\n *\n * better-auth의 모든 쿼리를 DB.getDB()를 통해 실행하여\n * Sonamu 테스트 트랜잭션과 동일한 커넥션을 공유합니다.\n */\nexport const sonamuKnexAdapter = () => {\n  let lazyOptions: BetterAuthOptions | null = null;\n\n  const createCustomAdapter = (\n    getDb: () => Knex | Knex.Transaction,\n  ): AdapterFactoryCustomizeAdapterCreator => {\n    return () => ({\n      create: async ({ model, data }) => {\n        const [row] = await getDb()(model).insert(data).returning(\"*\");\n        return row;\n      },\n\n      findOne: async ({ model, where }) => {\n        let query = getDb()(model);\n        query = applyWhere(query, where);\n        const row = await query.first();\n        return row ?? null;\n      },\n\n      findMany: async ({ model, where, limit, offset, sortBy }) => {\n        let query = getDb()(model);\n        if (where) {\n          query = applyWhere(query, where);\n        }\n        if (sortBy) {\n          query = query.orderBy(sortBy.field, sortBy.direction);\n        }\n        if (limit) {\n          query = query.limit(limit);\n        }\n        if (offset) {\n          query = query.offset(offset);\n        }\n        return await query;\n      },\n\n      update: async ({ model, where, update }) => {\n        let query = getDb()(model);\n        query = applyWhere(query, where);\n        const [row] = await query.update(update).returning(\"*\");\n        return row ?? null;\n      },\n\n      updateMany: async ({ model, where, update }) => {\n        let query = getDb()(model);\n        query = applyWhere(query, where);\n        const count = await query.update(update);\n        return count;\n      },\n\n      delete: async ({ model, where }) => {\n        let query = getDb()(model);\n        query = applyWhere(query, where);\n        await query.del();\n      },\n\n      deleteMany: async ({ model, where }) => {\n        let query = getDb()(model);\n        query = applyWhere(query, where);\n        const count = await query.del();\n        return count;\n      },\n\n      count: async ({ model, where }) => {\n        let query = getDb()(model);\n        if (where) {\n          query = applyWhere(query, where);\n        }\n        const [{ count }] = await query.count(\"* as count\");\n        return Number(count);\n      },\n    });\n  };\n\n  const adapterConfig = {\n    adapterId: \"sonamu-knex\",\n    adapterName: \"Sonamu Knex Adapter\",\n    usePlural: false,\n    supportsJSON: true,\n    supportsDates: true,\n    supportsBooleans: true,\n    supportsNumericIds: false,\n    transaction: async <R>(cb: (trx: DBTransactionAdapter) => Promise<R>): Promise<R> => {\n      const db = DB.getDB(\"w\");\n      return db.transaction(async (trx) => {\n        const options = lazyOptions;\n        if (!options) {\n          throw new Error(\"sonamuKnexAdapter: options not initialized\");\n        }\n        return cb(\n          createAdapterFactory({\n            config: adapterConfig,\n            adapter: createCustomAdapter(() => trx),\n          })(options),\n        );\n      });\n    },\n  };\n\n  const adapterCreator = createAdapterFactory({\n    config: adapterConfig,\n    adapter: createCustomAdapter(() => DB.getDB(\"w\")),\n  });\n\n  return (options: BetterAuthOptions) => {\n    lazyOptions = options;\n    return adapterCreator(options);\n  };\n};\n\n/**\n * Better Auth의 공식 어댑터(Kysely, Drizzle, Prisma, MongoDB) 패턴에 맞춰\n * AND 그룹과 OR 그룹을 분리한 뒤 top-level AND로 결합합니다.\n * 결과: (A AND B AND ...) AND (C OR D OR ...)\n */\nexport function applyWhere(\n  query: Knex.QueryBuilder,\n  conditions: CleanedWhere[],\n): Knex.QueryBuilder {\n  const andGroup = conditions.filter((c) => c.connector !== \"OR\");\n  const orGroup = conditions.filter((c) => c.connector === \"OR\");\n\n  if (andGroup.length > 0) {\n    for (const condition of andGroup) {\n      query = applyCondition(query, condition, \"where\");\n    }\n  }\n\n  if (orGroup.length > 0) {\n    query = query.where(function (this: Knex.QueryBuilder) {\n      for (let i = 0; i < orGroup.length; i++) {\n        applyCondition(this, orGroup[i], i === 0 ? \"where\" : \"orWhere\");\n      }\n    });\n  }\n\n  return query;\n}\n\nfunction applyCondition(\n  query: Knex.QueryBuilder,\n  condition: CleanedWhere,\n  method: \"where\" | \"orWhere\",\n): Knex.QueryBuilder {\n  const { field, value, operator } = condition;\n\n  switch (operator) {\n    case \"eq\":\n      if (value === null) {\n        return query[method === \"orWhere\" ? \"orWhereNull\" : \"whereNull\"](field);\n      }\n      return query[method](field, \"=\", value);\n    case \"ne\":\n      if (value === null) {\n        return query[method === \"orWhere\" ? \"orWhereNotNull\" : \"whereNotNull\"](field);\n      }\n      return query[method](field, \"!=\", value);\n    case \"lt\":\n      return query[method](field, \"<\", value);\n    case \"lte\":\n      return query[method](field, \"<=\", value);\n    case \"gt\":\n      return query[method](field, \">\", value);\n    case \"gte\":\n      return query[method](field, \">=\", value);\n    case \"in\":\n      return query[method === \"orWhere\" ? \"orWhereIn\" : \"whereIn\"](\n        field,\n        value as (string | number)[],\n      );\n    case \"not_in\":\n      return query[method === \"orWhere\" ? \"orWhereNotIn\" : \"whereNotIn\"](\n        field,\n        value as (string | number)[],\n      );\n    case \"contains\":\n      return query[method](field, \"like\", `%${value}%`);\n    case \"starts_with\":\n      return query[method](field, \"like\", `${value}%`);\n    case \"ends_with\":\n      return query[method](field, \"like\", `%${value}`);\n    default:\n      return query;\n  }\n}\n"],"names":["createAdapterFactory","DB","sonamuKnexAdapter","lazyOptions","createCustomAdapter","getDb","create","model","data","row","insert","returning","findOne","where","query","applyWhere","first","findMany","limit","offset","sortBy","orderBy","field","direction","update","updateMany","count","delete","del","deleteMany","Number","adapterConfig","adapterId","adapterName","usePlural","supportsJSON","supportsDates","supportsBooleans","supportsNumericIds","transaction","cb","db","getDB","trx","options","Error","config","adapter","adapterCreator","conditions","andGroup","filter","c","connector","orGroup","length","condition","applyCondition","i","method","value","operator"],"mappings":"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"}
163
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/auth/knex-adapter.ts"],"sourcesContent":["import type { BetterAuthOptions } from \"better-auth\";\nimport type {\n  AdapterFactoryCustomizeAdapterCreator,\n  DBTransactionAdapter,\n} from \"better-auth/adapters\";\nimport { createAdapterFactory } from \"better-auth/adapters\";\nimport type { Knex } from \"knex\";\nimport { DB } from \"../database/db\";\n\ninterface CleanedWhere {\n  field: string;\n  value: string | number | boolean | string[] | number[] | Date | null;\n  operator: string;\n  connector: string;\n}\n\n/**\n * better-auth용 Sonamu knex 어댑터\n *\n * better-auth의 모든 쿼리를 DB.getDB()를 통해 실행하여\n * Sonamu 테스트 트랜잭션과 동일한 커넥션을 공유합니다.\n */\nexport const sonamuKnexAdapter = () => {\n  let lazyOptions: BetterAuthOptions | null = null;\n\n  const createCustomAdapter = (\n    getDb: () => Knex | Knex.Transaction,\n  ): AdapterFactoryCustomizeAdapterCreator => {\n    return ({ getFieldName }) => ({\n      create: async ({ model, data }) => {\n        const [row] = await getDb()(model).insert(data).returning(\"*\");\n        return row;\n      },\n\n      findOne: async ({ model, where }) => {\n        let query = getDb()(model);\n        query = applyWhere(query, where);\n        const row = await query.first();\n        return row ?? null;\n      },\n\n      findMany: async ({ model, where, limit, offset, sortBy }) => {\n        let query = getDb()(model);\n        if (where) {\n          query = applyWhere(query, where);\n        }\n        if (sortBy) {\n          const dbField = getFieldName({ model, field: sortBy.field });\n          query = query.orderBy(dbField, sortBy.direction);\n        }\n        if (limit) {\n          query = query.limit(limit);\n        }\n        if (offset) {\n          query = query.offset(offset);\n        }\n        return await query;\n      },\n\n      update: async ({ model, where, update }) => {\n        let query = getDb()(model);\n        query = applyWhere(query, where);\n        const [row] = await query.update(update).returning(\"*\");\n        return row ?? null;\n      },\n\n      updateMany: async ({ model, where, update }) => {\n        let query = getDb()(model);\n        query = applyWhere(query, where);\n        const count = await query.update(update);\n        return count;\n      },\n\n      delete: async ({ model, where }) => {\n        let query = getDb()(model);\n        query = applyWhere(query, where);\n        await query.del();\n      },\n\n      deleteMany: async ({ model, where }) => {\n        let query = getDb()(model);\n        query = applyWhere(query, where);\n        const count = await query.del();\n        return count;\n      },\n\n      count: async ({ model, where }) => {\n        let query = getDb()(model);\n        if (where) {\n          query = applyWhere(query, where);\n        }\n        const [{ count }] = await query.count(\"* as count\");\n        return Number(count);\n      },\n    });\n  };\n\n  const adapterConfig = {\n    adapterId: \"sonamu-knex\",\n    adapterName: \"Sonamu Knex Adapter\",\n    usePlural: false,\n    supportsJSON: true,\n    supportsDates: true,\n    supportsBooleans: true,\n    supportsNumericIds: false,\n    transaction: async <R>(cb: (trx: DBTransactionAdapter) => Promise<R>): Promise<R> => {\n      const db = DB.getDB(\"w\");\n      return db.transaction(async (trx) => {\n        const options = lazyOptions;\n        if (!options) {\n          throw new Error(\"sonamuKnexAdapter: options not initialized\");\n        }\n        return cb(\n          createAdapterFactory({\n            config: adapterConfig,\n            adapter: createCustomAdapter(() => trx),\n          })(options),\n        );\n      });\n    },\n  };\n\n  const adapterCreator = createAdapterFactory({\n    config: adapterConfig,\n    adapter: createCustomAdapter(() => DB.getDB(\"w\")),\n  });\n\n  return (options: BetterAuthOptions) => {\n    lazyOptions = options;\n    return adapterCreator(options);\n  };\n};\n\n/**\n * Better Auth의 공식 어댑터(Kysely, Drizzle, Prisma, MongoDB) 패턴에 맞춰\n * AND 그룹과 OR 그룹을 분리한 뒤 top-level AND로 결합합니다.\n * 결과: (A AND B AND ...) AND (C OR D OR ...)\n */\nexport function applyWhere(\n  query: Knex.QueryBuilder,\n  conditions: CleanedWhere[],\n): Knex.QueryBuilder {\n  const andGroup = conditions.filter((c) => c.connector !== \"OR\");\n  const orGroup = conditions.filter((c) => c.connector === \"OR\");\n\n  if (andGroup.length > 0) {\n    for (const condition of andGroup) {\n      query = applyCondition(query, condition, \"where\");\n    }\n  }\n\n  if (orGroup.length > 0) {\n    query = query.where(function (this: Knex.QueryBuilder) {\n      for (let i = 0; i < orGroup.length; i++) {\n        applyCondition(this, orGroup[i], i === 0 ? \"where\" : \"orWhere\");\n      }\n    });\n  }\n\n  return query;\n}\n\nfunction applyCondition(\n  query: Knex.QueryBuilder,\n  condition: CleanedWhere,\n  method: \"where\" | \"orWhere\",\n): Knex.QueryBuilder {\n  const { field, value, operator } = condition;\n\n  switch (operator) {\n    case \"eq\":\n      if (value === null) {\n        return query[method === \"orWhere\" ? \"orWhereNull\" : \"whereNull\"](field);\n      }\n      return query[method](field, \"=\", value);\n    case \"ne\":\n      if (value === null) {\n        return query[method === \"orWhere\" ? \"orWhereNotNull\" : \"whereNotNull\"](field);\n      }\n      return query[method](field, \"!=\", value);\n    case \"lt\":\n      return query[method](field, \"<\", value);\n    case \"lte\":\n      return query[method](field, \"<=\", value);\n    case \"gt\":\n      return query[method](field, \">\", value);\n    case \"gte\":\n      return query[method](field, \">=\", value);\n    case \"in\":\n      return query[method === \"orWhere\" ? \"orWhereIn\" : \"whereIn\"](\n        field,\n        value as (string | number)[],\n      );\n    case \"not_in\":\n      return query[method === \"orWhere\" ? \"orWhereNotIn\" : \"whereNotIn\"](\n        field,\n        value as (string | number)[],\n      );\n    case \"contains\":\n      return query[method](field, \"like\", `%${value}%`);\n    case \"starts_with\":\n      return query[method](field, \"like\", `${value}%`);\n    case \"ends_with\":\n      return query[method](field, \"like\", `%${value}`);\n    default:\n      return query;\n  }\n}\n"],"names":["createAdapterFactory","DB","sonamuKnexAdapter","lazyOptions","createCustomAdapter","getDb","getFieldName","create","model","data","row","insert","returning","findOne","where","query","applyWhere","first","findMany","limit","offset","sortBy","dbField","field","orderBy","direction","update","updateMany","count","delete","del","deleteMany","Number","adapterConfig","adapterId","adapterName","usePlural","supportsJSON","supportsDates","supportsBooleans","supportsNumericIds","transaction","cb","db","getDB","trx","options","Error","config","adapter","adapterCreator","conditions","andGroup","filter","c","connector","orGroup","length","condition","applyCondition","i","method","value","operator"],"mappings":"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"}
@@ -29,7 +29,7 @@ import * as path from "node:path";
29
29
  try {
30
30
  // Sonamu가 초기화되어 있는 경우
31
31
  const { Sonamu } = require("../api");
32
- apiKey = Sonamu.secret?.anthropic_api_key;
32
+ apiKey = Sonamu.secrets?.anthropic_api_key;
33
33
  } catch {
34
34
  // Sonamu가 초기화되지 않은 경우 (테스트 등)
35
35
  apiKey = undefined;
@@ -221,7 +221,7 @@ IMPORTANT: Return pure JSON only. Do NOT wrap in markdown code blocks.`;
221
221
  apiKey
222
222
  });
223
223
  const { text, usage } = await generateText({
224
- model: anthropic("claude-sonnet-4-5"),
224
+ model: anthropic("claude-sonnet-4-6"),
225
225
  prompt
226
226
  });
227
227
  const tokensUsed = usage?.totalTokens || 0;
@@ -338,4 +338,4 @@ IMPORTANT: Return pure JSON only. Do NOT wrap in markdown code blocks.`;
338
338
  return result;
339
339
  }
340
340
 
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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"}
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.secrets?.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-6\"),\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","secrets","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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"}
@@ -40,6 +40,10 @@ describe("LoadersResult", ()=>{
40
40
  expectTypeOf(result).toHaveProperty("posts");
41
41
  expectTypeOf(result).toEqualTypeOf();
42
42
  });
43
+ it("중첩 로더의 as 경로도 hydrate한다", ()=>{
44
+ const result = {};
45
+ expectTypeOf(result).toEqualTypeOf();
46
+ });
43
47
  it("여러 로더를 처리한다", ()=>{
44
48
  const result = {};
45
49
  expectTypeOf(result).toHaveProperty("projects");
@@ -65,6 +69,10 @@ describe("InferAllSubsets", ()=>{
65
69
  const result = {};
66
70
  expectTypeOf(result).toEqualTypeOf();
67
71
  });
72
+ it("중첩 로더 경로 alias를 포함한 Subset 결과를 추론한다", ()=>{
73
+ const result = {};
74
+ expectTypeOf(result).toEqualTypeOf();
75
+ });
68
76
  });
69
77
  describe("서브셋이 여러 개인 경우", ()=>{
70
78
  it("로더 없이 기본 Subset 결과를 추론한다", ()=>{
@@ -78,4 +86,4 @@ describe("InferAllSubsets", ()=>{
78
86
  });
79
87
  });
80
88
 
81
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/puri-subset.test-d.ts"],"sourcesContent":["import { describe, expectTypeOf, it } from \"vitest\";\nimport type { DatabaseSchemaExtend } from \"../types/types\";\nimport type { Puri } from \"./puri\";\nimport type { Hydrate, InferAllSubsets, LoadersResult } from \"./puri-subset.types\";\nimport type { PuriWrapper } from \"./puri-wrapper\";\n\n/** biome-ignore lint/suspicious/noExplicitAny: Puri Subset 타입 시스템에서 any를 허용함 */\ntype MockPuri<T> = Puri<DatabaseSchemaExtend, any, T>;\n\ndescribe(\"Hydrate\", () => {\n  it(\"flat 객체를 그대로 유지한다 (__ 없는 경우)\", () => {\n    type Input = { id: number; name: string };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<Hydrate<Input>>();\n  });\n\n  it(\"단일 depth의 __ 키를 중첩 객체로 변환한다\", () => {\n    type Input = { id: number; user__name: string; user__email: string };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      id: number;\n      user: { name: string; email: string };\n    }>();\n  });\n\n  it(\"다중 depth의 __ 키를 재귀적으로 중첩 객체로 변환한다\", () => {\n    type Input = {\n      id: number;\n      user__profile__bio: string;\n      user__profile__avatar: string;\n      user__name: string;\n    };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      id: number;\n      user: {\n        name: string;\n        profile: { bio: string; avatar: string };\n      };\n    }>();\n  });\n\n  it(\"여러 관계를 동시에 처리한다\", () => {\n    type Input = {\n      id: number;\n      user__id: number;\n      user__name: string;\n      post__id: number;\n      post__title: string;\n    };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      id: number;\n      user: { id: number; name: string };\n      post: { id: number; title: string };\n    }>();\n  });\n\n  it(\"빈 객체를 처리한다\", () => {\n    type Input = {};\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{}>();\n  });\n\n  it(\"nullable 필드가 있는 중첩 객체를 처리한다\", () => {\n    type Input = {\n      id: number;\n      user__id: number | null;\n      user__name: string | null;\n    };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      id: number;\n      user: { id: number | null; name: string | null };\n    }>();\n  });\n\n  it(\"동일한 prefix를 가진 여러 필드를 올바르게 그룹화한다\", () => {\n    type Input = {\n      id: number;\n      user__id: number;\n      user__name: string;\n      user_count: number;\n    };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      id: number;\n      user: { id: number; name: string };\n      user_count: number;\n    }>();\n  });\n});\n\ndescribe(\"LoadersResult\", () => {\n  it(\"단일 로더의 결과 타입을 생성한다\", () => {\n    type MockLoaderQb = (\n      qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n      fromIds: number[] | string[],\n    ) => MockPuri<{ id: number; title: string; refId: number }>;\n\n    type Loaders = [\n      {\n        as: \"posts\";\n        refId: \"id\";\n        qb: MockLoaderQb;\n      },\n    ];\n    type Result = LoadersResult<Loaders>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toHaveProperty(\"posts\");\n    expectTypeOf(result).toEqualTypeOf<{ posts: { id: number; title: string }[] }>();\n  });\n\n  it(\"중첩 로더를 처리한다\", () => {\n    type CommentLoaderQb = (\n      qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n      fromIds: number[] | string[],\n    ) => MockPuri<{ id: number; content: string; refId: number }>;\n\n    type PostLoaderQb = (\n      qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n      fromIds: number[] | string[],\n    ) => MockPuri<{ id: number; title: string; refId: number }>;\n\n    type Loaders = [\n      {\n        as: \"posts\";\n        refId: \"id\";\n        qb: PostLoaderQb;\n        loaders: [\n          {\n            as: \"comments\";\n            refId: \"id\";\n            qb: CommentLoaderQb;\n          },\n        ];\n      },\n    ];\n    type Result = LoadersResult<Loaders>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toHaveProperty(\"posts\");\n    expectTypeOf(result).toEqualTypeOf<{\n      posts: { id: number; title: string; comments: { id: number; content: string }[] }[];\n    }>();\n  });\n\n  it(\"여러 로더를 처리한다\", () => {\n    type ProjectLoader = {\n      as: \"projects\";\n      refId: \"id\";\n      qb: (\n        qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n        fromIds: number[] | string[],\n      ) => MockPuri<{ id: number; name: string; refId: number }>;\n    };\n    type DepartmentLoader = {\n      as: \"department\";\n      refId: \"id\";\n      qb: (\n        qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n        fromIds: number[] | string[],\n      ) => MockPuri<{ id: number; name: string; company_name: string; refId: number }>;\n    };\n    type EmployeeLoader = {\n      as: \"employees\";\n      refId: \"id\";\n      qb: (\n        qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n        fromIds: number[] | string[],\n      ) => MockPuri<{ id: number; employee_number: string; refId: number }>;\n      loaders: [\n        {\n          as: \"user\";\n          refId: \"id\";\n          qb: (\n            qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n            fromIds: number[] | string[],\n          ) => MockPuri<{ id: number; name: string; email: string; refId: number }>;\n        },\n      ];\n    };\n\n    type Loaders = [ProjectLoader, DepartmentLoader, EmployeeLoader];\n    type Result = LoadersResult<Loaders>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toHaveProperty(\"projects\");\n    expectTypeOf(result).toHaveProperty(\"employees\");\n    expectTypeOf(result).toHaveProperty(\"department\");\n    expectTypeOf(result).toEqualTypeOf<{\n      projects: { id: number; name: string }[];\n      department: { id: number; name: string; company_name: string }[];\n      employees: {\n        id: number;\n        employee_number: string;\n        user: { id: number; name: string; email: string }[];\n      }[];\n    }>();\n  });\n});\n\ndescribe(\"InferAllSubsets\", () => {\n  describe(\"서브셋이 하나인 경우\", () => {\n    it(\"로더 없이 기본 Subset 결과를 추론한다\", () => {\n      // SubsetQuery\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        user__name: string;\n        user__email: string;\n        department__name: string;\n        department__company__name: string;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n      };\n\n      // LoaderQuery\n      type LoaderQueries = {\n        A: [];\n      };\n\n      // Result\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toHaveProperty(\"A\");\n      expectTypeOf(result.A).toEqualTypeOf<{\n        id: number;\n        user: { name: string; email: string };\n        department: { name: string; company: { name: string } };\n      }>();\n      expectTypeOf(result.A.department.company).toEqualTypeOf<{ name: string }>();\n    });\n\n    it(\"로더를 포함한 Subset 결과를 추론한다\", () => {\n      // SubsetQuery\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        company__id: number;\n        company__name: string;\n        department__id: number;\n        department__name: string;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n      };\n\n      // LoaderQuery\n      type CompanyDepartmentsLoader = {\n        as: \"company__departments\";\n        refId: \"company__id\";\n        qb: (\n          qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n          fromIds: number[] | string[],\n        ) => MockPuri<{ id: number; name: string; refId: number }>;\n      };\n      type DepartmentProjectsLoader = {\n        as: \"department__projects\";\n        refId: \"department__id\";\n        qb: (\n          qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n          fromIds: number[] | string[],\n        ) => MockPuri<{ id: number; name: string; status: string; refId: number }>;\n      };\n      type LoaderQueries = {\n        A: [CompanyDepartmentsLoader, DepartmentProjectsLoader];\n      };\n\n      // Result\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toHaveProperty(\"A\");\n      expectTypeOf(result).toEqualTypeOf<{\n        A: {\n          id: number;\n          company: { id: number; name: string; departments: { id: number; name: string }[] };\n          department: {\n            id: number;\n            name: string;\n            projects: { id: number; name: string; status: string }[];\n          };\n        };\n      }>();\n    });\n\n    it(\"중첩 로더를 포함한 Subset 결과를 추론한다\", () => {\n      // SubsetQuery\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        company__name: string;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n      };\n\n      // LoaderQuery\n      type CompanyDepartmentsLoader = {\n        as: \"company__departments\";\n        refId: \"company__id\";\n        qb: (\n          qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n          fromIds: number[] | string[],\n        ) => MockPuri<{ id: number; name: string; refId: number }>;\n        loaders: [\n          {\n            as: \"projects\";\n            refId: \"id\";\n            qb: (\n              qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n              fromIds: number[] | string[],\n            ) => MockPuri<{ id: number; name: string; status: string; refId: number }>;\n          },\n        ];\n      };\n      type LoaderQueries = {\n        A: [CompanyDepartmentsLoader];\n      };\n\n      // Result\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        A: {\n          id: number;\n          company: {\n            name: string;\n            departments: {\n              id: number;\n              name: string;\n              projects: { id: number; name: string; status: string }[];\n            }[];\n          };\n        };\n      }>();\n    });\n  });\n\n  describe(\"서브셋이 여러 개인 경우\", () => {\n    it(\"로더 없이 기본 Subset 결과를 추론한다\", () => {\n      // SubsetQuery\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        email: string;\n        department__name: string;\n      }>;\n      type SubsetFnP = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        department__name: string;\n      }>;\n      type SubsetFnSS = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        email: string;\n        password: string;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n        P: SubsetFnP;\n        SS: SubsetFnSS;\n      };\n\n      // LoaderQuery\n      type LoaderQueries = {\n        A: [];\n        P: [];\n        SS: [];\n      };\n\n      // Result\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        A: { id: number; name: string; email: string; department: { name: string } };\n        P: { id: number; name: string; department: { name: string } };\n        SS: { id: number; name: string; email: string; password: string };\n      }>();\n    });\n\n    it(\"로더를 포함한 Subset 결과를 추론한다\", () => {\n      // SubsetQuery\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        email: string;\n        department__name: string;\n      }>;\n      type SubsetFnP = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        department__name: string;\n      }>;\n      type SubsetFnSS = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        email: string;\n        password: string;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n        P: SubsetFnP;\n        SS: SubsetFnSS;\n      };\n\n      // LoaderQuery\n      type ProjectLoader = {\n        as: \"projects\";\n        refId: \"id\";\n        qb: (\n          qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n          fromIds: number[] | string[],\n        ) => MockPuri<{ id: number; name: string; status: string; refId: number }>;\n        loaders: [\n          {\n            as: \"tags\";\n            refId: \"id\";\n            qb: (\n              qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n              fromIds: number[] | string[],\n            ) => MockPuri<{ id: number; name: string; refId: number }>;\n          },\n        ];\n      };\n      type LoaderQueries = {\n        A: [ProjectLoader];\n        P: [];\n        SS: [];\n      };\n\n      // Result\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        A: {\n          id: number;\n          name: string;\n          email: string;\n          department: { name: string };\n          projects: {\n            id: number;\n            name: string;\n            status: string;\n            tags: { id: number; name: string }[];\n          }[];\n        };\n        P: { id: number; name: string; department: { name: string } };\n        SS: { id: number; name: string; email: string; password: string };\n      }>();\n    });\n  });\n});\n"],"names":["describe","expectTypeOf","it","result","toEqualTypeOf","toHaveProperty","A","department","company"],"mappings":"AAAA,SAASA,QAAQ,EAAEC,YAAY,EAAEC,EAAE,QAAQ,SAAS;AASpDF,SAAS,WAAW;IAClBE,GAAG,gCAAgC;QAIjC,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IACpC;IAEAF,GAAG,+BAA+B;QAIhC,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAIpC;IAEAF,GAAG,qCAAqC;QAStC,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAOpC;IAEAF,GAAG,mBAAmB;QAUpB,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAKpC;IAEAF,GAAG,cAAc;QAIf,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IACpC;IAEAF,GAAG,+BAA+B;QAQhC,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAIpC;IAEAF,GAAG,oCAAoC;QASrC,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAKpC;AACF;AAEAJ,SAAS,iBAAiB;IACxBE,GAAG,sBAAsB;QAevB,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQE,cAAc,CAAC;QACpCJ,aAAaE,QAAQC,aAAa;IACpC;IAEAF,GAAG,eAAe;QA2BhB,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQE,cAAc,CAAC;QACpCJ,aAAaE,QAAQC,aAAa;IAGpC;IAEAF,GAAG,eAAe;QAuChB,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQE,cAAc,CAAC;QACpCJ,aAAaE,QAAQE,cAAc,CAAC;QACpCJ,aAAaE,QAAQE,cAAc,CAAC;QACpCJ,aAAaE,QAAQC,aAAa;IASpC;AACF;AAEAJ,SAAS,mBAAmB;IAC1BA,SAAS,eAAe;QACtBE,GAAG,4BAA4B;YAqB7B,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQE,cAAc,CAAC;YACpCJ,aAAaE,OAAOG,CAAC,EAAEF,aAAa;YAKpCH,aAAaE,OAAOG,CAAC,CAACC,UAAU,CAACC,OAAO,EAAEJ,aAAa;QACzD;QAEAF,GAAG,2BAA2B;YAqC5B,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQE,cAAc,CAAC;YACpCJ,aAAaE,QAAQC,aAAa;QAWpC;QAEAF,GAAG,8BAA8B;YAoC/B,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQC,aAAa;QAapC;IACF;IAEAJ,SAAS,iBAAiB;QACxBE,GAAG,4BAA4B;YAmC7B,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQC,aAAa;QAKpC;QAEAF,GAAG,2BAA2B;YAqD5B,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQC,aAAa;QAgBpC;IACF;AACF"}
89
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/puri-subset.test-d.ts"],"sourcesContent":["import { describe, expectTypeOf, it } from \"vitest\";\nimport type { DatabaseSchemaExtend } from \"../types/types\";\nimport type { Puri } from \"./puri\";\nimport type { Hydrate, InferAllSubsets, LoadersResult } from \"./puri-subset.types\";\nimport type { PuriWrapper } from \"./puri-wrapper\";\n\n/** biome-ignore lint/suspicious/noExplicitAny: Puri Subset 타입 시스템에서 any를 허용함 */\ntype MockPuri<T> = Puri<DatabaseSchemaExtend, any, T>;\n\ndescribe(\"Hydrate\", () => {\n  it(\"flat 객체를 그대로 유지한다 (__ 없는 경우)\", () => {\n    type Input = { id: number; name: string };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<Hydrate<Input>>();\n  });\n\n  it(\"단일 depth의 __ 키를 중첩 객체로 변환한다\", () => {\n    type Input = { id: number; user__name: string; user__email: string };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      id: number;\n      user: { name: string; email: string };\n    }>();\n  });\n\n  it(\"다중 depth의 __ 키를 재귀적으로 중첩 객체로 변환한다\", () => {\n    type Input = {\n      id: number;\n      user__profile__bio: string;\n      user__profile__avatar: string;\n      user__name: string;\n    };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      id: number;\n      user: {\n        name: string;\n        profile: { bio: string; avatar: string };\n      };\n    }>();\n  });\n\n  it(\"여러 관계를 동시에 처리한다\", () => {\n    type Input = {\n      id: number;\n      user__id: number;\n      user__name: string;\n      post__id: number;\n      post__title: string;\n    };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      id: number;\n      user: { id: number; name: string };\n      post: { id: number; title: string };\n    }>();\n  });\n\n  it(\"빈 객체를 처리한다\", () => {\n    type Input = {};\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{}>();\n  });\n\n  it(\"nullable 필드가 있는 중첩 객체를 처리한다\", () => {\n    type Input = {\n      id: number;\n      user__id: number | null;\n      user__name: string | null;\n    };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      id: number;\n      user: { id: number | null; name: string | null };\n    }>();\n  });\n\n  it(\"동일한 prefix를 가진 여러 필드를 올바르게 그룹화한다\", () => {\n    type Input = {\n      id: number;\n      user__id: number;\n      user__name: string;\n      user_count: number;\n    };\n    type Result = Hydrate<Input>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      id: number;\n      user: { id: number; name: string };\n      user_count: number;\n    }>();\n  });\n});\n\ndescribe(\"LoadersResult\", () => {\n  it(\"단일 로더의 결과 타입을 생성한다\", () => {\n    type MockLoaderQb = (\n      qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n      fromIds: number[] | string[],\n    ) => MockPuri<{ id: number; title: string; refId: number }>;\n\n    type Loaders = [\n      {\n        as: \"posts\";\n        refId: \"id\";\n        qb: MockLoaderQb;\n      },\n    ];\n    type Result = LoadersResult<Loaders>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toHaveProperty(\"posts\");\n    expectTypeOf(result).toEqualTypeOf<{ posts: { id: number; title: string }[] }>();\n  });\n\n  it(\"중첩 로더를 처리한다\", () => {\n    type CommentLoaderQb = (\n      qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n      fromIds: number[] | string[],\n    ) => MockPuri<{ id: number; content: string; refId: number }>;\n\n    type PostLoaderQb = (\n      qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n      fromIds: number[] | string[],\n    ) => MockPuri<{ id: number; title: string; refId: number }>;\n\n    type Loaders = [\n      {\n        as: \"posts\";\n        refId: \"id\";\n        qb: PostLoaderQb;\n        loaders: [\n          {\n            as: \"comments\";\n            refId: \"id\";\n            qb: CommentLoaderQb;\n          },\n        ];\n      },\n    ];\n    type Result = LoadersResult<Loaders>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toHaveProperty(\"posts\");\n    expectTypeOf(result).toEqualTypeOf<{\n      posts: { id: number; title: string; comments: { id: number; content: string }[] }[];\n    }>();\n  });\n\n  it(\"중첩 로더의 as 경로도 hydrate한다\", () => {\n    type IngredientLoaderQb = (\n      qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n      fromIds: number[] | string[],\n    ) => MockPuri<{ id: number; name: string; refId: number }>;\n\n    type PrescriptionItemLoaderQb = (\n      qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n      fromIds: number[] | string[],\n    ) => MockPuri<{\n      id: number;\n      test__id: number;\n      test__name: string;\n      refId: number;\n    }>;\n\n    type Loaders = [\n      {\n        as: \"items\";\n        refId: \"id\";\n        qb: PrescriptionItemLoaderQb;\n        loaders: [\n          {\n            as: \"test__items\";\n            refId: \"test__id\";\n            qb: IngredientLoaderQb;\n          },\n        ];\n      },\n    ];\n    type Result = LoadersResult<Loaders>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toEqualTypeOf<{\n      items: {\n        id: number;\n        test: {\n          id: number;\n          name: string;\n          items: { id: number; name: string }[];\n        };\n      }[];\n    }>();\n  });\n\n  it(\"여러 로더를 처리한다\", () => {\n    type ProjectLoader = {\n      as: \"projects\";\n      refId: \"id\";\n      qb: (\n        qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n        fromIds: number[] | string[],\n      ) => MockPuri<{ id: number; name: string; refId: number }>;\n    };\n    type DepartmentLoader = {\n      as: \"department\";\n      refId: \"id\";\n      qb: (\n        qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n        fromIds: number[] | string[],\n      ) => MockPuri<{ id: number; name: string; company_name: string; refId: number }>;\n    };\n    type EmployeeLoader = {\n      as: \"employees\";\n      refId: \"id\";\n      qb: (\n        qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n        fromIds: number[] | string[],\n      ) => MockPuri<{ id: number; employee_number: string; refId: number }>;\n      loaders: [\n        {\n          as: \"user\";\n          refId: \"id\";\n          qb: (\n            qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n            fromIds: number[] | string[],\n          ) => MockPuri<{ id: number; name: string; email: string; refId: number }>;\n        },\n      ];\n    };\n\n    type Loaders = [ProjectLoader, DepartmentLoader, EmployeeLoader];\n    type Result = LoadersResult<Loaders>;\n\n    const result = {} as Result;\n    expectTypeOf(result).toHaveProperty(\"projects\");\n    expectTypeOf(result).toHaveProperty(\"employees\");\n    expectTypeOf(result).toHaveProperty(\"department\");\n    expectTypeOf(result).toEqualTypeOf<{\n      projects: { id: number; name: string }[];\n      department: { id: number; name: string; company_name: string }[];\n      employees: {\n        id: number;\n        employee_number: string;\n        user: { id: number; name: string; email: string }[];\n      }[];\n    }>();\n  });\n});\n\ndescribe(\"InferAllSubsets\", () => {\n  describe(\"서브셋이 하나인 경우\", () => {\n    it(\"로더 없이 기본 Subset 결과를 추론한다\", () => {\n      // SubsetQuery\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        user__name: string;\n        user__email: string;\n        department__name: string;\n        department__company__name: string;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n      };\n\n      // LoaderQuery\n      type LoaderQueries = {\n        A: [];\n      };\n\n      // Result\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toHaveProperty(\"A\");\n      expectTypeOf(result.A).toEqualTypeOf<{\n        id: number;\n        user: { name: string; email: string };\n        department: { name: string; company: { name: string } };\n      }>();\n      expectTypeOf(result.A.department.company).toEqualTypeOf<{ name: string }>();\n    });\n\n    it(\"로더를 포함한 Subset 결과를 추론한다\", () => {\n      // SubsetQuery\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        company__id: number;\n        company__name: string;\n        department__id: number;\n        department__name: string;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n      };\n\n      // LoaderQuery\n      type CompanyDepartmentsLoader = {\n        as: \"company__departments\";\n        refId: \"company__id\";\n        qb: (\n          qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n          fromIds: number[] | string[],\n        ) => MockPuri<{ id: number; name: string; refId: number }>;\n      };\n      type DepartmentProjectsLoader = {\n        as: \"department__projects\";\n        refId: \"department__id\";\n        qb: (\n          qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n          fromIds: number[] | string[],\n        ) => MockPuri<{ id: number; name: string; status: string; refId: number }>;\n      };\n      type LoaderQueries = {\n        A: [CompanyDepartmentsLoader, DepartmentProjectsLoader];\n      };\n\n      // Result\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toHaveProperty(\"A\");\n      expectTypeOf(result).toEqualTypeOf<{\n        A: {\n          id: number;\n          company: { id: number; name: string; departments: { id: number; name: string }[] };\n          department: {\n            id: number;\n            name: string;\n            projects: { id: number; name: string; status: string }[];\n          };\n        };\n      }>();\n    });\n\n    it(\"중첩 로더를 포함한 Subset 결과를 추론한다\", () => {\n      // SubsetQuery\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        company__name: string;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n      };\n\n      // LoaderQuery\n      type CompanyDepartmentsLoader = {\n        as: \"company__departments\";\n        refId: \"company__id\";\n        qb: (\n          qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n          fromIds: number[] | string[],\n        ) => MockPuri<{ id: number; name: string; refId: number }>;\n        loaders: [\n          {\n            as: \"projects\";\n            refId: \"id\";\n            qb: (\n              qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n              fromIds: number[] | string[],\n            ) => MockPuri<{ id: number; name: string; status: string; refId: number }>;\n          },\n        ];\n      };\n      type LoaderQueries = {\n        A: [CompanyDepartmentsLoader];\n      };\n\n      // Result\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        A: {\n          id: number;\n          company: {\n            name: string;\n            departments: {\n              id: number;\n              name: string;\n              projects: { id: number; name: string; status: string }[];\n            }[];\n          };\n        };\n      }>();\n    });\n\n    it(\"중첩 로더 경로 alias를 포함한 Subset 결과를 추론한다\", () => {\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        root__id: number;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n      };\n\n      type RootItemLoader = {\n        as: \"root__items\";\n        refId: \"root__id\";\n        qb: (\n          qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n          fromIds: number[] | string[],\n        ) => MockPuri<{\n          id: number;\n          test__id: number;\n          test__name: string;\n          refId: number;\n        }>;\n        loaders: [\n          {\n            as: \"test__inner_items\";\n            refId: \"test__id\";\n            qb: (\n              qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n              fromIds: number[] | string[],\n            ) => MockPuri<{ id: number; name: string; refId: number }>;\n          },\n        ];\n      };\n      type LoaderQueries = {\n        A: [RootItemLoader];\n      };\n\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        A: {\n          id: number;\n          root: {\n            id: number;\n            items: {\n              id: number;\n              test: {\n                id: number;\n                name: string;\n                inner_items: { id: number; name: string }[];\n              };\n            }[];\n          };\n        };\n      }>();\n    });\n  });\n\n  describe(\"서브셋이 여러 개인 경우\", () => {\n    it(\"로더 없이 기본 Subset 결과를 추론한다\", () => {\n      // SubsetQuery\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        email: string;\n        department__name: string;\n      }>;\n      type SubsetFnP = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        department__name: string;\n      }>;\n      type SubsetFnSS = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        email: string;\n        password: string;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n        P: SubsetFnP;\n        SS: SubsetFnSS;\n      };\n\n      // LoaderQuery\n      type LoaderQueries = {\n        A: [];\n        P: [];\n        SS: [];\n      };\n\n      // Result\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        A: { id: number; name: string; email: string; department: { name: string } };\n        P: { id: number; name: string; department: { name: string } };\n        SS: { id: number; name: string; email: string; password: string };\n      }>();\n    });\n\n    it(\"로더를 포함한 Subset 결과를 추론한다\", () => {\n      // SubsetQuery\n      type SubsetFnA = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        email: string;\n        department__name: string;\n      }>;\n      type SubsetFnP = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        department__name: string;\n      }>;\n      type SubsetFnSS = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => MockPuri<{\n        id: number;\n        name: string;\n        email: string;\n        password: string;\n      }>;\n      type SubsetQueries = {\n        A: SubsetFnA;\n        P: SubsetFnP;\n        SS: SubsetFnSS;\n      };\n\n      // LoaderQuery\n      type ProjectLoader = {\n        as: \"projects\";\n        refId: \"id\";\n        qb: (\n          qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n          fromIds: number[] | string[],\n        ) => MockPuri<{ id: number; name: string; status: string; refId: number }>;\n        loaders: [\n          {\n            as: \"tags\";\n            refId: \"id\";\n            qb: (\n              qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n              fromIds: number[] | string[],\n            ) => MockPuri<{ id: number; name: string; refId: number }>;\n          },\n        ];\n      };\n      type LoaderQueries = {\n        A: [ProjectLoader];\n        P: [];\n        SS: [];\n      };\n\n      // Result\n      type Result = InferAllSubsets<SubsetQueries, LoaderQueries>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        A: {\n          id: number;\n          name: string;\n          email: string;\n          department: { name: string };\n          projects: {\n            id: number;\n            name: string;\n            status: string;\n            tags: { id: number; name: string }[];\n          }[];\n        };\n        P: { id: number; name: string; department: { name: string } };\n        SS: { id: number; name: string; email: string; password: string };\n      }>();\n    });\n  });\n});\n"],"names":["describe","expectTypeOf","it","result","toEqualTypeOf","toHaveProperty","A","department","company"],"mappings":"AAAA,SAASA,QAAQ,EAAEC,YAAY,EAAEC,EAAE,QAAQ,SAAS;AASpDF,SAAS,WAAW;IAClBE,GAAG,gCAAgC;QAIjC,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IACpC;IAEAF,GAAG,+BAA+B;QAIhC,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAIpC;IAEAF,GAAG,qCAAqC;QAStC,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAOpC;IAEAF,GAAG,mBAAmB;QAUpB,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAKpC;IAEAF,GAAG,cAAc;QAIf,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IACpC;IAEAF,GAAG,+BAA+B;QAQhC,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAIpC;IAEAF,GAAG,oCAAoC;QASrC,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAKpC;AACF;AAEAJ,SAAS,iBAAiB;IACxBE,GAAG,sBAAsB;QAevB,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQE,cAAc,CAAC;QACpCJ,aAAaE,QAAQC,aAAa;IACpC;IAEAF,GAAG,eAAe;QA2BhB,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQE,cAAc,CAAC;QACpCJ,aAAaE,QAAQC,aAAa;IAGpC;IAEAF,GAAG,2BAA2B;QAgC5B,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQC,aAAa;IAUpC;IAEAF,GAAG,eAAe;QAuChB,MAAMC,SAAS,CAAC;QAChBF,aAAaE,QAAQE,cAAc,CAAC;QACpCJ,aAAaE,QAAQE,cAAc,CAAC;QACpCJ,aAAaE,QAAQE,cAAc,CAAC;QACpCJ,aAAaE,QAAQC,aAAa;IASpC;AACF;AAEAJ,SAAS,mBAAmB;IAC1BA,SAAS,eAAe;QACtBE,GAAG,4BAA4B;YAqB7B,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQE,cAAc,CAAC;YACpCJ,aAAaE,OAAOG,CAAC,EAAEF,aAAa;YAKpCH,aAAaE,OAAOG,CAAC,CAACC,UAAU,CAACC,OAAO,EAAEJ,aAAa;QACzD;QAEAF,GAAG,2BAA2B;YAqC5B,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQE,cAAc,CAAC;YACpCJ,aAAaE,QAAQC,aAAa;QAWpC;QAEAF,GAAG,8BAA8B;YAoC/B,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQC,aAAa;QAapC;QAEAF,GAAG,uCAAuC;YAsCxC,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQC,aAAa;QAgBpC;IACF;IAEAJ,SAAS,iBAAiB;QACxBE,GAAG,4BAA4B;YAmC7B,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQC,aAAa;QAKpC;QAEAF,GAAG,2BAA2B;YAqD5B,MAAMC,SAAS,CAAC;YAChBF,aAAaE,QAAQC,aAAa;QAgBpC;IACF;AACF"}
@@ -95,7 +95,7 @@ export type LoadersResult<TLoaders extends GenericPuriLoader[]> = Expand<{
95
95
  /**
96
96
  * 기본 결과와 Loader 결과 병합
97
97
  */
98
- type WithLoaders<TBase, TLoaders> = TLoaders extends GenericPuriLoader[] ? Expand<TBase & LoadersResult<TLoaders>> : TBase;
98
+ type WithLoaders<TBase, TLoaders> = TLoaders extends GenericPuriLoader[] ? Expand<Hydrate<TBase & LoadersResult<TLoaders>>> : TBase;
99
99
  /**
100
100
  * 단일 Subset 함수 + Loader에서 최종 결과 타입 추론
101
101
  */
@@ -1 +1 @@
1
- {"version":3,"file":"puri-subset.types.d.ts","sourceRoot":"","sources":["../../src/database/puri-subset.types.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"puri-subset.types.d.ts","sourceRoot":"","sources":["../../src/database/puri-subset.types.ts"],"names":[],"mappings":"AAAA,oFAAoF;AAEpF;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,gBAAgB,CAAC;AAC3D,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,QAAQ,CAAC;AACnC,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AAC3C,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAMlD;;;GAGG;AACH,MAAM,MAAM,YAAY,GAAG,CAAC,SAAS,EAAE,WAAW,CAAC,oBAAoB,CAAC,KAAK,IAAI,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC,CAAC;AAEjG;;GAEG;AACH,MAAM,MAAM,iBAAiB,CAAC,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,GAAG,EAAE,GAAG,EAAE,MAAM,CAAC,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;AAMjF;;;;GAIG;AACH,MAAM,MAAM,cAAc,GAAG,CAC3B,SAAS,EAAE,WAAW,CAAC,oBAAoB,CAAC,EAC5C,OAAO,EAAE,MAAM,EAAE,GAAG,MAAM,EAAE,KACzB,IAAI,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC,CAAC;AAEzB;;;GAGG;AACH,MAAM,MAAM,iBAAiB,GAAG;IAC9B,sBAAsB;IACtB,EAAE,EAAE,MAAM,CAAC;IACX,yBAAyB;IACzB,KAAK,EAAE,MAAM,CAAC;IACd,mBAAmB;IACnB,EAAE,EAAE,cAAc,CAAC;IACnB,wBAAwB;IACxB,OAAO,CAAC,EAAE,iBAAiB,EAAE,CAAC;CAC/B,CAAC;AAEF;;;GAGG;AACH,MAAM,MAAM,iBAAiB,CAAC,UAAU,SAAS,MAAM,IAAI,MAAM,CAAC,UAAU,EAAE,iBAAiB,EAAE,CAAC,CAAC;AAMnG;;;GAGG;AACH,KAAK,WAAW,CAAC,CAAC,SAAS,MAAM,IAAI,CAAC,SAAS,GAAG,MAAM,IAAI,KAAK,MAAM,EAAE,GAAG,IAAI,GAAG,KAAK,CAAC;AAEzF;;;GAGG;AACH,KAAK,WAAW,CAAC,CAAC,SAAS,MAAM,EAAE,IAAI,SAAS,MAAM,IAAI,CAAC,SAAS,GAAG,IAAI,KAAK,MAAM,IAAI,EAAE,GACxF,IAAI,GACJ,KAAK,CAAC;AAEV;;;;;;;;;;GAUG;AACH,MAAM,MAAM,OAAO,CAAC,CAAC,IAAI,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC;AACjD,KAAK,YAAY,CAAC,CAAC,IAAI;KACpB,CAAC,IAAI,MAAM,CAAC,IAAI,CAAC,SAAS,GAAG,MAAM,KAAK,MAAM,EAAE,GAAG,KAAK,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;CACrE,GAAG;KACD,CAAC,IAAI,WAAW,CAAC,MAAM,CAAC,GAAG,MAAM,CAAC,GAAG,MAAM,CAC1C,YAAY,CAAC;SACV,CAAC,IAAI,MAAM,CAAC,IAAI,CAAC,SAAS,GAAG,CAAC,KAAK,MAAM,EAAE,GAAG,WAAW,CAAC,CAAC,GAAG,MAAM,EAAE,CAAC,CAAC,GAAG,KAAK,GAAG,CAAC,CAAC,CAAC,CAAC;KACzF,CAAC,CACH;CACF,CAAC;AAMF;;GAEG;AACH,KAAK,mBAAmB,CAAC,SAAS,IAAI,SAAS,SAAS,cAAc,GAClE,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,GACxE,KAAK,CAAC;AAEV;;;;;;;;;GASG;AACH,MAAM,MAAM,aAAa,CAAC,QAAQ,SAAS,iBAAiB,EAAE,IAAI,MAAM,CAAC;KACtE,CAAC,IAAI,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,GAAG,WAAW,CAAC,mBAAmB,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE;CAC9F,CAAC,CAAC;AAEH;;GAEG;AACH,KAAK,WAAW,CAAC,KAAK,EAAE,QAAQ,IAAI,QAAQ,SAAS,iBAAiB,EAAE,GACpE,MAAM,CAAC,OAAO,CAAC,KAAK,GAAG,aAAa,CAAC,QAAQ,CAAC,CAAC,CAAC,GAChD,KAAK,CAAC;AAMV;;GAEG;AACH,KAAK,sBAAsB,CACzB,SAAS,SAAS,CAAC,GAAG,IAAI,EAAE,GAAG,KAAK,IAAI,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC,EACvD,QAAQ,SAAS,iBAAiB,EAAE,GAAG,SAAS,GAAG,SAAS,IAC1D,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC,iBAAiB,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,CAAC,CAAC;AAErF;;;;;;;;;;;;GAYG;AACH,MAAM,MAAM,eAAe,CACzB,UAAU,SAAS,MAAM,CAAC,MAAM,EAAE,CAAC,GAAG,IAAI,EAAE,GAAG,KAAK,GAAG,CAAC,EACxD,UAAU,SAAS,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,iBAAiB,EAAE,CAAC,CAAC,IAC7D;KACD,CAAC,IAAI,MAAM,UAAU,GAAG,sBAAsB,CAC7C,UAAU,CAAC,CAAC,CAAC,EACb,CAAC,SAAS,MAAM,UAAU,GAAG,UAAU,CAAC,CAAC,CAAC,GAAG,SAAS,CACvD;CACF,CAAC;AAMF;;GAEG;AACH,MAAM,MAAM,eAAe,GACvB,MAAM,GACN,QAAQ,GACR,SAAS,GACT,cAAc,GACd,OAAO,GACP,OAAO,GACP,OAAO,GACP,MAAM,GACN,OAAO,GACP,OAAO,GACP,QAAQ,GACR,OAAO,GACP,QAAQ,GACR,SAAS,GACT,UAAU,CAAC"}
@@ -13,4 +13,4 @@
13
13
  * Knex QueryBuilder에서 clear 가능한 statement 목록
14
14
  */ export { };
15
15
 
16
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/puri-subset.types.ts"],"sourcesContent":["/** biome-ignore-all lint/suspicious/noExplicitAny: Puri Subset 타입 시스템에서 any를 허용함 */\n\n/**\n * Puri Subset 타입 시스템\n *\n * SubsetQuery와 LoaderQuery를 기반으로 최종 결과 타입을 추론하는 타입 유틸리티들.\n * 핵심 개념:\n * - SubsetQuery: 기본 쿼리 (join, select)\n * - LoaderQuery: 1:N, N:M 관계 데이터 로딩\n * - Hydrate: flat한 결과를 중첩 객체로 변환 (예: user__name → { user: { name } })\n */\n\nimport type { DatabaseSchemaExtend } from \"../types/types\";\nimport type { Puri } from \"./puri\";\nimport type { Expand } from \"./puri.types\";\nimport type { PuriWrapper } from \"./puri-wrapper\";\n\n// ============================================================================\n// 기본 타입 정의\n// ============================================================================\n\n/**\n * SubsetQuery 함수 시그니처\n * PuriWrapper를 받아 Puri 쿼리 빌더를 반환\n */\nexport type PuriSubsetFn = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => Puri<any, any, any>;\n\n/**\n * Puri 인스턴스에서 TResult 타입 추출\n */\nexport type ExtractPuriResult<T> = T extends Puri<any, any, infer R> ? R : never;\n\n// ============================================================================\n// Loader 관련 타입\n// ============================================================================\n\n/**\n * Loader 쿼리 빌더 함수 시그니처\n * @param qbWrapper - Puri 래퍼\n * @param fromIds - 부모 레코드 ID 배열\n */\nexport type PuriLoaderQbFn = (\n  qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n  fromIds: number[] | string[],\n) => Puri<any, any, any>;\n\n/**\n * 단일 Loader 정의\n * 1:N 또는 N:M 관계 데이터를 로드하는 설정\n */\nexport type GenericPuriLoader = {\n  /** 결과 객체에서 사용할 필드명 */\n  as: string;\n  /** 부모 레코드와 연결할 참조 필드명 */\n  refId: string;\n  /** 데이터 로딩 쿼리 빌더 */\n  qb: PuriLoaderQbFn;\n  /** 중첩 로더 (재귀적 로딩 지원) */\n  loaders?: GenericPuriLoader[];\n};\n\n/**\n * 모델별 Loader 쿼리 컬렉션\n * 각 SubsetKey에 대해 Loader 배열을 정의\n */\nexport type PuriLoaderQueries<TSubsetKey extends string> = Record<TSubsetKey, GenericPuriLoader[]>;\n\n// ============================================================================\n// Hydrate 타입 시스템\n// ============================================================================\n\n/**\n * 구분자(__) 앞부분 추출\n * @example ExtractHead<\"user__name\"> = \"user\"\n */\ntype ExtractHead<K extends string> = K extends `${infer Head}__${string}` ? Head : never;\n\n/**\n * 구분자(__) 뒷부분 추출\n * @example ExtractTail<\"user__profile__name\", \"user\"> = \"profile__name\"\n */\ntype ExtractTail<K extends string, Head extends string> = K extends `${Head}__${infer Tail}`\n  ? Tail\n  : never;\n\n/**\n * Flat 객체를 중첩 객체로 변환\n *\n * 런타임 hydrate 함수와 동일한 동작을 타입 레벨에서 구현.\n * `__`로 구분된 키를 중첩 객체 구조로 변환.\n *\n * @example\n * type Input = { id: number; user__name: string; user__profile__bio: string }\n * type Output = Hydrate<Input>\n * // { id: number; user: { name: string; profile: { bio: string } } }\n */\nexport type Hydrate<T> = Expand<HydrateInner<T>>;\ntype HydrateInner<T> = {\n  [K in keyof T as K extends `${string}__${string}` ? never : K]: T[K];\n} & {\n  [K in ExtractHead<keyof T & string>]: Expand<\n    HydrateInner<{\n      [P in keyof T as P extends `${K}__${string}` ? ExtractTail<P & string, K> : never]: T[P];\n    }>\n  >;\n};\n\n// ============================================================================\n// Loader 결과 타입 추론\n// ============================================================================\n\n/**\n * Loader 쿼리 함수에서 결과 타입 추출 (refId 제외, Hydrate 적용)\n */\ntype ExtractLoaderResult<TLoaderQb> = TLoaderQb extends PuriLoaderQbFn\n  ? Expand<Hydrate<Omit<ExtractPuriResult<ReturnType<TLoaderQb>>, \"refId\">>>\n  : never;\n\n/**\n * Loader 배열에서 결과 객체 타입 빌드 (재귀적으로 중첩 loader 처리)\n *\n * @example\n * type Loaders = [\n *   { as: \"employees\"; refId: \"id\"; qb: ...; loaders: [{ as: \"projects\"; ... }] }\n * ]\n * type Result = LoadersResult<Loaders>\n * // { employees: Array<{ ...employee fields; projects: Array<...> }> }\n */\nexport type LoadersResult<TLoaders extends GenericPuriLoader[]> = Expand<{\n  [L in TLoaders[number] as L[\"as\"]]: WithLoaders<ExtractLoaderResult<L[\"qb\"]>, L[\"loaders\"]>[];\n}>;\n\n/**\n * 기본 결과와 Loader 결과 병합\n */\ntype WithLoaders<TBase, TLoaders> = TLoaders extends GenericPuriLoader[]\n  ? Expand<TBase & LoadersResult<TLoaders>>\n  : TBase;\n\n// ============================================================================\n// Subset 결과 타입 추론\n// ============================================================================\n\n/**\n * 단일 Subset 함수 + Loader에서 최종 결과 타입 추론\n */\ntype InferSubsetWithLoaders<\n  TSubsetFn extends (...args: any) => Puri<any, any, any>,\n  TLoaders extends GenericPuriLoader[] | undefined = undefined,\n> = Expand<Hydrate<WithLoaders<ExtractPuriResult<ReturnType<TSubsetFn>>, TLoaders>>>;\n\n/**\n * 전체 SubsetQueries + LoaderQueries 객체에서 전체 결과 맵 생성\n *\n * @template TSubsetMap - Subset 함수들의 모음 객체\n * @template TLoaderMap - Loader 배열들의 모음 객체\n *\n * @example\n * type Result = InferAllSubsets<\n *   { A: () => Puri<...>, P: () => Puri<...> },\n *   { A: [...loaders], P: [] }\n * >\n * // { A: InferredTypeA; P: InferredTypeP }\n */\nexport type InferAllSubsets<\n  TSubsetMap extends Record<string, (...args: any) => any>,\n  TLoaderMap extends Partial<Record<string, GenericPuriLoader[]>>,\n> = {\n  [K in keyof TSubsetMap]: InferSubsetWithLoaders<\n    TSubsetMap[K],\n    K extends keyof TLoaderMap ? TLoaderMap[K] : undefined\n  >;\n};\n\n// ============================================================================\n// Clear 문\n// ============================================================================\n\n/**\n * Knex QueryBuilder에서 clear 가능한 statement 목록\n */\nexport type ClearStatements =\n  | \"with\"\n  | \"select\"\n  | \"columns\"\n  | \"hintComments\"\n  | \"where\"\n  | \"union\"\n  | \"using\"\n  | \"join\"\n  | \"group\"\n  | \"order\"\n  | \"having\"\n  | \"limit\"\n  | \"offset\"\n  | \"counter\"\n  | \"counters\";\n"],"names":[],"mappings":"AAAA,kFAAkF,GAElF;;;;;;;;CAQC,GAoKD,+EAA+E;AAC/E,UAAU;AACV,+EAA+E;AAE/E;;CAEC,GACD,WAee"}
16
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/puri-subset.types.ts"],"sourcesContent":["/** biome-ignore-all lint/suspicious/noExplicitAny: Puri Subset 타입 시스템에서 any를 허용함 */\n\n/**\n * Puri Subset 타입 시스템\n *\n * SubsetQuery와 LoaderQuery를 기반으로 최종 결과 타입을 추론하는 타입 유틸리티들.\n * 핵심 개념:\n * - SubsetQuery: 기본 쿼리 (join, select)\n * - LoaderQuery: 1:N, N:M 관계 데이터 로딩\n * - Hydrate: flat한 결과를 중첩 객체로 변환 (예: user__name → { user: { name } })\n */\n\nimport type { DatabaseSchemaExtend } from \"../types/types\";\nimport type { Puri } from \"./puri\";\nimport type { Expand } from \"./puri.types\";\nimport type { PuriWrapper } from \"./puri-wrapper\";\n\n// ============================================================================\n// 기본 타입 정의\n// ============================================================================\n\n/**\n * SubsetQuery 함수 시그니처\n * PuriWrapper를 받아 Puri 쿼리 빌더를 반환\n */\nexport type PuriSubsetFn = (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => Puri<any, any, any>;\n\n/**\n * Puri 인스턴스에서 TResult 타입 추출\n */\nexport type ExtractPuriResult<T> = T extends Puri<any, any, infer R> ? R : never;\n\n// ============================================================================\n// Loader 관련 타입\n// ============================================================================\n\n/**\n * Loader 쿼리 빌더 함수 시그니처\n * @param qbWrapper - Puri 래퍼\n * @param fromIds - 부모 레코드 ID 배열\n */\nexport type PuriLoaderQbFn = (\n  qbWrapper: PuriWrapper<DatabaseSchemaExtend>,\n  fromIds: number[] | string[],\n) => Puri<any, any, any>;\n\n/**\n * 단일 Loader 정의\n * 1:N 또는 N:M 관계 데이터를 로드하는 설정\n */\nexport type GenericPuriLoader = {\n  /** 결과 객체에서 사용할 필드명 */\n  as: string;\n  /** 부모 레코드와 연결할 참조 필드명 */\n  refId: string;\n  /** 데이터 로딩 쿼리 빌더 */\n  qb: PuriLoaderQbFn;\n  /** 중첩 로더 (재귀적 로딩 지원) */\n  loaders?: GenericPuriLoader[];\n};\n\n/**\n * 모델별 Loader 쿼리 컬렉션\n * 각 SubsetKey에 대해 Loader 배열을 정의\n */\nexport type PuriLoaderQueries<TSubsetKey extends string> = Record<TSubsetKey, GenericPuriLoader[]>;\n\n// ============================================================================\n// Hydrate 타입 시스템\n// ============================================================================\n\n/**\n * 구분자(__) 앞부분 추출\n * @example ExtractHead<\"user__name\"> = \"user\"\n */\ntype ExtractHead<K extends string> = K extends `${infer Head}__${string}` ? Head : never;\n\n/**\n * 구분자(__) 뒷부분 추출\n * @example ExtractTail<\"user__profile__name\", \"user\"> = \"profile__name\"\n */\ntype ExtractTail<K extends string, Head extends string> = K extends `${Head}__${infer Tail}`\n  ? Tail\n  : never;\n\n/**\n * Flat 객체를 중첩 객체로 변환\n *\n * 런타임 hydrate 함수와 동일한 동작을 타입 레벨에서 구현.\n * `__`로 구분된 키를 중첩 객체 구조로 변환.\n *\n * @example\n * type Input = { id: number; user__name: string; user__profile__bio: string }\n * type Output = Hydrate<Input>\n * // { id: number; user: { name: string; profile: { bio: string } } }\n */\nexport type Hydrate<T> = Expand<HydrateInner<T>>;\ntype HydrateInner<T> = {\n  [K in keyof T as K extends `${string}__${string}` ? never : K]: T[K];\n} & {\n  [K in ExtractHead<keyof T & string>]: Expand<\n    HydrateInner<{\n      [P in keyof T as P extends `${K}__${string}` ? ExtractTail<P & string, K> : never]: T[P];\n    }>\n  >;\n};\n\n// ============================================================================\n// Loader 결과 타입 추론\n// ============================================================================\n\n/**\n * Loader 쿼리 함수에서 결과 타입 추출 (refId 제외, Hydrate 적용)\n */\ntype ExtractLoaderResult<TLoaderQb> = TLoaderQb extends PuriLoaderQbFn\n  ? Expand<Hydrate<Omit<ExtractPuriResult<ReturnType<TLoaderQb>>, \"refId\">>>\n  : never;\n\n/**\n * Loader 배열에서 결과 객체 타입 빌드 (재귀적으로 중첩 loader 처리)\n *\n * @example\n * type Loaders = [\n *   { as: \"employees\"; refId: \"id\"; qb: ...; loaders: [{ as: \"projects\"; ... }] }\n * ]\n * type Result = LoadersResult<Loaders>\n * // { employees: Array<{ ...employee fields; projects: Array<...> }> }\n */\nexport type LoadersResult<TLoaders extends GenericPuriLoader[]> = Expand<{\n  [L in TLoaders[number] as L[\"as\"]]: WithLoaders<ExtractLoaderResult<L[\"qb\"]>, L[\"loaders\"]>[];\n}>;\n\n/**\n * 기본 결과와 Loader 결과 병합\n */\ntype WithLoaders<TBase, TLoaders> = TLoaders extends GenericPuriLoader[]\n  ? Expand<Hydrate<TBase & LoadersResult<TLoaders>>>\n  : TBase;\n\n// ============================================================================\n// Subset 결과 타입 추론\n// ============================================================================\n\n/**\n * 단일 Subset 함수 + Loader에서 최종 결과 타입 추론\n */\ntype InferSubsetWithLoaders<\n  TSubsetFn extends (...args: any) => Puri<any, any, any>,\n  TLoaders extends GenericPuriLoader[] | undefined = undefined,\n> = Expand<Hydrate<WithLoaders<ExtractPuriResult<ReturnType<TSubsetFn>>, TLoaders>>>;\n\n/**\n * 전체 SubsetQueries + LoaderQueries 객체에서 전체 결과 맵 생성\n *\n * @template TSubsetMap - Subset 함수들의 모음 객체\n * @template TLoaderMap - Loader 배열들의 모음 객체\n *\n * @example\n * type Result = InferAllSubsets<\n *   { A: () => Puri<...>, P: () => Puri<...> },\n *   { A: [...loaders], P: [] }\n * >\n * // { A: InferredTypeA; P: InferredTypeP }\n */\nexport type InferAllSubsets<\n  TSubsetMap extends Record<string, (...args: any) => any>,\n  TLoaderMap extends Partial<Record<string, GenericPuriLoader[]>>,\n> = {\n  [K in keyof TSubsetMap]: InferSubsetWithLoaders<\n    TSubsetMap[K],\n    K extends keyof TLoaderMap ? TLoaderMap[K] : undefined\n  >;\n};\n\n// ============================================================================\n// Clear 문\n// ============================================================================\n\n/**\n * Knex QueryBuilder에서 clear 가능한 statement 목록\n */\nexport type ClearStatements =\n  | \"with\"\n  | \"select\"\n  | \"columns\"\n  | \"hintComments\"\n  | \"where\"\n  | \"union\"\n  | \"using\"\n  | \"join\"\n  | \"group\"\n  | \"order\"\n  | \"having\"\n  | \"limit\"\n  | \"offset\"\n  | \"counter\"\n  | \"counters\";\n"],"names":[],"mappings":"AAAA,kFAAkF,GAElF;;;;;;;;CAQC,GAoKD,+EAA+E;AAC/E,UAAU;AACV,+EAA+E;AAE/E;;CAEC,GACD,WAee"}