@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,633 @@
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
+ var logger = createLogger("App");
62
+
63
+ // src/infrastructure/cache/cache.ts
64
+ var MemoryCache = class {
65
+ constructor(options) {
66
+ this.store = /* @__PURE__ */ new Map();
67
+ this.name = options.name;
68
+ this.defaultTtl = options.ttl || 3600;
69
+ this.maxSize = options.max || 1e3;
70
+ this.prefix = options.prefix || "";
71
+ }
72
+ getFullKey(key) {
73
+ return this.prefix ? `${this.prefix}:${this.name}:${key}` : `${this.name}:${key}`;
74
+ }
75
+ isExpired(entry) {
76
+ if (entry.expireAt === null) return false;
77
+ return Date.now() > entry.expireAt;
78
+ }
79
+ cleanup() {
80
+ for (const [key, entry] of this.store.entries()) {
81
+ if (this.isExpired(entry)) {
82
+ this.store.delete(key);
83
+ }
84
+ }
85
+ }
86
+ async get(key) {
87
+ const fullKey = this.getFullKey(key);
88
+ const entry = this.store.get(fullKey);
89
+ if (!entry) return void 0;
90
+ if (this.isExpired(entry)) {
91
+ this.store.delete(fullKey);
92
+ return void 0;
93
+ }
94
+ return entry.value;
95
+ }
96
+ async set(key, value, ttl) {
97
+ const fullKey = this.getFullKey(key);
98
+ const ttlSeconds = ttl ?? this.defaultTtl;
99
+ if (this.store.size >= this.maxSize) {
100
+ this.cleanup();
101
+ if (this.store.size >= this.maxSize) {
102
+ const firstKey = this.store.keys().next().value;
103
+ if (firstKey) this.store.delete(firstKey);
104
+ }
105
+ }
106
+ this.store.set(fullKey, {
107
+ value,
108
+ expireAt: ttlSeconds > 0 ? Date.now() + ttlSeconds * 1e3 : null
109
+ });
110
+ }
111
+ async del(key) {
112
+ const fullKey = this.getFullKey(key);
113
+ this.store.delete(fullKey);
114
+ }
115
+ async has(key) {
116
+ const value = await this.get(key);
117
+ return value !== void 0;
118
+ }
119
+ async reset() {
120
+ const prefix = this.getFullKey("");
121
+ for (const key of this.store.keys()) {
122
+ if (key.startsWith(prefix)) {
123
+ this.store.delete(key);
124
+ }
125
+ }
126
+ }
127
+ async keys() {
128
+ this.cleanup();
129
+ const prefix = this.getFullKey("");
130
+ const result = [];
131
+ for (const key of this.store.keys()) {
132
+ if (key.startsWith(prefix)) {
133
+ result.push(key.slice(prefix.length));
134
+ }
135
+ }
136
+ return result;
137
+ }
138
+ /** Get cache info for admin dashboard */
139
+ getInfo() {
140
+ return {
141
+ name: this.name,
142
+ ttl: this.defaultTtl,
143
+ maxSize: this.maxSize
144
+ };
145
+ }
146
+ };
147
+
148
+ // src/infrastructure/cache/cache-manager.ts
149
+ var CacheManagerImpl = class {
150
+ constructor(options = {}) {
151
+ this.caches = /* @__PURE__ */ new Map();
152
+ this.options = {
153
+ defaultTtl: options.defaultTtl || 3600,
154
+ prefix: options.prefix || ""
155
+ };
156
+ }
157
+ create(options) {
158
+ const existing = this.caches.get(options.name);
159
+ if (existing) {
160
+ return existing;
161
+ }
162
+ const cache = new MemoryCache({
163
+ ...options,
164
+ ttl: options.ttl || this.options.defaultTtl,
165
+ prefix: this.options.prefix
166
+ });
167
+ this.caches.set(options.name, cache);
168
+ return cache;
169
+ }
170
+ get(name) {
171
+ return this.caches.get(name);
172
+ }
173
+ async flushAll() {
174
+ const promises = [];
175
+ for (const cache of this.caches.values()) {
176
+ promises.push(cache.reset());
177
+ }
178
+ await Promise.all(promises);
179
+ }
180
+ /** Get all cache names */
181
+ getAllCaches() {
182
+ return Array.from(this.caches.keys());
183
+ }
184
+ /** Get stats for all caches (for admin dashboard) */
185
+ async getStats() {
186
+ const stats = [];
187
+ for (const [name, cache] of this.caches.entries()) {
188
+ const keys = await cache.keys();
189
+ stats.push({
190
+ name,
191
+ itemCount: keys.length,
192
+ ttl: cache.getInfo().ttl
193
+ });
194
+ }
195
+ return stats;
196
+ }
197
+ /** Clear a specific cache by name */
198
+ async clearCache(name) {
199
+ const cache = this.caches.get(name);
200
+ if (cache) {
201
+ await cache.reset();
202
+ return true;
203
+ }
204
+ return false;
205
+ }
206
+ };
207
+ var globalForCache = globalThis;
208
+ var cacheManager = globalForCache.coreCacheManager ?? new CacheManagerImpl();
209
+ if (process.env.NODE_ENV !== "production") {
210
+ globalForCache.coreCacheManager = cacheManager;
211
+ }
212
+
213
+ // src/infrastructure/event-bus/event-bus.ts
214
+ var logger2 = createLogger("EventBus");
215
+ var EventBusImpl = class {
216
+ constructor() {
217
+ this.handlers = /* @__PURE__ */ new Map();
218
+ this.onceHandlers = /* @__PURE__ */ new Map();
219
+ }
220
+ on(event, handler) {
221
+ if (!this.handlers.has(event)) {
222
+ this.handlers.set(event, /* @__PURE__ */ new Set());
223
+ }
224
+ this.handlers.get(event).add(handler);
225
+ logger2.debug(`Subscribed to event: ${event}`);
226
+ return {
227
+ unsubscribe: () => {
228
+ this.handlers.get(event)?.delete(handler);
229
+ }
230
+ };
231
+ }
232
+ once(event, handler) {
233
+ if (!this.onceHandlers.has(event)) {
234
+ this.onceHandlers.set(event, /* @__PURE__ */ new Set());
235
+ }
236
+ this.onceHandlers.get(event).add(handler);
237
+ return {
238
+ unsubscribe: () => {
239
+ this.onceHandlers.get(event)?.delete(handler);
240
+ }
241
+ };
242
+ }
243
+ emit(event, data) {
244
+ logger2.debug(`Emitting event: ${event}`);
245
+ const handlers = this.handlers.get(event);
246
+ const onceHandlers = this.onceHandlers.get(event);
247
+ if (handlers) {
248
+ for (const handler of handlers) {
249
+ try {
250
+ handler(data);
251
+ } catch (error) {
252
+ logger2.error(`Error in event handler for ${event}`, {
253
+ error: String(error)
254
+ });
255
+ }
256
+ }
257
+ }
258
+ if (onceHandlers) {
259
+ for (const handler of onceHandlers) {
260
+ try {
261
+ handler(data);
262
+ } catch (error) {
263
+ logger2.error(`Error in once handler for ${event}`, {
264
+ error: String(error)
265
+ });
266
+ }
267
+ }
268
+ this.onceHandlers.delete(event);
269
+ }
270
+ }
271
+ async emitAsync(event, data) {
272
+ logger2.debug(`Emitting async event: ${event}`);
273
+ const handlers = this.handlers.get(event);
274
+ const onceHandlers = this.onceHandlers.get(event);
275
+ const promises = [];
276
+ if (handlers) {
277
+ for (const handler of handlers) {
278
+ promises.push(
279
+ Promise.resolve(handler(data)).catch((error) => {
280
+ logger2.error(`Error in async handler for ${event}`, {
281
+ error: String(error)
282
+ });
283
+ })
284
+ );
285
+ }
286
+ }
287
+ if (onceHandlers) {
288
+ for (const handler of onceHandlers) {
289
+ promises.push(
290
+ Promise.resolve(handler(data)).catch((error) => {
291
+ logger2.error(`Error in async once handler for ${event}`, {
292
+ error: String(error)
293
+ });
294
+ })
295
+ );
296
+ }
297
+ this.onceHandlers.delete(event);
298
+ }
299
+ await Promise.all(promises);
300
+ }
301
+ off(event) {
302
+ this.handlers.delete(event);
303
+ this.onceHandlers.delete(event);
304
+ logger2.debug(`Removed all handlers for event: ${event}`);
305
+ }
306
+ removeAllListeners() {
307
+ this.handlers.clear();
308
+ this.onceHandlers.clear();
309
+ logger2.debug("Removed all event listeners");
310
+ }
311
+ };
312
+ var eventBus = new EventBusImpl();
313
+
314
+ // src/infrastructure/cron/cron-manager.ts
315
+ var logger3 = createLogger("CronJobManager");
316
+ var SimpleCronJob = class {
317
+ constructor(options) {
318
+ this.intervalId = null;
319
+ this.running = false;
320
+ this.name = options.name;
321
+ this.cronTime = options.cronTime;
322
+ this.onTick = options.onTick;
323
+ this.intervalMs = this.parseInterval(options.cronTime);
324
+ if (options.start) {
325
+ this.start();
326
+ }
327
+ }
328
+ /**
329
+ * Parse simple cron expressions to interval
330
+ * Supports:
331
+ * - '* * * * *' - every minute
332
+ * - '0 * * * *' - every hour
333
+ * - '0 0 * * *' - every day at midnight
334
+ * - Also supports simple intervals: '5m', '1h', '1d'
335
+ */
336
+ parseInterval(cronTime) {
337
+ const simpleMatch = cronTime.match(/^(\d+)([smhd])$/);
338
+ if (simpleMatch) {
339
+ const value = parseInt(simpleMatch[1], 10);
340
+ const unit = simpleMatch[2];
341
+ switch (unit) {
342
+ case "s":
343
+ return value * 1e3;
344
+ case "m":
345
+ return value * 60 * 1e3;
346
+ case "h":
347
+ return value * 60 * 60 * 1e3;
348
+ case "d":
349
+ return value * 24 * 60 * 60 * 1e3;
350
+ }
351
+ }
352
+ const parts = cronTime.split(" ");
353
+ if (parts.length >= 5) {
354
+ const [minute, hour, dayOfMonth, ,] = parts;
355
+ if (minute === "*" && hour === "*") {
356
+ return 60 * 1e3;
357
+ }
358
+ if (minute !== "*" && hour === "*") {
359
+ return 60 * 60 * 1e3;
360
+ }
361
+ if (minute !== "*" && hour !== "*" && dayOfMonth === "*") {
362
+ return 24 * 60 * 60 * 1e3;
363
+ }
364
+ }
365
+ logger3.warn(
366
+ `Could not parse cron expression "${cronTime}", defaulting to every minute`
367
+ );
368
+ return 60 * 1e3;
369
+ }
370
+ start() {
371
+ if (this.running) return;
372
+ logger3.info(`Starting cron job: ${this.name}`, {
373
+ interval: `${this.intervalMs}ms`
374
+ });
375
+ this.running = true;
376
+ this.intervalId = setInterval(async () => {
377
+ try {
378
+ await this.onTick();
379
+ } catch (error) {
380
+ logger3.error(`Cron job "${this.name}" failed`, {
381
+ error: String(error)
382
+ });
383
+ }
384
+ }, this.intervalMs);
385
+ }
386
+ stop() {
387
+ if (!this.running || !this.intervalId) return;
388
+ logger3.info(`Stopping cron job: ${this.name}`);
389
+ clearInterval(this.intervalId);
390
+ this.intervalId = null;
391
+ this.running = false;
392
+ }
393
+ isRunning() {
394
+ return this.running;
395
+ }
396
+ nextDate() {
397
+ if (!this.running) return null;
398
+ return new Date(Date.now() + this.intervalMs);
399
+ }
400
+ };
401
+ var CronJobManagerImpl = class {
402
+ constructor() {
403
+ this.jobs = /* @__PURE__ */ new Map();
404
+ }
405
+ addJob(options) {
406
+ if (this.jobs.has(options.name)) {
407
+ logger3.warn(`Job "${options.name}" already exists, removing old one`);
408
+ this.removeJob(options.name);
409
+ }
410
+ const job = new SimpleCronJob(options);
411
+ this.jobs.set(options.name, job);
412
+ logger3.info(`Added cron job: ${options.name}`, {
413
+ cronTime: options.cronTime
414
+ });
415
+ return job;
416
+ }
417
+ removeJob(name) {
418
+ const job = this.jobs.get(name);
419
+ if (job) {
420
+ job.stop();
421
+ this.jobs.delete(name);
422
+ logger3.info(`Removed cron job: ${name}`);
423
+ }
424
+ }
425
+ getJob(name) {
426
+ return this.jobs.get(name);
427
+ }
428
+ getAllJobs() {
429
+ return Array.from(this.jobs.values());
430
+ }
431
+ startAll() {
432
+ for (const job of this.jobs.values()) {
433
+ job.start();
434
+ }
435
+ logger3.info(`Started ${this.jobs.size} cron jobs`);
436
+ }
437
+ stopAll() {
438
+ for (const job of this.jobs.values()) {
439
+ job.stop();
440
+ }
441
+ logger3.info(`Stopped ${this.jobs.size} cron jobs`);
442
+ }
443
+ /** Get info for all jobs (for admin dashboard) */
444
+ getJobsInfo() {
445
+ const info = [];
446
+ for (const job of this.jobs.values()) {
447
+ info.push({
448
+ name: job.name,
449
+ cronTime: job.cronTime,
450
+ isRunning: job.isRunning(),
451
+ nextDate: job.nextDate()
452
+ });
453
+ }
454
+ return info;
455
+ }
456
+ /** Toggle a job on/off */
457
+ toggleJob(name) {
458
+ const job = this.jobs.get(name);
459
+ if (!job) return false;
460
+ if (job.isRunning()) {
461
+ job.stop();
462
+ } else {
463
+ job.start();
464
+ }
465
+ return true;
466
+ }
467
+ };
468
+ var cronJobManager = new CronJobManagerImpl();
469
+
470
+ // src/infrastructure/api-service.ts
471
+ var APIService = class {
472
+ constructor(baseURL, options = {}) {
473
+ this.baseURL = baseURL;
474
+ this.options = options;
475
+ }
476
+ // --------------------------------------------------------
477
+ // Core HTTP Methods
478
+ // --------------------------------------------------------
479
+ /**
480
+ * Makes a GET request to the specified endpoint
481
+ */
482
+ async get(endpoint = "", config = {}) {
483
+ const url = this.buildURL(endpoint, config.params);
484
+ return this.request("GET", url, void 0, config);
485
+ }
486
+ /**
487
+ * Makes a POST request with JSON body
488
+ */
489
+ async post(endpoint = "", data, config = {}) {
490
+ const url = this.buildURL(endpoint);
491
+ return this.request("POST", url, data, config);
492
+ }
493
+ /**
494
+ * Makes a PUT request with JSON body
495
+ */
496
+ async put(endpoint = "", data, config = {}) {
497
+ const url = this.buildURL(endpoint);
498
+ return this.request("PUT", url, data, config);
499
+ }
500
+ /**
501
+ * Makes a PATCH request with JSON body
502
+ */
503
+ async patch(endpoint = "", data, config = {}) {
504
+ const url = this.buildURL(endpoint);
505
+ return this.request("PATCH", url, data, config);
506
+ }
507
+ /**
508
+ * Makes a DELETE request
509
+ */
510
+ async delete(endpoint = "", data, config = {}) {
511
+ const url = this.buildURL(endpoint);
512
+ return this.request("DELETE", url, data, config);
513
+ }
514
+ // --------------------------------------------------------
515
+ // Internal Helpers
516
+ // --------------------------------------------------------
517
+ /**
518
+ * Build full URL with query params
519
+ */
520
+ buildURL(endpoint, params) {
521
+ const url = `${this.baseURL}${endpoint}`;
522
+ if (!params || Object.keys(params).length === 0) return url;
523
+ const searchParams = new URLSearchParams();
524
+ for (const [key, value] of Object.entries(params)) {
525
+ if (value !== void 0 && value !== null && value !== "") {
526
+ searchParams.append(key, String(value));
527
+ }
528
+ }
529
+ const queryString = searchParams.toString();
530
+ return queryString ? `${url}?${queryString}` : url;
531
+ }
532
+ /**
533
+ * Core request method — all HTTP methods funnel through here.
534
+ * Handles: headers, body, timeout, error handling, 401 redirect.
535
+ */
536
+ async request(method, url, data, config = {}) {
537
+ this.options.onRequest?.(method, url);
538
+ const headers = {
539
+ "Content-Type": "application/json",
540
+ ...this.options.defaultHeaders,
541
+ ...config.headers
542
+ };
543
+ const fetchOptions = {
544
+ method,
545
+ headers,
546
+ credentials: "include",
547
+ // Send cookies (same as axios withCredentials)
548
+ signal: config.signal
549
+ };
550
+ if (data !== void 0 && method !== "GET") {
551
+ fetchOptions.body = JSON.stringify(data);
552
+ }
553
+ try {
554
+ const response = await fetch(url, fetchOptions);
555
+ this.options.onResponse?.(method, url, response.status);
556
+ if (response.status === 401) {
557
+ const error = {
558
+ message: "Unauthorized",
559
+ status: 401,
560
+ code: "UNAUTHORIZED"
561
+ };
562
+ this.options.onUnauthorized?.();
563
+ this.options.onError?.(error);
564
+ throw error;
565
+ }
566
+ if (!response.ok) {
567
+ let errorBody;
568
+ try {
569
+ errorBody = await response.json();
570
+ } catch {
571
+ errorBody = await response.text();
572
+ }
573
+ const error = {
574
+ message: typeof errorBody === "object" && errorBody !== null && "message" in errorBody ? String(errorBody.message) : `HTTP Error ${response.status}`,
575
+ status: response.status,
576
+ code: typeof errorBody === "object" && errorBody !== null && "code" in errorBody ? String(errorBody.code) : void 0,
577
+ details: errorBody
578
+ };
579
+ this.options.onError?.(error);
580
+ throw error;
581
+ }
582
+ if (response.status === 204) {
583
+ return void 0;
584
+ }
585
+ const responseData = await response.json();
586
+ return responseData;
587
+ } catch (error) {
588
+ if (this.isAPIError(error)) {
589
+ throw error;
590
+ }
591
+ const apiError = {
592
+ message: error instanceof Error ? error.message : "Network error",
593
+ status: 0,
594
+ code: "NETWORK_ERROR"
595
+ };
596
+ this.options.onError?.(apiError);
597
+ throw apiError;
598
+ }
599
+ }
600
+ /**
601
+ * Type guard for APIError
602
+ */
603
+ isAPIError(error) {
604
+ return typeof error === "object" && error !== null && "message" in error && "status" in error;
605
+ }
606
+ };
607
+ var defaultClientOptions = {
608
+ onUnauthorized: () => {
609
+ if (typeof window !== "undefined") {
610
+ const currentPath = window.location.pathname;
611
+ window.location.replace(`/login?next_path=${encodeURIComponent(currentPath)}`);
612
+ }
613
+ }
614
+ };
615
+ function isAPIError(error) {
616
+ return typeof error === "object" && error !== null && "message" in error && "status" in error && typeof error.status === "number";
617
+ }
618
+
619
+ exports.APIService = APIService;
620
+ exports.CacheManagerImpl = CacheManagerImpl;
621
+ exports.CronJobManagerImpl = CronJobManagerImpl;
622
+ exports.EventBusImpl = EventBusImpl;
623
+ exports.MemoryCache = MemoryCache;
624
+ exports.SimpleCronJob = SimpleCronJob;
625
+ exports.cacheManager = cacheManager;
626
+ exports.createLogger = createLogger;
627
+ exports.cronJobManager = cronJobManager;
628
+ exports.defaultClientOptions = defaultClientOptions;
629
+ exports.eventBus = eventBus;
630
+ exports.isAPIError = isAPIError;
631
+ exports.logger = logger;
632
+ //# sourceMappingURL=index.js.map
633
+ //# sourceMappingURL=index.js.map