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.
- package/dist/database/base-model.d.ts.map +1 -1
- package/dist/database/base-model.js +3 -4
- package/dist/database/knex.d.ts.map +1 -1
- package/dist/database/knex.js +10 -18
- package/dist/database/puri.d.ts +6 -6
- package/dist/database/puri.d.ts.map +1 -1
- package/dist/database/puri.js +1 -1
- package/dist/database/puri.types.d.ts +2 -1
- package/dist/database/puri.types.d.ts.map +1 -1
- package/dist/database/puri.types.js +1 -1
- package/dist/database/upsert-builder.js +4 -4
- package/dist/tasks/workflow-manager.d.ts.map +1 -1
- package/dist/tasks/workflow-manager.js +3 -5
- package/dist/testing/bootstrap.d.ts.map +1 -1
- package/dist/testing/bootstrap.js +4 -8
- package/dist/utils/utils.d.ts +0 -1
- package/dist/utils/utils.d.ts.map +1 -1
- package/dist/utils/utils.js +1 -6
- package/package.json +5 -5
- package/src/database/base-model.ts +2 -3
- package/src/database/knex.ts +9 -19
- package/src/database/puri.ts +6 -6
- package/src/database/puri.types.ts +2 -1
- package/src/database/upsert-builder.ts +4 -4
- package/src/tasks/workflow-manager.ts +3 -5
- package/src/testing/bootstrap.ts +2 -6
- package/src/utils/utils.ts +0 -5
|
@@ -37,14 +37,10 @@ export function bootstrap(vi, options) {
|
|
|
37
37
|
}
|
|
38
38
|
function getMockContext() {
|
|
39
39
|
return {
|
|
40
|
-
|
|
41
|
-
session: {},
|
|
40
|
+
session: null,
|
|
42
41
|
user: null,
|
|
43
|
-
|
|
44
|
-
|
|
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":"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"}
|
package/dist/utils/utils.d.ts
CHANGED
|
@@ -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;
|
|
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"}
|
package/dist/utils/utils.js
CHANGED
|
@@ -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.
|
|
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.
|
|
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/
|
|
131
|
+
"@sonamu-kit/tasks": "^0.2.0",
|
|
132
132
|
"@sonamu-kit/ts-loader": "^2.1.3",
|
|
133
|
-
"@sonamu-kit/
|
|
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.
|
|
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
|
|
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);
|
package/src/database/knex.ts
CHANGED
|
@@ -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
|
-
|
|
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
|
}
|
package/src/database/puri.ts
CHANGED
|
@@ -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:
|
|
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:
|
|
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:
|
|
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:
|
|
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:
|
|
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
|
|
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:
|
|
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,
|
|
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 ?
|
|
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에서 동시 실행할 태스크 수, 기본은
|
|
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
|
-
|
|
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;
|
package/src/testing/bootstrap.ts
CHANGED
|
@@ -64,14 +64,10 @@ export function bootstrap(vi: VitestUtils, options?: BootstrapOptions) {
|
|
|
64
64
|
|
|
65
65
|
function getMockContext(): Context {
|
|
66
66
|
return {
|
|
67
|
-
|
|
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
|
|
package/src/utils/utils.ts
CHANGED
|
@@ -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[],
|