@hazeljs/core 0.2.0-beta.26 → 0.2.0-beta.27

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.
@@ -1 +1 @@
1
- {"version":3,"file":"hazel-app.d.ts","sourceRoot":"","sources":["../src/hazel-app.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"hazel-app.d.ts","sourceRoot":"","sources":["../src/hazel-app.ts"],"names":[],"mappings":"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"}
package/dist/hazel-app.js CHANGED
@@ -94,7 +94,7 @@ class HazelApp {
94
94
  if (visited.has(moduleType))
95
95
  return [];
96
96
  visited.add(moduleType);
97
- const metadata = Reflect.getMetadata(MODULE_METADATA_KEY, moduleType) || {};
97
+ const metadata = (0, hazel_module_1.getModuleMetadata)(moduleType) || {};
98
98
  const controllers = [];
99
99
  // Collect from imported modules first
100
100
  if (metadata.imports) {
@@ -160,7 +160,7 @@ class HazelApp {
160
160
  return;
161
161
  }
162
162
  const { method, url, headers } = req;
163
- logger_1.default.info('Incoming request:', { method, url, headers });
163
+ logger_1.default.debug('Incoming request:', { method, url, headers });
164
164
  // Handle CORS
165
165
  if (this.corsEnabled) {
166
166
  const origin = headers['origin'] || '*';
@@ -183,9 +183,11 @@ class HazelApp {
183
183
  return;
184
184
  }
185
185
  }
186
- // Parse request body for POST/PUT/PATCH requests
186
+ // Parse request body for POST/PUT/PATCH requests (skip multipart - let route handle it)
187
187
  let body = undefined;
188
- if (method === 'POST' || method === 'PUT' || method === 'PATCH') {
188
+ const contentType = headers['content-type'] || '';
189
+ const isMultipart = contentType.includes('multipart/form-data');
190
+ if ((method === 'POST' || method === 'PUT' || method === 'PATCH') && !isMultipart) {
189
191
  try {
190
192
  const chunks = [];
191
193
  req.on('data', (chunk) => chunks.push(chunk));
@@ -194,7 +196,6 @@ class HazelApp {
194
196
  try {
195
197
  const bodyStr = Buffer.concat(chunks).toString();
196
198
  if (bodyStr) {
197
- const contentType = headers['content-type'] || '';
198
199
  if (contentType.includes('application/json')) {
199
200
  body = JSON.parse(bodyStr);
200
201
  }
@@ -1 +1 @@
1
- {"version":3,"file":"hazel-module.d.ts","sourceRoot":"","sources":["../src/hazel-module.ts"],"names":[],"mappings":"AAAA,OAAO,kBAAkB,CAAC;AAC1B,OAAO,EAAE,IAAI,EAAE,MAAM,SAAS,CAAC;AAC/B,OAAO,EAAE,SAAS,EAAS,MAAM,aAAa,CAAC;AAM/C,MAAM,WAAW,aAAa;IAC5B,OAAO,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;IAC1B,WAAW,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;IAC9B,SAAS,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;IAC5B,OAAO,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;CAC3B;AAED,wBAAgB,WAAW,CAAC,OAAO,EAAE,aAAa,GAAG,cAAc,CAIlE;AAGD,eAAO,MAAM,MAAM,oBAAc,CAAC;AAElC,wBAAgB,iBAAiB,CAAC,MAAM,EAAE,MAAM,GAAG,aAAa,GAAG,SAAS,CAE3E;AAED,qBAAa,mBAAmB;IAGlB,OAAO,CAAC,QAAQ,CAAC,UAAU;IAFvC,OAAO,CAAC,SAAS,CAAY;gBAEA,UAAU,EAAE,IAAI,CAAC,OAAO,CAAC;IAMtD,OAAO,CAAC,UAAU;IA8FlB,YAAY,IAAI,SAAS;CAG1B"}
1
+ {"version":3,"file":"hazel-module.d.ts","sourceRoot":"","sources":["../src/hazel-module.ts"],"names":[],"mappings":"AAAA,OAAO,kBAAkB,CAAC;AAC1B,OAAO,EAAE,IAAI,EAAE,MAAM,SAAS,CAAC;AAC/B,OAAO,EAAE,SAAS,EAAS,MAAM,aAAa,CAAC;AAM/C,MAAM,WAAW,aAAa;IAC5B,OAAO,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;IAC1B,WAAW,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;IAC9B,SAAS,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;IAC5B,OAAO,CAAC,EAAE,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;CAC3B;AAED,wBAAgB,WAAW,CAAC,OAAO,EAAE,aAAa,GAAG,cAAc,CAIlE;AAGD,eAAO,MAAM,MAAM,oBAAc,CAAC;AAElC,wBAAgB,iBAAiB,CAAC,MAAM,EAAE,MAAM,GAAG,aAAa,GAAG,SAAS,CAa3E;AAED,qBAAa,mBAAmB;IAGlB,OAAO,CAAC,QAAQ,CAAC,UAAU;IAFvC,OAAO,CAAC,SAAS,CAAY;gBAEA,UAAU,EAAE,IAAI,CAAC,OAAO,CAAC;IAOtD,OAAO,CAAC,UAAU;IA0GlB,YAAY,IAAI,SAAS;CAG1B"}
@@ -19,45 +19,77 @@ function HazelModule(options) {
19
19
  // Alias for backward compatibility
20
20
  exports.Module = HazelModule;
21
21
  function getModuleMetadata(target) {
22
- return Reflect.getMetadata(MODULE_METADATA_KEY, target);
22
+ const fromDecorator = Reflect.getMetadata(MODULE_METADATA_KEY, target);
23
+ if (fromDecorator)
24
+ return fromDecorator;
25
+ // Support dynamic modules: { module, providers?, controllers?, imports? }
26
+ if (target && typeof target === 'object' && 'module' in target) {
27
+ const dyn = target;
28
+ return {
29
+ providers: dyn.providers,
30
+ controllers: dyn.controllers,
31
+ imports: dyn.imports,
32
+ };
33
+ }
34
+ return undefined;
23
35
  }
24
36
  class HazelModuleInstance {
25
37
  constructor(moduleType) {
26
38
  this.moduleType = moduleType;
27
- logger_1.default.debug(`Initializing HazelModule: ${moduleType.name}`);
39
+ const name = moduleType?.name ?? moduleType?.module?.name ?? 'DynamicModule';
40
+ logger_1.default.debug(`Initializing HazelModule: ${name}`);
28
41
  this.container = container_1.Container.getInstance();
29
42
  this.initialize();
30
43
  }
31
44
  initialize() {
32
45
  const metadata = getModuleMetadata(this.moduleType) || {};
33
46
  logger_1.default.debug('Module metadata:', metadata);
47
+ // Initialize imported modules first (so their providers are available)
48
+ if (metadata.imports) {
49
+ logger_1.default.debug('Initializing imported modules:', metadata.imports.map((m) => (m && typeof m === 'object' && 'module' in m ? m.module?.name : m?.name)));
50
+ metadata.imports.forEach((moduleType) => {
51
+ new HazelModuleInstance(moduleType);
52
+ });
53
+ }
34
54
  // Register providers
35
55
  if (metadata.providers) {
36
- logger_1.default.debug('Registering providers:', metadata.providers.map((p) => p.name));
56
+ logger_1.default.debug('Registering providers:', metadata.providers.map((p) => (p && typeof p === 'object' && 'provide' in p ? p.provide : p?.name)));
37
57
  metadata.providers.forEach((provider) => {
38
- logger_1.default.debug(`Registering provider: ${provider.name}`);
58
+ // Dynamic module provider: { provide, useFactory?, useClass?, useValue? } (NestJS-style)
59
+ if (provider && typeof provider === 'object' && ('provide' in provider || 'token' in provider)) {
60
+ const p = provider;
61
+ const token = p.token ?? p.provide;
62
+ logger_1.default.debug(`Registering provider config for: ${token}`);
63
+ this.container.registerProvider({
64
+ token,
65
+ useFactory: p.useFactory,
66
+ useClass: p.useClass,
67
+ useValue: p.useValue,
68
+ inject: p.inject,
69
+ });
70
+ return;
71
+ }
72
+ const cls = provider;
73
+ logger_1.default.debug(`Registering provider: ${cls?.name}`);
39
74
  // Check if provider is request-scoped
40
- const scope = Reflect.getMetadata('hazel:scope', provider);
75
+ const scope = Reflect.getMetadata('hazel:scope', cls);
41
76
  if (scope === 'request') {
42
77
  // Don't eagerly resolve request-scoped providers
43
- // They will be resolved per-request by the container
44
- logger_1.default.debug(`Skipping eager resolution for request-scoped provider: ${provider.name}`);
45
- // Just register the class itself, not an instance
46
78
  this.container.registerProvider({
47
- token: provider,
48
- useClass: provider,
79
+ token: cls,
80
+ useClass: cls,
49
81
  scope: container_1.Scope.REQUEST,
50
82
  });
51
83
  }
52
84
  else {
53
85
  // Eagerly resolve singleton and transient providers
54
- this.container.register(provider, this.container.resolve(provider));
86
+ this.container.register(cls, this.container.resolve(cls));
55
87
  }
56
88
  });
57
89
  }
58
90
  // Register controllers
59
91
  if (metadata.controllers) {
60
- logger_1.default.debug('Registering controllers:', metadata.controllers.map((c) => c.name));
92
+ logger_1.default.debug('Registering controllers:', metadata.controllers.map((c) => c?.name));
61
93
  metadata.controllers.forEach((controller) => {
62
94
  logger_1.default.debug(`Registering controller: ${controller.name}`);
63
95
  // Check if controller has request-scoped dependencies
@@ -94,13 +126,6 @@ class HazelModuleInstance {
94
126
  });
95
127
  });
96
128
  }
97
- // Initialize imported modules
98
- if (metadata.imports) {
99
- logger_1.default.debug('Initializing imported modules:', metadata.imports.map((m) => m.name));
100
- metadata.imports.forEach((moduleType) => {
101
- new HazelModuleInstance(moduleType);
102
- });
103
- }
104
129
  }
105
130
  getContainer() {
106
131
  return this.container;
@@ -5,6 +5,8 @@ export interface HazelResponse {
5
5
  end(): void;
6
6
  status(code: number): HazelResponse;
7
7
  json(data: unknown): void;
8
+ /** Send a Buffer as binary (e.g. audio, PDF). Sets Content-Type if provided. */
9
+ sendBuffer?(buffer: Buffer, contentType?: string): void;
8
10
  }
9
11
  export declare class HazelExpressResponse implements HazelResponse {
10
12
  private res;
@@ -15,6 +17,7 @@ export declare class HazelExpressResponse implements HazelResponse {
15
17
  write(chunk: string): void;
16
18
  end(): void;
17
19
  status(code: number): HazelResponse;
20
+ sendBuffer(buffer: Buffer, contentType?: string): void;
18
21
  json(data: unknown): void;
19
22
  }
20
23
  //# sourceMappingURL=hazel-response.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"hazel-response.d.ts","sourceRoot":"","sources":["../src/hazel-response.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAEnC,MAAM,WAAW,aAAa;IAC5B,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7C,KAAK,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,GAAG,IAAI,IAAI,CAAC;IACZ,MAAM,CAAC,IAAI,EAAE,MAAM,GAAG,aAAa,CAAC;IACpC,IAAI,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI,CAAC;CAC3B;AAED,qBAAa,oBAAqB,YAAW,aAAa;IAI5C,OAAO,CAAC,GAAG;IAHvB,OAAO,CAAC,WAAW,CAAkB;IACrC,OAAO,CAAC,WAAW,CAAkB;gBAEjB,GAAG,EAAE,QAAQ;IAEjC,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,IAAI;IAM5C,KAAK,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAY1B,GAAG,IAAI,IAAI;IAMX,MAAM,CAAC,IAAI,EAAE,MAAM,GAAG,aAAa;IAOnC,IAAI,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI;CA+B1B"}
1
+ {"version":3,"file":"hazel-response.d.ts","sourceRoot":"","sources":["../src/hazel-response.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAEnC,MAAM,WAAW,aAAa;IAC5B,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7C,KAAK,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,GAAG,IAAI,IAAI,CAAC;IACZ,MAAM,CAAC,IAAI,EAAE,MAAM,GAAG,aAAa,CAAC;IACpC,IAAI,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI,CAAC;IAC1B,gFAAgF;IAChF,UAAU,CAAC,CAAC,MAAM,EAAE,MAAM,EAAE,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;CACzD;AAED,qBAAa,oBAAqB,YAAW,aAAa;IAI5C,OAAO,CAAC,GAAG;IAHvB,OAAO,CAAC,WAAW,CAAkB;IACrC,OAAO,CAAC,WAAW,CAAkB;gBAEjB,GAAG,EAAE,QAAQ;IAEjC,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,IAAI;IAM5C,KAAK,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAY1B,GAAG,IAAI,IAAI;IAMX,MAAM,CAAC,IAAI,EAAE,MAAM,GAAG,aAAa;IAOnC,UAAU,CAAC,MAAM,EAAE,MAAM,EAAE,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI;IAWtD,IAAI,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI;CA+B1B"}
@@ -35,6 +35,16 @@ class HazelExpressResponse {
35
35
  }
36
36
  return this;
37
37
  }
38
+ sendBuffer(buffer, contentType) {
39
+ if (this.isStreaming || this.headersSent) {
40
+ return;
41
+ }
42
+ if (contentType) {
43
+ this.res.setHeader('Content-Type', contentType);
44
+ }
45
+ this.res.send(buffer);
46
+ this.headersSent = true;
47
+ }
38
48
  json(data) {
39
49
  if (this.isStreaming || this.headersSent) {
40
50
  return; // Don't try to send JSON if we're already streaming or headers are sent
@@ -1 +1 @@
1
- {"version":3,"file":"logger.d.ts","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,MAAM,SAAS,CAAC;AAK9B,OAAO,EAAE,eAAe,EAAE,cAAc,EAAE,MAAM,MAAM,CAAC;AAuOvD,eAAO,MAAM,aAAa,GACxB,KAAK,eAAe,EACpB,KAAK,cAAc,EACnB,MAAM,MAAM,IAAI,KACf,IAuBF,CAAC;AAQF,QAAA,MAAM,cAAc;0BALO,OAAO;CAOhC,CAAC;AAGH,eAAe,cAAc,CAAC"}
1
+ {"version":3,"file":"logger.d.ts","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,MAAM,SAAS,CAAC;AAK9B,OAAO,EAAE,eAAe,EAAE,cAAc,EAAE,MAAM,MAAM,CAAC;AAmQvD,eAAO,MAAM,aAAa,GACxB,KAAK,eAAe,EACpB,KAAK,cAAc,EACnB,MAAM,MAAM,IAAI,KACf,IAuBF,CAAC;AAQF,QAAA,MAAM,cAAc;0BALO,OAAO;CAOhC,CAAC;AAGH,eAAe,cAAc,CAAC"}
package/dist/logger.js CHANGED
@@ -13,8 +13,10 @@ const chalk_1 = __importDefault(require("chalk"));
13
13
  dotenv_1.default.config();
14
14
  const logLevel = process.env.LOG_LEVEL || 'info';
15
15
  const logDir = process.env.LOG_DIR || 'logs';
16
+ const logEnabled = process.env.LOG_ENABLED !== 'false'; // default: true
17
+ const logPackage = process.env.LOG_PACKAGE || ''; // e.g. "http" = only HTTP request logs
16
18
  // Ensure log directory exists
17
- if (!fs_1.default.existsSync(logDir)) {
19
+ if (logDir && !fs_1.default.existsSync(logDir)) {
18
20
  fs_1.default.mkdirSync(logDir, { recursive: true });
19
21
  }
20
22
  // Professional color scheme for different log levels
@@ -151,17 +153,25 @@ const customFormat = winston_1.default.format.printf(({ level, message, timestam
151
153
  }
152
154
  return `${time} ${levelStr} ${msg}${metaStr}`;
153
155
  });
154
- // Create logger instance
155
- const logger = winston_1.default.createLogger({
156
- level: logLevel,
157
- format: winston_1.default.format.combine(winston_1.default.format.timestamp({ format: 'YYYY-MM-DD HH:mm:ss' }), customFormat),
158
- transports: [
159
- // Console transport with colors
160
- new winston_1.default.transports.Console({
156
+ // When LOG_PACKAGE=http, only allow logs that look like HTTP requests
157
+ const isHttpLog = (info) => {
158
+ const msg = String(info.message ?? '');
159
+ return /^(GET|POST|PUT|PATCH|DELETE|OPTIONS|HEAD)\s+/i.test(msg) || msg.includes(' ') || msg.includes(' ');
160
+ };
161
+ const transports = [];
162
+ if (logEnabled) {
163
+ if (logPackage === 'http') {
164
+ transports.push(new winston_1.default.transports.Console({
165
+ format: winston_1.default.format.combine(winston_1.default.format((info) => (isHttpLog(info) ? info : false))(), customFormat),
166
+ }));
167
+ }
168
+ else {
169
+ transports.push(new winston_1.default.transports.Console({
161
170
  format: winston_1.default.format.combine(customFormat),
162
- }),
163
- // File transport for all logs (without colors)
164
- new winston_1.default.transports.File({
171
+ }));
172
+ }
173
+ if (logDir) {
174
+ transports.push(new winston_1.default.transports.File({
165
175
  filename: path_1.default.join(logDir, 'combined.log'),
166
176
  format: winston_1.default.format.combine(winston_1.default.format.timestamp(), winston_1.default.format.printf(({ level, message, timestamp, ...metadata }) => {
167
177
  let msg = `${timestamp} [${level.toUpperCase()}] ${message}`;
@@ -175,9 +185,7 @@ const logger = winston_1.default.createLogger({
175
185
  }
176
186
  return msg;
177
187
  })),
178
- }),
179
- // File transport for errors only (without colors)
180
- new winston_1.default.transports.File({
188
+ }), new winston_1.default.transports.File({
181
189
  filename: path_1.default.join(logDir, 'error.log'),
182
190
  level: 'error',
183
191
  format: winston_1.default.format.combine(winston_1.default.format.timestamp(), winston_1.default.format.printf(({ level, message, timestamp, ...metadata }) => {
@@ -192,8 +200,17 @@ const logger = winston_1.default.createLogger({
192
200
  }
193
201
  return msg;
194
202
  })),
195
- }),
196
- ],
203
+ }));
204
+ }
205
+ }
206
+ else {
207
+ transports.push(new winston_1.default.transports.Console({ silent: true }));
208
+ }
209
+ // Create logger instance
210
+ const logger = winston_1.default.createLogger({
211
+ level: logLevel,
212
+ format: winston_1.default.format.combine(winston_1.default.format.timestamp({ format: 'YYYY-MM-DD HH:mm:ss' }), customFormat),
213
+ transports,
197
214
  });
198
215
  // Log application info when starting
199
216
  const appInfo = {
@@ -1 +1 @@
1
- {"version":3,"file":"router.d.ts","sourceRoot":"","sources":["../src/router.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,SAAS,CAAC;AAC/B,OAAO,EAAE,cAAc,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAC5D,OAAO,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAMxC,OAAO,kBAAkB,CAAC;AAY1B,UAAU,UAAU;IAClB,OAAO,EAAE,YAAY,CAAC;IACtB,OAAO,EAAE,cAAc,CAAC;CACzB;AAED,KAAK,YAAY,GAAG,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,EAAE,OAAO,CAAC,EAAE,cAAc,KAAK,IAAI,CAAC;AAEpF,qBAAa,MAAM;IAKL,OAAO,CAAC,SAAS;IAJ7B,OAAO,CAAC,MAAM,CAA0C;IACxD,OAAO,CAAC,cAAc,CAAuD;IAC7E,OAAO,CAAC,iBAAiB,CAAoB;gBAEzB,SAAS,EAAE,SAAS;IAUxC,kBAAkB,CAAC,UAAU,EAAE,IAAI,CAAC,OAAO,CAAC,GAAG,IAAI;YA4DrC,UAAU;YAsBV,iBAAiB;IAe/B,OAAO,CAAC,kBAAkB;IAqQ1B,OAAO,CAAC,aAAa;IAiBrB,OAAO,CAAC,SAAS;IAiBjB,OAAO,CAAC,kBAAkB;IAW1B,OAAO,CAAC,aAAa;IAOf,KAAK,CAAC,MAAM,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,EAAE,OAAO,EAAE,cAAc,GAAG,OAAO,CAAC,UAAU,GAAG,IAAI,CAAC;IA0C7F,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,YAAY,EAAE,GAAG,IAAI;IAIjD,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,YAAY,EAAE,GAAG,IAAI;IAIlD,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,YAAY,EAAE,GAAG,IAAI;IAIjD,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,YAAY,EAAE,GAAG,IAAI;IAIpD,OAAO,CAAC,QAAQ;IAKV,aAAa,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,GAAG,OAAO,CAAC,IAAI,CAAC;CAkChE"}
1
+ {"version":3,"file":"router.d.ts","sourceRoot":"","sources":["../src/router.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,SAAS,CAAC;AAC/B,OAAO,EAAE,cAAc,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAC5D,OAAO,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAMxC,OAAO,kBAAkB,CAAC;AAY1B,UAAU,UAAU;IAClB,OAAO,EAAE,YAAY,CAAC;IACtB,OAAO,EAAE,cAAc,CAAC;CACzB;AAED,KAAK,YAAY,GAAG,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,EAAE,OAAO,CAAC,EAAE,cAAc,KAAK,IAAI,CAAC;AAEpF,qBAAa,MAAM;IAKL,OAAO,CAAC,SAAS;IAJ7B,OAAO,CAAC,MAAM,CAA0C;IACxD,OAAO,CAAC,cAAc,CAAuD;IAC7E,OAAO,CAAC,iBAAiB,CAAoB;gBAEzB,SAAS,EAAE,SAAS;IAUxC,kBAAkB,CAAC,UAAU,EAAE,IAAI,CAAC,OAAO,CAAC,GAAG,IAAI;YA4DrC,UAAU;YAsBV,iBAAiB;IAe/B,OAAO,CAAC,kBAAkB;IAwQ1B,OAAO,CAAC,aAAa;IAiBrB,OAAO,CAAC,SAAS;IAiBjB,OAAO,CAAC,kBAAkB;IAW1B,OAAO,CAAC,aAAa;IAOf,KAAK,CAAC,MAAM,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,EAAE,OAAO,EAAE,cAAc,GAAG,OAAO,CAAC,UAAU,GAAG,IAAI,CAAC;IA0C7F,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,YAAY,EAAE,GAAG,IAAI;IAIjD,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,YAAY,EAAE,GAAG,IAAI;IAIlD,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,YAAY,EAAE,GAAG,IAAI;IAIjD,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,YAAY,EAAE,GAAG,IAAI;IAIpD,OAAO,CAAC,QAAQ;IAKV,aAAa,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,GAAG,OAAO,CAAC,IAAI,CAAC;CAkChE"}
package/dist/router.js CHANGED
@@ -195,6 +195,10 @@ class Router {
195
195
  args[i] = context.headers;
196
196
  }
197
197
  }
198
+ else if (injection.type === 'request') {
199
+ // Handle @Req() / @Request() decorator - raw request for multipart, etc.
200
+ args[i] = req;
201
+ }
198
202
  else if (injection.type === 'response') {
199
203
  // Handle @Res decorator
200
204
  args[i] = new hazel_response_1.HazelExpressResponse(res);
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@hazeljs/core",
3
- "version": "0.2.0-beta.26",
3
+ "version": "0.2.0-beta.27",
4
4
  "description": "Core HazelJS framework - Dependency injection, routing, decorators, and base functionality",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
@@ -63,5 +63,5 @@
63
63
  "url": "https://github.com/hazeljs/hazel-js/issues"
64
64
  },
65
65
  "homepage": "https://hazeljs.com",
66
- "gitHead": "9b164a3441a9f1ecccad9e5cb6d978d2f9b2accd"
66
+ "gitHead": "7ea9c4fdc8acf9a6d91722b7fd92b6bbd376e9e8"
67
67
  }