@goplusvn/core 0.1.0

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 (223) hide show
  1. package/dist/audit/index.d.mts +115 -0
  2. package/dist/audit/index.d.ts +115 -0
  3. package/dist/audit/index.js +204 -0
  4. package/dist/audit/index.js.map +1 -0
  5. package/dist/audit/index.mjs +200 -0
  6. package/dist/audit/index.mjs.map +1 -0
  7. package/dist/auth/index.d.mts +86 -0
  8. package/dist/auth/index.d.ts +86 -0
  9. package/dist/auth/index.js +210 -0
  10. package/dist/auth/index.js.map +1 -0
  11. package/dist/auth/index.mjs +198 -0
  12. package/dist/auth/index.mjs.map +1 -0
  13. package/dist/button-1dWvP9Ib.d.mts +30 -0
  14. package/dist/button-1dWvP9Ib.d.ts +30 -0
  15. package/dist/calendar-2QzdEo1z.d.mts +20 -0
  16. package/dist/calendar-2QzdEo1z.d.ts +20 -0
  17. package/dist/code-generation/index.d.mts +30 -0
  18. package/dist/code-generation/index.d.ts +30 -0
  19. package/dist/code-generation/index.js +31 -0
  20. package/dist/code-generation/index.js.map +1 -0
  21. package/dist/code-generation/index.mjs +28 -0
  22. package/dist/code-generation/index.mjs.map +1 -0
  23. package/dist/configs/index.d.mts +175 -0
  24. package/dist/configs/index.d.ts +175 -0
  25. package/dist/configs/index.js +254 -0
  26. package/dist/configs/index.js.map +1 -0
  27. package/dist/configs/index.mjs +233 -0
  28. package/dist/configs/index.mjs.map +1 -0
  29. package/dist/crud/index.d.mts +646 -0
  30. package/dist/crud/index.d.ts +646 -0
  31. package/dist/crud/index.js +11772 -0
  32. package/dist/crud/index.js.map +1 -0
  33. package/dist/crud/index.mjs +11665 -0
  34. package/dist/crud/index.mjs.map +1 -0
  35. package/dist/crud/server.d.mts +20 -0
  36. package/dist/crud/server.d.ts +20 -0
  37. package/dist/crud/server.js +123 -0
  38. package/dist/crud/server.js.map +1 -0
  39. package/dist/crud/server.mjs +120 -0
  40. package/dist/crud/server.mjs.map +1 -0
  41. package/dist/data-table-skeleton-12NA8Mjx.d.mts +39 -0
  42. package/dist/data-table-skeleton-12NA8Mjx.d.ts +39 -0
  43. package/dist/dialog-bKfjZMTd.d.mts +22 -0
  44. package/dist/dialog-bKfjZMTd.d.ts +22 -0
  45. package/dist/dynamic-icon-DrGIiu2N.d.mts +10 -0
  46. package/dist/dynamic-icon-DrGIiu2N.d.ts +10 -0
  47. package/dist/home/index.d.mts +269 -0
  48. package/dist/home/index.d.ts +269 -0
  49. package/dist/home/index.js +1678 -0
  50. package/dist/home/index.js.map +1 -0
  51. package/dist/home/index.mjs +1635 -0
  52. package/dist/home/index.mjs.map +1 -0
  53. package/dist/hooks/index.d.mts +7 -0
  54. package/dist/hooks/index.d.ts +7 -0
  55. package/dist/hooks/index.js +8316 -0
  56. package/dist/hooks/index.js.map +1 -0
  57. package/dist/hooks/index.mjs +8255 -0
  58. package/dist/hooks/index.mjs.map +1 -0
  59. package/dist/index-50hpiPrV.d.ts +116 -0
  60. package/dist/index-B9zQVEVi.d.mts +116 -0
  61. package/dist/index.d.mts +5 -0
  62. package/dist/index.d.ts +5 -0
  63. package/dist/index.js +123 -0
  64. package/dist/index.js.map +1 -0
  65. package/dist/index.mjs +118 -0
  66. package/dist/index.mjs.map +1 -0
  67. package/dist/infrastructure/index.d.mts +423 -0
  68. package/dist/infrastructure/index.d.ts +423 -0
  69. package/dist/infrastructure/index.js +633 -0
  70. package/dist/infrastructure/index.js.map +1 -0
  71. package/dist/infrastructure/index.mjs +619 -0
  72. package/dist/infrastructure/index.mjs.map +1 -0
  73. package/dist/label-DWTEkNPo.d.ts +226 -0
  74. package/dist/label-LPpdcoBx.d.mts +226 -0
  75. package/dist/layout/index.d.mts +48 -0
  76. package/dist/layout/index.d.ts +48 -0
  77. package/dist/layout/index.js +117 -0
  78. package/dist/layout/index.js.map +1 -0
  79. package/dist/layout/index.mjs +90 -0
  80. package/dist/layout/index.mjs.map +1 -0
  81. package/dist/navigation/index.d.mts +16 -0
  82. package/dist/navigation/index.d.ts +16 -0
  83. package/dist/navigation/index.js +53 -0
  84. package/dist/navigation/index.js.map +1 -0
  85. package/dist/navigation/index.mjs +50 -0
  86. package/dist/navigation/index.mjs.map +1 -0
  87. package/dist/notification/index.d.mts +105 -0
  88. package/dist/notification/index.d.ts +105 -0
  89. package/dist/notification/index.js +278 -0
  90. package/dist/notification/index.js.map +1 -0
  91. package/dist/notification/index.mjs +274 -0
  92. package/dist/notification/index.mjs.map +1 -0
  93. package/dist/organization/index.d.mts +99 -0
  94. package/dist/organization/index.d.ts +99 -0
  95. package/dist/organization/index.js +360 -0
  96. package/dist/organization/index.js.map +1 -0
  97. package/dist/organization/index.mjs +352 -0
  98. package/dist/organization/index.mjs.map +1 -0
  99. package/dist/plugin/index.d.mts +83 -0
  100. package/dist/plugin/index.d.ts +83 -0
  101. package/dist/plugin/index.js +86 -0
  102. package/dist/plugin/index.js.map +1 -0
  103. package/dist/plugin/index.mjs +84 -0
  104. package/dist/plugin/index.mjs.map +1 -0
  105. package/dist/providers/index.d.mts +25 -0
  106. package/dist/providers/index.d.ts +25 -0
  107. package/dist/providers/index.js +84 -0
  108. package/dist/providers/index.js.map +1 -0
  109. package/dist/providers/index.mjs +77 -0
  110. package/dist/providers/index.mjs.map +1 -0
  111. package/dist/rbac/index.d.mts +226 -0
  112. package/dist/rbac/index.d.ts +226 -0
  113. package/dist/rbac/index.js +4784 -0
  114. package/dist/rbac/index.js.map +1 -0
  115. package/dist/rbac/index.mjs +4722 -0
  116. package/dist/rbac/index.mjs.map +1 -0
  117. package/dist/rbac/permissions.d.mts +26 -0
  118. package/dist/rbac/permissions.d.ts +26 -0
  119. package/dist/rbac/permissions.js +94 -0
  120. package/dist/rbac/permissions.js.map +1 -0
  121. package/dist/rbac/permissions.mjs +90 -0
  122. package/dist/rbac/permissions.mjs.map +1 -0
  123. package/dist/rbac/server.d.mts +1 -0
  124. package/dist/rbac/server.d.ts +1 -0
  125. package/dist/rbac/server.js +128 -0
  126. package/dist/rbac/server.js.map +1 -0
  127. package/dist/rbac/server.mjs +124 -0
  128. package/dist/rbac/server.mjs.map +1 -0
  129. package/dist/schemas/index.d.mts +1257 -0
  130. package/dist/schemas/index.d.ts +1257 -0
  131. package/dist/schemas/index.js +572 -0
  132. package/dist/schemas/index.js.map +1 -0
  133. package/dist/schemas/index.mjs +523 -0
  134. package/dist/schemas/index.mjs.map +1 -0
  135. package/dist/server-QuYCTa89.d.mts +83 -0
  136. package/dist/server-QuYCTa89.d.ts +83 -0
  137. package/dist/sonner-C74GlRDQ.d.mts +71 -0
  138. package/dist/sonner-C74GlRDQ.d.ts +71 -0
  139. package/dist/status-BOXZgIqX.d.mts +12 -0
  140. package/dist/status-BOXZgIqX.d.ts +12 -0
  141. package/dist/system/index.d.mts +77 -0
  142. package/dist/system/index.d.ts +77 -0
  143. package/dist/system/index.js +102 -0
  144. package/dist/system/index.js.map +1 -0
  145. package/dist/system/index.mjs +100 -0
  146. package/dist/system/index.mjs.map +1 -0
  147. package/dist/tabs-C6FfBwPY.d.mts +18 -0
  148. package/dist/tabs-C6FfBwPY.d.ts +18 -0
  149. package/dist/tenant-provider-B8eC_Wpb.d.mts +27 -0
  150. package/dist/tenant-provider-B8eC_Wpb.d.ts +27 -0
  151. package/dist/types/index.d.mts +469 -0
  152. package/dist/types/index.d.ts +469 -0
  153. package/dist/types/index.js +25 -0
  154. package/dist/types/index.js.map +1 -0
  155. package/dist/types/index.mjs +21 -0
  156. package/dist/types/index.mjs.map +1 -0
  157. package/dist/ui/auth.d.mts +39 -0
  158. package/dist/ui/auth.d.ts +39 -0
  159. package/dist/ui/auth.js +4941 -0
  160. package/dist/ui/auth.js.map +1 -0
  161. package/dist/ui/auth.mjs +4896 -0
  162. package/dist/ui/auth.mjs.map +1 -0
  163. package/dist/ui/crud.d.mts +2 -0
  164. package/dist/ui/crud.d.ts +2 -0
  165. package/dist/ui/crud.js +4 -0
  166. package/dist/ui/crud.js.map +1 -0
  167. package/dist/ui/crud.mjs +3 -0
  168. package/dist/ui/crud.mjs.map +1 -0
  169. package/dist/ui/data-display.d.mts +596 -0
  170. package/dist/ui/data-display.d.ts +596 -0
  171. package/dist/ui/data-display.js +5307 -0
  172. package/dist/ui/data-display.js.map +1 -0
  173. package/dist/ui/data-display.mjs +5212 -0
  174. package/dist/ui/data-display.mjs.map +1 -0
  175. package/dist/ui/feedback.d.mts +55 -0
  176. package/dist/ui/feedback.d.ts +55 -0
  177. package/dist/ui/feedback.js +2608 -0
  178. package/dist/ui/feedback.js.map +1 -0
  179. package/dist/ui/feedback.mjs +2526 -0
  180. package/dist/ui/feedback.mjs.map +1 -0
  181. package/dist/ui/forms.d.mts +309 -0
  182. package/dist/ui/forms.d.ts +309 -0
  183. package/dist/ui/forms.js +4656 -0
  184. package/dist/ui/forms.js.map +1 -0
  185. package/dist/ui/forms.mjs +4571 -0
  186. package/dist/ui/forms.mjs.map +1 -0
  187. package/dist/ui/index.d.mts +331 -0
  188. package/dist/ui/index.d.ts +331 -0
  189. package/dist/ui/index.js +16953 -0
  190. package/dist/ui/index.js.map +1 -0
  191. package/dist/ui/index.mjs +16598 -0
  192. package/dist/ui/index.mjs.map +1 -0
  193. package/dist/ui/primitives/client.d.mts +61 -0
  194. package/dist/ui/primitives/client.d.ts +61 -0
  195. package/dist/ui/primitives/client.js +3408 -0
  196. package/dist/ui/primitives/client.js.map +1 -0
  197. package/dist/ui/primitives/client.mjs +3256 -0
  198. package/dist/ui/primitives/client.mjs.map +1 -0
  199. package/dist/ui/primitives.d.mts +113 -0
  200. package/dist/ui/primitives.d.ts +113 -0
  201. package/dist/ui/primitives.js +3356 -0
  202. package/dist/ui/primitives.js.map +1 -0
  203. package/dist/ui/primitives.mjs +3227 -0
  204. package/dist/ui/primitives.mjs.map +1 -0
  205. package/dist/user/index.d.mts +228 -0
  206. package/dist/user/index.d.ts +228 -0
  207. package/dist/user/index.js +4306 -0
  208. package/dist/user/index.js.map +1 -0
  209. package/dist/user/index.mjs +4260 -0
  210. package/dist/user/index.mjs.map +1 -0
  211. package/dist/utils/index.d.mts +205 -0
  212. package/dist/utils/index.d.ts +205 -0
  213. package/dist/utils/index.js +574 -0
  214. package/dist/utils/index.js.map +1 -0
  215. package/dist/utils/index.mjs +514 -0
  216. package/dist/utils/index.mjs.map +1 -0
  217. package/dist/workflow/index.d.mts +40 -0
  218. package/dist/workflow/index.d.ts +40 -0
  219. package/dist/workflow/index.js +3710 -0
  220. package/dist/workflow/index.js.map +1 -0
  221. package/dist/workflow/index.mjs +3677 -0
  222. package/dist/workflow/index.mjs.map +1 -0
  223. package/package.json +311 -0
@@ -0,0 +1,115 @@
1
+ interface AuditLog {
2
+ id: string;
3
+ /** Action type: create, update, delete, login, logout, custom */
4
+ action: string;
5
+ /** Resource/entity name (e.g., 'purchase-order', 'user') */
6
+ resource: string;
7
+ /** Resource ID */
8
+ resourceId?: string;
9
+ /** User who performed the action */
10
+ userId?: string;
11
+ /** User's role at time of action */
12
+ roleName?: string;
13
+ /** Changes made (for update actions) */
14
+ changes?: Record<string, {
15
+ old: unknown;
16
+ new: unknown;
17
+ }>;
18
+ /** Additional metadata */
19
+ metadata?: Record<string, unknown>;
20
+ /** Request IP address */
21
+ ip?: string;
22
+ /** User agent */
23
+ userAgent?: string;
24
+ /** HTTP status code */
25
+ status?: number;
26
+ /** Timestamp */
27
+ createdAt: Date;
28
+ }
29
+ interface CreateAuditInput {
30
+ action: string;
31
+ resource: string;
32
+ resourceId?: string;
33
+ userId?: string;
34
+ roleName?: string;
35
+ changes?: Record<string, {
36
+ old: unknown;
37
+ new: unknown;
38
+ }>;
39
+ metadata?: Record<string, unknown>;
40
+ ip?: string;
41
+ userAgent?: string;
42
+ status?: number;
43
+ }
44
+ interface AuditLogger {
45
+ log(entry: AuditLog): Promise<void>;
46
+ }
47
+ interface AuditManagerOptions {
48
+ /** Custom logger implementation */
49
+ logger?: AuditLogger;
50
+ /** Actions to automatically audit */
51
+ actions?: string[];
52
+ }
53
+
54
+ /**
55
+ * Console-based audit logger (default)
56
+ * In production, replace with database or external service
57
+ */
58
+ declare class ConsoleAuditLogger implements AuditLogger {
59
+ log(entry: AuditLog): Promise<void>;
60
+ }
61
+ /**
62
+ * AuditManager - tracks actions for compliance and debugging
63
+ *
64
+ * @example
65
+ * ```typescript
66
+ * import { auditManager } from '@goerp/core/audit';
67
+ *
68
+ * // Log an action
69
+ * await auditManager.log({
70
+ * action: 'update',
71
+ * resource: 'purchase-order',
72
+ * resourceId: '123',
73
+ * userId: session.user.id,
74
+ * changes: { status: { old: 'pending', new: 'approved' } }
75
+ * });
76
+ * ```
77
+ */
78
+ declare class AuditManagerImpl {
79
+ private auditLogger;
80
+ private registeredActions;
81
+ constructor(options?: AuditManagerOptions);
82
+ /** Set custom audit logger */
83
+ setLogger(auditLogger: AuditLogger): void;
84
+ /** Register action for automatic auditing */
85
+ registerAction(action: string): void;
86
+ /** Check if action is registered for auditing */
87
+ isRegistered(action: string): boolean;
88
+ /** Log an audit entry */
89
+ log(input: CreateAuditInput): Promise<void>;
90
+ /**
91
+ * Get logs if the logger supports it (specifically MemoryAuditLogger)
92
+ */
93
+ getLogs(filter?: any): AuditLog[];
94
+ /** Create middleware for automatic API route auditing */
95
+ middleware(): (ctx: {
96
+ action?: {
97
+ actionName: string;
98
+ resourceName: string;
99
+ };
100
+ state?: {
101
+ currentUser?: {
102
+ id: string;
103
+ };
104
+ currentRole?: string;
105
+ };
106
+ request?: {
107
+ ip?: string;
108
+ header?: Record<string, string>;
109
+ };
110
+ status?: number;
111
+ }, next: () => Promise<void>) => Promise<void>;
112
+ }
113
+ declare const auditManager: AuditManagerImpl;
114
+
115
+ export { type AuditLog, type AuditLogger, AuditManagerImpl, type AuditManagerOptions, ConsoleAuditLogger, type CreateAuditInput, auditManager };
@@ -0,0 +1,115 @@
1
+ interface AuditLog {
2
+ id: string;
3
+ /** Action type: create, update, delete, login, logout, custom */
4
+ action: string;
5
+ /** Resource/entity name (e.g., 'purchase-order', 'user') */
6
+ resource: string;
7
+ /** Resource ID */
8
+ resourceId?: string;
9
+ /** User who performed the action */
10
+ userId?: string;
11
+ /** User's role at time of action */
12
+ roleName?: string;
13
+ /** Changes made (for update actions) */
14
+ changes?: Record<string, {
15
+ old: unknown;
16
+ new: unknown;
17
+ }>;
18
+ /** Additional metadata */
19
+ metadata?: Record<string, unknown>;
20
+ /** Request IP address */
21
+ ip?: string;
22
+ /** User agent */
23
+ userAgent?: string;
24
+ /** HTTP status code */
25
+ status?: number;
26
+ /** Timestamp */
27
+ createdAt: Date;
28
+ }
29
+ interface CreateAuditInput {
30
+ action: string;
31
+ resource: string;
32
+ resourceId?: string;
33
+ userId?: string;
34
+ roleName?: string;
35
+ changes?: Record<string, {
36
+ old: unknown;
37
+ new: unknown;
38
+ }>;
39
+ metadata?: Record<string, unknown>;
40
+ ip?: string;
41
+ userAgent?: string;
42
+ status?: number;
43
+ }
44
+ interface AuditLogger {
45
+ log(entry: AuditLog): Promise<void>;
46
+ }
47
+ interface AuditManagerOptions {
48
+ /** Custom logger implementation */
49
+ logger?: AuditLogger;
50
+ /** Actions to automatically audit */
51
+ actions?: string[];
52
+ }
53
+
54
+ /**
55
+ * Console-based audit logger (default)
56
+ * In production, replace with database or external service
57
+ */
58
+ declare class ConsoleAuditLogger implements AuditLogger {
59
+ log(entry: AuditLog): Promise<void>;
60
+ }
61
+ /**
62
+ * AuditManager - tracks actions for compliance and debugging
63
+ *
64
+ * @example
65
+ * ```typescript
66
+ * import { auditManager } from '@goerp/core/audit';
67
+ *
68
+ * // Log an action
69
+ * await auditManager.log({
70
+ * action: 'update',
71
+ * resource: 'purchase-order',
72
+ * resourceId: '123',
73
+ * userId: session.user.id,
74
+ * changes: { status: { old: 'pending', new: 'approved' } }
75
+ * });
76
+ * ```
77
+ */
78
+ declare class AuditManagerImpl {
79
+ private auditLogger;
80
+ private registeredActions;
81
+ constructor(options?: AuditManagerOptions);
82
+ /** Set custom audit logger */
83
+ setLogger(auditLogger: AuditLogger): void;
84
+ /** Register action for automatic auditing */
85
+ registerAction(action: string): void;
86
+ /** Check if action is registered for auditing */
87
+ isRegistered(action: string): boolean;
88
+ /** Log an audit entry */
89
+ log(input: CreateAuditInput): Promise<void>;
90
+ /**
91
+ * Get logs if the logger supports it (specifically MemoryAuditLogger)
92
+ */
93
+ getLogs(filter?: any): AuditLog[];
94
+ /** Create middleware for automatic API route auditing */
95
+ middleware(): (ctx: {
96
+ action?: {
97
+ actionName: string;
98
+ resourceName: string;
99
+ };
100
+ state?: {
101
+ currentUser?: {
102
+ id: string;
103
+ };
104
+ currentRole?: string;
105
+ };
106
+ request?: {
107
+ ip?: string;
108
+ header?: Record<string, string>;
109
+ };
110
+ status?: number;
111
+ }, next: () => Promise<void>) => Promise<void>;
112
+ }
113
+ declare const auditManager: AuditManagerImpl;
114
+
115
+ export { type AuditLog, type AuditLogger, AuditManagerImpl, type AuditManagerOptions, ConsoleAuditLogger, type CreateAuditInput, auditManager };
@@ -0,0 +1,204 @@
1
+ 'use strict';
2
+
3
+ // src/infrastructure/logger/logger.ts
4
+ var LOG_LEVELS = {
5
+ trace: 0,
6
+ debug: 1,
7
+ info: 2,
8
+ warn: 3,
9
+ error: 4
10
+ };
11
+ function getLogLevel() {
12
+ const envLevel = process.env.LOG_LEVEL?.toLowerCase();
13
+ if (envLevel && LOG_LEVELS[envLevel] !== void 0) {
14
+ return envLevel;
15
+ }
16
+ return process.env.NODE_ENV === "production" ? "info" : "debug";
17
+ }
18
+ function formatMessage(level, name, message, context) {
19
+ const timestamp = (/* @__PURE__ */ new Date()).toISOString();
20
+ const contextStr = context ? ` ${JSON.stringify(context)}` : "";
21
+ return `${timestamp} [${level.toUpperCase()}] [${name}] ${message}${contextStr}`;
22
+ }
23
+ function shouldLog(level, minLevel) {
24
+ return LOG_LEVELS[level] >= LOG_LEVELS[minLevel];
25
+ }
26
+ var ConsoleLogger = class {
27
+ constructor(options = {}) {
28
+ this.name = options.name || "App";
29
+ this.level = options.level || getLogLevel();
30
+ }
31
+ trace(message, context) {
32
+ if (shouldLog("trace", this.level)) {
33
+ console.log(formatMessage("trace", this.name, message, context));
34
+ }
35
+ }
36
+ debug(message, context) {
37
+ if (shouldLog("debug", this.level)) {
38
+ console.log(formatMessage("debug", this.name, message, context));
39
+ }
40
+ }
41
+ info(message, context) {
42
+ if (shouldLog("info", this.level)) {
43
+ console.info(formatMessage("info", this.name, message, context));
44
+ }
45
+ }
46
+ warn(message, context) {
47
+ if (shouldLog("warn", this.level)) {
48
+ console.warn(formatMessage("warn", this.name, message, context));
49
+ }
50
+ }
51
+ error(message, context) {
52
+ if (shouldLog("error", this.level)) {
53
+ console.error(formatMessage("error", this.name, message, context));
54
+ }
55
+ }
56
+ };
57
+ function createLogger(nameOrOptions = {}) {
58
+ const options = typeof nameOrOptions === "string" ? { name: nameOrOptions } : nameOrOptions;
59
+ return new ConsoleLogger(options);
60
+ }
61
+ createLogger("App");
62
+
63
+ // src/audit/memory-audit-logger.ts
64
+ var MemoryAuditLogger = class {
65
+ constructor(maxLogs = 1e3) {
66
+ this.logs = [];
67
+ this.maxLogs = maxLogs;
68
+ }
69
+ async log(entry) {
70
+ this.logs.unshift(entry);
71
+ if (this.logs.length > this.maxLogs) {
72
+ this.logs = this.logs.slice(0, this.maxLogs);
73
+ }
74
+ }
75
+ /**
76
+ * Get logs with filtering
77
+ */
78
+ getLogs(filter = {}) {
79
+ let filtered = this.logs;
80
+ if (filter.fromDate) {
81
+ filtered = filtered.filter((log) => log.createdAt >= filter.fromDate);
82
+ }
83
+ if (filter.toDate) {
84
+ filtered = filtered.filter((log) => log.createdAt <= filter.toDate);
85
+ }
86
+ if (filter.userId) {
87
+ filtered = filtered.filter((log) => log.userId === filter.userId);
88
+ }
89
+ if (filter.action) {
90
+ filtered = filtered.filter((log) => log.action === filter.action);
91
+ }
92
+ if (filter.resource) {
93
+ filtered = filtered.filter((log) => log.resource === filter.resource);
94
+ }
95
+ if (filter.type) {
96
+ filtered = filtered.filter((log) => {
97
+ if (filter.type === "error") return (log.status || 200) >= 400;
98
+ if (filter.type === "warning")
99
+ return (log.status || 200) >= 300 && (log.status || 200) < 400;
100
+ return (log.status || 200) < 300;
101
+ });
102
+ }
103
+ const offset = filter.offset || 0;
104
+ const limit = filter.limit || 50;
105
+ return filtered.slice(offset, offset + limit);
106
+ }
107
+ /**
108
+ * Clear all logs
109
+ */
110
+ clear() {
111
+ this.logs = [];
112
+ }
113
+ };
114
+
115
+ // src/audit/audit-manager.ts
116
+ var logger2 = createLogger("AuditManager");
117
+ var ConsoleAuditLogger = class {
118
+ async log(entry) {
119
+ logger2.info(`AUDIT: ${entry.action} ${entry.resource}`, {
120
+ id: entry.id,
121
+ resourceId: entry.resourceId,
122
+ userId: entry.userId,
123
+ changes: entry.changes
124
+ });
125
+ }
126
+ };
127
+ var AuditManagerImpl = class {
128
+ constructor(options = {}) {
129
+ this.auditLogger = options.logger || new MemoryAuditLogger();
130
+ this.registeredActions = new Set(
131
+ options.actions || ["create", "update", "delete", "login", "logout"]
132
+ );
133
+ }
134
+ /** Set custom audit logger */
135
+ setLogger(auditLogger) {
136
+ this.auditLogger = auditLogger;
137
+ }
138
+ /** Register action for automatic auditing */
139
+ registerAction(action) {
140
+ this.registeredActions.add(action);
141
+ }
142
+ /** Check if action is registered for auditing */
143
+ isRegistered(action) {
144
+ return this.registeredActions.has(action);
145
+ }
146
+ /** Log an audit entry */
147
+ async log(input) {
148
+ const entry = {
149
+ id: crypto.randomUUID(),
150
+ ...input,
151
+ createdAt: /* @__PURE__ */ new Date()
152
+ };
153
+ try {
154
+ await this.auditLogger.log(entry);
155
+ } catch (error) {
156
+ logger2.error("Failed to log audit entry", { error: String(error) });
157
+ }
158
+ }
159
+ /**
160
+ * Get logs if the logger supports it (specifically MemoryAuditLogger)
161
+ */
162
+ getLogs(filter = {}) {
163
+ if ("getLogs" in this.auditLogger) {
164
+ return this.auditLogger.getLogs(filter);
165
+ }
166
+ return [];
167
+ }
168
+ /** Create middleware for automatic API route auditing */
169
+ middleware() {
170
+ return async (ctx, next) => {
171
+ const startTime = Date.now();
172
+ let error = null;
173
+ try {
174
+ await next();
175
+ } catch (e) {
176
+ error = e;
177
+ throw e;
178
+ } finally {
179
+ if (ctx.action && this.isRegistered(ctx.action.actionName)) {
180
+ await this.log({
181
+ action: ctx.action.actionName,
182
+ resource: ctx.action.resourceName,
183
+ userId: ctx.state?.currentUser?.id,
184
+ roleName: ctx.state?.currentRole,
185
+ ip: ctx.request?.ip,
186
+ userAgent: ctx.request?.header?.["user-agent"],
187
+ status: ctx.status || (error ? 500 : 200),
188
+ metadata: {
189
+ duration: Date.now() - startTime,
190
+ ...error && { error: error.message }
191
+ }
192
+ });
193
+ }
194
+ }
195
+ };
196
+ }
197
+ };
198
+ var auditManager = new AuditManagerImpl();
199
+
200
+ exports.AuditManagerImpl = AuditManagerImpl;
201
+ exports.ConsoleAuditLogger = ConsoleAuditLogger;
202
+ exports.auditManager = auditManager;
203
+ //# sourceMappingURL=index.js.map
204
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/infrastructure/logger/logger.ts","../../src/audit/memory-audit-logger.ts","../../src/audit/audit-manager.ts"],"names":["logger"],"mappings":"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","file":"index.js","sourcesContent":["import type { Logger, LoggerOptions, LogContext, LogLevel } from \"./types\";\n\nconst LOG_LEVELS: Record<LogLevel, number> = {\n trace: 0,\n debug: 1,\n info: 2,\n warn: 3,\n error: 4,\n};\n\nfunction getLogLevel(): LogLevel {\n const envLevel = process.env.LOG_LEVEL?.toLowerCase() as LogLevel;\n if (envLevel && LOG_LEVELS[envLevel] !== undefined) {\n return envLevel;\n }\n return process.env.NODE_ENV === \"production\" ? \"info\" : \"debug\";\n}\n\nfunction formatMessage(\n level: LogLevel,\n name: string,\n message: string,\n context?: LogContext,\n): string {\n const timestamp = new Date().toISOString();\n const contextStr = context ? ` ${JSON.stringify(context)}` : \"\";\n return `${timestamp} [${level.toUpperCase()}] [${name}] ${message}${contextStr}`;\n}\n\nfunction shouldLog(level: LogLevel, minLevel: LogLevel): boolean {\n return LOG_LEVELS[level] >= LOG_LEVELS[minLevel];\n}\n\nclass ConsoleLogger implements Logger {\n private name: string;\n private level: LogLevel;\n\n constructor(options: LoggerOptions = {}) {\n this.name = options.name || \"App\";\n this.level = options.level || getLogLevel();\n }\n\n trace(message: string, context?: LogContext): void {\n if (shouldLog(\"trace\", this.level)) {\n console.log(formatMessage(\"trace\", this.name, message, context));\n }\n }\n\n debug(message: string, context?: LogContext): void {\n if (shouldLog(\"debug\", this.level)) {\n console.log(formatMessage(\"debug\", this.name, message, context));\n }\n }\n\n info(message: string, context?: LogContext): void {\n if (shouldLog(\"info\", this.level)) {\n console.info(formatMessage(\"info\", this.name, message, context));\n }\n }\n\n warn(message: string, context?: LogContext): void {\n if (shouldLog(\"warn\", this.level)) {\n console.warn(formatMessage(\"warn\", this.name, message, context));\n }\n }\n\n error(message: string, context?: LogContext): void {\n if (shouldLog(\"error\", this.level)) {\n console.error(formatMessage(\"error\", this.name, message, context));\n }\n }\n}\n\n/**\n * Create a logger instance with the given name/options\n *\n * @example\n * ```typescript\n * import { createLogger } from '@goerp/core/infrastructure';\n *\n * const logger = createLogger('OrderService');\n * logger.info('Order created', { orderId: '123' });\n * logger.error('Payment failed', { error: err.message });\n * ```\n */\nexport function createLogger(\n nameOrOptions: string | LoggerOptions = {},\n): Logger {\n const options =\n typeof nameOrOptions === \"string\" ? { name: nameOrOptions } : nameOrOptions;\n\n return new ConsoleLogger(options);\n}\n\n// Default app logger\nexport const logger = createLogger(\"App\");\n","import type { AuditLog, AuditLogger } from \"./types\";\n\n/**\n * Filter options for retrieving audit logs\n */\nexport interface AuditLogFilter {\n fromDate?: Date;\n toDate?: Date;\n userId?: string;\n action?: string;\n resource?: string;\n type?: \"info\" | \"warning\" | \"error\";\n limit?: number;\n offset?: number;\n}\n\n/**\n * In-memory audit logger that supports retrieval\n */\nexport class MemoryAuditLogger implements AuditLogger {\n private logs: AuditLog[] = [];\n private readonly maxLogs: number;\n\n constructor(maxLogs = 1000) {\n this.maxLogs = maxLogs;\n }\n\n async log(entry: AuditLog): Promise<void> {\n // Add to beginning of array\n this.logs.unshift(entry);\n\n // Trim if exceeds max size\n if (this.logs.length > this.maxLogs) {\n this.logs = this.logs.slice(0, this.maxLogs);\n }\n }\n\n /**\n * Get logs with filtering\n */\n getLogs(filter: AuditLogFilter = {}): AuditLog[] {\n let filtered = this.logs;\n\n if (filter.fromDate) {\n filtered = filtered.filter((log) => log.createdAt >= filter.fromDate!);\n }\n\n if (filter.toDate) {\n filtered = filtered.filter((log) => log.createdAt <= filter.toDate!);\n }\n\n if (filter.userId) {\n filtered = filtered.filter((log) => log.userId === filter.userId);\n }\n\n if (filter.action) {\n filtered = filtered.filter((log) => log.action === filter.action);\n }\n\n if (filter.resource) {\n filtered = filtered.filter((log) => log.resource === filter.resource);\n }\n\n // \"type\" filter is a fuzzy mapping based on status code or action name for demo purposes\n if (filter.type) {\n filtered = filtered.filter((log) => {\n if (filter.type === \"error\") return (log.status || 200) >= 400;\n if (filter.type === \"warning\")\n return (log.status || 200) >= 300 && (log.status || 200) < 400;\n return (log.status || 200) < 300;\n });\n }\n\n const offset = filter.offset || 0;\n const limit = filter.limit || 50;\n\n return filtered.slice(offset, offset + limit);\n }\n\n /**\n * Clear all logs\n */\n clear(): void {\n this.logs = [];\n }\n}\n","import type {\n AuditLog,\n AuditLogger,\n AuditManagerOptions,\n CreateAuditInput,\n} from \"./types\";\nimport { createLogger } from \"../infrastructure/logger\";\nimport { MemoryAuditLogger } from \"./memory-audit-logger\";\n\nconst logger = createLogger(\"AuditManager\");\n\n/**\n * Console-based audit logger (default)\n * In production, replace with database or external service\n */\nclass ConsoleAuditLogger implements AuditLogger {\n async log(entry: AuditLog): Promise<void> {\n logger.info(`AUDIT: ${entry.action} ${entry.resource}`, {\n id: entry.id,\n resourceId: entry.resourceId,\n userId: entry.userId,\n changes: entry.changes,\n });\n }\n}\n\n/**\n * AuditManager - tracks actions for compliance and debugging\n *\n * @example\n * ```typescript\n * import { auditManager } from '@goerp/core/audit';\n *\n * // Log an action\n * await auditManager.log({\n * action: 'update',\n * resource: 'purchase-order',\n * resourceId: '123',\n * userId: session.user.id,\n * changes: { status: { old: 'pending', new: 'approved' } }\n * });\n * ```\n */\nclass AuditManagerImpl {\n private auditLogger: AuditLogger;\n private registeredActions: Set<string>;\n\n constructor(options: AuditManagerOptions = {}) {\n this.auditLogger = options.logger || new MemoryAuditLogger();\n this.registeredActions = new Set(\n options.actions || [\"create\", \"update\", \"delete\", \"login\", \"logout\"],\n );\n }\n\n /** Set custom audit logger */\n setLogger(auditLogger: AuditLogger): void {\n this.auditLogger = auditLogger;\n }\n\n /** Register action for automatic auditing */\n registerAction(action: string): void {\n this.registeredActions.add(action);\n }\n\n /** Check if action is registered for auditing */\n isRegistered(action: string): boolean {\n return this.registeredActions.has(action);\n }\n\n /** Log an audit entry */\n async log(input: CreateAuditInput): Promise<void> {\n const entry: AuditLog = {\n id: crypto.randomUUID(),\n ...input,\n createdAt: new Date(),\n };\n\n try {\n await this.auditLogger.log(entry);\n } catch (error) {\n logger.error(\"Failed to log audit entry\", { error: String(error) });\n }\n }\n\n /**\n * Get logs if the logger supports it (specifically MemoryAuditLogger)\n */\n getLogs(filter: any = {}): AuditLog[] {\n if (\"getLogs\" in this.auditLogger) {\n return (this.auditLogger as any).getLogs(filter);\n }\n return [];\n }\n\n /** Create middleware for automatic API route auditing */\n middleware() {\n return async (\n ctx: {\n action?: { actionName: string; resourceName: string };\n state?: { currentUser?: { id: string }; currentRole?: string };\n request?: { ip?: string; header?: Record<string, string> };\n status?: number;\n },\n next: () => Promise<void>,\n ): Promise<void> => {\n const startTime = Date.now();\n let error: Error | null = null;\n\n try {\n await next();\n } catch (e) {\n error = e as Error;\n throw e;\n } finally {\n if (ctx.action && this.isRegistered(ctx.action.actionName)) {\n await this.log({\n action: ctx.action.actionName,\n resource: ctx.action.resourceName,\n userId: ctx.state?.currentUser?.id,\n roleName: ctx.state?.currentRole,\n ip: ctx.request?.ip,\n userAgent: ctx.request?.header?.[\"user-agent\"],\n status: ctx.status || (error ? 500 : 200),\n metadata: {\n duration: Date.now() - startTime,\n ...(error && { error: error.message }),\n },\n });\n }\n }\n };\n }\n}\n\n// Singleton instance\nexport const auditManager = new AuditManagerImpl();\n\n// Export class for testing\nexport { AuditManagerImpl, ConsoleAuditLogger, MemoryAuditLogger };\n"]}
@@ -0,0 +1,200 @@
1
+ // src/infrastructure/logger/logger.ts
2
+ var LOG_LEVELS = {
3
+ trace: 0,
4
+ debug: 1,
5
+ info: 2,
6
+ warn: 3,
7
+ error: 4
8
+ };
9
+ function getLogLevel() {
10
+ const envLevel = process.env.LOG_LEVEL?.toLowerCase();
11
+ if (envLevel && LOG_LEVELS[envLevel] !== void 0) {
12
+ return envLevel;
13
+ }
14
+ return process.env.NODE_ENV === "production" ? "info" : "debug";
15
+ }
16
+ function formatMessage(level, name, message, context) {
17
+ const timestamp = (/* @__PURE__ */ new Date()).toISOString();
18
+ const contextStr = context ? ` ${JSON.stringify(context)}` : "";
19
+ return `${timestamp} [${level.toUpperCase()}] [${name}] ${message}${contextStr}`;
20
+ }
21
+ function shouldLog(level, minLevel) {
22
+ return LOG_LEVELS[level] >= LOG_LEVELS[minLevel];
23
+ }
24
+ var ConsoleLogger = class {
25
+ constructor(options = {}) {
26
+ this.name = options.name || "App";
27
+ this.level = options.level || getLogLevel();
28
+ }
29
+ trace(message, context) {
30
+ if (shouldLog("trace", this.level)) {
31
+ console.log(formatMessage("trace", this.name, message, context));
32
+ }
33
+ }
34
+ debug(message, context) {
35
+ if (shouldLog("debug", this.level)) {
36
+ console.log(formatMessage("debug", this.name, message, context));
37
+ }
38
+ }
39
+ info(message, context) {
40
+ if (shouldLog("info", this.level)) {
41
+ console.info(formatMessage("info", this.name, message, context));
42
+ }
43
+ }
44
+ warn(message, context) {
45
+ if (shouldLog("warn", this.level)) {
46
+ console.warn(formatMessage("warn", this.name, message, context));
47
+ }
48
+ }
49
+ error(message, context) {
50
+ if (shouldLog("error", this.level)) {
51
+ console.error(formatMessage("error", this.name, message, context));
52
+ }
53
+ }
54
+ };
55
+ function createLogger(nameOrOptions = {}) {
56
+ const options = typeof nameOrOptions === "string" ? { name: nameOrOptions } : nameOrOptions;
57
+ return new ConsoleLogger(options);
58
+ }
59
+ createLogger("App");
60
+
61
+ // src/audit/memory-audit-logger.ts
62
+ var MemoryAuditLogger = class {
63
+ constructor(maxLogs = 1e3) {
64
+ this.logs = [];
65
+ this.maxLogs = maxLogs;
66
+ }
67
+ async log(entry) {
68
+ this.logs.unshift(entry);
69
+ if (this.logs.length > this.maxLogs) {
70
+ this.logs = this.logs.slice(0, this.maxLogs);
71
+ }
72
+ }
73
+ /**
74
+ * Get logs with filtering
75
+ */
76
+ getLogs(filter = {}) {
77
+ let filtered = this.logs;
78
+ if (filter.fromDate) {
79
+ filtered = filtered.filter((log) => log.createdAt >= filter.fromDate);
80
+ }
81
+ if (filter.toDate) {
82
+ filtered = filtered.filter((log) => log.createdAt <= filter.toDate);
83
+ }
84
+ if (filter.userId) {
85
+ filtered = filtered.filter((log) => log.userId === filter.userId);
86
+ }
87
+ if (filter.action) {
88
+ filtered = filtered.filter((log) => log.action === filter.action);
89
+ }
90
+ if (filter.resource) {
91
+ filtered = filtered.filter((log) => log.resource === filter.resource);
92
+ }
93
+ if (filter.type) {
94
+ filtered = filtered.filter((log) => {
95
+ if (filter.type === "error") return (log.status || 200) >= 400;
96
+ if (filter.type === "warning")
97
+ return (log.status || 200) >= 300 && (log.status || 200) < 400;
98
+ return (log.status || 200) < 300;
99
+ });
100
+ }
101
+ const offset = filter.offset || 0;
102
+ const limit = filter.limit || 50;
103
+ return filtered.slice(offset, offset + limit);
104
+ }
105
+ /**
106
+ * Clear all logs
107
+ */
108
+ clear() {
109
+ this.logs = [];
110
+ }
111
+ };
112
+
113
+ // src/audit/audit-manager.ts
114
+ var logger2 = createLogger("AuditManager");
115
+ var ConsoleAuditLogger = class {
116
+ async log(entry) {
117
+ logger2.info(`AUDIT: ${entry.action} ${entry.resource}`, {
118
+ id: entry.id,
119
+ resourceId: entry.resourceId,
120
+ userId: entry.userId,
121
+ changes: entry.changes
122
+ });
123
+ }
124
+ };
125
+ var AuditManagerImpl = class {
126
+ constructor(options = {}) {
127
+ this.auditLogger = options.logger || new MemoryAuditLogger();
128
+ this.registeredActions = new Set(
129
+ options.actions || ["create", "update", "delete", "login", "logout"]
130
+ );
131
+ }
132
+ /** Set custom audit logger */
133
+ setLogger(auditLogger) {
134
+ this.auditLogger = auditLogger;
135
+ }
136
+ /** Register action for automatic auditing */
137
+ registerAction(action) {
138
+ this.registeredActions.add(action);
139
+ }
140
+ /** Check if action is registered for auditing */
141
+ isRegistered(action) {
142
+ return this.registeredActions.has(action);
143
+ }
144
+ /** Log an audit entry */
145
+ async log(input) {
146
+ const entry = {
147
+ id: crypto.randomUUID(),
148
+ ...input,
149
+ createdAt: /* @__PURE__ */ new Date()
150
+ };
151
+ try {
152
+ await this.auditLogger.log(entry);
153
+ } catch (error) {
154
+ logger2.error("Failed to log audit entry", { error: String(error) });
155
+ }
156
+ }
157
+ /**
158
+ * Get logs if the logger supports it (specifically MemoryAuditLogger)
159
+ */
160
+ getLogs(filter = {}) {
161
+ if ("getLogs" in this.auditLogger) {
162
+ return this.auditLogger.getLogs(filter);
163
+ }
164
+ return [];
165
+ }
166
+ /** Create middleware for automatic API route auditing */
167
+ middleware() {
168
+ return async (ctx, next) => {
169
+ const startTime = Date.now();
170
+ let error = null;
171
+ try {
172
+ await next();
173
+ } catch (e) {
174
+ error = e;
175
+ throw e;
176
+ } finally {
177
+ if (ctx.action && this.isRegistered(ctx.action.actionName)) {
178
+ await this.log({
179
+ action: ctx.action.actionName,
180
+ resource: ctx.action.resourceName,
181
+ userId: ctx.state?.currentUser?.id,
182
+ roleName: ctx.state?.currentRole,
183
+ ip: ctx.request?.ip,
184
+ userAgent: ctx.request?.header?.["user-agent"],
185
+ status: ctx.status || (error ? 500 : 200),
186
+ metadata: {
187
+ duration: Date.now() - startTime,
188
+ ...error && { error: error.message }
189
+ }
190
+ });
191
+ }
192
+ }
193
+ };
194
+ }
195
+ };
196
+ var auditManager = new AuditManagerImpl();
197
+
198
+ export { AuditManagerImpl, ConsoleAuditLogger, auditManager };
199
+ //# sourceMappingURL=index.mjs.map
200
+ //# sourceMappingURL=index.mjs.map