@buenojs/bueno 0.8.3 → 0.8.5

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 (218) hide show
  1. package/README.md +136 -16
  2. package/dist/cli/{index.js → bin.js} +3036 -1421
  3. package/dist/container/index.js +250 -0
  4. package/dist/context/index.js +219 -0
  5. package/dist/database/index.js +493 -0
  6. package/dist/frontend/index.js +7697 -0
  7. package/dist/health/index.js +364 -0
  8. package/dist/i18n/index.js +345 -0
  9. package/dist/index.js +11043 -6482
  10. package/dist/jobs/index.js +819 -0
  11. package/dist/lock/index.js +367 -0
  12. package/dist/logger/index.js +281 -0
  13. package/dist/metrics/index.js +289 -0
  14. package/dist/middleware/index.js +77 -0
  15. package/dist/migrations/index.js +571 -0
  16. package/dist/modules/index.js +3346 -0
  17. package/dist/notification/index.js +484 -0
  18. package/dist/observability/index.js +331 -0
  19. package/dist/openapi/index.js +776 -0
  20. package/dist/orm/index.js +1356 -0
  21. package/dist/router/index.js +886 -0
  22. package/dist/rpc/index.js +691 -0
  23. package/dist/schema/index.js +400 -0
  24. package/dist/telemetry/index.js +595 -0
  25. package/dist/template/index.js +640 -0
  26. package/dist/templates/index.js +640 -0
  27. package/dist/testing/index.js +1111 -0
  28. package/dist/types/index.js +60 -0
  29. package/package.json +121 -27
  30. package/src/cache/index.ts +2 -1
  31. package/src/cli/bin.ts +2 -2
  32. package/src/cli/commands/build.ts +183 -165
  33. package/src/cli/commands/dev.ts +96 -89
  34. package/src/cli/commands/generate.ts +142 -111
  35. package/src/cli/commands/help.ts +20 -16
  36. package/src/cli/commands/index.ts +3 -6
  37. package/src/cli/commands/migration.ts +124 -105
  38. package/src/cli/commands/new.ts +392 -438
  39. package/src/cli/commands/start.ts +81 -79
  40. package/src/cli/core/args.ts +68 -50
  41. package/src/cli/core/console.ts +89 -95
  42. package/src/cli/core/index.ts +4 -4
  43. package/src/cli/core/prompt.ts +65 -62
  44. package/src/cli/core/spinner.ts +23 -20
  45. package/src/cli/index.ts +46 -38
  46. package/src/cli/templates/database/index.ts +61 -0
  47. package/src/cli/templates/database/mysql.ts +14 -0
  48. package/src/cli/templates/database/none.ts +16 -0
  49. package/src/cli/templates/database/postgresql.ts +14 -0
  50. package/src/cli/templates/database/sqlite.ts +14 -0
  51. package/src/cli/templates/deploy.ts +29 -26
  52. package/src/cli/templates/docker.ts +41 -30
  53. package/src/cli/templates/frontend/index.ts +63 -0
  54. package/src/cli/templates/frontend/none.ts +17 -0
  55. package/src/cli/templates/frontend/react.ts +140 -0
  56. package/src/cli/templates/frontend/solid.ts +134 -0
  57. package/src/cli/templates/frontend/svelte.ts +131 -0
  58. package/src/cli/templates/frontend/vue.ts +130 -0
  59. package/src/cli/templates/generators/index.ts +339 -0
  60. package/src/cli/templates/generators/types.ts +56 -0
  61. package/src/cli/templates/index.ts +35 -2
  62. package/src/cli/templates/project/api.ts +81 -0
  63. package/src/cli/templates/project/default.ts +140 -0
  64. package/src/cli/templates/project/fullstack.ts +111 -0
  65. package/src/cli/templates/project/index.ts +95 -0
  66. package/src/cli/templates/project/minimal.ts +45 -0
  67. package/src/cli/templates/project/types.ts +94 -0
  68. package/src/cli/templates/project/website.ts +263 -0
  69. package/src/cli/utils/fs.ts +55 -41
  70. package/src/cli/utils/index.ts +3 -2
  71. package/src/cli/utils/strings.ts +47 -33
  72. package/src/cli/utils/version.ts +47 -0
  73. package/src/config/env-validation.ts +100 -0
  74. package/src/config/env.ts +169 -41
  75. package/src/config/index.ts +28 -20
  76. package/src/config/loader.ts +25 -16
  77. package/src/config/merge.ts +21 -10
  78. package/src/config/types.ts +545 -25
  79. package/src/config/validation.ts +215 -7
  80. package/src/container/forward-ref.ts +22 -22
  81. package/src/container/index.ts +34 -12
  82. package/src/context/index.ts +11 -1
  83. package/src/database/index.ts +7 -190
  84. package/src/database/orm/builder.ts +457 -0
  85. package/src/database/orm/casts/index.ts +130 -0
  86. package/src/database/orm/casts/types.ts +25 -0
  87. package/src/database/orm/compiler.ts +304 -0
  88. package/src/database/orm/hooks/index.ts +114 -0
  89. package/src/database/orm/index.ts +61 -0
  90. package/src/database/orm/model-registry.ts +59 -0
  91. package/src/database/orm/model.ts +821 -0
  92. package/src/database/orm/relationships/base.ts +146 -0
  93. package/src/database/orm/relationships/belongs-to-many.ts +179 -0
  94. package/src/database/orm/relationships/belongs-to.ts +56 -0
  95. package/src/database/orm/relationships/has-many.ts +45 -0
  96. package/src/database/orm/relationships/has-one.ts +41 -0
  97. package/src/database/orm/relationships/index.ts +11 -0
  98. package/src/database/orm/scopes/index.ts +55 -0
  99. package/src/events/__tests__/event-system.test.ts +235 -0
  100. package/src/events/config.ts +238 -0
  101. package/src/events/example-usage.ts +185 -0
  102. package/src/events/index.ts +278 -0
  103. package/src/events/manager.ts +385 -0
  104. package/src/events/registry.ts +182 -0
  105. package/src/events/types.ts +124 -0
  106. package/src/frontend/api-routes.ts +65 -23
  107. package/src/frontend/bundler.ts +76 -34
  108. package/src/frontend/console-client.ts +2 -2
  109. package/src/frontend/console-stream.ts +94 -38
  110. package/src/frontend/dev-server.ts +94 -46
  111. package/src/frontend/file-router.ts +61 -19
  112. package/src/frontend/frameworks/index.ts +37 -10
  113. package/src/frontend/frameworks/react.ts +10 -8
  114. package/src/frontend/frameworks/solid.ts +11 -9
  115. package/src/frontend/frameworks/svelte.ts +15 -9
  116. package/src/frontend/frameworks/vue.ts +13 -11
  117. package/src/frontend/hmr-client.ts +12 -10
  118. package/src/frontend/hmr.ts +146 -103
  119. package/src/frontend/index.ts +14 -5
  120. package/src/frontend/islands.ts +41 -22
  121. package/src/frontend/isr.ts +59 -37
  122. package/src/frontend/layout.ts +36 -21
  123. package/src/frontend/ssr/react.ts +74 -27
  124. package/src/frontend/ssr/solid.ts +54 -20
  125. package/src/frontend/ssr/svelte.ts +48 -14
  126. package/src/frontend/ssr/vue.ts +50 -18
  127. package/src/frontend/ssr.ts +83 -39
  128. package/src/frontend/types.ts +91 -56
  129. package/src/health/index.ts +21 -9
  130. package/src/i18n/engine.ts +305 -0
  131. package/src/i18n/index.ts +38 -0
  132. package/src/i18n/loader.ts +218 -0
  133. package/src/i18n/middleware.ts +164 -0
  134. package/src/i18n/negotiator.ts +162 -0
  135. package/src/i18n/types.ts +158 -0
  136. package/src/index.ts +179 -27
  137. package/src/jobs/drivers/memory.ts +315 -0
  138. package/src/jobs/drivers/redis.ts +459 -0
  139. package/src/jobs/index.ts +30 -0
  140. package/src/jobs/queue.ts +281 -0
  141. package/src/jobs/types.ts +295 -0
  142. package/src/jobs/worker.ts +380 -0
  143. package/src/logger/index.ts +1 -3
  144. package/src/logger/transports/index.ts +62 -22
  145. package/src/metrics/index.ts +25 -16
  146. package/src/migrations/index.ts +9 -0
  147. package/src/modules/filters.ts +13 -17
  148. package/src/modules/guards.ts +49 -26
  149. package/src/modules/index.ts +409 -298
  150. package/src/modules/interceptors.ts +58 -20
  151. package/src/modules/lazy.ts +11 -19
  152. package/src/modules/lifecycle.ts +15 -7
  153. package/src/modules/metadata.ts +15 -5
  154. package/src/modules/pipes.ts +94 -72
  155. package/src/notification/channels/base.ts +68 -0
  156. package/src/notification/channels/email.ts +105 -0
  157. package/src/notification/channels/push.ts +104 -0
  158. package/src/notification/channels/sms.ts +105 -0
  159. package/src/notification/channels/whatsapp.ts +104 -0
  160. package/src/notification/index.ts +48 -0
  161. package/src/notification/service.ts +354 -0
  162. package/src/notification/types.ts +344 -0
  163. package/src/observability/__tests__/observability.test.ts +483 -0
  164. package/src/observability/breadcrumbs.ts +114 -0
  165. package/src/observability/index.ts +136 -0
  166. package/src/observability/interceptor.ts +85 -0
  167. package/src/observability/service.ts +303 -0
  168. package/src/observability/trace.ts +37 -0
  169. package/src/observability/types.ts +196 -0
  170. package/src/openapi/__tests__/decorators.test.ts +335 -0
  171. package/src/openapi/__tests__/document-builder.test.ts +285 -0
  172. package/src/openapi/__tests__/route-scanner.test.ts +334 -0
  173. package/src/openapi/__tests__/schema-generator.test.ts +275 -0
  174. package/src/openapi/decorators.ts +328 -0
  175. package/src/openapi/document-builder.ts +274 -0
  176. package/src/openapi/index.ts +112 -0
  177. package/src/openapi/metadata.ts +112 -0
  178. package/src/openapi/route-scanner.ts +289 -0
  179. package/src/openapi/schema-generator.ts +256 -0
  180. package/src/openapi/swagger-module.ts +166 -0
  181. package/src/openapi/types.ts +398 -0
  182. package/src/orm/index.ts +10 -0
  183. package/src/rpc/index.ts +3 -1
  184. package/src/schema/index.ts +9 -0
  185. package/src/security/index.ts +15 -6
  186. package/src/ssg/index.ts +9 -8
  187. package/src/telemetry/index.ts +76 -22
  188. package/src/template/index.ts +7 -0
  189. package/src/templates/engine.ts +224 -0
  190. package/src/templates/index.ts +9 -0
  191. package/src/templates/loader.ts +331 -0
  192. package/src/templates/renderers/markdown.ts +212 -0
  193. package/src/templates/renderers/simple.ts +269 -0
  194. package/src/templates/types.ts +154 -0
  195. package/src/testing/index.ts +100 -27
  196. package/src/types/optional-deps.d.ts +347 -187
  197. package/src/validation/index.ts +92 -2
  198. package/src/validation/schemas.ts +536 -0
  199. package/tests/integration/fullstack.test.ts +4 -4
  200. package/tests/unit/database.test.ts +2 -72
  201. package/tests/unit/env-validation.test.ts +166 -0
  202. package/tests/unit/events.test.ts +910 -0
  203. package/tests/unit/i18n.test.ts +455 -0
  204. package/tests/unit/jobs.test.ts +493 -0
  205. package/tests/unit/notification.test.ts +988 -0
  206. package/tests/unit/observability.test.ts +453 -0
  207. package/tests/unit/orm/builder.test.ts +323 -0
  208. package/tests/unit/orm/casts.test.ts +179 -0
  209. package/tests/unit/orm/compiler.test.ts +220 -0
  210. package/tests/unit/orm/eager-loading.test.ts +285 -0
  211. package/tests/unit/orm/hooks.test.ts +191 -0
  212. package/tests/unit/orm/model.test.ts +373 -0
  213. package/tests/unit/orm/relationships.test.ts +303 -0
  214. package/tests/unit/orm/scopes.test.ts +74 -0
  215. package/tests/unit/templates-simple.test.ts +53 -0
  216. package/tests/unit/templates.test.ts +454 -0
  217. package/tests/unit/validation.test.ts +18 -24
  218. package/tsconfig.json +11 -3
@@ -0,0 +1,331 @@
1
+ // @bun
2
+ var __defProp = Object.defineProperty;
3
+ var __export = (target, all) => {
4
+ for (var name in all)
5
+ __defProp(target, name, {
6
+ get: all[name],
7
+ enumerable: true,
8
+ configurable: true,
9
+ set: (newValue) => all[name] = () => newValue
10
+ });
11
+ };
12
+ var __legacyDecorateClassTS = function(decorators, target, key, desc) {
13
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
14
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function")
15
+ r = Reflect.decorate(decorators, target, key, desc);
16
+ else
17
+ for (var i = decorators.length - 1;i >= 0; i--)
18
+ if (d = decorators[i])
19
+ r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
20
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
21
+ };
22
+ var __legacyMetadataTS = (k, v) => {
23
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function")
24
+ return Reflect.metadata(k, v);
25
+ };
26
+ var __esm = (fn, res) => () => (fn && (res = fn(fn = 0)), res);
27
+ var __require = import.meta.require;
28
+
29
+ // src/observability/breadcrumbs.ts
30
+ class BreadcrumbCollector {
31
+ _buffer;
32
+ _maxSize;
33
+ _head = 0;
34
+ _size = 0;
35
+ constructor(maxSize = 20) {
36
+ if (maxSize < 1)
37
+ throw new RangeError("maxSize must be at least 1");
38
+ this._maxSize = maxSize;
39
+ this._buffer = new Array(maxSize);
40
+ }
41
+ add(entry) {
42
+ const index = (this._head + this._size) % this._maxSize;
43
+ this._buffer[index] = entry;
44
+ if (this._size < this._maxSize) {
45
+ this._size++;
46
+ } else {
47
+ this._head = (this._head + 1) % this._maxSize;
48
+ }
49
+ }
50
+ getAll() {
51
+ const result = [];
52
+ for (let i = 0;i < this._size; i++) {
53
+ result.push(this._buffer[(this._head + i) % this._maxSize]);
54
+ }
55
+ return result;
56
+ }
57
+ clear() {
58
+ this._head = 0;
59
+ this._size = 0;
60
+ }
61
+ get size() {
62
+ return this._size;
63
+ }
64
+ get maxSize() {
65
+ return this._maxSize;
66
+ }
67
+ }
68
+ function httpBreadcrumb(method, path, statusCode, durationMs) {
69
+ const level = statusCode !== undefined && statusCode >= 400 ? "error" : "info";
70
+ const data = { method, path };
71
+ if (statusCode !== undefined)
72
+ data.statusCode = statusCode;
73
+ if (durationMs !== undefined)
74
+ data.durationMs = durationMs;
75
+ return {
76
+ timestamp: new Date,
77
+ type: "http",
78
+ level,
79
+ message: `${method} ${path}${statusCode !== undefined ? ` ${statusCode}` : ""}`,
80
+ data
81
+ };
82
+ }
83
+ function logBreadcrumb(level, message, data) {
84
+ return {
85
+ timestamp: new Date,
86
+ type: "log",
87
+ level,
88
+ message,
89
+ data
90
+ };
91
+ }
92
+
93
+ // src/observability/service.ts
94
+ function generateId() {
95
+ const bytes = new Uint8Array(16);
96
+ crypto.getRandomValues(bytes);
97
+ return Array.from(bytes).map((b) => b.toString(16).padStart(2, "0")).join("");
98
+ }
99
+ function extractSafeHeaders(req) {
100
+ const safe = {};
101
+ const sensitiveHeaders = new Set([
102
+ "authorization",
103
+ "cookie",
104
+ "x-api-key",
105
+ "x-auth-token",
106
+ "proxy-authorization"
107
+ ]);
108
+ req.headers.forEach((value, key) => {
109
+ if (!sensitiveHeaders.has(key.toLowerCase())) {
110
+ safe[key] = value;
111
+ }
112
+ });
113
+ return safe;
114
+ }
115
+ function getStatusCode(error) {
116
+ const e = error;
117
+ return e.statusCode ?? e.status;
118
+ }
119
+
120
+ class ObservabilityService {
121
+ reporter;
122
+ breadcrumbs;
123
+ options;
124
+ constructor(options) {
125
+ this.reporter = options.reporter;
126
+ this.options = {
127
+ breadcrumbsSize: options.breadcrumbsSize ?? 20,
128
+ ignoreErrors: options.ignoreErrors ?? [],
129
+ ignoreStatusCodes: options.ignoreStatusCodes ?? [],
130
+ tags: options.tags ?? {},
131
+ captureUnhandled: options.captureUnhandled ?? false
132
+ };
133
+ this.breadcrumbs = new BreadcrumbCollector(this.options.breadcrumbsSize);
134
+ if (this.options.captureUnhandled) {
135
+ this.setupUnhandledCapture();
136
+ }
137
+ }
138
+ captureFromContext(context, error) {
139
+ if (this.shouldIgnore(error))
140
+ return;
141
+ this.breadcrumbs.add(httpBreadcrumb(context.method, context.path, undefined));
142
+ const event = this.assembleEvent(error, "error", context);
143
+ this.dispatchAsync(event);
144
+ }
145
+ registerShutdown() {
146
+ const flush = async () => {
147
+ if (this.reporter.flush) {
148
+ try {
149
+ await this.reporter.flush();
150
+ } catch (err) {
151
+ console.error("[ObservabilityService] Reporter flush failed:", err);
152
+ }
153
+ }
154
+ };
155
+ process.once("beforeExit", () => {
156
+ flush().catch(console.error);
157
+ });
158
+ }
159
+ captureError(error, level = "error") {
160
+ if (this.shouldIgnore(error))
161
+ return;
162
+ const event = this.assembleEvent(error, level);
163
+ this.dispatchAsync(event);
164
+ }
165
+ captureMessage(message, level = "info", extra) {
166
+ if (!this.reporter.captureMessage)
167
+ return;
168
+ const event = {
169
+ id: generateId(),
170
+ timestamp: new Date,
171
+ message,
172
+ level,
173
+ extra
174
+ };
175
+ this.dispatchMessageAsync(event);
176
+ }
177
+ addBreadcrumb(entry) {
178
+ this.breadcrumbs.add({
179
+ ...entry,
180
+ timestamp: entry.timestamp ?? new Date
181
+ });
182
+ }
183
+ getBreadcrumbCollector() {
184
+ return this.breadcrumbs;
185
+ }
186
+ assembleEvent(error, level, context) {
187
+ const user = context?.get("user");
188
+ const traceId = context?.get("traceId") ?? undefined;
189
+ const spanId = context?.get("spanId") ?? undefined;
190
+ return {
191
+ id: generateId(),
192
+ timestamp: new Date,
193
+ error,
194
+ level,
195
+ request: context ? {
196
+ method: context.method,
197
+ path: context.path,
198
+ headers: extractSafeHeaders(context.req),
199
+ ip: context.ip ?? "",
200
+ userAgent: context.getHeader("user-agent")
201
+ } : undefined,
202
+ traceId,
203
+ spanId,
204
+ user: user ?? undefined,
205
+ breadcrumbs: this.breadcrumbs.getAll(),
206
+ tags: { ...this.options.tags }
207
+ };
208
+ }
209
+ shouldIgnore(error) {
210
+ for (const ErrorClass of this.options.ignoreErrors) {
211
+ if (error instanceof ErrorClass)
212
+ return true;
213
+ }
214
+ if (this.options.ignoreStatusCodes.length > 0) {
215
+ const code = getStatusCode(error);
216
+ if (code !== undefined && this.options.ignoreStatusCodes.includes(code)) {
217
+ return true;
218
+ }
219
+ }
220
+ return false;
221
+ }
222
+ dispatchAsync(event) {
223
+ Promise.resolve(this.reporter.captureError(event)).catch((err) => {
224
+ console.error("[ObservabilityService] Reporter.captureError failed:", err);
225
+ });
226
+ }
227
+ dispatchMessageAsync(event) {
228
+ if (!this.reporter.captureMessage)
229
+ return;
230
+ Promise.resolve(this.reporter.captureMessage(event)).catch((err) => {
231
+ console.error("[ObservabilityService] Reporter.captureMessage failed:", err);
232
+ });
233
+ }
234
+ setupUnhandledCapture() {
235
+ process.on("unhandledRejection", (reason) => {
236
+ const error = reason instanceof Error ? reason : new Error(String(reason ?? "Unhandled rejection"));
237
+ this.captureError(error, "error");
238
+ });
239
+ process.on("uncaughtException", (error) => {
240
+ this.captureError(error, "fatal");
241
+ });
242
+ }
243
+ }
244
+ function extractTraceContext(context) {
245
+ const traceparent = context.getHeader("traceparent");
246
+ if (!traceparent)
247
+ return {};
248
+ const parts = traceparent.split("-");
249
+ if (parts.length < 4)
250
+ return {};
251
+ return { traceId: parts[1], spanId: parts[2] };
252
+ }
253
+
254
+ // src/observability/trace.ts
255
+ function generateTraceId() {
256
+ const bytes = new Uint8Array(16);
257
+ crypto.getRandomValues(bytes);
258
+ return Array.from(bytes).map((b) => b.toString(16).padStart(2, "0")).join("");
259
+ }
260
+ function generateSpanId() {
261
+ const bytes = new Uint8Array(8);
262
+ crypto.getRandomValues(bytes);
263
+ return Array.from(bytes).map((b) => b.toString(16).padStart(2, "0")).join("");
264
+ }
265
+ function buildTraceparent(traceId, spanId) {
266
+ return `00-${traceId}-${spanId}-01`;
267
+ }
268
+
269
+ // src/observability/interceptor.ts
270
+ function resolveTraceIds(context) {
271
+ const existing = extractTraceContext(context);
272
+ if (existing.traceId && existing.spanId) {
273
+ return { traceId: existing.traceId, spanId: existing.spanId };
274
+ }
275
+ return { traceId: generateTraceId(), spanId: generateSpanId() };
276
+ }
277
+
278
+ class ObservabilityInterceptor {
279
+ service;
280
+ constructor(service) {
281
+ this.service = service;
282
+ }
283
+ async intercept(context, next) {
284
+ const startMs = Date.now();
285
+ const { traceId, spanId } = resolveTraceIds(context);
286
+ context.set("traceId", traceId);
287
+ context.set("spanId", spanId);
288
+ this.service.addBreadcrumb({
289
+ type: "navigation",
290
+ level: "info",
291
+ message: `${context.method} ${context.path}`,
292
+ data: { traceId, spanId }
293
+ });
294
+ try {
295
+ const result = await next.handle();
296
+ const durationMs = Date.now() - startMs;
297
+ const response = result;
298
+ const statusCode = response instanceof Response ? response.status : undefined;
299
+ this.service.getBreadcrumbCollector().add(httpBreadcrumb(context.method, context.path, statusCode, durationMs));
300
+ return result;
301
+ } catch (error) {
302
+ this.service.captureFromContext(context, error);
303
+ const durationMs = Date.now() - startMs;
304
+ this.service.getBreadcrumbCollector().add(httpBreadcrumb(context.method, context.path, undefined, durationMs));
305
+ throw error;
306
+ }
307
+ }
308
+ }
309
+
310
+ // src/observability/index.ts
311
+ class ObservabilityModule {
312
+ static setup(app, options) {
313
+ const service = new ObservabilityService(options);
314
+ const interceptor = new ObservabilityInterceptor(service);
315
+ app.useGlobalInterceptors(interceptor);
316
+ service.registerShutdown();
317
+ return service;
318
+ }
319
+ }
320
+ export {
321
+ logBreadcrumb,
322
+ httpBreadcrumb,
323
+ generateTraceId,
324
+ generateSpanId,
325
+ extractTraceContext,
326
+ buildTraceparent,
327
+ ObservabilityService,
328
+ ObservabilityModule,
329
+ ObservabilityInterceptor,
330
+ BreadcrumbCollector
331
+ };