sonamu 0.8.26 → 0.8.28

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.
@@ -37,14 +37,10 @@ export function bootstrap(vi, options) {
37
37
  }
38
38
  function getMockContext() {
39
39
  return {
40
- ip: "127.0.0.1",
41
- session: {},
40
+ session: null,
42
41
  user: null,
43
- passport: {
44
- login: async ()=>{},
45
- logout: ()=>{}
46
- },
47
- naiteStore: Naite.createStore()
42
+ naiteStore: Naite.createStore(),
43
+ locale: ""
48
44
  };
49
45
  }
50
46
  export async function runWithContext(context, fn) {
@@ -122,4 +118,4 @@ export const testAs = Object.assign(async (user, title, fn, options)=>{
122
118
  todo: (title)=>vitestTest.todo(title)
123
119
  });
124
120
 
125
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/testing/bootstrap.ts"],"sourcesContent":["import {\n  afterAll,\n  afterEach,\n  beforeAll,\n  beforeEach,\n  type TestFunction,\n  type TestOptions,\n  type VitestUtils,\n  test as vitestTest,\n} from \"vitest\";\nimport type { Context } from \"../api/context\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { DB } from \"../database/db\";\nimport { Naite, type SerializedTrace } from \"../naite/naite\";\nimport { NaiteReporter } from \"../naite/naite-reporter\";\n\nexport interface BootstrapOptions {\n  /**\n   * Sonamu 초기화 모드를 지정합니다.\n   * - true (기본값): 테스팅 모드로 초기화 (빠름, Syncer/Task 생략)\n   * - false: 전체 초기화 (Syncer, Task, EntityManager 등 모두 로드)\n   *\n   * migrator, syncer, template 등의 테스트에서는 false를 사용합니다.\n   */\n  forTesting?: boolean;\n}\n\nexport function bootstrap(vi: VitestUtils, options?: BootstrapOptions) {\n  const forTesting = options?.forTesting ?? true;\n\n  beforeAll(async () => {\n    if (!forTesting) {\n      // forTesting: false인 경우 재초기화가 필요하므로 플래그 해제\n      Sonamu.isInitialized = false;\n    }\n    await Sonamu.init(true, false, undefined, forTesting);\n  });\n  beforeEach(async () => {\n    await DB.createTestTransaction();\n  });\n  afterEach(async ({ task }) => {\n    vi.useRealTimers();\n    await DB.clearTestTransaction();\n\n    await NaiteReporter.reportTestResult({\n      suiteName: task.suite?.name ?? \"(no suite)\",\n      suiteFilePath: task.file?.filepath,\n      testName: task.name,\n      testFilePath: task.file?.filepath ?? \"\",\n      testLine: task.location?.line ?? 0,\n      status: task.result?.state ?? \"pass\",\n      duration: task.result?.duration ?? 0,\n      error: task.result?.errors?.[0]\n        ? {\n            message: task.result.errors[0].message,\n            stack: task.result.errors[0].stack,\n          }\n        : undefined,\n      traces: task.meta?.traces ?? [],\n    });\n  });\n  afterAll(() => {});\n}\n\nfunction getMockContext(): Context {\n  return {\n    ip: \"127.0.0.1\",\n    session: {},\n    user: null,\n    passport: {\n      login: async () => {},\n      logout: () => {},\n    },\n    naiteStore: Naite.createStore(),\n  } as unknown as Context;\n}\n\nexport async function runWithContext(context: Context | null, fn: () => Promise<void>) {\n  // Sonamu.asyncLocalStorage.run으로 context 설정\n  await Sonamu.asyncLocalStorage.run({ context: context ?? getMockContext() }, fn);\n}\n\nexport async function runWithMockContext(fn: () => Promise<void>) {\n  await runWithContext(getMockContext(), fn);\n}\n\ndeclare module \"vitest\" {\n  interface TaskMeta {\n    traces: SerializedTrace[];\n  }\n}\n\nexport const test = Object.assign(\n  async (title: string, fn: TestFunction<object>, options?: TestOptions) => {\n    return vitestTest(title, options, async (context) => {\n      await runWithMockContext(async () => {\n        try {\n          await fn(context);\n          context.task.meta.traces = Naite.getAllTraces();\n        } catch (e: unknown) {\n          context.task.meta.traces = Naite.getAllTraces();\n          throw e;\n        }\n      });\n    });\n  },\n  {\n    skip: async (title: string, fn: TestFunction<object>, options?: TestOptions) =>\n      vitestTest.skip(title, options, fn),\n    only: async (title: string, fn: TestFunction<object>, options?: TestOptions) => {\n      return vitestTest.only(title, options, async (context) => {\n        await runWithMockContext(async () => {\n          try {\n            await fn(context);\n            context.task.meta.traces = Naite.getAllTraces();\n          } catch (e: unknown) {\n            context.task.meta.traces = Naite.getAllTraces();\n            throw e;\n          }\n        });\n      });\n    },\n    todo: (title: string) => vitestTest.todo(title),\n    each: vitestTest.each.bind(vitestTest) as typeof vitestTest.each,\n  },\n);\n\nexport const testAs = Object.assign(\n  async <User extends Context[\"user\"]>(\n    user: User,\n    title: string,\n    fn: TestFunction<object>,\n    options?: TestOptions,\n  ) => {\n    return vitestTest(title, options, async (context) => {\n      await runWithContext(\n        {\n          ...getMockContext(),\n          user,\n        },\n        async () => {\n          try {\n            await fn(context);\n            context.task.meta.traces = Naite.getAllTraces();\n          } catch (e: unknown) {\n            context.task.meta.traces = Naite.getAllTraces();\n            throw e;\n          }\n        },\n      );\n    });\n  },\n  {\n    skip: async <User extends Context[\"user\"]>(\n      _user: User,\n      title: string,\n      fn: TestFunction<object>,\n      options?: TestOptions,\n    ) => vitestTest.skip(title, options, fn),\n    only: async <User extends Context[\"user\"]>(\n      user: User,\n      title: string,\n      fn: TestFunction<object>,\n      options?: TestOptions,\n    ) => {\n      return vitestTest.only(title, options, async (context) => {\n        await runWithContext(\n          {\n            ...getMockContext(),\n            user,\n          },\n          async () => {\n            try {\n              await fn(context);\n              context.task.meta.traces = Naite.getAllTraces();\n            } catch (e: unknown) {\n              context.task.meta.traces = Naite.getAllTraces();\n              throw e;\n            }\n          },\n        );\n      });\n    },\n    todo: (title: string) => vitestTest.todo(title),\n  },\n);\n"],"names":["afterAll","afterEach","beforeAll","beforeEach","test","vitestTest","Sonamu","DB","Naite","NaiteReporter","bootstrap","vi","options","forTesting","isInitialized","init","undefined","createTestTransaction","task","useRealTimers","clearTestTransaction","reportTestResult","suiteName","suite","name","suiteFilePath","file","filepath","testName","testFilePath","testLine","location","line","status","result","state","duration","error","errors","message","stack","traces","meta","getMockContext","ip","session","user","passport","login","logout","naiteStore","createStore","runWithContext","context","fn","asyncLocalStorage","run","runWithMockContext","Object","assign","title","getAllTraces","e","skip","only","todo","each","bind","testAs","_user"],"mappings":"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"}
121
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/testing/bootstrap.ts"],"sourcesContent":["import {\n  afterAll,\n  afterEach,\n  beforeAll,\n  beforeEach,\n  type TestFunction,\n  type TestOptions,\n  type VitestUtils,\n  test as vitestTest,\n} from \"vitest\";\nimport type { Context } from \"../api/context\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { DB } from \"../database/db\";\nimport { Naite, type SerializedTrace } from \"../naite/naite\";\nimport { NaiteReporter } from \"../naite/naite-reporter\";\n\nexport interface BootstrapOptions {\n  /**\n   * Sonamu 초기화 모드를 지정합니다.\n   * - true (기본값): 테스팅 모드로 초기화 (빠름, Syncer/Task 생략)\n   * - false: 전체 초기화 (Syncer, Task, EntityManager 등 모두 로드)\n   *\n   * migrator, syncer, template 등의 테스트에서는 false를 사용합니다.\n   */\n  forTesting?: boolean;\n}\n\nexport function bootstrap(vi: VitestUtils, options?: BootstrapOptions) {\n  const forTesting = options?.forTesting ?? true;\n\n  beforeAll(async () => {\n    if (!forTesting) {\n      // forTesting: false인 경우 재초기화가 필요하므로 플래그 해제\n      Sonamu.isInitialized = false;\n    }\n    await Sonamu.init(true, false, undefined, forTesting);\n  });\n  beforeEach(async () => {\n    await DB.createTestTransaction();\n  });\n  afterEach(async ({ task }) => {\n    vi.useRealTimers();\n    await DB.clearTestTransaction();\n\n    await NaiteReporter.reportTestResult({\n      suiteName: task.suite?.name ?? \"(no suite)\",\n      suiteFilePath: task.file?.filepath,\n      testName: task.name,\n      testFilePath: task.file?.filepath ?? \"\",\n      testLine: task.location?.line ?? 0,\n      status: task.result?.state ?? \"pass\",\n      duration: task.result?.duration ?? 0,\n      error: task.result?.errors?.[0]\n        ? {\n            message: task.result.errors[0].message,\n            stack: task.result.errors[0].stack,\n          }\n        : undefined,\n      traces: task.meta?.traces ?? [],\n    });\n  });\n  afterAll(() => {});\n}\n\nfunction getMockContext(): Context {\n  return {\n    session: null,\n    user: null,\n    naiteStore: Naite.createStore(),\n    locale: \"\",\n  } as unknown as Context;\n}\n\nexport async function runWithContext(context: Context | null, fn: () => Promise<void>) {\n  // Sonamu.asyncLocalStorage.run으로 context 설정\n  await Sonamu.asyncLocalStorage.run({ context: context ?? getMockContext() }, fn);\n}\n\nexport async function runWithMockContext(fn: () => Promise<void>) {\n  await runWithContext(getMockContext(), fn);\n}\n\ndeclare module \"vitest\" {\n  interface TaskMeta {\n    traces: SerializedTrace[];\n  }\n}\n\nexport const test = Object.assign(\n  async (title: string, fn: TestFunction<object>, options?: TestOptions) => {\n    return vitestTest(title, options, async (context) => {\n      await runWithMockContext(async () => {\n        try {\n          await fn(context);\n          context.task.meta.traces = Naite.getAllTraces();\n        } catch (e: unknown) {\n          context.task.meta.traces = Naite.getAllTraces();\n          throw e;\n        }\n      });\n    });\n  },\n  {\n    skip: async (title: string, fn: TestFunction<object>, options?: TestOptions) =>\n      vitestTest.skip(title, options, fn),\n    only: async (title: string, fn: TestFunction<object>, options?: TestOptions) => {\n      return vitestTest.only(title, options, async (context) => {\n        await runWithMockContext(async () => {\n          try {\n            await fn(context);\n            context.task.meta.traces = Naite.getAllTraces();\n          } catch (e: unknown) {\n            context.task.meta.traces = Naite.getAllTraces();\n            throw e;\n          }\n        });\n      });\n    },\n    todo: (title: string) => vitestTest.todo(title),\n    each: vitestTest.each.bind(vitestTest) as typeof vitestTest.each,\n  },\n);\n\nexport const testAs = Object.assign(\n  async <User extends Context[\"user\"]>(\n    user: User,\n    title: string,\n    fn: TestFunction<object>,\n    options?: TestOptions,\n  ) => {\n    return vitestTest(title, options, async (context) => {\n      await runWithContext(\n        {\n          ...getMockContext(),\n          user,\n        },\n        async () => {\n          try {\n            await fn(context);\n            context.task.meta.traces = Naite.getAllTraces();\n          } catch (e: unknown) {\n            context.task.meta.traces = Naite.getAllTraces();\n            throw e;\n          }\n        },\n      );\n    });\n  },\n  {\n    skip: async <User extends Context[\"user\"]>(\n      _user: User,\n      title: string,\n      fn: TestFunction<object>,\n      options?: TestOptions,\n    ) => vitestTest.skip(title, options, fn),\n    only: async <User extends Context[\"user\"]>(\n      user: User,\n      title: string,\n      fn: TestFunction<object>,\n      options?: TestOptions,\n    ) => {\n      return vitestTest.only(title, options, async (context) => {\n        await runWithContext(\n          {\n            ...getMockContext(),\n            user,\n          },\n          async () => {\n            try {\n              await fn(context);\n              context.task.meta.traces = Naite.getAllTraces();\n            } catch (e: unknown) {\n              context.task.meta.traces = Naite.getAllTraces();\n              throw e;\n            }\n          },\n        );\n      });\n    },\n    todo: (title: string) => vitestTest.todo(title),\n  },\n);\n"],"names":["afterAll","afterEach","beforeAll","beforeEach","test","vitestTest","Sonamu","DB","Naite","NaiteReporter","bootstrap","vi","options","forTesting","isInitialized","init","undefined","createTestTransaction","task","useRealTimers","clearTestTransaction","reportTestResult","suiteName","suite","name","suiteFilePath","file","filepath","testName","testFilePath","testLine","location","line","status","result","state","duration","error","errors","message","stack","traces","meta","getMockContext","session","user","naiteStore","createStore","locale","runWithContext","context","fn","asyncLocalStorage","run","runWithMockContext","Object","assign","title","getAllTraces","e","skip","only","todo","each","bind","testAs","_user"],"mappings":"AAAA,SACEA,QAAQ,EACRC,SAAS,EACTC,SAAS,EACTC,UAAU,EAIVC,QAAQC,UAAU,QACb,SAAS;AAEhB,SAASC,MAAM,QAAQ,mBAAgB;AACvC,SAASC,EAAE,QAAQ,oBAAiB;AACpC,SAASC,KAAK,QAA8B,oBAAiB;AAC7D,SAASC,aAAa,QAAQ,6BAA0B;AAaxD,OAAO,SAASC,UAAUC,EAAe,EAAEC,OAA0B;IACnE,MAAMC,aAAaD,SAASC,cAAc;IAE1CX,UAAU;QACR,IAAI,CAACW,YAAY;YACf,2CAA2C;YAC3CP,OAAOQ,aAAa,GAAG;QACzB;QACA,MAAMR,OAAOS,IAAI,CAAC,MAAM,OAAOC,WAAWH;IAC5C;IACAV,WAAW;QACT,MAAMI,GAAGU,qBAAqB;IAChC;IACAhB,UAAU,OAAO,EAAEiB,IAAI,EAAE;QACvBP,GAAGQ,aAAa;QAChB,MAAMZ,GAAGa,oBAAoB;QAE7B,MAAMX,cAAcY,gBAAgB,CAAC;YACnCC,WAAWJ,KAAKK,KAAK,EAAEC,QAAQ;YAC/BC,eAAeP,KAAKQ,IAAI,EAAEC;YAC1BC,UAAUV,KAAKM,IAAI;YACnBK,cAAcX,KAAKQ,IAAI,EAAEC,YAAY;YACrCG,UAAUZ,KAAKa,QAAQ,EAAEC,QAAQ;YACjCC,QAAQf,KAAKgB,MAAM,EAAEC,SAAS;YAC9BC,UAAUlB,KAAKgB,MAAM,EAAEE,YAAY;YACnCC,OAAOnB,KAAKgB,MAAM,EAAEI,QAAQ,CAAC,EAAE,GAC3B;gBACEC,SAASrB,KAAKgB,MAAM,CAACI,MAAM,CAAC,EAAE,CAACC,OAAO;gBACtCC,OAAOtB,KAAKgB,MAAM,CAACI,MAAM,CAAC,EAAE,CAACE,KAAK;YACpC,IACAxB;YACJyB,QAAQvB,KAAKwB,IAAI,EAAED,UAAU,EAAE;QACjC;IACF;IACAzC,SAAS,KAAO;AAClB;AAEA,SAAS2C;IACP,OAAO;QACLC,SAAS;QACTC,MAAM;QACNC,YAAYtC,MAAMuC,WAAW;QAC7BC,QAAQ;IACV;AACF;AAEA,OAAO,eAAeC,eAAeC,OAAuB,EAAEC,EAAuB;IACnF,4CAA4C;IAC5C,MAAM7C,OAAO8C,iBAAiB,CAACC,GAAG,CAAC;QAAEH,SAASA,WAAWP;IAAiB,GAAGQ;AAC/E;AAEA,OAAO,eAAeG,mBAAmBH,EAAuB;IAC9D,MAAMF,eAAeN,kBAAkBQ;AACzC;AAQA,OAAO,MAAM/C,OAAOmD,OAAOC,MAAM,CAC/B,OAAOC,OAAeN,IAA0BvC;IAC9C,OAAOP,WAAWoD,OAAO7C,SAAS,OAAOsC;QACvC,MAAMI,mBAAmB;YACvB,IAAI;gBACF,MAAMH,GAAGD;gBACTA,QAAQhC,IAAI,CAACwB,IAAI,CAACD,MAAM,GAAGjC,MAAMkD,YAAY;YAC/C,EAAE,OAAOC,GAAY;gBACnBT,QAAQhC,IAAI,CAACwB,IAAI,CAACD,MAAM,GAAGjC,MAAMkD,YAAY;gBAC7C,MAAMC;YACR;QACF;IACF;AACF,GACA;IACEC,MAAM,OAAOH,OAAeN,IAA0BvC,UACpDP,WAAWuD,IAAI,CAACH,OAAO7C,SAASuC;IAClCU,MAAM,OAAOJ,OAAeN,IAA0BvC;QACpD,OAAOP,WAAWwD,IAAI,CAACJ,OAAO7C,SAAS,OAAOsC;YAC5C,MAAMI,mBAAmB;gBACvB,IAAI;oBACF,MAAMH,GAAGD;oBACTA,QAAQhC,IAAI,CAACwB,IAAI,CAACD,MAAM,GAAGjC,MAAMkD,YAAY;gBAC/C,EAAE,OAAOC,GAAY;oBACnBT,QAAQhC,IAAI,CAACwB,IAAI,CAACD,MAAM,GAAGjC,MAAMkD,YAAY;oBAC7C,MAAMC;gBACR;YACF;QACF;IACF;IACAG,MAAM,CAACL,QAAkBpD,WAAWyD,IAAI,CAACL;IACzCM,MAAM1D,WAAW0D,IAAI,CAACC,IAAI,CAAC3D;AAC7B,GACA;AAEF,OAAO,MAAM4D,SAASV,OAAOC,MAAM,CACjC,OACEX,MACAY,OACAN,IACAvC;IAEA,OAAOP,WAAWoD,OAAO7C,SAAS,OAAOsC;QACvC,MAAMD,eACJ;YACE,GAAGN,gBAAgB;YACnBE;QACF,GACA;YACE,IAAI;gBACF,MAAMM,GAAGD;gBACTA,QAAQhC,IAAI,CAACwB,IAAI,CAACD,MAAM,GAAGjC,MAAMkD,YAAY;YAC/C,EAAE,OAAOC,GAAY;gBACnBT,QAAQhC,IAAI,CAACwB,IAAI,CAACD,MAAM,GAAGjC,MAAMkD,YAAY;gBAC7C,MAAMC;YACR;QACF;IAEJ;AACF,GACA;IACEC,MAAM,OACJM,OACAT,OACAN,IACAvC,UACGP,WAAWuD,IAAI,CAACH,OAAO7C,SAASuC;IACrCU,MAAM,OACJhB,MACAY,OACAN,IACAvC;QAEA,OAAOP,WAAWwD,IAAI,CAACJ,OAAO7C,SAAS,OAAOsC;YAC5C,MAAMD,eACJ;gBACE,GAAGN,gBAAgB;gBACnBE;YACF,GACA;gBACE,IAAI;oBACF,MAAMM,GAAGD;oBACTA,QAAQhC,IAAI,CAACwB,IAAI,CAACD,MAAM,GAAGjC,MAAMkD,YAAY;gBAC/C,EAAE,OAAOC,GAAY;oBACnBT,QAAQhC,IAAI,CAACwB,IAAI,CAACD,MAAM,GAAGjC,MAAMkD,YAAY;oBAC7C,MAAMC;gBACR;YACF;QAEJ;IACF;IACAG,MAAM,CAACL,QAAkBpD,WAAWyD,IAAI,CAACL;AAC3C,GACA"}
@@ -7,7 +7,6 @@ export declare function exhaustive(_param: never): void;
7
7
  export declare function assertExists<T>(value: T | null | undefined, message?: string): T;
8
8
  export declare function assertNotNull<T>(value: T | null, message?: string): T;
9
9
  export declare function assertDefined<T>(value: T | undefined, message?: string): T;
10
- export declare function chunk<T>(array: T[], size: number): T[][];
11
10
  export declare function intersectionBy<T, K>(arr1: readonly T[], arr2: readonly T[], iteratee: (item: T) => K): T[];
12
11
  export declare function differenceWith<T>(arr1: readonly T[], arr2: readonly T[], comparator: (a: T, b: T) => boolean): T[];
13
12
  export declare function merge<T extends Record<string, any>>(defaultObj: T, userObj: T): T;
@@ -1 +1 @@
1
- {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../src/utils/utils.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,SAAS,CAAC;AAI9C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAEjD,wBAAgB,eAAe,IAAI,YAAY,CAG9C;AAED,wBAAgB,eAAe,IAAI,YAAY,CAyB9C;AAED,wBAAgB,WAAW,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,KAAK,IAAI,WAAW,CAAC,CAAC,CAAC,CAEhE;AAED,wBAAgB,UAAU,CAAC,MAAM,EAAE,KAAK,QAEvC;AAGD,wBAAgB,YAAY,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,IAAI,GAAG,SAAS,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,CAAC,CAKhF;AAGD,wBAAgB,aAAa,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,IAAI,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,CAAC,CAKrE;AAED,wBAAgB,aAAa,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,SAAS,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,CAAC,CAK1E;AAGD,wBAAgB,KAAK,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,EAAE,IAAI,EAAE,MAAM,GAAG,CAAC,EAAE,EAAE,CAExD;AAED,wBAAgB,cAAc,CAAC,CAAC,EAAE,CAAC,EACjC,IAAI,EAAE,SAAS,CAAC,EAAE,EAClB,IAAI,EAAE,SAAS,CAAC,EAAE,EAClB,QAAQ,EAAE,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GACvB,CAAC,EAAE,CAGL;AAED,wBAAgB,cAAc,CAAC,CAAC,EAC9B,IAAI,EAAE,SAAS,CAAC,EAAE,EAClB,IAAI,EAAE,SAAS,CAAC,EAAE,EAClB,UAAU,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,KAAK,OAAO,GAClC,CAAC,EAAE,CAEL;AAGD,wBAAgB,KAAK,CAAC,CAAC,SAAS,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE,OAAO,EAAE,CAAC,GAAG,CAAC,CAsBjF;AAcD,wBAAgB,+BAA+B,CAAC,OAAO,EAAE,cAAc,CAAC,SAAS,CAAC,GAAG,OAAO,CAM3F"}
1
+ {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../src/utils/utils.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,SAAS,CAAC;AAG9C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAEjD,wBAAgB,eAAe,IAAI,YAAY,CAG9C;AAED,wBAAgB,eAAe,IAAI,YAAY,CAyB9C;AAED,wBAAgB,WAAW,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,KAAK,IAAI,WAAW,CAAC,CAAC,CAAC,CAEhE;AAED,wBAAgB,UAAU,CAAC,MAAM,EAAE,KAAK,QAEvC;AAGD,wBAAgB,YAAY,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,IAAI,GAAG,SAAS,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,CAAC,CAKhF;AAGD,wBAAgB,aAAa,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,IAAI,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,CAAC,CAKrE;AAED,wBAAgB,aAAa,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,SAAS,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,CAAC,CAK1E;AAGD,wBAAgB,cAAc,CAAC,CAAC,EAAE,CAAC,EACjC,IAAI,EAAE,SAAS,CAAC,EAAE,EAClB,IAAI,EAAE,SAAS,CAAC,EAAE,EAClB,QAAQ,EAAE,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GACvB,CAAC,EAAE,CAGL;AAED,wBAAgB,cAAc,CAAC,CAAC,EAC9B,IAAI,EAAE,SAAS,CAAC,EAAE,EAClB,IAAI,EAAE,SAAS,CAAC,EAAE,EAClB,UAAU,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,KAAK,OAAO,GAClC,CAAC,EAAE,CAEL;AAGD,wBAAgB,KAAK,CAAC,CAAC,SAAS,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE,OAAO,EAAE,CAAC,GAAG,CAAC,CAsBjF;AAcD,wBAAgB,+BAA+B,CAAC,OAAO,EAAE,cAAc,CAAC,SAAS,CAAC,GAAG,OAAO,CAM3F"}
@@ -1,6 +1,5 @@
1
1
  import fs from "fs";
2
2
  import path from "path";
3
- import { cluster } from "radashi";
4
3
  export function findAppRootPath() {
5
4
  const apiRootPath = findApiRootPath();
6
5
  return path.dirname(apiRootPath);
@@ -55,10 +54,6 @@ export function assertDefined(value, message) {
55
54
  }
56
55
  return value;
57
56
  }
58
- // lodash chunk 대체 (radash cluster 사용)
59
- export function chunk(array, size) {
60
- return cluster(array, Math.ceil(array.length / size));
61
- }
62
57
  // lodash intersectionBy 대체
63
58
  export function intersectionBy(arr1, arr2, iteratee) {
64
59
  const arr2Keys = new Set(arr2.map(iteratee));
@@ -105,4 +100,4 @@ export function convertFastifyHeadersToStandard(headers) {
105
100
  return headersObj;
106
101
  }
107
102
 
108
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/utils/utils.ts"],"sourcesContent":["import type { FastifyRequest } from \"fastify\";\nimport fs from \"fs\";\nimport path from \"path\";\nimport { cluster } from \"radashi\";\nimport type { AbsolutePath } from \"./path-utils\";\n\nexport function findAppRootPath(): AbsolutePath {\n  const apiRootPath = findApiRootPath();\n  return path.dirname(apiRootPath) as AbsolutePath;\n}\n\nexport function findApiRootPath(): AbsolutePath {\n  // NOTE: for support npm / yarn / pnpm workspaces\n  // 하지만 workspace 쓰면 process.cwd() 하면 되는데... 이건 나중에 협의 후 수정하는걸로\n  const workspacePath = process.env.PNPM_SCRIPT_SRC_DIR ?? process.env.INIT_CWD;\n  if (nonNullable(workspacePath)) {\n    return workspacePath as AbsolutePath;\n  }\n\n  if (nonNullable(process.env.PNPM_PACKAGE_NAME)) {\n    return process.cwd().split(path.sep).join(path.sep) as AbsolutePath;\n  }\n\n  const basePath = import.meta.filename;\n  let dir = path.dirname(basePath);\n  if (dir.includes(\"/.yarn/\")) {\n    dir = dir.split(\"/.yarn/\")[0];\n  }\n\n  do {\n    if (fs.existsSync(path.join(dir, \"/package.json\"))) {\n      return dir.split(path.sep).join(path.sep) as AbsolutePath;\n    }\n    dir = dir.split(path.sep).slice(0, -1).join(path.sep);\n  } while (dir.split(path.sep).length > 1);\n  throw new Error(\"Cannot find AppRoot using Sonamu -2\");\n}\n\nexport function nonNullable<T>(value: T): value is NonNullable<T> {\n  return value !== null && value !== undefined;\n}\n\nexport function exhaustive(_param: never) {\n  throw new Error(`exhaustive`);\n}\n\n// 일반 버전\nexport function assertExists<T>(value: T | null | undefined, message?: string): T {\n  if (value === null || value === undefined) {\n    throw new Error(message ?? \"Value must exist\");\n  }\n  return value;\n}\n\n// null만 체크\nexport function assertNotNull<T>(value: T | null, message?: string): T {\n  if (value === null) {\n    throw new Error(message ?? \"Value must not be null\");\n  }\n  return value;\n}\n// undefined만 체크\nexport function assertDefined<T>(value: T | undefined, message?: string): T {\n  if (value === undefined) {\n    throw new Error(message ?? \"Value must be defined\");\n  }\n  return value;\n}\n\n// lodash chunk 대체 (radash cluster 사용)\nexport function chunk<T>(array: T[], size: number): T[][] {\n  return cluster(array, Math.ceil(array.length / size));\n}\n// lodash intersectionBy 대체\nexport function intersectionBy<T, K>(\n  arr1: readonly T[],\n  arr2: readonly T[],\n  iteratee: (item: T) => K,\n): T[] {\n  const arr2Keys = new Set(arr2.map(iteratee));\n  return arr1.filter((item) => arr2Keys.has(iteratee(item)));\n}\n// lodash differenceWith 대체\nexport function differenceWith<T>(\n  arr1: readonly T[],\n  arr2: readonly T[],\n  comparator: (a: T, b: T) => boolean,\n): T[] {\n  return arr1.filter((itemA) => !arr2.some((itemB) => comparator(itemA, itemB)));\n}\n\n// biome-ignore lint/suspicious/noExplicitAny: dynamic property access\nexport function merge<T extends Record<string, any>>(defaultObj: T, userObj: T): T {\n  // 원본 보존을 위해 defaultObj 복사\n  const result = { ...defaultObj };\n\n  // userObj의 각 속성을 순회\n  for (const key in userObj) {\n    // userObj의 own property만 처리 (프로토타입 체인 제외)\n    if (Object.hasOwn(userObj, key)) {\n      const userValue = userObj[key];\n      const defaultValue = result[key];\n\n      // 두 값이 모두 객체이고, 배열이 아닌 경우 재귀적으로 병합\n      if (isPlainObject(userValue) && isPlainObject(defaultValue)) {\n        result[key] = merge(defaultValue, userValue);\n      } else {\n        // 그 외의 경우 userObj의 값으로 덮어쓰기\n        result[key] = userValue;\n      }\n    }\n  }\n\n  return result;\n}\n\n// plain object 판별 헬퍼 함수\n// (배열, null, Date 등을 제외한 순수 객체만 true)\nfunction isPlainObject(value: unknown): value is Record<string, unknown> {\n  return (\n    value !== null &&\n    typeof value === \"object\" &&\n    !Array.isArray(value) &&\n    Object.prototype.toString.call(value) === \"[object Object]\"\n  );\n}\n\n// Convert Fastify headers to standard Headers object\nexport function convertFastifyHeadersToStandard(headers: FastifyRequest[\"headers\"]): Headers {\n  const headersObj = new Headers();\n  Object.entries(headers).forEach(([key, value]) => {\n    if (value) headersObj.append(key, value.toString());\n  });\n  return headersObj;\n}\n"],"names":["fs","path","cluster","findAppRootPath","apiRootPath","findApiRootPath","dirname","workspacePath","process","env","PNPM_SCRIPT_SRC_DIR","INIT_CWD","nonNullable","PNPM_PACKAGE_NAME","cwd","split","sep","join","basePath","filename","dir","includes","existsSync","slice","length","Error","value","undefined","exhaustive","_param","assertExists","message","assertNotNull","assertDefined","chunk","array","size","Math","ceil","intersectionBy","arr1","arr2","iteratee","arr2Keys","Set","map","filter","item","has","differenceWith","comparator","itemA","some","itemB","merge","defaultObj","userObj","result","key","Object","hasOwn","userValue","defaultValue","isPlainObject","Array","isArray","prototype","toString","call","convertFastifyHeadersToStandard","headers","headersObj","Headers","entries","forEach","append"],"mappings":"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"}
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/utils/utils.ts"],"sourcesContent":["import type { FastifyRequest } from \"fastify\";\nimport fs from \"fs\";\nimport path from \"path\";\nimport type { AbsolutePath } from \"./path-utils\";\n\nexport function findAppRootPath(): AbsolutePath {\n  const apiRootPath = findApiRootPath();\n  return path.dirname(apiRootPath) as AbsolutePath;\n}\n\nexport function findApiRootPath(): AbsolutePath {\n  // NOTE: for support npm / yarn / pnpm workspaces\n  // 하지만 workspace 쓰면 process.cwd() 하면 되는데... 이건 나중에 협의 후 수정하는걸로\n  const workspacePath = process.env.PNPM_SCRIPT_SRC_DIR ?? process.env.INIT_CWD;\n  if (nonNullable(workspacePath)) {\n    return workspacePath as AbsolutePath;\n  }\n\n  if (nonNullable(process.env.PNPM_PACKAGE_NAME)) {\n    return process.cwd().split(path.sep).join(path.sep) as AbsolutePath;\n  }\n\n  const basePath = import.meta.filename;\n  let dir = path.dirname(basePath);\n  if (dir.includes(\"/.yarn/\")) {\n    dir = dir.split(\"/.yarn/\")[0];\n  }\n\n  do {\n    if (fs.existsSync(path.join(dir, \"/package.json\"))) {\n      return dir.split(path.sep).join(path.sep) as AbsolutePath;\n    }\n    dir = dir.split(path.sep).slice(0, -1).join(path.sep);\n  } while (dir.split(path.sep).length > 1);\n  throw new Error(\"Cannot find AppRoot using Sonamu -2\");\n}\n\nexport function nonNullable<T>(value: T): value is NonNullable<T> {\n  return value !== null && value !== undefined;\n}\n\nexport function exhaustive(_param: never) {\n  throw new Error(`exhaustive`);\n}\n\n// 일반 버전\nexport function assertExists<T>(value: T | null | undefined, message?: string): T {\n  if (value === null || value === undefined) {\n    throw new Error(message ?? \"Value must exist\");\n  }\n  return value;\n}\n\n// null만 체크\nexport function assertNotNull<T>(value: T | null, message?: string): T {\n  if (value === null) {\n    throw new Error(message ?? \"Value must not be null\");\n  }\n  return value;\n}\n// undefined만 체크\nexport function assertDefined<T>(value: T | undefined, message?: string): T {\n  if (value === undefined) {\n    throw new Error(message ?? \"Value must be defined\");\n  }\n  return value;\n}\n\n// lodash intersectionBy 대체\nexport function intersectionBy<T, K>(\n  arr1: readonly T[],\n  arr2: readonly T[],\n  iteratee: (item: T) => K,\n): T[] {\n  const arr2Keys = new Set(arr2.map(iteratee));\n  return arr1.filter((item) => arr2Keys.has(iteratee(item)));\n}\n// lodash differenceWith 대체\nexport function differenceWith<T>(\n  arr1: readonly T[],\n  arr2: readonly T[],\n  comparator: (a: T, b: T) => boolean,\n): T[] {\n  return arr1.filter((itemA) => !arr2.some((itemB) => comparator(itemA, itemB)));\n}\n\n// biome-ignore lint/suspicious/noExplicitAny: dynamic property access\nexport function merge<T extends Record<string, any>>(defaultObj: T, userObj: T): T {\n  // 원본 보존을 위해 defaultObj 복사\n  const result = { ...defaultObj };\n\n  // userObj의 각 속성을 순회\n  for (const key in userObj) {\n    // userObj의 own property만 처리 (프로토타입 체인 제외)\n    if (Object.hasOwn(userObj, key)) {\n      const userValue = userObj[key];\n      const defaultValue = result[key];\n\n      // 두 값이 모두 객체이고, 배열이 아닌 경우 재귀적으로 병합\n      if (isPlainObject(userValue) && isPlainObject(defaultValue)) {\n        result[key] = merge(defaultValue, userValue);\n      } else {\n        // 그 외의 경우 userObj의 값으로 덮어쓰기\n        result[key] = userValue;\n      }\n    }\n  }\n\n  return result;\n}\n\n// plain object 판별 헬퍼 함수\n// (배열, null, Date 등을 제외한 순수 객체만 true)\nfunction isPlainObject(value: unknown): value is Record<string, unknown> {\n  return (\n    value !== null &&\n    typeof value === \"object\" &&\n    !Array.isArray(value) &&\n    Object.prototype.toString.call(value) === \"[object Object]\"\n  );\n}\n\n// Convert Fastify headers to standard Headers object\nexport function convertFastifyHeadersToStandard(headers: FastifyRequest[\"headers\"]): Headers {\n  const headersObj = new Headers();\n  Object.entries(headers).forEach(([key, value]) => {\n    if (value) headersObj.append(key, value.toString());\n  });\n  return headersObj;\n}\n"],"names":["fs","path","findAppRootPath","apiRootPath","findApiRootPath","dirname","workspacePath","process","env","PNPM_SCRIPT_SRC_DIR","INIT_CWD","nonNullable","PNPM_PACKAGE_NAME","cwd","split","sep","join","basePath","filename","dir","includes","existsSync","slice","length","Error","value","undefined","exhaustive","_param","assertExists","message","assertNotNull","assertDefined","intersectionBy","arr1","arr2","iteratee","arr2Keys","Set","map","filter","item","has","differenceWith","comparator","itemA","some","itemB","merge","defaultObj","userObj","result","key","Object","hasOwn","userValue","defaultValue","isPlainObject","Array","isArray","prototype","toString","call","convertFastifyHeadersToStandard","headers","headersObj","Headers","entries","forEach","append"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "sonamu",
3
- "version": "0.8.26",
3
+ "version": "0.8.28",
4
4
  "description": "Sonamu — TypeScript Fullstack API Framework",
5
5
  "keywords": [
6
6
  "typescript",
@@ -115,7 +115,7 @@
115
115
  "fastify-sse-v2": "^4.2.1",
116
116
  "flydrive": "^1.3.0",
117
117
  "inflection": "^3.0.2",
118
- "knex": "^3.1.0",
118
+ "knex": "^3.2.9",
119
119
  "mime-types": "^3.0.1",
120
120
  "minimatch": "^10.0.3",
121
121
  "node-cron": "^4.2.1",
@@ -128,9 +128,9 @@
128
128
  "vite": "7.3.0",
129
129
  "vitest": "^4.0.10",
130
130
  "@sonamu-kit/hmr-hook": "^0.4.1",
131
- "@sonamu-kit/hmr-runner": "^0.1.1",
131
+ "@sonamu-kit/tasks": "^0.2.0",
132
132
  "@sonamu-kit/ts-loader": "^2.1.3",
133
- "@sonamu-kit/tasks": "^0.2.0"
133
+ "@sonamu-kit/hmr-runner": "^0.1.1"
134
134
  },
135
135
  "devDependencies": {
136
136
  "@biomejs/biome": "^2.3.13",
@@ -159,7 +159,7 @@
159
159
  "better-auth": "~1.4.18",
160
160
  "fastify": "^4",
161
161
  "ioredis": "^5.8.2",
162
- "knex": "^3.1.0",
162
+ "knex": "^3.2.9",
163
163
  "pgvector": "^0.2.1",
164
164
  "typescript": "^5.9.3",
165
165
  "voyageai": "^0.0.8",
@@ -1,7 +1,7 @@
1
1
  /** biome-ignore-all lint/suspicious/noExplicitAny: Puri의 타입은 개별 모델에서 확정되므로 BaseModel에서는 any를 허용함 */
2
2
  import { getLogger, type Logger } from "@logtape/logtape";
3
3
  import type { Knex } from "knex";
4
- import { cloneDeep, group, isObject, omit, set } from "radashi";
4
+ import { cloneDeep, cluster, group, isObject, omit, set } from "radashi";
5
5
  import { type ListResult, normalizeFilterQuery, validateSonamuFilters } from "..";
6
6
  import { Sonamu } from "../api";
7
7
  import { EntityManager } from "../entity/entity-manager";
@@ -9,7 +9,6 @@ import type { FilterOperator, FilterQuery } from "../filter/types";
9
9
  import { convertDomainToCategory } from "../logger/category";
10
10
  import type { DatabaseSchemaExtend, SonamuQueryMode } from "../types/types";
11
11
  import { getJoinTables, getTableNamesFromWhere } from "../utils/sql-parser";
12
- import { chunk } from "../utils/utils";
13
12
  import type { EnhancerMap, ResolveSubsetIntersection } from "./base-model.types";
14
13
  import type { DBPreset } from "./db";
15
14
  import { DB } from "./db";
@@ -91,7 +90,7 @@ export class BaseModelClass<
91
90
  }
92
91
 
93
92
  let resultIds: number[] = [];
94
- for (const items of chunk(unqKeys, chunkSize)) {
93
+ for (const items of cluster(unqKeys, chunkSize)) {
95
94
  const dbRows = await wdb(tableName)
96
95
  .select("id", wdb.raw(selectField))
97
96
  .whereIn(whereInField as string, items);
@@ -12,12 +12,20 @@ export function createKnexInstance(config: Knex.Config): Knex {
12
12
  }
13
13
 
14
14
  config.pool = {
15
+ maxConnectionLifetimeMillis: 1800000,
16
+ maxConnectionLifetimeJitterMillis: 300000,
15
17
  ...(config.pool ?? {}),
16
18
  propagateCreateError: false,
17
19
  idleTimeoutMillis: 10000,
18
20
  reapIntervalMillis: 1000,
19
21
  acquireTimeoutMillis: 30000,
20
22
  createTimeoutMillis: 30000,
23
+ validate: (connection: unknown) => {
24
+ if (typeof connection !== "object" || connection === null) return false;
25
+ const conn = connection as Record<string, unknown>;
26
+ if (conn._ending === true || conn._closed === true) return false;
27
+ return true;
28
+ },
21
29
  afterCreate: ((
22
30
  conn: Knex.Client & Record<string, unknown>,
23
31
  done: (err: Error | null, conn: Knex.Client) => void,
@@ -30,27 +38,9 @@ export function createKnexInstance(config: Knex.Config): Knex {
30
38
  stream.stream.setKeepAlive(true, 10000);
31
39
  }
32
40
 
33
- conn.on("error", (err: Error) => {
34
- Object.defineProperty(conn, "__knex__disposed", {
35
- value: err,
36
- writable: true,
37
- configurable: true,
38
- enumerable: true,
39
- });
40
- });
41
-
42
41
  done(null, conn);
43
42
  }) satisfies Knex.PoolConfig["afterCreate"],
44
43
  };
45
44
 
46
- const knexInstance = knex(config);
47
- knexInstance.client.validateConnection = (connection: unknown) => {
48
- if (typeof connection !== "object" || connection === null) return false;
49
- if ("__knex__disposed" in connection) return false;
50
- const conn = connection as Record<string, unknown>;
51
- if (conn._ending === true || conn._closed === true) return false;
52
- return true;
53
- };
54
-
55
- return knexInstance;
45
+ return knex(config);
56
46
  }
@@ -223,23 +223,23 @@ export class Puri<TSchema, TTables extends Record<string, any>, TResult> {
223
223
  }
224
224
 
225
225
  // Raw functions for SELECT
226
- static rawString(sql: string, params: unknown[] = []): SqlExpression<"string"> {
226
+ static rawString(sql: string, params: Knex.RawBinding[] = []): SqlExpression<"string"> {
227
227
  return { _type: "sql_expression", _return: "string", _sql: sql, _params: params };
228
228
  }
229
229
 
230
- static rawStringArray(sql: string, params: unknown[] = []): SqlExpression<"string[]"> {
230
+ static rawStringArray(sql: string, params: Knex.RawBinding[] = []): SqlExpression<"string[]"> {
231
231
  return { _type: "sql_expression", _return: "string[]", _sql: sql, _params: params };
232
232
  }
233
233
 
234
- static rawNumber(sql: string, params: unknown[] = []): SqlExpression<"number"> {
234
+ static rawNumber(sql: string, params: Knex.RawBinding[] = []): SqlExpression<"number"> {
235
235
  return { _type: "sql_expression", _return: "number", _sql: sql, _params: params };
236
236
  }
237
237
 
238
- static rawBoolean(sql: string, params: unknown[] = []): SqlExpression<"boolean"> {
238
+ static rawBoolean(sql: string, params: Knex.RawBinding[] = []): SqlExpression<"boolean"> {
239
239
  return { _type: "sql_expression", _return: "boolean", _sql: sql, _params: params };
240
240
  }
241
241
 
242
- static rawDate(sql: string, params: unknown[] = []): SqlExpression<"date"> {
242
+ static rawDate(sql: string, params: Knex.RawBinding[] = []): SqlExpression<"date"> {
243
243
  return { _type: "sql_expression", _return: "date", _sql: sql, _params: params };
244
244
  }
245
245
 
@@ -812,7 +812,7 @@ export class Puri<TSchema, TTables extends Record<string, any>, TResult> {
812
812
  }
813
813
 
814
814
  // WHERE RAW
815
- whereRaw(sql: string, bindings?: readonly unknown[]): this {
815
+ whereRaw(sql: string, bindings?: readonly Knex.RawBinding[]): this {
816
816
  this.knexQuery.whereRaw(sql, bindings);
817
817
  return this;
818
818
  }
@@ -1,5 +1,6 @@
1
1
  /** biome-ignore-all lint/suspicious/noExplicitAny: Puri.types.ts는 다양한 타입을 사용하고 있습니다. */
2
2
 
3
+ import type { Knex } from "knex";
3
4
  import type { QueryResult } from "pg";
4
5
  import type { DatabaseForeignKeys, DatabaseSchemaExtend } from "../types/types";
5
6
  import type { Puri } from "./puri";
@@ -311,7 +312,7 @@ export type SqlExpression<
311
312
  _type: "sql_expression"; // 또는 "computed_value"
312
313
  _return: T;
313
314
  _sql: string;
314
- _params: unknown[];
315
+ _params: Knex.RawBinding[];
315
316
  };
316
317
 
317
318
  // 결과 타입 가독성을 위한 타입 확장
@@ -1,11 +1,11 @@
1
1
  import { getLogger } from "@logtape/logtape";
2
2
  import { randomUUID } from "crypto";
3
3
  import type { Knex } from "knex";
4
- import { isArray, unique } from "radashi";
4
+ import { cluster, isArray, unique } from "radashi";
5
5
  import { EntityManager } from "../entity/entity-manager";
6
6
  import { Naite } from "../naite/naite";
7
7
  import type { DatabaseForeignKeys, DatabaseSchemaExtend, EntityIndex } from "../types/types";
8
- import { assertDefined, chunk, nonNullable } from "../utils/utils";
8
+ import { assertDefined, nonNullable } from "../utils/utils";
9
9
  import { batchUpdate, type RowWithId } from "./_batch_update";
10
10
  import type { ColumnKeys, ForeignKeyColumns, IdType, TableName } from "./puri.types";
11
11
 
@@ -290,7 +290,7 @@ export class UpsertBuilder {
290
290
 
291
291
  // 현재 레벨 upsert
292
292
  const chunkSize = options?.chunkSize;
293
- const levelChunks = chunkSize ? chunk(resolvedRows, chunkSize) : [resolvedRows];
293
+ const levelChunks = chunkSize ? cluster(resolvedRows, chunkSize) : [resolvedRows];
294
294
  const selectFields = unique(["id", ...extractFields]);
295
295
 
296
296
  for (let index = 0; index < levelChunks.length; index++) {
@@ -451,7 +451,7 @@ export class UpsertBuilder {
451
451
 
452
452
  // 각 FK 컬럼에 대한 WHERE IN 조건 추가
453
453
  for (const { column, values } of fkConditions) {
454
- deleteQuery = deleteQuery.whereIn(column, values);
454
+ deleteQuery = deleteQuery.whereIn(column, values as Knex.Value[]);
455
455
  }
456
456
 
457
457
  // 방금 upsert한 ID는 제외
@@ -24,7 +24,7 @@ import type { WorkflowMetadata } from "./decorator";
24
24
  import { StepWrapper } from "./step-wrapper";
25
25
 
26
26
  export interface WorkflowOptions {
27
- // worker에서 동시 실행할 태스크 수, 기본은 CPU 코어 수 - 1
27
+ // worker에서 동시 실행할 태스크 수, 기본은 1
28
28
  concurrency?: number;
29
29
 
30
30
  // worker에서 사용할 pub/sub 여부, 기본은 true
@@ -288,11 +288,9 @@ export class WorkflowManager {
288
288
  headers: {},
289
289
  createSSE: (schema: ZodObject) => createMockSSEFactory(schema),
290
290
  naiteStore: Naite.createStore(),
291
+ locale: "",
291
292
  user: null,
292
- passport: {
293
- login: async () => {},
294
- logout: () => {},
295
- },
293
+ session: null,
296
294
  } as unknown as Context;
297
295
 
298
296
  const contextProvider = Sonamu.config.tasks?.contextProvider;
@@ -64,14 +64,10 @@ export function bootstrap(vi: VitestUtils, options?: BootstrapOptions) {
64
64
 
65
65
  function getMockContext(): Context {
66
66
  return {
67
- ip: "127.0.0.1",
68
- session: {},
67
+ session: null,
69
68
  user: null,
70
- passport: {
71
- login: async () => {},
72
- logout: () => {},
73
- },
74
69
  naiteStore: Naite.createStore(),
70
+ locale: "",
75
71
  } as unknown as Context;
76
72
  }
77
73
 
@@ -1,7 +1,6 @@
1
1
  import type { FastifyRequest } from "fastify";
2
2
  import fs from "fs";
3
3
  import path from "path";
4
- import { cluster } from "radashi";
5
4
  import type { AbsolutePath } from "./path-utils";
6
5
 
7
6
  export function findAppRootPath(): AbsolutePath {
@@ -67,10 +66,6 @@ export function assertDefined<T>(value: T | undefined, message?: string): T {
67
66
  return value;
68
67
  }
69
68
 
70
- // lodash chunk 대체 (radash cluster 사용)
71
- export function chunk<T>(array: T[], size: number): T[][] {
72
- return cluster(array, Math.ceil(array.length / size));
73
- }
74
69
  // lodash intersectionBy 대체
75
70
  export function intersectionBy<T, K>(
76
71
  arr1: readonly T[],