sonamu 0.9.3 → 0.9.4

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.
Files changed (82) hide show
  1. package/dist/api/config.d.ts +0 -8
  2. package/dist/api/config.d.ts.map +1 -1
  3. package/dist/api/config.js +1 -1
  4. package/dist/api/sonamu.d.ts +0 -1
  5. package/dist/api/sonamu.d.ts.map +1 -1
  6. package/dist/api/sonamu.js +2 -41
  7. package/dist/auth/audit-log/builders.d.ts +216 -0
  8. package/dist/auth/audit-log/builders.d.ts.map +1 -0
  9. package/dist/auth/audit-log/builders.js +307 -0
  10. package/dist/auth/audit-log/events.d.ts +143 -0
  11. package/dist/auth/audit-log/events.d.ts.map +1 -0
  12. package/dist/auth/audit-log/events.js +74 -0
  13. package/dist/auth/audit-log/plugin.d.ts +11 -0
  14. package/dist/auth/audit-log/plugin.d.ts.map +1 -0
  15. package/dist/auth/audit-log/plugin.js +427 -0
  16. package/dist/auth/audit-log-ingestor.d.ts +3 -3
  17. package/dist/auth/audit-log-ingestor.d.ts.map +1 -1
  18. package/dist/auth/audit-log-ingestor.js +44 -50
  19. package/dist/auth/index.d.ts +2 -0
  20. package/dist/auth/index.d.ts.map +1 -1
  21. package/dist/auth/index.js +4 -4
  22. package/dist/auth/plugins/entity-definitions/admin.d.ts +1 -1
  23. package/dist/auth/plugins/entity-definitions/admin.js +4 -4
  24. package/dist/auth/plugins/entity-definitions/audit-log.d.ts +2 -2
  25. package/dist/auth/plugins/entity-definitions/audit-log.js +3 -3
  26. package/dist/auth/plugins/wrappers/admin.d.ts +2 -2
  27. package/dist/auth/plugins/wrappers/sso.d.ts +1 -1
  28. package/dist/bin/fixture.d.ts.map +1 -1
  29. package/dist/bin/fixture.js +111 -1
  30. package/dist/database/_batch_update.d.ts +1 -1
  31. package/dist/database/_batch_update.js +2 -2
  32. package/dist/entity/entity-manager.d.ts +2 -2
  33. package/dist/entity/entity-manager.d.ts.map +1 -1
  34. package/dist/entity/entity-manager.js +14 -4
  35. package/dist/index.js +4 -3
  36. package/dist/syncer/syncer.d.ts.map +1 -1
  37. package/dist/syncer/syncer.js +2 -9
  38. package/dist/template/implementations/entry-server.template.js +3 -2
  39. package/dist/template/implementations/generated.template.d.ts.map +1 -1
  40. package/dist/template/implementations/generated.template.js +2 -1
  41. package/dist/template/implementations/generated_sso.template.d.ts.map +1 -1
  42. package/dist/template/implementations/generated_sso.template.js +2 -1
  43. package/dist/template/implementations/queries.template.d.ts.map +1 -1
  44. package/dist/template/implementations/queries.template.js +3 -1
  45. package/dist/template/implementations/sd.template.js +3 -2
  46. package/dist/template/implementations/services.template.d.ts.map +1 -1
  47. package/dist/template/implementations/services.template.js +44 -7
  48. package/dist/template/zod-converter.d.ts.map +1 -1
  49. package/dist/template/zod-converter.js +2 -2
  50. package/dist/types/types.d.ts +14 -14
  51. package/dist/ui-web/assets/{index-DrTfl0Ts.js → index-C5KUjXm0.js} +46 -46
  52. package/dist/ui-web/index.html +1 -1
  53. package/dist/utils/fs-utils.d.ts.map +1 -1
  54. package/dist/utils/fs-utils.js +4 -4
  55. package/package.json +1 -2
  56. package/src/api/config.ts +0 -8
  57. package/src/api/sonamu.ts +1 -51
  58. package/src/auth/audit-log/builders.ts +791 -0
  59. package/src/auth/audit-log/events.ts +149 -0
  60. package/src/auth/audit-log/plugin.ts +913 -0
  61. package/src/auth/audit-log-ingestor.ts +3 -4
  62. package/src/auth/index.ts +2 -0
  63. package/src/auth/plugins/entity-definitions/admin.ts +3 -3
  64. package/src/auth/plugins/entity-definitions/audit-log.ts +2 -2
  65. package/src/bin/fixture.ts +143 -0
  66. package/src/database/_batch_update.ts +1 -1
  67. package/src/entity/entity-manager.ts +10 -3
  68. package/src/shared/app.shared.ts.txt +2 -3
  69. package/src/shared/web.shared.ts.txt +2 -2
  70. package/src/syncer/syncer.ts +1 -11
  71. package/src/template/implementations/entry-server.template.ts +1 -1
  72. package/src/template/implementations/generated.template.ts +1 -0
  73. package/src/template/implementations/generated_sso.template.ts +1 -0
  74. package/src/template/implementations/queries.template.ts +10 -1
  75. package/src/template/implementations/sd.template.ts +1 -1
  76. package/src/template/implementations/services.template.ts +62 -6
  77. package/src/template/zod-converter.ts +2 -1
  78. package/src/utils/fs-utils.ts +6 -4
  79. package/dist/auth/audit-log-proxy-types.d.ts +0 -23
  80. package/dist/auth/audit-log-proxy-types.d.ts.map +0 -1
  81. package/dist/auth/audit-log-proxy-types.js +0 -1
  82. package/src/auth/audit-log-proxy-types.ts +0 -23
@@ -1,7 +1,46 @@
1
- import { __esmMin } from "../_virtual/rolldown_runtime.js";
2
1
  import { createHash } from "node:crypto";
3
2
 
4
3
  //#region src/auth/audit-log-ingestor.ts
4
+ const AUDIT_EVENT_SOURCE = "better_auth";
5
+ const AUDIT_EVENT_SOURCE_VERSION = "better-auth|@better-auth/infra";
6
+ const SESSION_EVENT_TYPES = new Set([
7
+ "user_signed_in",
8
+ "user_signed_out",
9
+ "user_sign_in_failed",
10
+ "session_created",
11
+ "session_revoked",
12
+ "all_sessions_revoked",
13
+ "user_impersonated",
14
+ "user_impersonated_stopped"
15
+ ]);
16
+ const ACCOUNT_EVENT_TYPES = new Set([
17
+ "account_linked",
18
+ "account_unlinked",
19
+ "password_changed",
20
+ "two_factor_enabled",
21
+ "two_factor_disabled",
22
+ "two_factor_verified"
23
+ ]);
24
+ const VERIFICATION_EVENT_TYPES = new Set([
25
+ "email_verification_sent",
26
+ "password_reset_requested",
27
+ "password_reset_completed"
28
+ ]);
29
+ const SECURITY_EVENT_TYPES = new Set([
30
+ "security_blocked",
31
+ "security_allowed",
32
+ "security_challenged",
33
+ "security_stale_account"
34
+ ]);
35
+ const USER_EVENT_TYPES = new Set([
36
+ "user_created",
37
+ "profile_updated",
38
+ "profile_image_updated",
39
+ "email_verified",
40
+ "user_banned",
41
+ "user_unbanned",
42
+ "user_deleted"
43
+ ]);
5
44
  function pickString(source, key) {
6
45
  const value = source[key];
7
46
  return typeof value === "string" ? value : null;
@@ -67,9 +106,9 @@ function computeDedupeKey(parts) {
67
106
  return createHash("sha256").update(raw).digest("hex");
68
107
  }
69
108
  /**
70
- * Better Auth dash() 이벤트를 audit_events 테이블에 적재합니다.
109
+ * sonamuAuditLog 플러그인이 구성한 AuditLogEvent를 audit_events 테이블에 적재합니다.
71
110
  * ON CONFLICT (dedupe_key) DO NOTHING으로 중복을 silent 무시합니다.
72
- * auth.auditLog: true 설정 시 sonamu 내부에서 자동으로 호출됩니다.
111
+ * auth.plugins에 sonamuAuditLog() 추가 시 sonamu 내부에서 자동으로 호출됩니다.
73
112
  */
74
113
  async function ingestAuditEvent(db, event) {
75
114
  const eventData = event.eventData;
@@ -149,52 +188,7 @@ async function ingestAuditEvent(db, event) {
149
188
  return;
150
189
  }
151
190
  }
152
- var AUDIT_EVENT_SOURCE, AUDIT_EVENT_SOURCE_VERSION, SESSION_EVENT_TYPES, ACCOUNT_EVENT_TYPES, VERIFICATION_EVENT_TYPES, SECURITY_EVENT_TYPES, USER_EVENT_TYPES;
153
- var init_audit_log_ingestor = __esmMin((() => {
154
- AUDIT_EVENT_SOURCE = "better_auth";
155
- AUDIT_EVENT_SOURCE_VERSION = "better-auth|@better-auth/infra";
156
- SESSION_EVENT_TYPES = new Set([
157
- "user_signed_in",
158
- "user_signed_out",
159
- "user_sign_in_failed",
160
- "session_created",
161
- "session_revoked",
162
- "all_sessions_revoked",
163
- "user_impersonated",
164
- "user_impersonated_stopped"
165
- ]);
166
- ACCOUNT_EVENT_TYPES = new Set([
167
- "account_linked",
168
- "account_unlinked",
169
- "password_changed",
170
- "two_factor_enabled",
171
- "two_factor_disabled",
172
- "two_factor_verified"
173
- ]);
174
- VERIFICATION_EVENT_TYPES = new Set([
175
- "email_verification_sent",
176
- "password_reset_requested",
177
- "password_reset_completed"
178
- ]);
179
- SECURITY_EVENT_TYPES = new Set([
180
- "security_blocked",
181
- "security_allowed",
182
- "security_challenged",
183
- "security_stale_account"
184
- ]);
185
- USER_EVENT_TYPES = new Set([
186
- "user_created",
187
- "profile_updated",
188
- "profile_image_updated",
189
- "email_verified",
190
- "email_changed",
191
- "user_banned",
192
- "user_unbanned",
193
- "user_deleted"
194
- ]);
195
- }));
196
191
 
197
192
  //#endregion
198
- init_audit_log_ingestor();
199
- export { ingestAuditEvent, init_audit_log_ingestor };
200
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"audit-log-ingestor.js","names":[],"sources":["../../src/auth/audit-log-ingestor.ts"],"sourcesContent":["import { createHash } from \"node:crypto\";\n\nimport { type Knex } from \"knex\";\n\nimport { type AuditLogEvent } from \"./audit-log-proxy-types\";\n\nconst AUDIT_EVENT_SOURCE = \"better_auth\";\nconst AUDIT_EVENT_SOURCE_VERSION = \"better-auth|@better-auth/infra\";\n\nconst SESSION_EVENT_TYPES = new Set<string>([\n  \"user_signed_in\",\n  \"user_signed_out\",\n  \"user_sign_in_failed\",\n  \"session_created\",\n  \"session_revoked\",\n  \"all_sessions_revoked\",\n  \"user_impersonated\",\n  \"user_impersonated_stopped\",\n]);\n\nconst ACCOUNT_EVENT_TYPES = new Set<string>([\n  \"account_linked\",\n  \"account_unlinked\",\n  \"password_changed\",\n\n  // @better-auth/infra@0.1.14 기준 EVENT_TYPES 상수에는 정의되어 있으나 실제 trackEvent() 호출이 없는 미구현 이벤트임\n  // 향후 버전에서 emit될 경우를 대비해 account로 임시 등록\n  \"two_factor_enabled\",\n  \"two_factor_disabled\",\n  \"two_factor_verified\",\n]);\n\nconst VERIFICATION_EVENT_TYPES = new Set<string>([\n  \"email_verification_sent\",\n  \"password_reset_requested\",\n  \"password_reset_completed\",\n]);\n\nconst SECURITY_EVENT_TYPES = new Set<string>([\n  \"security_blocked\",\n  \"security_allowed\",\n  \"security_challenged\",\n  \"security_stale_account\",\n]);\n\nconst USER_EVENT_TYPES = new Set<string>([\n  \"user_created\",\n  \"profile_updated\",\n  \"profile_image_updated\",\n  \"email_verified\",\n  \"email_changed\",\n  \"user_banned\",\n  \"user_unbanned\",\n  \"user_deleted\",\n]);\n\nfunction pickString(source: Record<string, unknown>, key: string): string | null {\n  const value = source[key];\n  return typeof value === \"string\" ? value : null;\n}\n\nfunction pickFirstString(source: Record<string, unknown>, keys: readonly string[]): string | null {\n  for (const key of keys) {\n    const value = pickString(source, key);\n    if (value !== null) {\n      return value;\n    }\n  }\n  return null;\n}\n\nfunction parseOccurredAt(raw: unknown): Date {\n  if (raw instanceof Date && !Number.isNaN(raw.getTime())) {\n    return raw;\n  }\n  if (typeof raw === \"string\") {\n    const parsed = new Date(raw);\n    if (!Number.isNaN(parsed.getTime())) {\n      return parsed;\n    }\n  }\n  return new Date();\n}\n\nfunction classifyCategory(\n  eventType: string,\n): \"user\" | \"session\" | \"account\" | \"verification\" | \"organization\" | \"security\" {\n  if (eventType.startsWith(\"organization_\")) {\n    return \"organization\";\n  }\n  if (SESSION_EVENT_TYPES.has(eventType)) {\n    return \"session\";\n  }\n  if (ACCOUNT_EVENT_TYPES.has(eventType)) {\n    return \"account\";\n  }\n  if (VERIFICATION_EVENT_TYPES.has(eventType)) {\n    return \"verification\";\n  }\n  if (SECURITY_EVENT_TYPES.has(eventType)) {\n    return \"security\";\n  }\n  if (USER_EVENT_TYPES.has(eventType)) {\n    return \"user\";\n  }\n  return \"user\";\n}\n\nfunction computeDedupeKey(parts: {\n  source: string;\n  event_type: string;\n  event_key: string;\n  actor_user_id: string | null;\n  subject_user_id: string | null;\n  organization_id: string | null;\n  team_id: string | null;\n  session_id: string | null;\n  identifier: string | null;\n  reason: string | null;\n  action: string | null;\n  occurred_at: Date;\n}): string {\n  const norm = (v: string | null): string => v ?? \"\";\n  const raw = [\n    parts.source,\n    parts.event_type,\n    parts.event_key,\n    norm(parts.actor_user_id),\n    norm(parts.subject_user_id),\n    norm(parts.organization_id),\n    norm(parts.team_id),\n    norm(parts.session_id),\n    norm(parts.identifier),\n    norm(parts.reason),\n    norm(parts.action),\n    parts.occurred_at.toISOString(),\n  ].join(\"|\");\n  return createHash(\"sha256\").update(raw).digest(\"hex\");\n}\n\n/**\n * Better Auth dash() 이벤트를 audit_events 테이블에 적재합니다.\n * ON CONFLICT (dedupe_key) DO NOTHING으로 중복을 silent 무시합니다.\n * auth.auditLog: true 설정 시 sonamu 내부에서 자동으로 호출됩니다.\n */\nexport async function ingestAuditEvent(db: Knex, event: AuditLogEvent): Promise<void> {\n  const eventData = event.eventData;\n  const occurred_at = parseOccurredAt(eventData[\"occurredAt\"]);\n\n  const actor_user_id = pickString(eventData, \"triggeredBy\");\n  const subject_user_id = pickFirstString(eventData, [\n    \"userId\",\n    \"userid\",\n    \"acceptedById\",\n    \"rejectedById\",\n  ]);\n  const organization_id = pickString(eventData, \"organizationId\");\n  const team_id = pickFirstString(eventData, [\"teamId\", \"inviteeTeamId\"]);\n  const session_id = pickString(eventData, \"sessionId\");\n  const provider_id = pickString(eventData, \"providerId\");\n  const login_method = pickString(eventData, \"loginMethod\");\n  const identifier = pickFirstString(eventData, [\n    \"identifier\",\n    \"userEmail\",\n    \"memberEmail\",\n    \"inviteeEmail\",\n    \"acceptedByEmail\",\n    \"rejectedByEmail\",\n  ]);\n  const visitor_id = pickString(eventData, \"visitorId\");\n  const reason = pickFirstString(eventData, [\"reason\", \"banReason\"]);\n  const action = pickString(eventData, \"action\");\n  const trigger_context = pickString(eventData, \"triggerContext\");\n  const user_agent = pickString(eventData, \"userAgent\");\n\n  const ip_address = event.ipAddress ?? null;\n  const city = event.city ?? null;\n  const country = event.country ?? null;\n  const country_code = event.countryCode ?? null;\n\n  const category = classifyCategory(event.eventType);\n  const dedupe_key = computeDedupeKey({\n    source: AUDIT_EVENT_SOURCE,\n    event_type: event.eventType,\n    event_key: event.eventKey,\n    actor_user_id,\n    subject_user_id,\n    organization_id,\n    team_id,\n    session_id,\n    identifier,\n    reason,\n    action,\n    occurred_at,\n  });\n\n  // ON CONFLICT DO NOTHING: dedupe_key UNIQUE 위반을 silent 무시.\n  // try/catch 방식은 PG 커넥션을 aborted 상태로 만드므로 사용하지 않는다.\n  const inserted = await db(\"audit_events\")\n    .insert({\n      source: AUDIT_EVENT_SOURCE,\n      source_version: AUDIT_EVENT_SOURCE_VERSION,\n      category,\n      event_type: event.eventType,\n      event_key: event.eventKey,\n      dedupe_key,\n      actor_user_id,\n      subject_user_id,\n      organization_id,\n      team_id,\n      session_id,\n      provider_id,\n      login_method,\n      identifier,\n      visitor_id,\n      reason,\n      action,\n      trigger_context,\n      ip_address,\n      country_code,\n      country,\n      city,\n      user_agent,\n      payload_json: eventData,\n      occurred_at,\n    })\n    .onConflict(\"dedupe_key\")\n    .ignore()\n    .returning(\"dedupe_key\");\n\n  if (inserted.length === 0) {\n    return; // silent dedupe\n  }\n}\n"],"mappings":"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"}
193
+ export { ingestAuditEvent };
194
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"audit-log-ingestor.js","names":[],"sources":["../../src/auth/audit-log-ingestor.ts"],"sourcesContent":["import { createHash } from \"node:crypto\";\n\nimport { type Knex } from \"knex\";\n\nimport { type AuditLogEvent } from \"./audit-log/events\";\n\nconst AUDIT_EVENT_SOURCE = \"better_auth\";\nconst AUDIT_EVENT_SOURCE_VERSION = \"better-auth|@better-auth/infra\";\n\nconst SESSION_EVENT_TYPES = new Set<string>([\n  \"user_signed_in\",\n  \"user_signed_out\",\n  \"user_sign_in_failed\",\n  \"session_created\",\n  \"session_revoked\",\n  \"all_sessions_revoked\",\n  \"user_impersonated\",\n  \"user_impersonated_stopped\",\n]);\n\nconst ACCOUNT_EVENT_TYPES = new Set<string>([\n  \"account_linked\",\n  \"account_unlinked\",\n  \"password_changed\",\n\n  // @better-auth/infra@0.1.14 기준 EVENT_TYPES 상수에는 정의되어 있으나 실제 trackEvent() 호출이 없는 미구현 이벤트임\n  // 향후 버전에서 emit될 경우를 대비해 account로 임시 등록\n  \"two_factor_enabled\",\n  \"two_factor_disabled\",\n  \"two_factor_verified\",\n]);\n\nconst VERIFICATION_EVENT_TYPES = new Set<string>([\n  \"email_verification_sent\",\n  \"password_reset_requested\",\n  \"password_reset_completed\",\n]);\n\nconst SECURITY_EVENT_TYPES = new Set<string>([\n  \"security_blocked\",\n  \"security_allowed\",\n  \"security_challenged\",\n  \"security_stale_account\",\n]);\n\nconst USER_EVENT_TYPES = new Set<string>([\n  \"user_created\",\n  \"profile_updated\",\n  \"profile_image_updated\",\n  \"email_verified\",\n  \"user_banned\",\n  \"user_unbanned\",\n  \"user_deleted\",\n]);\n\nfunction pickString(source: Record<string, unknown>, key: string): string | null {\n  const value = source[key];\n  return typeof value === \"string\" ? value : null;\n}\n\nfunction pickFirstString(source: Record<string, unknown>, keys: readonly string[]): string | null {\n  for (const key of keys) {\n    const value = pickString(source, key);\n    if (value !== null) {\n      return value;\n    }\n  }\n  return null;\n}\n\nfunction parseOccurredAt(raw: unknown): Date {\n  if (raw instanceof Date && !Number.isNaN(raw.getTime())) {\n    return raw;\n  }\n  if (typeof raw === \"string\") {\n    const parsed = new Date(raw);\n    if (!Number.isNaN(parsed.getTime())) {\n      return parsed;\n    }\n  }\n  return new Date();\n}\n\nfunction classifyCategory(\n  eventType: string,\n): \"user\" | \"session\" | \"account\" | \"verification\" | \"organization\" | \"security\" {\n  if (eventType.startsWith(\"organization_\")) {\n    return \"organization\";\n  }\n  if (SESSION_EVENT_TYPES.has(eventType)) {\n    return \"session\";\n  }\n  if (ACCOUNT_EVENT_TYPES.has(eventType)) {\n    return \"account\";\n  }\n  if (VERIFICATION_EVENT_TYPES.has(eventType)) {\n    return \"verification\";\n  }\n  if (SECURITY_EVENT_TYPES.has(eventType)) {\n    return \"security\";\n  }\n  if (USER_EVENT_TYPES.has(eventType)) {\n    return \"user\";\n  }\n  return \"user\";\n}\n\nfunction computeDedupeKey(parts: {\n  source: string;\n  event_type: string;\n  event_key: string;\n  actor_user_id: string | null;\n  subject_user_id: string | null;\n  organization_id: string | null;\n  team_id: string | null;\n  session_id: string | null;\n  identifier: string | null;\n  reason: string | null;\n  action: string | null;\n  occurred_at: Date;\n}): string {\n  const norm = (v: string | null): string => v ?? \"\";\n  const raw = [\n    parts.source,\n    parts.event_type,\n    parts.event_key,\n    norm(parts.actor_user_id),\n    norm(parts.subject_user_id),\n    norm(parts.organization_id),\n    norm(parts.team_id),\n    norm(parts.session_id),\n    norm(parts.identifier),\n    norm(parts.reason),\n    norm(parts.action),\n    parts.occurred_at.toISOString(),\n  ].join(\"|\");\n  return createHash(\"sha256\").update(raw).digest(\"hex\");\n}\n\n/**\n * sonamuAuditLog 플러그인이 구성한 AuditLogEvent를 audit_events 테이블에 적재합니다.\n * ON CONFLICT (dedupe_key) DO NOTHING으로 중복을 silent 무시합니다.\n * auth.plugins에 sonamuAuditLog() 추가 시 sonamu 내부에서 자동으로 호출됩니다.\n */\nexport async function ingestAuditEvent(db: Knex, event: AuditLogEvent): Promise<void> {\n  const eventData = event.eventData;\n  const occurred_at = parseOccurredAt(eventData[\"occurredAt\"]);\n\n  const actor_user_id = pickString(eventData, \"triggeredBy\");\n  const subject_user_id = pickFirstString(eventData, [\n    \"userId\",\n    \"userid\",\n    \"acceptedById\",\n    \"rejectedById\",\n  ]);\n  const organization_id = pickString(eventData, \"organizationId\");\n  const team_id = pickFirstString(eventData, [\"teamId\", \"inviteeTeamId\"]);\n  const session_id = pickString(eventData, \"sessionId\");\n  const provider_id = pickString(eventData, \"providerId\");\n  const login_method = pickString(eventData, \"loginMethod\");\n  const identifier = pickFirstString(eventData, [\n    \"identifier\",\n    \"userEmail\",\n    \"memberEmail\",\n    \"inviteeEmail\",\n    \"acceptedByEmail\",\n    \"rejectedByEmail\",\n  ]);\n  const visitor_id = pickString(eventData, \"visitorId\");\n  const reason = pickFirstString(eventData, [\"reason\", \"banReason\"]);\n  const action = pickString(eventData, \"action\");\n  const trigger_context = pickString(eventData, \"triggerContext\");\n  const user_agent = pickString(eventData, \"userAgent\");\n\n  const ip_address = event.ipAddress ?? null;\n  const city = event.city ?? null;\n  const country = event.country ?? null;\n  const country_code = event.countryCode ?? null;\n\n  const category = classifyCategory(event.eventType);\n  const dedupe_key = computeDedupeKey({\n    source: AUDIT_EVENT_SOURCE,\n    event_type: event.eventType,\n    event_key: event.eventKey,\n    actor_user_id,\n    subject_user_id,\n    organization_id,\n    team_id,\n    session_id,\n    identifier,\n    reason,\n    action,\n    occurred_at,\n  });\n\n  // ON CONFLICT DO NOTHING: dedupe_key UNIQUE 위반을 silent 무시.\n  // try/catch 방식은 PG 커넥션을 aborted 상태로 만드므로 사용하지 않는다.\n  const inserted = await db(\"audit_events\")\n    .insert({\n      source: AUDIT_EVENT_SOURCE,\n      source_version: AUDIT_EVENT_SOURCE_VERSION,\n      category,\n      event_type: event.eventType,\n      event_key: event.eventKey,\n      dedupe_key,\n      actor_user_id,\n      subject_user_id,\n      organization_id,\n      team_id,\n      session_id,\n      provider_id,\n      login_method,\n      identifier,\n      visitor_id,\n      reason,\n      action,\n      trigger_context,\n      ip_address,\n      country_code,\n      country,\n      city,\n      user_agent,\n      payload_json: eventData,\n      occurred_at,\n    })\n    .onConflict(\"dedupe_key\")\n    .ignore()\n    .returning(\"dedupe_key\");\n\n  if (inserted.length === 0) {\n    return; // silent dedupe\n  }\n}\n"],"mappings":"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"}
@@ -1,6 +1,8 @@
1
1
  export type { GenerateBetterAuthEntitiesOptions } from "./auth-generator";
2
2
  export { generateBetterAuthEntities } from "./auth-generator";
3
3
  export { ingestAuditEvent } from "./audit-log-ingestor";
4
+ export { type AuditLogEvent } from "./audit-log/events";
5
+ export { sonamuAuditLog } from "./audit-log/plugin";
4
6
  export { BASE_FIELD_MAPPINGS, betterAuthV1 } from "./better-auth-entities";
5
7
  export { sonamuKnexAdapter } from "./knex-adapter";
6
8
  export * from "./plugins";
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/auth/index.ts"],"names":[],"mappings":"AAAA,YAAY,EAAE,iCAAiC,EAAE,MAAM,kBAAkB,CAAC;AAC1E,OAAO,EAAE,0BAA0B,EAAE,MAAM,kBAAkB,CAAC;AAC9D,OAAO,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AACxD,OAAO,EAAE,mBAAmB,EAAE,YAAY,EAAE,MAAM,wBAAwB,CAAC;AAC3E,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AAGnD,cAAc,WAAW,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/auth/index.ts"],"names":[],"mappings":"AAAA,YAAY,EAAE,iCAAiC,EAAE,MAAM,kBAAkB,CAAC;AAC1E,OAAO,EAAE,0BAA0B,EAAE,MAAM,kBAAkB,CAAC;AAC9D,OAAO,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AACxD,OAAO,EAAE,KAAK,aAAa,EAAE,MAAM,oBAAoB,CAAC;AACxD,OAAO,EAAE,cAAc,EAAE,MAAM,oBAAoB,CAAC;AACpD,OAAO,EAAE,mBAAmB,EAAE,YAAY,EAAE,MAAM,wBAAwB,CAAC;AAC3E,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AAGnD,cAAc,WAAW,CAAC"}
@@ -1,5 +1,6 @@
1
- import { ingestAuditEvent, init_audit_log_ingestor } from "./audit-log-ingestor.js";
2
1
  import { BASE_FIELD_MAPPINGS, betterAuthV1, init_better_auth_entities } from "./better-auth-entities.js";
2
+ import { ingestAuditEvent } from "./audit-log-ingestor.js";
3
+ import { sonamuAuditLog } from "./audit-log/plugin.js";
3
4
  import { generateBetterAuthEntities } from "./auth-generator.js";
4
5
  import { init_knex_adapter, sonamuKnexAdapter } from "./knex-adapter.js";
5
6
  import { ADMIN_SCHEMA, admin } from "./plugins/wrappers/admin.js";
@@ -15,10 +16,9 @@ import { USERNAME_SCHEMA, username } from "./plugins/wrappers/username.js";
15
16
  import "./plugins/index.js";
16
17
 
17
18
  //#region src/auth/index.ts
18
- init_audit_log_ingestor();
19
19
  init_better_auth_entities();
20
20
  init_knex_adapter();
21
21
 
22
22
  //#endregion
23
- export { ADMIN_SCHEMA, ANONYMOUS_SCHEMA, API_KEY_SCHEMA, BASE_FIELD_MAPPINGS, JWT_SCHEMA, ORGANIZATION_SCHEMA, PASSKEY_SCHEMA, PHONE_NUMBER_SCHEMA, SSO_SCHEMA, TWO_FACTOR_SCHEMA, USERNAME_SCHEMA, admin, anonymous, apiKey, betterAuthV1, generateBetterAuthEntities, ingestAuditEvent, jwt, organization, passkey, phoneNumber, sonamuKnexAdapter, sso, twoFactor, username };
24
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vLi4vc3JjL2F1dGgvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgeyBHZW5lcmF0ZUJldHRlckF1dGhFbnRpdGllc09wdGlvbnMgfSBmcm9tIFwiLi9hdXRoLWdlbmVyYXRvclwiO1xuZXhwb3J0IHsgZ2VuZXJhdGVCZXR0ZXJBdXRoRW50aXRpZXMgfSBmcm9tIFwiLi9hdXRoLWdlbmVyYXRvclwiO1xuZXhwb3J0IHsgaW5nZXN0QXVkaXRFdmVudCB9IGZyb20gXCIuL2F1ZGl0LWxvZy1pbmdlc3RvclwiO1xuZXhwb3J0IHsgQkFTRV9GSUVMRF9NQVBQSU5HUywgYmV0dGVyQXV0aFYxIH0gZnJvbSBcIi4vYmV0dGVyLWF1dGgtZW50aXRpZXNcIjtcbmV4cG9ydCB7IHNvbmFtdUtuZXhBZGFwdGVyIH0gZnJvbSBcIi4va25leC1hZGFwdGVyXCI7XG5cbi8vIOyZuOu2gOuhnOuKlCB3cmFwcGVyc+unjCBleHBvcnQgKGFkbWluLCB0d29GYWN0b3Ig65OxIOuemO2NvCDtlajsiJjsmYAgU0NIRU1BKVxuZXhwb3J0ICogZnJvbSBcIi4vcGx1Z2luc1wiO1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozt5QkFFd0Q7MkJBQ21CO21CQUN4QiJ9
23
+ export { ADMIN_SCHEMA, ANONYMOUS_SCHEMA, API_KEY_SCHEMA, BASE_FIELD_MAPPINGS, JWT_SCHEMA, ORGANIZATION_SCHEMA, PASSKEY_SCHEMA, PHONE_NUMBER_SCHEMA, SSO_SCHEMA, TWO_FACTOR_SCHEMA, USERNAME_SCHEMA, admin, anonymous, apiKey, betterAuthV1, generateBetterAuthEntities, ingestAuditEvent, jwt, organization, passkey, phoneNumber, sonamuAuditLog, sonamuKnexAdapter, sso, twoFactor, username };
24
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vLi4vc3JjL2F1dGgvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHR5cGUgeyBHZW5lcmF0ZUJldHRlckF1dGhFbnRpdGllc09wdGlvbnMgfSBmcm9tIFwiLi9hdXRoLWdlbmVyYXRvclwiO1xuZXhwb3J0IHsgZ2VuZXJhdGVCZXR0ZXJBdXRoRW50aXRpZXMgfSBmcm9tIFwiLi9hdXRoLWdlbmVyYXRvclwiO1xuZXhwb3J0IHsgaW5nZXN0QXVkaXRFdmVudCB9IGZyb20gXCIuL2F1ZGl0LWxvZy1pbmdlc3RvclwiO1xuZXhwb3J0IHsgdHlwZSBBdWRpdExvZ0V2ZW50IH0gZnJvbSBcIi4vYXVkaXQtbG9nL2V2ZW50c1wiO1xuZXhwb3J0IHsgc29uYW11QXVkaXRMb2cgfSBmcm9tIFwiLi9hdWRpdC1sb2cvcGx1Z2luXCI7XG5leHBvcnQgeyBCQVNFX0ZJRUxEX01BUFBJTkdTLCBiZXR0ZXJBdXRoVjEgfSBmcm9tIFwiLi9iZXR0ZXItYXV0aC1lbnRpdGllc1wiO1xuZXhwb3J0IHsgc29uYW11S25leEFkYXB0ZXIgfSBmcm9tIFwiLi9rbmV4LWFkYXB0ZXJcIjtcblxuLy8g7Jm467aA66Gc64qUIHdyYXBwZXJz66eMIGV4cG9ydCAoYWRtaW4sIHR3b0ZhY3RvciDrk7Eg656Y7Y28IO2VqOyImOyZgCBTQ0hFTUEpXG5leHBvcnQgKiBmcm9tIFwiLi9wbHVnaW5zXCI7XG4iXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7OzsyQkFLMkU7bUJBQ3hCIn0=
@@ -9,7 +9,7 @@ import { type BetterAuthEntityDef } from "./types";
9
9
  * - role: 사용자 역할 (기본값: "user")
10
10
  * - banned: 차단 여부
11
11
  * - ban_reason: 차단 사유
12
- * - ban_expires: 차단 만료 시간 (Unix timestamp)
12
+ * - ban_expires: 차단 만료 시간 (Date)
13
13
  *
14
14
  * Session 테이블:
15
15
  * - impersonated_by: 대리 로그인한 관리자 ID
@@ -9,7 +9,7 @@
9
9
  * - role: 사용자 역할 (기본값: "user")
10
10
  * - banned: 차단 여부
11
11
  * - ban_reason: 차단 사유
12
- * - ban_expires: 차단 만료 시간 (Unix timestamp)
12
+ * - ban_expires: 차단 만료 시간 (Date)
13
13
  *
14
14
  * Session 테이블:
15
15
  * - impersonated_by: 대리 로그인한 관리자 ID
@@ -42,9 +42,9 @@ const adminEntityDef = {
42
42
  },
43
43
  {
44
44
  name: "ban_expires",
45
- type: "bigInteger",
45
+ type: "date",
46
46
  nullable: true,
47
- desc: "차단 만료 (Unix timestamp)"
47
+ desc: "차단 만료"
48
48
  }
49
49
  ],
50
50
  Session: [{
@@ -58,4 +58,4 @@ const adminEntityDef = {
58
58
 
59
59
  //#endregion
60
60
  export { adminEntityDef };
61
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWRtaW4uanMiLCJuYW1lcyI6WyJhZG1pbkVudGl0eURlZjogQmV0dGVyQXV0aEVudGl0eURlZiJdLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9hdXRoL3BsdWdpbnMvZW50aXR5LWRlZmluaXRpb25zL2FkbWluLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHR5cGUgQmV0dGVyQXV0aEVudGl0eURlZiB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbi8qKlxuICogYmV0dGVyLWF1dGggYWRtaW4g7ZSM65+s6re47J24IOyXlO2LsO2LsCDsoJXsnZhcbiAqIGh0dHBzOi8vd3d3LmJldHRlci1hdXRoLmNvbS9kb2NzL3BsdWdpbnMvYWRtaW5cbiAqXG4gKiDqtIDrpqzsnpAg6riw64ql7J2EIOychO2VnCDtlYTrk5zrpbwg7LaU6rCA7ZWp64uI64ukOlxuICpcbiAqIFVzZXIg7YWM7J2067iUOlxuICogLSByb2xlOiDsgqzsmqnsnpAg7Jet7ZWgICjquLDrs7jqsJI6IFwidXNlclwiKVxuICogLSBiYW5uZWQ6IOywqOuLqCDsl6zrtoBcbiAqIC0gYmFuX3JlYXNvbjog7LCo64uoIOyCrOycoFxuICogLSBiYW5fZXhwaXJlczog7LCo64uoIOunjOujjCDsi5zqsIQgKFVuaXggdGltZXN0YW1wKVxuICpcbiAqIFNlc3Npb24g7YWM7J2067iUOlxuICogLSBpbXBlcnNvbmF0ZWRfYnk6IOuMgOumrCDroZzqt7jsnbjtlZwg6rSA66as7J6QIElEXG4gKi9cbmV4cG9ydCBjb25zdCBhZG1pbkVudGl0eURlZjogQmV0dGVyQXV0aEVudGl0eURlZiA9IHtcbiAgaWQ6IFwiYWRtaW5cIixcbiAgbmFtZTogXCJBZG1pblwiLFxuICBlbnRpdGllczogW10sXG4gIGFkZGl0aW9uYWxQcm9wczoge1xuICAgIFVzZXI6IFtcbiAgICAgIHtcbiAgICAgICAgbmFtZTogXCJyb2xlXCIsXG4gICAgICAgIHR5cGU6IFwic3RyaW5nXCIsXG4gICAgICAgIG51bGxhYmxlOiB0cnVlLFxuICAgICAgICBkYkRlZmF1bHQ6ICdcInVzZXJcIicsXG4gICAgICAgIGRlc2M6IFwi7IKs7Jqp7J6QIOyXre2VoFwiLFxuICAgICAgfSxcbiAgICAgIHtcbiAgICAgICAgbmFtZTogXCJiYW5uZWRcIixcbiAgICAgICAgdHlwZTogXCJib29sZWFuXCIsXG4gICAgICAgIG51bGxhYmxlOiB0cnVlLFxuICAgICAgICBkYkRlZmF1bHQ6IFwiZmFsc2VcIixcbiAgICAgICAgZGVzYzogXCLssKjri6gg7Jes67aAXCIsXG4gICAgICB9LFxuICAgICAge1xuICAgICAgICBuYW1lOiBcImJhbl9yZWFzb25cIixcbiAgICAgICAgdHlwZTogXCJzdHJpbmdcIixcbiAgICAgICAgbnVsbGFibGU6IHRydWUsXG4gICAgICAgIGRlc2M6IFwi7LCo64uoIOyCrOycoFwiLFxuICAgICAgfSxcbiAgICAgIHtcbiAgICAgICAgbmFtZTogXCJiYW5fZXhwaXJlc1wiLFxuICAgICAgICB0eXBlOiBcImJpZ0ludGVnZXJcIixcbiAgICAgICAgbnVsbGFibGU6IHRydWUsXG4gICAgICAgIGRlc2M6IFwi7LCo64uoIOunjOujjCAoVW5peCB0aW1lc3RhbXApXCIsXG4gICAgICB9LFxuICAgIF0sXG4gICAgU2Vzc2lvbjogW1xuICAgICAge1xuICAgICAgICBuYW1lOiBcImltcGVyc29uYXRlZF9ieVwiLFxuICAgICAgICB0eXBlOiBcInN0cmluZ1wiLFxuICAgICAgICBudWxsYWJsZTogdHJ1ZSxcbiAgICAgICAgZGVzYzogXCLrjIDrpqwg66Gc6re47J247ZWcIOq0gOumrOyekCBJRFwiLFxuICAgICAgfSxcbiAgICBdLFxuICB9LFxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQWlCQSxNQUFhQSxpQkFBc0M7Q0FDakQsSUFBSTtDQUNKLE1BQU07Q0FDTixVQUFVLEVBQUU7Q0FDWixpQkFBaUI7RUFDZixNQUFNO0dBQ0o7SUFDRSxNQUFNO0lBQ04sTUFBTTtJQUNOLFVBQVU7SUFDVixXQUFXO0lBQ1gsTUFBTTtJQUNQO0dBQ0Q7SUFDRSxNQUFNO0lBQ04sTUFBTTtJQUNOLFVBQVU7SUFDVixXQUFXO0lBQ1gsTUFBTTtJQUNQO0dBQ0Q7SUFDRSxNQUFNO0lBQ04sTUFBTTtJQUNOLFVBQVU7SUFDVixNQUFNO0lBQ1A7R0FDRDtJQUNFLE1BQU07SUFDTixNQUFNO0lBQ04sVUFBVTtJQUNWLE1BQU07SUFDUDtHQUNGO0VBQ0QsU0FBUyxDQUNQO0dBQ0UsTUFBTTtHQUNOLE1BQU07R0FDTixVQUFVO0dBQ1YsTUFBTTtHQUNQLENBQ0Y7RUFDRjtDQUNGIn0=
61
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWRtaW4uanMiLCJuYW1lcyI6WyJhZG1pbkVudGl0eURlZjogQmV0dGVyQXV0aEVudGl0eURlZiJdLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9hdXRoL3BsdWdpbnMvZW50aXR5LWRlZmluaXRpb25zL2FkbWluLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHR5cGUgQmV0dGVyQXV0aEVudGl0eURlZiB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbi8qKlxuICogYmV0dGVyLWF1dGggYWRtaW4g7ZSM65+s6re47J24IOyXlO2LsO2LsCDsoJXsnZhcbiAqIGh0dHBzOi8vd3d3LmJldHRlci1hdXRoLmNvbS9kb2NzL3BsdWdpbnMvYWRtaW5cbiAqXG4gKiDqtIDrpqzsnpAg6riw64ql7J2EIOychO2VnCDtlYTrk5zrpbwg7LaU6rCA7ZWp64uI64ukOlxuICpcbiAqIFVzZXIg7YWM7J2067iUOlxuICogLSByb2xlOiDsgqzsmqnsnpAg7Jet7ZWgICjquLDrs7jqsJI6IFwidXNlclwiKVxuICogLSBiYW5uZWQ6IOywqOuLqCDsl6zrtoBcbiAqIC0gYmFuX3JlYXNvbjog7LCo64uoIOyCrOycoFxuICogLSBiYW5fZXhwaXJlczog7LCo64uoIOunjOujjCDsi5zqsIQgKERhdGUpXG4gKlxuICogU2Vzc2lvbiDthYzsnbTruJQ6XG4gKiAtIGltcGVyc29uYXRlZF9ieTog64yA66asIOuhnOq3uOyduO2VnCDqtIDrpqzsnpAgSURcbiAqL1xuZXhwb3J0IGNvbnN0IGFkbWluRW50aXR5RGVmOiBCZXR0ZXJBdXRoRW50aXR5RGVmID0ge1xuICBpZDogXCJhZG1pblwiLFxuICBuYW1lOiBcIkFkbWluXCIsXG4gIGVudGl0aWVzOiBbXSxcbiAgYWRkaXRpb25hbFByb3BzOiB7XG4gICAgVXNlcjogW1xuICAgICAge1xuICAgICAgICBuYW1lOiBcInJvbGVcIixcbiAgICAgICAgdHlwZTogXCJzdHJpbmdcIixcbiAgICAgICAgbnVsbGFibGU6IHRydWUsXG4gICAgICAgIGRiRGVmYXVsdDogJ1widXNlclwiJyxcbiAgICAgICAgZGVzYzogXCLsgqzsmqnsnpAg7Jet7ZWgXCIsXG4gICAgICB9LFxuICAgICAge1xuICAgICAgICBuYW1lOiBcImJhbm5lZFwiLFxuICAgICAgICB0eXBlOiBcImJvb2xlYW5cIixcbiAgICAgICAgbnVsbGFibGU6IHRydWUsXG4gICAgICAgIGRiRGVmYXVsdDogXCJmYWxzZVwiLFxuICAgICAgICBkZXNjOiBcIuywqOuLqCDsl6zrtoBcIixcbiAgICAgIH0sXG4gICAgICB7XG4gICAgICAgIG5hbWU6IFwiYmFuX3JlYXNvblwiLFxuICAgICAgICB0eXBlOiBcInN0cmluZ1wiLFxuICAgICAgICBudWxsYWJsZTogdHJ1ZSxcbiAgICAgICAgZGVzYzogXCLssKjri6gg7IKs7JygXCIsXG4gICAgICB9LFxuICAgICAge1xuICAgICAgICBuYW1lOiBcImJhbl9leHBpcmVzXCIsXG4gICAgICAgIHR5cGU6IFwiZGF0ZVwiLFxuICAgICAgICBudWxsYWJsZTogdHJ1ZSxcbiAgICAgICAgZGVzYzogXCLssKjri6gg66eM66OMXCIsXG4gICAgICB9LFxuICAgIF0sXG4gICAgU2Vzc2lvbjogW1xuICAgICAge1xuICAgICAgICBuYW1lOiBcImltcGVyc29uYXRlZF9ieVwiLFxuICAgICAgICB0eXBlOiBcInN0cmluZ1wiLFxuICAgICAgICBudWxsYWJsZTogdHJ1ZSxcbiAgICAgICAgZGVzYzogXCLrjIDrpqwg66Gc6re47J247ZWcIOq0gOumrOyekCBJRFwiLFxuICAgICAgfSxcbiAgICBdLFxuICB9LFxufTtcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQWlCQSxNQUFhQSxpQkFBc0M7Q0FDakQsSUFBSTtDQUNKLE1BQU07Q0FDTixVQUFVLEVBQUU7Q0FDWixpQkFBaUI7RUFDZixNQUFNO0dBQ0o7SUFDRSxNQUFNO0lBQ04sTUFBTTtJQUNOLFVBQVU7SUFDVixXQUFXO0lBQ1gsTUFBTTtJQUNQO0dBQ0Q7SUFDRSxNQUFNO0lBQ04sTUFBTTtJQUNOLFVBQVU7SUFDVixXQUFXO0lBQ1gsTUFBTTtJQUNQO0dBQ0Q7SUFDRSxNQUFNO0lBQ04sTUFBTTtJQUNOLFVBQVU7SUFDVixNQUFNO0lBQ1A7R0FDRDtJQUNFLE1BQU07SUFDTixNQUFNO0lBQ04sVUFBVTtJQUNWLE1BQU07SUFDUDtHQUNGO0VBQ0QsU0FBUyxDQUNQO0dBQ0UsTUFBTTtHQUNOLE1BQU07R0FDTixVQUFVO0dBQ1YsTUFBTTtHQUNQLENBQ0Y7RUFDRjtDQUNGIn0=
@@ -2,8 +2,8 @@ import { type BetterAuthEntityDef } from "./types";
2
2
  /**
3
3
  * better-auth AuditLog 플러그인 엔티티 정의
4
4
  *
5
- * auth.auditLog: true 활성화되며 audit_events 테이블을 생성합니다.
6
- * - Better Auth dash() 플러그인이 전송하는 이벤트를 1건씩 적재합니다.
5
+ * auth.plugins에 sonamuAuditLog() 추가 audit_events 테이블이 사용됩니다.
6
+ * - sonamuAuditLog 플러그인이 Better Auth databaseHooks/organizationHooks/middleware에서 수신한 이벤트를 1건씩 적재합니다.
7
7
  * - dedupe_key(sha256 hex)로 중복 적재를 방지합니다.
8
8
  *
9
9
  * 생성 방법: pnpm sonamu auth generate --plugins audit-log
@@ -2,8 +2,8 @@
2
2
  /**
3
3
  * better-auth AuditLog 플러그인 엔티티 정의
4
4
  *
5
- * auth.auditLog: true 활성화되며 audit_events 테이블을 생성합니다.
6
- * - Better Auth dash() 플러그인이 전송하는 이벤트를 1건씩 적재합니다.
5
+ * auth.plugins에 sonamuAuditLog() 추가 audit_events 테이블이 사용됩니다.
6
+ * - sonamuAuditLog 플러그인이 Better Auth databaseHooks/organizationHooks/middleware에서 수신한 이벤트를 1건씩 적재합니다.
7
7
  * - dedupe_key(sha256 hex)로 중복 적재를 방지합니다.
8
8
  *
9
9
  * 생성 방법: pnpm sonamu auth generate --plugins audit-log
@@ -288,4 +288,4 @@ const auditLogEntityDef = {
288
288
 
289
289
  //#endregion
290
290
  export { auditLogEntityDef };
291
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"audit-log.js","names":["auditLogEntityDef: BetterAuthEntityDef"],"sources":["../../../../src/auth/plugins/entity-definitions/audit-log.ts"],"sourcesContent":["import { type BetterAuthEntityDef } from \"./types\";\n\n/**\n * better-auth AuditLog 플러그인 엔티티 정의\n *\n * auth.auditLog: true 로 활성화되며 audit_events 테이블을 생성합니다.\n * - Better Auth dash() 플러그인이 전송하는 이벤트를 1건씩 적재합니다.\n * - dedupe_key(sha256 hex)로 중복 적재를 방지합니다.\n *\n * 생성 방법: pnpm sonamu auth generate --plugins audit-log\n */\nexport const auditLogEntityDef: BetterAuthEntityDef = {\n  id: \"audit-log\",\n  name: \"AuditLog\",\n  entities: [\n    {\n      id: \"AuditEvent\",\n      table: \"audit_events\",\n      title: \"감사이벤트\",\n      props: [\n        { name: \"id\", type: \"integer\", desc: \"ID\" },\n        { name: \"source\", type: \"string\", length: 32, desc: \"이벤트 소스\" },\n        { name: \"source_version\", type: \"string\", length: 96, nullable: true, desc: \"소스 버전\" },\n        { name: \"category\", type: \"enum\", id: \"AuditEventCategory\", desc: \"카테고리\" },\n        { name: \"event_type\", type: \"string\", length: 64, desc: \"이벤트 타입\" },\n        { name: \"event_key\", type: \"string\", length: 191, desc: \"이벤트 키\" },\n        { name: \"dedupe_key\", type: \"string\", length: 64, desc: \"중복 제거 키\" },\n        {\n          name: \"actor_user_id\",\n          type: \"string\",\n          length: 191,\n          nullable: true,\n          desc: \"액터 사용자 ID\",\n        },\n        {\n          name: \"subject_user_id\",\n          type: \"string\",\n          length: 191,\n          nullable: true,\n          desc: \"대상 사용자 ID\",\n        },\n        {\n          name: \"organization_id\",\n          type: \"string\",\n          length: 191,\n          nullable: true,\n          desc: \"조직 ID\",\n        },\n        { name: \"team_id\", type: \"string\", length: 191, nullable: true, desc: \"팀 ID\" },\n        { name: \"session_id\", type: \"string\", length: 191, nullable: true, desc: \"세션 ID\" },\n        { name: \"provider_id\", type: \"string\", length: 64, nullable: true, desc: \"프로바이더 ID\" },\n        { name: \"login_method\", type: \"string\", length: 64, nullable: true, desc: \"로그인 방식\" },\n        { name: \"identifier\", type: \"string\", length: 255, nullable: true, desc: \"식별자\" },\n        { name: \"visitor_id\", type: \"string\", length: 191, nullable: true, desc: \"방문자 ID\" },\n        { name: \"reason\", type: \"string\", length: 128, nullable: true, desc: \"사유\" },\n        { name: \"action\", type: \"string\", length: 64, nullable: true, desc: \"액션\" },\n        {\n          name: \"trigger_context\",\n          type: \"string\",\n          length: 64,\n          nullable: true,\n          desc: \"트리거 컨텍스트\",\n        },\n        { name: \"ip_address\", type: \"string\", length: 45, nullable: true, desc: \"IP 주소\" },\n        { name: \"country_code\", type: \"string\", length: 8, nullable: true, desc: \"국가 코드\" },\n        { name: \"country\", type: \"string\", length: 100, nullable: true, desc: \"국가\" },\n        { name: \"city\", type: \"string\", length: 100, nullable: true, desc: \"도시\" },\n        { name: \"user_agent\", type: \"string\", nullable: true, desc: \"User-Agent\" },\n        { name: \"payload_json\", type: \"json\", id: \"AuditEventPayload\", desc: \"원본 payload\" },\n        { name: \"occurred_at\", type: \"date\", desc: \"발생 시각\" },\n        {\n          name: \"ingested_at\",\n          type: \"date\",\n          dbDefault: \"CURRENT_TIMESTAMP\",\n          desc: \"적재 시각\",\n        },\n      ],\n      indexes: [\n        {\n          type: \"unique\",\n          name: \"audit_events_dedupe_key_unique\",\n          columns: [{ name: \"dedupe_key\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_occurred_at_index\",\n          columns: [{ name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_event_type_occurred_at_index\",\n          columns: [{ name: \"event_type\" }, { name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_subject_user_id_occurred_at_index\",\n          columns: [{ name: \"subject_user_id\" }, { name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_actor_user_id_occurred_at_index\",\n          columns: [{ name: \"actor_user_id\" }, { name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_organization_id_occurred_at_index\",\n          columns: [{ name: \"organization_id\" }, { name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_team_id_occurred_at_index\",\n          columns: [{ name: \"team_id\" }, { name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_session_id_index\",\n          columns: [{ name: \"session_id\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_reason_occurred_at_index\",\n          columns: [{ name: \"reason\" }, { name: \"occurred_at\" }],\n        },\n      ],\n      subsets: {\n        A: [\n          \"id\",\n          \"source\",\n          \"source_version\",\n          \"category\",\n          \"event_type\",\n          \"event_key\",\n          \"dedupe_key\",\n          \"actor_user_id\",\n          \"subject_user_id\",\n          \"organization_id\",\n          \"team_id\",\n          \"session_id\",\n          \"provider_id\",\n          \"login_method\",\n          \"identifier\",\n          \"visitor_id\",\n          \"reason\",\n          \"action\",\n          \"trigger_context\",\n          \"ip_address\",\n          \"country_code\",\n          \"country\",\n          \"city\",\n          \"user_agent\",\n          \"payload_json\",\n          \"occurred_at\",\n          \"ingested_at\",\n        ],\n      },\n      enums: {\n        AuditEventOrderBy: { \"id-desc\": \"ID최신순\" },\n        AuditEventSearchField: { id: \"ID\" },\n        AuditEventCategory: {\n          user: \"사용자\",\n          session: \"세션\",\n          account: \"계정\",\n          verification: \"인증\",\n          organization: \"조직\",\n          security: \"보안\",\n        },\n      },\n    },\n  ],\n  additionalProps: {},\n};\n"],"mappings":"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"}
291
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"audit-log.js","names":["auditLogEntityDef: BetterAuthEntityDef"],"sources":["../../../../src/auth/plugins/entity-definitions/audit-log.ts"],"sourcesContent":["import { type BetterAuthEntityDef } from \"./types\";\n\n/**\n * better-auth AuditLog 플러그인 엔티티 정의\n *\n * auth.plugins에 sonamuAuditLog() 추가 시 audit_events 테이블이 사용됩니다.\n * - sonamuAuditLog 플러그인이 Better Auth databaseHooks/organizationHooks/middleware에서 수신한 이벤트를 1건씩 적재합니다.\n * - dedupe_key(sha256 hex)로 중복 적재를 방지합니다.\n *\n * 생성 방법: pnpm sonamu auth generate --plugins audit-log\n */\nexport const auditLogEntityDef: BetterAuthEntityDef = {\n  id: \"audit-log\",\n  name: \"AuditLog\",\n  entities: [\n    {\n      id: \"AuditEvent\",\n      table: \"audit_events\",\n      title: \"감사이벤트\",\n      props: [\n        { name: \"id\", type: \"integer\", desc: \"ID\" },\n        { name: \"source\", type: \"string\", length: 32, desc: \"이벤트 소스\" },\n        { name: \"source_version\", type: \"string\", length: 96, nullable: true, desc: \"소스 버전\" },\n        { name: \"category\", type: \"enum\", id: \"AuditEventCategory\", desc: \"카테고리\" },\n        { name: \"event_type\", type: \"string\", length: 64, desc: \"이벤트 타입\" },\n        { name: \"event_key\", type: \"string\", length: 191, desc: \"이벤트 키\" },\n        { name: \"dedupe_key\", type: \"string\", length: 64, desc: \"중복 제거 키\" },\n        {\n          name: \"actor_user_id\",\n          type: \"string\",\n          length: 191,\n          nullable: true,\n          desc: \"액터 사용자 ID\",\n        },\n        {\n          name: \"subject_user_id\",\n          type: \"string\",\n          length: 191,\n          nullable: true,\n          desc: \"대상 사용자 ID\",\n        },\n        {\n          name: \"organization_id\",\n          type: \"string\",\n          length: 191,\n          nullable: true,\n          desc: \"조직 ID\",\n        },\n        { name: \"team_id\", type: \"string\", length: 191, nullable: true, desc: \"팀 ID\" },\n        { name: \"session_id\", type: \"string\", length: 191, nullable: true, desc: \"세션 ID\" },\n        { name: \"provider_id\", type: \"string\", length: 64, nullable: true, desc: \"프로바이더 ID\" },\n        { name: \"login_method\", type: \"string\", length: 64, nullable: true, desc: \"로그인 방식\" },\n        { name: \"identifier\", type: \"string\", length: 255, nullable: true, desc: \"식별자\" },\n        { name: \"visitor_id\", type: \"string\", length: 191, nullable: true, desc: \"방문자 ID\" },\n        { name: \"reason\", type: \"string\", length: 128, nullable: true, desc: \"사유\" },\n        { name: \"action\", type: \"string\", length: 64, nullable: true, desc: \"액션\" },\n        {\n          name: \"trigger_context\",\n          type: \"string\",\n          length: 64,\n          nullable: true,\n          desc: \"트리거 컨텍스트\",\n        },\n        { name: \"ip_address\", type: \"string\", length: 45, nullable: true, desc: \"IP 주소\" },\n        { name: \"country_code\", type: \"string\", length: 8, nullable: true, desc: \"국가 코드\" },\n        { name: \"country\", type: \"string\", length: 100, nullable: true, desc: \"국가\" },\n        { name: \"city\", type: \"string\", length: 100, nullable: true, desc: \"도시\" },\n        { name: \"user_agent\", type: \"string\", nullable: true, desc: \"User-Agent\" },\n        { name: \"payload_json\", type: \"json\", id: \"AuditEventPayload\", desc: \"원본 payload\" },\n        { name: \"occurred_at\", type: \"date\", desc: \"발생 시각\" },\n        {\n          name: \"ingested_at\",\n          type: \"date\",\n          dbDefault: \"CURRENT_TIMESTAMP\",\n          desc: \"적재 시각\",\n        },\n      ],\n      indexes: [\n        {\n          type: \"unique\",\n          name: \"audit_events_dedupe_key_unique\",\n          columns: [{ name: \"dedupe_key\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_occurred_at_index\",\n          columns: [{ name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_event_type_occurred_at_index\",\n          columns: [{ name: \"event_type\" }, { name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_subject_user_id_occurred_at_index\",\n          columns: [{ name: \"subject_user_id\" }, { name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_actor_user_id_occurred_at_index\",\n          columns: [{ name: \"actor_user_id\" }, { name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_organization_id_occurred_at_index\",\n          columns: [{ name: \"organization_id\" }, { name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_team_id_occurred_at_index\",\n          columns: [{ name: \"team_id\" }, { name: \"occurred_at\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_session_id_index\",\n          columns: [{ name: \"session_id\" }],\n        },\n        {\n          type: \"index\",\n          name: \"audit_events_reason_occurred_at_index\",\n          columns: [{ name: \"reason\" }, { name: \"occurred_at\" }],\n        },\n      ],\n      subsets: {\n        A: [\n          \"id\",\n          \"source\",\n          \"source_version\",\n          \"category\",\n          \"event_type\",\n          \"event_key\",\n          \"dedupe_key\",\n          \"actor_user_id\",\n          \"subject_user_id\",\n          \"organization_id\",\n          \"team_id\",\n          \"session_id\",\n          \"provider_id\",\n          \"login_method\",\n          \"identifier\",\n          \"visitor_id\",\n          \"reason\",\n          \"action\",\n          \"trigger_context\",\n          \"ip_address\",\n          \"country_code\",\n          \"country\",\n          \"city\",\n          \"user_agent\",\n          \"payload_json\",\n          \"occurred_at\",\n          \"ingested_at\",\n        ],\n      },\n      enums: {\n        AuditEventOrderBy: { \"id-desc\": \"ID최신순\" },\n        AuditEventSearchField: { id: \"ID\" },\n        AuditEventCategory: {\n          user: \"사용자\",\n          session: \"세션\",\n          account: \"계정\",\n          verification: \"인증\",\n          organization: \"조직\",\n          security: \"보안\",\n        },\n      },\n    },\n  ],\n  additionalProps: {},\n};\n"],"mappings":"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"}
@@ -826,8 +826,8 @@ export declare const admin: (options?: AdminOptions) => {
826
826
  $Infer: {
827
827
  body: {
828
828
  permissions: {
829
- readonly user?: ("update" | "delete" | "create" | "get" | "set-password" | "list" | "set-role" | "ban" | "impersonate" | "impersonate-admins")[] | undefined;
830
- readonly session?: ("delete" | "revoke" | "list")[] | undefined;
829
+ readonly user?: ("update" | "delete" | "create" | "get" | "list" | "set-role" | "ban" | "impersonate" | "impersonate-admins" | "set-password")[] | undefined;
830
+ readonly session?: ("delete" | "list" | "revoke")[] | undefined;
831
831
  };
832
832
  } & {
833
833
  userId?: string | undefined;
@@ -316,7 +316,7 @@ export declare const sso: (options?: SSOOptions) => {
316
316
  providerId: string;
317
317
  organizationId?: string | undefined;
318
318
  domain: string;
319
- }, "samlConfig" | "oidcConfig">>;
319
+ }, "oidcConfig" | "samlConfig">>;
320
320
  signInSSO: ReturnType<(options?: SSOOptions) => import("better-auth").StrictEndpoint<"/sign-in/sso", {
321
321
  method: "POST";
322
322
  body: import("zod").ZodObject<{
@@ -1 +1 @@
1
- {"version":3,"file":"fixture.d.ts","sourceRoot":"","sources":["../../src/bin/fixture.ts"],"names":[],"mappings":"AAQA,OAAO,EAAE,KAAK,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAGrE,UAAU,qBAAqB;IAC7B,CAAC,CAAC,EAAE,MAAM,EAAE,CAAC;IACb,GAAG,CAAC,EAAE,OAAO,CAAC;IACd,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,CAAC,EAAE,oBAAoB,CAAC;IAChC,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,UAAU,CAAC,EAAE,OAAO,CAAC;CACtB;AAED;;GAEG;AACH,wBAAsB,iBAAiB,CAAC,OAAO,EAAE,qBAAqB,iBAyLrE;AAED;;;GAGG;AACH,wBAAsB,mBAAmB,CAAC,OAAO,EAAE,qBAAqB,iBAgEvE;AAED;;;GAGG;AACH,wBAAsB,qBAAqB,CAAC,OAAO,EAAE,qBAAqB,iBAgDzE"}
1
+ {"version":3,"file":"fixture.d.ts","sourceRoot":"","sources":["../../src/bin/fixture.ts"],"names":[],"mappings":"AAQA,OAAO,EAAE,KAAK,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAGrE,UAAU,qBAAqB;IAC7B,CAAC,CAAC,EAAE,MAAM,EAAE,CAAC;IACb,GAAG,CAAC,EAAE,OAAO,CAAC;IACd,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,CAAC,EAAE,oBAAoB,CAAC;IAChC,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,UAAU,CAAC,EAAE,OAAO,CAAC;CACtB;AAmBD;;GAEG;AACH,wBAAsB,iBAAiB,CAAC,OAAO,EAAE,qBAAqB,iBAuTrE;AAED;;;GAGG;AACH,wBAAsB,mBAAmB,CAAC,OAAO,EAAE,qBAAqB,iBAgEvE;AAED;;;GAGG;AACH,wBAAsB,qBAAqB,CAAC,OAAO,EAAE,qBAAqB,iBAgDzE"}
@@ -15,6 +15,19 @@ init_entity_manager();
15
15
  init_data_explorer();
16
16
  init_fixture_generator();
17
17
  /**
18
+ * username을 일반적인 규칙(영문자 시작, 영문자/숫자만, 1-20자)에 맞도록 정규화합니다.
19
+ */
20
+ function sanitizeUsername(raw) {
21
+ let username = raw.toLowerCase().replace(/[^a-z0-9]/g, "");
22
+ if (username.length === 0) {
23
+ username = "user";
24
+ }
25
+ if (/^[0-9]/.test(username)) {
26
+ username = `u${username}`;
27
+ }
28
+ return username.slice(0, 20);
29
+ }
30
+ /**
18
31
  * fixture gen 명령어 - cone 메타데이터를 활용하여 자동으로 fixture를 생성합니다.
19
32
  */
20
33
  async function fixtureGenCommand(options) {
@@ -63,6 +76,103 @@ async function fixtureGenCommand(options) {
63
76
  }
64
77
  count = result.count;
65
78
  }
79
+ const hasUser = entityNames.includes("User");
80
+ if (hasUser) {
81
+ const userModeResult = await prompts({
82
+ type: "select",
83
+ name: "userMode",
84
+ message: "User 엔티티가 포함되어 있습니다. 생성 방식을 선택하세요:",
85
+ choices: [{
86
+ title: "1. 로그인 가능한 사용자 fixture 생성",
87
+ value: "login"
88
+ }, {
89
+ title: "2. 확인용 데이터(로그인 불가) 생성",
90
+ value: "dummy"
91
+ }]
92
+ });
93
+ if (!userModeResult.userMode) {
94
+ console.log(chalk.yellow("취소되었습니다."));
95
+ return;
96
+ }
97
+ if (userModeResult.userMode === "login") {
98
+ let useLLM$1 = options["use-llm"] ?? false;
99
+ if (!options["use-llm"]) {
100
+ const llmResult = await prompts({
101
+ type: "confirm",
102
+ name: "useLLM",
103
+ message: "LLM으로 더 현실적인 데이터를 생성할까요? (fixtureHint 기반, ANTHROPIC_API_KEY 필요)",
104
+ initial: false
105
+ });
106
+ useLLM$1 = llmResult.useLLM ?? false;
107
+ }
108
+ const enableLLMCache$1 = !options["no-cache"];
109
+ const DEFAULT_PASSWORD = "Test1234!";
110
+ const sourceDb = DB.getDB("r");
111
+ const generator$1 = new FixtureGenerator(sourceDb, sourceDb, "production_master", EntityManager, {
112
+ useLLM: useLLM$1,
113
+ enableLLMCache: enableLLMCache$1
114
+ });
115
+ const createdCredentials = [];
116
+ const basePath = Sonamu.config.server.auth?.basePath ?? "/api/auth";
117
+ if (useLLM$1) {
118
+ console.log(chalk.cyan(`\nLLM 모드로 로그인 가능한 사용자 ${count}명 생성 중... (캐싱: ${enableLLMCache$1 ? "ON" : "OFF"})`));
119
+ } else {
120
+ console.log(chalk.cyan(`\n로그인 가능한 사용자 ${count}명 생성 중...`));
121
+ }
122
+ for (let i = 0; i < count; i++) {
123
+ const userData = await generator$1.generate("User");
124
+ const name = String(userData.name ?? "");
125
+ const email = String(userData.email ?? "");
126
+ const username = sanitizeUsername(String(userData.username ?? ""));
127
+ const displayUsername = userData.display_username !== undefined ? String(userData.display_username) : undefined;
128
+ const body = {
129
+ name,
130
+ email,
131
+ username,
132
+ password: DEFAULT_PASSWORD
133
+ };
134
+ if (displayUsername !== undefined) {
135
+ body.display_username = displayUsername;
136
+ }
137
+ const req = new Request(`http://localhost${basePath}/sign-up/email`, {
138
+ method: "POST",
139
+ headers: { "Content-Type": "application/json" },
140
+ body: JSON.stringify(body)
141
+ });
142
+ const response = await Sonamu.auth.handler(req);
143
+ if (!response.ok) {
144
+ const responseData = await response.json();
145
+ const code = typeof responseData.code === "string" ? responseData.code : undefined;
146
+ if (code === "USER_ALREADY_EXISTS") {
147
+ console.log(chalk.yellow(` ⚠️ ${email} 이미 존재 - 건너뜁니다.`));
148
+ continue;
149
+ }
150
+ console.log(chalk.red(` ❌ ${email} 생성 실패: ${JSON.stringify(responseData)}`));
151
+ continue;
152
+ }
153
+ createdCredentials.push({
154
+ email,
155
+ password: DEFAULT_PASSWORD
156
+ });
157
+ console.log(chalk.green(` ✅ ${email} 생성 완료`));
158
+ }
159
+ if (createdCredentials.length > 0) {
160
+ const emails = createdCredentials.map((c) => c.email);
161
+ await DB.getDB("w")("users").whereIn("email", emails).update({ email_verified: true });
162
+ console.log(chalk.green(`\nemail_verified = true 업데이트 완료`));
163
+ }
164
+ if (useLLM$1) {
165
+ const stats = generator$1.getLLMCacheStats();
166
+ console.log(chalk.cyan(`[LLM Cache] 캐시 크기: ${stats.size}`));
167
+ }
168
+ console.log(chalk.green(`\n✅ ${createdCredentials.length}명의 로그인 가능한 사용자 생성 완료`));
169
+ if (createdCredentials.length > 0) {
170
+ console.log("\n생성된 계정 목록:");
171
+ console.table(createdCredentials);
172
+ }
173
+ return;
174
+ }
175
+ }
66
176
  let saveTarget = options["save-to"] || "db";
67
177
  if (!options["save-to"]) {
68
178
  const result = await prompts({
@@ -272,4 +382,4 @@ async function fixtureExploreCommand(options) {
272
382
 
273
383
  //#endregion
274
384
  export { fixtureExploreCommand, fixtureFetchCommand, fixtureGenCommand };
275
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"fixture.js","names":["entityNames: string[]","filename: string","strategy: DataExplorerStrategy"],"sources":["../../src/bin/fixture.ts"],"sourcesContent":["import chalk from \"chalk\";\nimport prompts from \"prompts\";\n\nimport { Sonamu } from \"../api/sonamu\";\nimport { DB } from \"../database/db\";\nimport { createKnexInstance } from \"../database/knex\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { DataExplorer } from \"../testing/data-explorer\";\nimport { type DataExplorerStrategy } from \"../testing/data-explorer\";\nimport { FixtureGenerator } from \"../testing/fixture-generator\";\n\ninterface FixtureCommandOptions {\n  _?: string[];\n  all?: boolean;\n  include?: string;\n  exclude?: string;\n  count?: string;\n  \"save-to\"?: string;\n  strategy?: DataExplorerStrategy;\n  limit?: string;\n  \"use-llm\"?: boolean;\n  \"no-cache\"?: boolean;\n}\n\n/**\n * fixture gen 명령어 - cone 메타데이터를 활용하여 자동으로 fixture를 생성합니다.\n */\nexport async function fixtureGenCommand(options: FixtureCommandOptions) {\n  try {\n    if (!EntityManager.isAutoloaded) {\n      await EntityManager.autoload();\n    }\n\n    let entityNames: string[];\n\n    if (options.all) {\n      entityNames = EntityManager.getAllIds();\n      if (options.exclude) {\n        const excludeList = options.exclude.split(\",\").map((s: string) => s.trim());\n        entityNames = entityNames.filter((name) => !excludeList.includes(name));\n      }\n    } else if (options.include) {\n      entityNames = options.include.split(\",\").map((s: string) => s.trim());\n    } else {\n      const result = await prompts({\n        type: \"multiselect\",\n        name: \"entities\",\n        message: \"Fixture를 생성할 Entity를 선택하세요:\",\n        choices: EntityManager.getAllIds().map((id) => ({ title: id, value: id })),\n        min: 1,\n      });\n\n      if (!result.entities || result.entities.length === 0) {\n        console.log(chalk.yellow(\"취소되었습니다.\"));\n        return;\n      }\n\n      entityNames = result.entities;\n    }\n\n    let count = options.count ? Number.parseInt(options.count, 10) : 5;\n    if (!options.count) {\n      const result = await prompts({\n        type: \"number\",\n        name: \"count\",\n        message: \"각 Entity별 생성 개수:\",\n        initial: 5,\n        min: 1,\n      });\n\n      if (!result.count) {\n        console.log(chalk.yellow(\"취소되었습니다.\"));\n        return;\n      }\n\n      count = result.count;\n    }\n\n    let saveTarget = options[\"save-to\"] || \"db\";\n    if (!options[\"save-to\"]) {\n      const result = await prompts({\n        type: \"select\",\n        name: \"saveTarget\",\n        message: \"저장 방식:\",\n        choices: [\n          { title: \"Fixture DB에 저장\", value: \"db\" },\n          { title: \"파일로 저장 (자동 파일명)\", value: \"file\" },\n          { title: \"파일로 저장 (파일명 지정)\", value: \"file:custom\" },\n          { title: \"저장 안 함 (출력만)\", value: \"none\" },\n        ],\n      });\n\n      saveTarget = result.saveTarget;\n\n      if (saveTarget === \"file:custom\") {\n        const filenameResult = await prompts({\n          type: \"text\",\n          name: \"filename\",\n          message: \"파일명:\",\n          initial: \"fixtures.json\",\n        });\n\n        if (!filenameResult.filename) {\n          console.log(chalk.yellow(\"취소되었습니다.\"));\n          return;\n        }\n\n        saveTarget = `file:${filenameResult.filename}`;\n      }\n    }\n\n    // LLM 사용 여부 결정\n    let useLLM = options[\"use-llm\"] ?? false;\n    if (!options[\"use-llm\"]) {\n      const llmResult = await prompts({\n        type: \"confirm\",\n        name: \"useLLM\",\n        message:\n          \"LLM으로 더 현실적인 데이터를 생성할까요? (fixtureHint 기반, ANTHROPIC_API_KEY 필요)\",\n        initial: false,\n      });\n      useLLM = llmResult.useLLM ?? false;\n    }\n\n    const enableLLMCache = !options[\"no-cache\"];\n\n    // fixture gen: fixture DB 내에서 참조 관계를 해결하고 저장합니다\n    const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);\n    const generator = new FixtureGenerator(fixtureDb, fixtureDb, \"fixture\", EntityManager, {\n      useLLM,\n      enableLLMCache,\n    });\n\n    if (useLLM) {\n      console.log(\n        chalk.cyan(\n          `\\nLLM 모드로 ${entityNames.join(\", \")} 생성 중... (캐싱: ${enableLLMCache ? \"ON\" : \"OFF\"})`,\n        ),\n      );\n    } else {\n      console.log(chalk.cyan(`\\n${entityNames.join(\", \")} 생성 중...`));\n    }\n\n    const specs = entityNames.map((entityName) => ({\n      entity: entityName,\n      count,\n      overrides: {},\n    }));\n\n    const results = await generator.generateBatch(specs);\n\n    if (useLLM) {\n      const stats = generator.getLLMCacheStats();\n      console.log(chalk.cyan(`[LLM Cache] 캐시 크기: ${stats.size}`));\n    }\n\n    console.log(chalk.green(`\\n✅ ${results.length}개 fixture 생성 완료`));\n\n    if (saveTarget === \"none\") {\n      console.log(JSON.stringify(results, null, 2));\n    } else if (saveTarget === \"db\") {\n      // generateBatch가 이미 DB에 저장했으므로 별도 저장이 불필요합니다.\n      console.log(chalk.green(\"Fixture DB에 저장되었습니다.\"));\n    } else if (saveTarget.startsWith(\"file\")) {\n      const fs = await import(\"node:fs/promises\");\n      const path = await import(\"node:path\");\n\n      const fixturesDir = path.join(process.cwd(), \"test\", \"fixtures\");\n      try {\n        await fs.access(fixturesDir);\n      } catch {\n        await fs.mkdir(fixturesDir, { recursive: true });\n      }\n\n      for (const entityName of entityNames) {\n        const entityResults = results.filter((r) => r.entityId === entityName);\n\n        if (entityResults.length === 0) {\n          continue;\n        }\n\n        let filename: string;\n        if (saveTarget === \"file\") {\n          const entity = EntityManager.get(entityName);\n          filename = `${entity.table}.json`;\n        } else {\n          filename = saveTarget.replace(\"file:\", \"\");\n        }\n\n        const filepath = path.join(fixturesDir, filename);\n        await fs.writeFile(\n          filepath,\n          JSON.stringify(\n            entityResults.map((r) => r.data),\n            null,\n            2,\n          ),\n        );\n        console.log(chalk.green(`✅ ${filepath} 저장 완료`));\n      }\n    }\n  } catch (error) {\n    console.error(\n      chalk.red(\n        \"Fixture 생성 중 오류가 발생했습니다.\\n\" +\n          \"원인: Entity 정의나 DB 연결을 확인해주세요.\\n\" +\n          \"자세한 내용:\",\n      ),\n      error,\n    );\n    throw error;\n  }\n}\n\n/**\n * fixture fetch 명령어 - 실제 운영 DB에서 데이터를 가져와 fixture로 저장합니다.\n * 관계된 데이터도 함께 가져오므로 현실적인 테스트 데이터를 확보할 수 있습니다.\n */\nexport async function fixtureFetchCommand(options: FixtureCommandOptions) {\n  try {\n    if (!EntityManager.isAutoloaded) {\n      await EntityManager.autoload();\n    }\n\n    let entityNames: string[];\n\n    if (options.all) {\n      entityNames = EntityManager.getAllIds();\n      if (options.exclude) {\n        const excludeList = options.exclude.split(\",\").map((s: string) => s.trim());\n        entityNames = entityNames.filter((name) => !excludeList.includes(name));\n      }\n    } else if (options.include) {\n      entityNames = options.include.split(\",\").map((s: string) => s.trim());\n    } else {\n      const result = await prompts({\n        type: \"multiselect\",\n        name: \"entities\",\n        message: \"Import할 Entity를 선택하세요:\",\n        choices: EntityManager.getAllIds().map((id) => ({ title: id, value: id })),\n        min: 1,\n      });\n\n      if (!result.entities || result.entities.length === 0) {\n        console.log(chalk.yellow(\"취소되었습니다.\"));\n        return;\n      }\n\n      entityNames = result.entities;\n    }\n\n    const strategy: DataExplorerStrategy = options.strategy ?? \"recent\";\n    const limit = options.limit ? Number.parseInt(options.limit, 10) : 10;\n\n    // fixture fetch: production 데이터를 fixture DB로 import합니다\n    const sourceDb = DB.getDB(\"r\"); // production_master (또는 development_master)\n    const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);\n    const generator = new FixtureGenerator(sourceDb, fixtureDb, \"fixture\", EntityManager);\n\n    console.log(chalk.cyan(`\\n${entityNames.join(\", \")} import 중...`));\n\n    for (const entityName of entityNames) {\n      const results = await generator.importFromSource(entityName, {\n        strategy,\n        limit,\n        includeRelations: true,\n        maxDepth: 2,\n      });\n\n      console.log(chalk.green(`✅ ${entityName}: ${results.length}개 import 완료`));\n    }\n  } catch (error) {\n    console.error(\n      chalk.red(\n        \"실제 DB에서 데이터를 가져오는 중 오류가 발생했습니다.\\n\" +\n          \"원인: 소스 DB 연결 설정(sonamu.config.ts)이나 Entity 관계 정의를 확인해주세요.\\n\" +\n          \"자세한 내용:\",\n      ),\n      error,\n    );\n    throw error;\n  }\n}\n\n/**\n * fixture explore 명령어 - DB의 실제 데이터를 조회하여 확인합니다.\n * 저장하지 않고 조회만 하므로 데이터를 빠르게 확인할 때 유용합니다.\n */\nexport async function fixtureExploreCommand(options: FixtureCommandOptions) {\n  try {\n    if (!EntityManager.isAutoloaded) {\n      await EntityManager.autoload();\n    }\n\n    let entityName = options.include;\n\n    if (!entityName) {\n      const result = await prompts({\n        type: \"select\",\n        name: \"entity\",\n        message: \"탐색할 Entity:\",\n        choices: EntityManager.getAllIds().map((id) => ({ title: id, value: id })),\n      });\n\n      if (!result.entity) {\n        console.log(chalk.yellow(\"취소되었습니다.\"));\n        return;\n      }\n\n      entityName = result.entity;\n    }\n\n    if (!entityName) {\n      throw new Error(\"Entity name is required\");\n    }\n\n    const strategy: DataExplorerStrategy = options.strategy ?? \"sample\";\n    const limit = options.limit ? Number.parseInt(options.limit, 10) : 10;\n\n    const db = DB.getDB(\"r\");\n    const explorer = new DataExplorer(db, EntityManager);\n    const data = await explorer.explore(entityName, { strategy, limit });\n\n    console.log(chalk.cyan(`\\n${entityName} ${data.length}개 조회 완료:`));\n    console.table(data);\n  } catch (error) {\n    console.error(\n      chalk.red(\n        \"데이터 조회 중 오류가 발생했습니다.\\n\" +\n          \"원인: DB 연결이나 Entity 정의를 확인해주세요.\\n\" +\n          \"자세한 내용:\",\n      ),\n      error,\n    );\n    throw error;\n  }\n}\n"],"mappings":"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"}
385
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"fixture.js","names":["entityNames: string[]","useLLM","enableLLMCache","generator","createdCredentials: Array<{ email: string; password: string }>","body: Record<string, unknown>","filename: string","strategy: DataExplorerStrategy"],"sources":["../../src/bin/fixture.ts"],"sourcesContent":["import chalk from \"chalk\";\nimport prompts from \"prompts\";\n\nimport { Sonamu } from \"../api/sonamu\";\nimport { DB } from \"../database/db\";\nimport { createKnexInstance } from \"../database/knex\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { DataExplorer } from \"../testing/data-explorer\";\nimport { type DataExplorerStrategy } from \"../testing/data-explorer\";\nimport { FixtureGenerator } from \"../testing/fixture-generator\";\n\ninterface FixtureCommandOptions {\n  _?: string[];\n  all?: boolean;\n  include?: string;\n  exclude?: string;\n  count?: string;\n  \"save-to\"?: string;\n  strategy?: DataExplorerStrategy;\n  limit?: string;\n  \"use-llm\"?: boolean;\n  \"no-cache\"?: boolean;\n}\n\n/**\n * username을 일반적인 규칙(영문자 시작, 영문자/숫자만, 1-20자)에 맞도록 정규화합니다.\n */\nfunction sanitizeUsername(raw: string): string {\n  // 소문자 변환 후 영문자/숫자 외 문자 제거\n  let username = raw.toLowerCase().replace(/[^a-z0-9]/g, \"\");\n  if (username.length === 0) {\n    username = \"user\";\n  }\n  // 첫 글자가 숫자면 'u' 접두어 추가\n  if (/^[0-9]/.test(username)) {\n    username = `u${username}`;\n  }\n  // 20자 초과 시 truncate\n  return username.slice(0, 20);\n}\n\n/**\n * fixture gen 명령어 - cone 메타데이터를 활용하여 자동으로 fixture를 생성합니다.\n */\nexport async function fixtureGenCommand(options: FixtureCommandOptions) {\n  try {\n    if (!EntityManager.isAutoloaded) {\n      await EntityManager.autoload();\n    }\n\n    let entityNames: string[];\n\n    if (options.all) {\n      entityNames = EntityManager.getAllIds();\n      if (options.exclude) {\n        const excludeList = options.exclude.split(\",\").map((s: string) => s.trim());\n        entityNames = entityNames.filter((name) => !excludeList.includes(name));\n      }\n    } else if (options.include) {\n      entityNames = options.include.split(\",\").map((s: string) => s.trim());\n    } else {\n      const result = await prompts({\n        type: \"multiselect\",\n        name: \"entities\",\n        message: \"Fixture를 생성할 Entity를 선택하세요:\",\n        choices: EntityManager.getAllIds().map((id) => ({ title: id, value: id })),\n        min: 1,\n      });\n\n      if (!result.entities || result.entities.length === 0) {\n        console.log(chalk.yellow(\"취소되었습니다.\"));\n        return;\n      }\n\n      entityNames = result.entities;\n    }\n\n    let count = options.count ? Number.parseInt(options.count, 10) : 5;\n    if (!options.count) {\n      const result = await prompts({\n        type: \"number\",\n        name: \"count\",\n        message: \"각 Entity별 생성 개수:\",\n        initial: 5,\n        min: 1,\n      });\n\n      if (!result.count) {\n        console.log(chalk.yellow(\"취소되었습니다.\"));\n        return;\n      }\n\n      count = result.count;\n    }\n\n    // User 엔티티가 포함된 경우: 로그인 가능/확인용 분기 선택\n    const hasUser = entityNames.includes(\"User\");\n\n    if (hasUser) {\n      const userModeResult = await prompts({\n        type: \"select\",\n        name: \"userMode\",\n        message: \"User 엔티티가 포함되어 있습니다. 생성 방식을 선택하세요:\",\n        choices: [\n          { title: \"1. 로그인 가능한 사용자 fixture 생성\", value: \"login\" },\n          { title: \"2. 확인용 데이터(로그인 불가) 생성\", value: \"dummy\" },\n        ],\n      });\n\n      if (!userModeResult.userMode) {\n        console.log(chalk.yellow(\"취소되었습니다.\"));\n        return;\n      }\n\n      if (userModeResult.userMode === \"login\") {\n        // LLM 사용 여부\n        let useLLM = options[\"use-llm\"] ?? false;\n        if (!options[\"use-llm\"]) {\n          const llmResult = await prompts({\n            type: \"confirm\",\n            name: \"useLLM\",\n            message:\n              \"LLM으로 더 현실적인 데이터를 생성할까요? (fixtureHint 기반, ANTHROPIC_API_KEY 필요)\",\n            initial: false,\n          });\n          useLLM = llmResult.useLLM ?? false;\n        }\n\n        const enableLLMCache = !options[\"no-cache\"];\n        const DEFAULT_PASSWORD = \"Test1234!\";\n\n        // 로그인 가능 경로에서는 sourceDb로 development_master 사용\n        const sourceDb = DB.getDB(\"r\");\n        const generator = new FixtureGenerator(\n          sourceDb,\n          sourceDb,\n          \"production_master\",\n          EntityManager,\n          { useLLM, enableLLMCache },\n        );\n\n        const createdCredentials: Array<{ email: string; password: string }> = [];\n        const basePath = Sonamu.config.server.auth?.basePath ?? \"/api/auth\";\n\n        if (useLLM) {\n          console.log(\n            chalk.cyan(\n              `\\nLLM 모드로 로그인 가능한 사용자 ${count}명 생성 중... (캐싱: ${enableLLMCache ? \"ON\" : \"OFF\"})`,\n            ),\n          );\n        } else {\n          console.log(chalk.cyan(`\\n로그인 가능한 사용자 ${count}명 생성 중...`));\n        }\n\n        for (let i = 0; i < count; i++) {\n          const userData = await generator.generate(\"User\");\n\n          const name = String(userData.name ?? \"\");\n          const email = String(userData.email ?? \"\");\n          const username = sanitizeUsername(String(userData.username ?? \"\"));\n          const displayUsername =\n            userData.display_username !== undefined ? String(userData.display_username) : undefined;\n\n          const body: Record<string, unknown> = {\n            name,\n            email,\n            username,\n            password: DEFAULT_PASSWORD,\n          };\n          if (displayUsername !== undefined) {\n            body.display_username = displayUsername;\n          }\n\n          const req = new Request(`http://localhost${basePath}/sign-up/email`, {\n            method: \"POST\",\n            headers: { \"Content-Type\": \"application/json\" },\n            body: JSON.stringify(body),\n          });\n\n          const response = await Sonamu.auth.handler(req);\n\n          if (!response.ok) {\n            const responseData = (await response.json()) as Record<string, unknown>;\n            const code = typeof responseData.code === \"string\" ? responseData.code : undefined;\n            if (code === \"USER_ALREADY_EXISTS\") {\n              console.log(chalk.yellow(`  ⚠️  ${email} 이미 존재 - 건너뜁니다.`));\n              continue;\n            }\n            console.log(chalk.red(`  ❌ ${email} 생성 실패: ${JSON.stringify(responseData)}`));\n            continue;\n          }\n\n          createdCredentials.push({ email, password: DEFAULT_PASSWORD });\n          console.log(chalk.green(`  ✅ ${email} 생성 완료`));\n        }\n\n        // email_verified = true 직접 업데이트 (dev 편의)\n        if (createdCredentials.length > 0) {\n          const emails = createdCredentials.map((c) => c.email);\n          await DB.getDB(\"w\")(\"users\").whereIn(\"email\", emails).update({ email_verified: true });\n          console.log(chalk.green(`\\nemail_verified = true 업데이트 완료`));\n        }\n\n        if (useLLM) {\n          const stats = generator.getLLMCacheStats();\n          console.log(chalk.cyan(`[LLM Cache] 캐시 크기: ${stats.size}`));\n        }\n\n        console.log(\n          chalk.green(`\\n✅ ${createdCredentials.length}명의 로그인 가능한 사용자 생성 완료`),\n        );\n        if (createdCredentials.length > 0) {\n          console.log(\"\\n생성된 계정 목록:\");\n          console.table(createdCredentials);\n        }\n\n        return;\n      }\n      // userMode === \"dummy\": 기존 generateBatch() 흐름 계속\n    }\n\n    let saveTarget = options[\"save-to\"] || \"db\";\n    if (!options[\"save-to\"]) {\n      const result = await prompts({\n        type: \"select\",\n        name: \"saveTarget\",\n        message: \"저장 방식:\",\n        choices: [\n          { title: \"Fixture DB에 저장\", value: \"db\" },\n          { title: \"파일로 저장 (자동 파일명)\", value: \"file\" },\n          { title: \"파일로 저장 (파일명 지정)\", value: \"file:custom\" },\n          { title: \"저장 안 함 (출력만)\", value: \"none\" },\n        ],\n      });\n\n      saveTarget = result.saveTarget;\n\n      if (saveTarget === \"file:custom\") {\n        const filenameResult = await prompts({\n          type: \"text\",\n          name: \"filename\",\n          message: \"파일명:\",\n          initial: \"fixtures.json\",\n        });\n\n        if (!filenameResult.filename) {\n          console.log(chalk.yellow(\"취소되었습니다.\"));\n          return;\n        }\n\n        saveTarget = `file:${filenameResult.filename}`;\n      }\n    }\n\n    // LLM 사용 여부 결정\n    let useLLM = options[\"use-llm\"] ?? false;\n    if (!options[\"use-llm\"]) {\n      const llmResult = await prompts({\n        type: \"confirm\",\n        name: \"useLLM\",\n        message:\n          \"LLM으로 더 현실적인 데이터를 생성할까요? (fixtureHint 기반, ANTHROPIC_API_KEY 필요)\",\n        initial: false,\n      });\n      useLLM = llmResult.useLLM ?? false;\n    }\n\n    const enableLLMCache = !options[\"no-cache\"];\n\n    // fixture gen: fixture DB 내에서 참조 관계를 해결하고 저장합니다\n    const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);\n    const generator = new FixtureGenerator(fixtureDb, fixtureDb, \"fixture\", EntityManager, {\n      useLLM,\n      enableLLMCache,\n    });\n\n    if (useLLM) {\n      console.log(\n        chalk.cyan(\n          `\\nLLM 모드로 ${entityNames.join(\", \")} 생성 중... (캐싱: ${enableLLMCache ? \"ON\" : \"OFF\"})`,\n        ),\n      );\n    } else {\n      console.log(chalk.cyan(`\\n${entityNames.join(\", \")} 생성 중...`));\n    }\n\n    const specs = entityNames.map((entityName) => ({\n      entity: entityName,\n      count,\n      overrides: {},\n    }));\n\n    const results = await generator.generateBatch(specs);\n\n    if (useLLM) {\n      const stats = generator.getLLMCacheStats();\n      console.log(chalk.cyan(`[LLM Cache] 캐시 크기: ${stats.size}`));\n    }\n\n    console.log(chalk.green(`\\n✅ ${results.length}개 fixture 생성 완료`));\n\n    if (saveTarget === \"none\") {\n      console.log(JSON.stringify(results, null, 2));\n    } else if (saveTarget === \"db\") {\n      // generateBatch가 이미 DB에 저장했으므로 별도 저장이 불필요합니다.\n      console.log(chalk.green(\"Fixture DB에 저장되었습니다.\"));\n    } else if (saveTarget.startsWith(\"file\")) {\n      const fs = await import(\"node:fs/promises\");\n      const path = await import(\"node:path\");\n\n      const fixturesDir = path.join(process.cwd(), \"test\", \"fixtures\");\n      try {\n        await fs.access(fixturesDir);\n      } catch {\n        await fs.mkdir(fixturesDir, { recursive: true });\n      }\n\n      for (const entityName of entityNames) {\n        const entityResults = results.filter((r) => r.entityId === entityName);\n\n        if (entityResults.length === 0) {\n          continue;\n        }\n\n        let filename: string;\n        if (saveTarget === \"file\") {\n          const entity = EntityManager.get(entityName);\n          filename = `${entity.table}.json`;\n        } else {\n          filename = saveTarget.replace(\"file:\", \"\");\n        }\n\n        const filepath = path.join(fixturesDir, filename);\n        await fs.writeFile(\n          filepath,\n          JSON.stringify(\n            entityResults.map((r) => r.data),\n            null,\n            2,\n          ),\n        );\n        console.log(chalk.green(`✅ ${filepath} 저장 완료`));\n      }\n    }\n  } catch (error) {\n    console.error(\n      chalk.red(\n        \"Fixture 생성 중 오류가 발생했습니다.\\n\" +\n          \"원인: Entity 정의나 DB 연결을 확인해주세요.\\n\" +\n          \"자세한 내용:\",\n      ),\n      error,\n    );\n    throw error;\n  }\n}\n\n/**\n * fixture fetch 명령어 - 실제 운영 DB에서 데이터를 가져와 fixture로 저장합니다.\n * 관계된 데이터도 함께 가져오므로 현실적인 테스트 데이터를 확보할 수 있습니다.\n */\nexport async function fixtureFetchCommand(options: FixtureCommandOptions) {\n  try {\n    if (!EntityManager.isAutoloaded) {\n      await EntityManager.autoload();\n    }\n\n    let entityNames: string[];\n\n    if (options.all) {\n      entityNames = EntityManager.getAllIds();\n      if (options.exclude) {\n        const excludeList = options.exclude.split(\",\").map((s: string) => s.trim());\n        entityNames = entityNames.filter((name) => !excludeList.includes(name));\n      }\n    } else if (options.include) {\n      entityNames = options.include.split(\",\").map((s: string) => s.trim());\n    } else {\n      const result = await prompts({\n        type: \"multiselect\",\n        name: \"entities\",\n        message: \"Import할 Entity를 선택하세요:\",\n        choices: EntityManager.getAllIds().map((id) => ({ title: id, value: id })),\n        min: 1,\n      });\n\n      if (!result.entities || result.entities.length === 0) {\n        console.log(chalk.yellow(\"취소되었습니다.\"));\n        return;\n      }\n\n      entityNames = result.entities;\n    }\n\n    const strategy: DataExplorerStrategy = options.strategy ?? \"recent\";\n    const limit = options.limit ? Number.parseInt(options.limit, 10) : 10;\n\n    // fixture fetch: production 데이터를 fixture DB로 import합니다\n    const sourceDb = DB.getDB(\"r\"); // production_master (또는 development_master)\n    const fixtureDb = createKnexInstance(Sonamu.dbConfig.fixture);\n    const generator = new FixtureGenerator(sourceDb, fixtureDb, \"fixture\", EntityManager);\n\n    console.log(chalk.cyan(`\\n${entityNames.join(\", \")} import 중...`));\n\n    for (const entityName of entityNames) {\n      const results = await generator.importFromSource(entityName, {\n        strategy,\n        limit,\n        includeRelations: true,\n        maxDepth: 2,\n      });\n\n      console.log(chalk.green(`✅ ${entityName}: ${results.length}개 import 완료`));\n    }\n  } catch (error) {\n    console.error(\n      chalk.red(\n        \"실제 DB에서 데이터를 가져오는 중 오류가 발생했습니다.\\n\" +\n          \"원인: 소스 DB 연결 설정(sonamu.config.ts)이나 Entity 관계 정의를 확인해주세요.\\n\" +\n          \"자세한 내용:\",\n      ),\n      error,\n    );\n    throw error;\n  }\n}\n\n/**\n * fixture explore 명령어 - DB의 실제 데이터를 조회하여 확인합니다.\n * 저장하지 않고 조회만 하므로 데이터를 빠르게 확인할 때 유용합니다.\n */\nexport async function fixtureExploreCommand(options: FixtureCommandOptions) {\n  try {\n    if (!EntityManager.isAutoloaded) {\n      await EntityManager.autoload();\n    }\n\n    let entityName = options.include;\n\n    if (!entityName) {\n      const result = await prompts({\n        type: \"select\",\n        name: \"entity\",\n        message: \"탐색할 Entity:\",\n        choices: EntityManager.getAllIds().map((id) => ({ title: id, value: id })),\n      });\n\n      if (!result.entity) {\n        console.log(chalk.yellow(\"취소되었습니다.\"));\n        return;\n      }\n\n      entityName = result.entity;\n    }\n\n    if (!entityName) {\n      throw new Error(\"Entity name is required\");\n    }\n\n    const strategy: DataExplorerStrategy = options.strategy ?? \"sample\";\n    const limit = options.limit ? Number.parseInt(options.limit, 10) : 10;\n\n    const db = DB.getDB(\"r\");\n    const explorer = new DataExplorer(db, EntityManager);\n    const data = await explorer.explore(entityName, { strategy, limit });\n\n    console.log(chalk.cyan(`\\n${entityName} ${data.length}개 조회 완료:`));\n    console.table(data);\n  } catch (error) {\n    console.error(\n      chalk.red(\n        \"데이터 조회 중 오류가 발생했습니다.\\n\" +\n          \"원인: DB 연결이나 Entity 정의를 확인해주세요.\\n\" +\n          \"자세한 내용:\",\n      ),\n      error,\n    );\n    throw error;\n  }\n}\n"],"mappings":"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"}