@eggjs/koa 2.16.0 → 2.18.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.
@@ -0,0 +1,453 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ const node_assert_1 = __importDefault(require("node:assert"));
7
+ const node_path_1 = require("node:path");
8
+ const node_util_1 = __importDefault(require("node:util"));
9
+ const node_stream_1 = __importDefault(require("node:stream"));
10
+ const content_disposition_1 = __importDefault(require("content-disposition"));
11
+ const cache_content_type_1 = __importDefault(require("cache-content-type"));
12
+ const on_finished_1 = __importDefault(require("on-finished"));
13
+ const escape_html_1 = __importDefault(require("escape-html"));
14
+ const type_is_1 = require("type-is");
15
+ const statuses_1 = __importDefault(require("statuses"));
16
+ const destroy_1 = __importDefault(require("destroy"));
17
+ const vary_1 = __importDefault(require("vary"));
18
+ const encodeurl_1 = __importDefault(require("encodeurl"));
19
+ class Response {
20
+ app;
21
+ req;
22
+ res;
23
+ ctx;
24
+ request;
25
+ constructor(app, ctx, req, res) {
26
+ this.app = app;
27
+ this.req = req;
28
+ this.res = res;
29
+ this.ctx = ctx;
30
+ }
31
+ /**
32
+ * Return the request socket.
33
+ */
34
+ get socket() {
35
+ return this.res.socket;
36
+ }
37
+ /**
38
+ * Return response header.
39
+ */
40
+ get header() {
41
+ return this.res.getHeaders() || {};
42
+ }
43
+ /**
44
+ * Return response header, alias as response.header
45
+ */
46
+ get headers() {
47
+ return this.header;
48
+ }
49
+ _explicitStatus;
50
+ /**
51
+ * Get response status code.
52
+ */
53
+ get status() {
54
+ return this.res.statusCode;
55
+ }
56
+ /**
57
+ * Set response status code.
58
+ */
59
+ set status(code) {
60
+ if (this.headerSent)
61
+ return;
62
+ (0, node_assert_1.default)(Number.isInteger(code), 'status code must be a number');
63
+ (0, node_assert_1.default)(code >= 100 && code <= 999, `invalid status code: ${code}`);
64
+ this._explicitStatus = true;
65
+ this.res.statusCode = code;
66
+ if (this.req.httpVersionMajor < 2) {
67
+ this.res.statusMessage = statuses_1.default.message[code];
68
+ }
69
+ if (this.body && statuses_1.default.empty[code])
70
+ this.body = null;
71
+ }
72
+ /**
73
+ * Get response status message
74
+ */
75
+ get message() {
76
+ return this.res.statusMessage || statuses_1.default.message[this.status];
77
+ }
78
+ /**
79
+ * Set response status message
80
+ */
81
+ set message(msg) {
82
+ this.res.statusMessage = msg;
83
+ }
84
+ _body;
85
+ _explicitNullBody;
86
+ /**
87
+ * Get response body.
88
+ */
89
+ get body() {
90
+ return this._body;
91
+ }
92
+ /**
93
+ * Set response body.
94
+ */
95
+ set body(val) {
96
+ const original = this._body;
97
+ this._body = val;
98
+ // no content
99
+ if (val == null) {
100
+ if (!statuses_1.default.empty[this.status])
101
+ this.status = 204;
102
+ if (val === null)
103
+ this._explicitNullBody = true;
104
+ this.remove('Content-Type');
105
+ this.remove('Content-Length');
106
+ this.remove('Transfer-Encoding');
107
+ return;
108
+ }
109
+ // set the status
110
+ if (!this._explicitStatus)
111
+ this.status = 200;
112
+ // set the content-type only if not yet set
113
+ const setType = !this.has('Content-Type');
114
+ // string
115
+ if (typeof val === 'string') {
116
+ if (setType)
117
+ this.type = /^\s*</.test(val) ? 'html' : 'text';
118
+ this.length = Buffer.byteLength(val);
119
+ return;
120
+ }
121
+ // buffer
122
+ if (Buffer.isBuffer(val)) {
123
+ if (setType)
124
+ this.type = 'bin';
125
+ this.length = val.length;
126
+ return;
127
+ }
128
+ // stream
129
+ if (val instanceof node_stream_1.default) {
130
+ (0, on_finished_1.default)(this.res, destroy_1.default.bind(null, val));
131
+ // eslint-disable-next-line eqeqeq
132
+ if (original != val) {
133
+ val.once('error', err => this.ctx.onerror(err));
134
+ // overwriting
135
+ if (original != null)
136
+ this.remove('Content-Length');
137
+ }
138
+ if (setType)
139
+ this.type = 'bin';
140
+ return;
141
+ }
142
+ // json
143
+ this.remove('Content-Length');
144
+ this.type = 'json';
145
+ }
146
+ /**
147
+ * Set Content-Length field to `n`.
148
+ */
149
+ set length(n) {
150
+ if (n === undefined)
151
+ return;
152
+ if (!this.has('Transfer-Encoding')) {
153
+ this.set('Content-Length', n);
154
+ }
155
+ }
156
+ /**
157
+ * Return parsed response Content-Length when present.
158
+ */
159
+ get length() {
160
+ if (this.has('Content-Length')) {
161
+ return parseInt(this.get('Content-Length'), 10) || 0;
162
+ }
163
+ const { body } = this;
164
+ if (!body || body instanceof node_stream_1.default)
165
+ return undefined;
166
+ if (typeof body === 'string')
167
+ return Buffer.byteLength(body);
168
+ if (Buffer.isBuffer(body))
169
+ return body.length;
170
+ return Buffer.byteLength(JSON.stringify(body));
171
+ }
172
+ /**
173
+ * Check if a header has been written to the socket.
174
+ */
175
+ get headerSent() {
176
+ return this.res.headersSent;
177
+ }
178
+ /**
179
+ * Vary on `field`.
180
+ */
181
+ vary(field) {
182
+ if (this.headerSent)
183
+ return;
184
+ (0, vary_1.default)(this.res, field);
185
+ }
186
+ /**
187
+ * Perform a 302 redirect to `url`.
188
+ *
189
+ * The string "back" is special-cased
190
+ * to provide Referrer support, when Referrer
191
+ * is not present `alt` or "/" is used.
192
+ *
193
+ * Examples:
194
+ *
195
+ * this.redirect('back');
196
+ * this.redirect('back', '/index.html');
197
+ * this.redirect('/login');
198
+ * this.redirect('http://google.com'); // will format to 'http://google.com/'
199
+ */
200
+ redirect(url, alt) {
201
+ // location
202
+ if (url === 'back')
203
+ url = this.ctx.get('Referrer') || alt || '/';
204
+ if (url.startsWith('https://') || url.startsWith('http://')) {
205
+ // formatting url again avoid security escapes
206
+ url = new URL(url).toString();
207
+ }
208
+ this.set('Location', (0, encodeurl_1.default)(url));
209
+ // status
210
+ if (!statuses_1.default.redirect[this.status])
211
+ this.status = 302;
212
+ // html
213
+ if (this.ctx.accepts('html')) {
214
+ url = (0, escape_html_1.default)(url);
215
+ this.type = 'text/html; charset=utf-8';
216
+ this.body = `Redirecting to <a href="${url}">${url}</a>.`;
217
+ return;
218
+ }
219
+ // text
220
+ this.type = 'text/plain; charset=utf-8';
221
+ this.body = `Redirecting to ${url}.`;
222
+ }
223
+ /**
224
+ * Set Content-Disposition header to "attachment" with optional `filename`.
225
+ */
226
+ attachment(filename, options) {
227
+ if (filename)
228
+ this.type = (0, node_path_1.extname)(filename);
229
+ this.set('Content-Disposition', (0, content_disposition_1.default)(filename, options));
230
+ }
231
+ /**
232
+ * Set Content-Type response header with `type` through `mime.lookup()`
233
+ * when it does not contain a charset.
234
+ *
235
+ * Examples:
236
+ *
237
+ * this.type = '.html';
238
+ * this.type = 'html';
239
+ * this.type = 'json';
240
+ * this.type = 'application/json';
241
+ * this.type = 'png';
242
+ */
243
+ set type(type) {
244
+ if (!type) {
245
+ this.remove('Content-Type');
246
+ return;
247
+ }
248
+ const mimeType = (0, cache_content_type_1.default)(type);
249
+ if (mimeType) {
250
+ this.set('Content-Type', mimeType);
251
+ }
252
+ }
253
+ /**
254
+ * Return the response mime type void of
255
+ * parameters such as "charset".
256
+ */
257
+ get type() {
258
+ const type = this.get('Content-Type');
259
+ if (!type)
260
+ return '';
261
+ return type.split(';', 1)[0];
262
+ }
263
+ /**
264
+ * Check whether the response is one of the listed types.
265
+ * Pretty much the same as `this.request.is()`.
266
+ *
267
+ * this.response.is('html')
268
+ * this.response.is('html', 'json')
269
+ */
270
+ is(type, ...types) {
271
+ const testTypes = Array.isArray(type) ? type :
272
+ (type ? [type] : []);
273
+ return (0, type_is_1.is)(this.type, [...testTypes, ...types]);
274
+ }
275
+ /**
276
+ * Set the Last-Modified date using a string or a Date.
277
+ *
278
+ * this.response.lastModified = new Date();
279
+ * this.response.lastModified = '2013-09-13';
280
+ */
281
+ set lastModified(val) {
282
+ if (typeof val === 'string')
283
+ val = new Date(val);
284
+ if (val) {
285
+ this.set('Last-Modified', val.toUTCString());
286
+ }
287
+ }
288
+ /**
289
+ * Get the Last-Modified date in Date form, if it exists.
290
+ */
291
+ get lastModified() {
292
+ const date = this.get('last-modified');
293
+ if (date)
294
+ return new Date(date);
295
+ }
296
+ /**
297
+ * Set the ETag of a response.
298
+ * This will normalize the quotes if necessary.
299
+ *
300
+ * this.response.etag = 'md5-hash-sum';
301
+ * this.response.etag = '"md5-hash-sum"';
302
+ * this.response.etag = 'W/"123456789"';
303
+ */
304
+ set etag(val) {
305
+ if (!/^(W\/)?"/.test(val))
306
+ val = `"${val}"`;
307
+ this.set('ETag', val);
308
+ }
309
+ /**
310
+ * Get the ETag of a response.
311
+ */
312
+ get etag() {
313
+ return this.get('ETag');
314
+ }
315
+ /**
316
+ * Return response header.
317
+ *
318
+ * Examples:
319
+ *
320
+ * this.get('Content-Type');
321
+ * // => "text/plain"
322
+ *
323
+ * this.get('content-type');
324
+ * // => "text/plain"
325
+ */
326
+ get(field) {
327
+ return (this.header[field.toLowerCase()] || '');
328
+ }
329
+ /**
330
+ * Returns true if the header identified by name is currently set in the outgoing headers.
331
+ * The header name matching is case-insensitive.
332
+ *
333
+ * Examples:
334
+ *
335
+ * this.has('Content-Type');
336
+ * // => true
337
+ *
338
+ * this.get('content-type');
339
+ * // => true
340
+ */
341
+ has(field) {
342
+ return this.res.hasHeader(field);
343
+ }
344
+ /**
345
+ * Set header `field` to `val` or pass
346
+ * an object of header fields.
347
+ *
348
+ * Examples:
349
+ *
350
+ * this.set('Foo', ['bar', 'baz']);
351
+ * this.set('Accept', 'application/json');
352
+ * this.set({ Accept: 'text/plain', 'X-API-Key': 'tobi' });
353
+ */
354
+ set(field, val) {
355
+ if (this.headerSent)
356
+ return;
357
+ if (typeof field === 'string') {
358
+ if (Array.isArray(val)) {
359
+ val = val.map(v => {
360
+ return typeof v === 'string' ? v : String(v);
361
+ });
362
+ }
363
+ else if (typeof val !== 'string') {
364
+ val = String(val);
365
+ }
366
+ this.res.setHeader(field, val);
367
+ }
368
+ else {
369
+ for (const key in field) {
370
+ this.set(key, field[key]);
371
+ }
372
+ }
373
+ }
374
+ /**
375
+ * Append additional header `field` with value `val`.
376
+ *
377
+ * Examples:
378
+ *
379
+ * ```
380
+ * this.append('Link', ['<http://localhost/>', '<http://localhost:3000/>']);
381
+ * this.append('Set-Cookie', 'foo=bar; Path=/; HttpOnly');
382
+ * this.append('Warning', '199 Miscellaneous warning');
383
+ */
384
+ append(field, val) {
385
+ const prev = this.get(field);
386
+ let value = val;
387
+ if (prev) {
388
+ value = Array.isArray(prev)
389
+ ? prev.concat(value)
390
+ : [prev].concat(val);
391
+ }
392
+ return this.set(field, value);
393
+ }
394
+ /**
395
+ * Remove header `field`.
396
+ */
397
+ remove(field) {
398
+ if (this.headerSent)
399
+ return;
400
+ this.res.removeHeader(field);
401
+ }
402
+ /**
403
+ * Checks if the request is writable.
404
+ * Tests for the existence of the socket
405
+ * as node sometimes does not set it.
406
+ */
407
+ get writable() {
408
+ // can't write any more after response finished
409
+ // response.writableEnded is available since Node > 12.9
410
+ // https://nodejs.org/api/http.html#http_response_writableended
411
+ // response.finished is undocumented feature of previous Node versions
412
+ // https://stackoverflow.com/questions/16254385/undocumented-response-finished-in-node-js
413
+ if (this.res.writableEnded || this.res.finished)
414
+ return false;
415
+ const socket = this.res.socket;
416
+ // There are already pending outgoing res, but still writable
417
+ // https://github.com/nodejs/node/blob/v4.4.7/lib/_http_server.js#L486
418
+ if (!socket)
419
+ return true;
420
+ return socket.writable;
421
+ }
422
+ /**
423
+ * Inspect implementation.
424
+ */
425
+ inspect() {
426
+ if (!this.res)
427
+ return;
428
+ const o = this.toJSON();
429
+ Reflect.set(o, 'body', this.body);
430
+ return o;
431
+ }
432
+ [node_util_1.default.inspect.custom]() {
433
+ return this.inspect();
434
+ }
435
+ /**
436
+ * Return JSON representation.
437
+ */
438
+ toJSON() {
439
+ return {
440
+ status: this.status,
441
+ message: this.message,
442
+ header: this.header,
443
+ };
444
+ }
445
+ /**
446
+ * Flush any set headers and begin the body
447
+ */
448
+ flushHeaders() {
449
+ this.res.flushHeaders();
450
+ }
451
+ }
452
+ exports.default = Response;
453
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVzcG9uc2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcmVzcG9uc2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFBQSw4REFBaUM7QUFDakMseUNBQW9DO0FBQ3BDLDBEQUE2QjtBQUM3Qiw4REFBaUM7QUFFakMsOEVBQXFEO0FBQ3JELDRFQUF5QztBQUN6Qyw4REFBbUM7QUFDbkMsOERBQWlDO0FBQ2pDLHFDQUF1QztBQUN2Qyx3REFBZ0M7QUFDaEMsc0RBQThCO0FBQzlCLGdEQUF3QjtBQUN4QiwwREFBa0M7QUFLbEMsTUFBcUIsUUFBUTtJQUMzQixHQUFHLENBQWM7SUFDakIsR0FBRyxDQUFrQjtJQUNyQixHQUFHLENBQWlCO0lBQ3BCLEdBQUcsQ0FBb0I7SUFDdkIsT0FBTyxDQUFVO0lBRWpCLFlBQVksR0FBZ0IsRUFBRSxHQUFzQixFQUFFLEdBQW9CLEVBQUUsR0FBbUI7UUFDN0YsSUFBSSxDQUFDLEdBQUcsR0FBRyxHQUFHLENBQUM7UUFDZixJQUFJLENBQUMsR0FBRyxHQUFHLEdBQUcsQ0FBQztRQUNmLElBQUksQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDO1FBQ2YsSUFBSSxDQUFDLEdBQUcsR0FBRyxHQUFHLENBQUM7SUFDakIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsSUFBSSxNQUFNO1FBQ1IsT0FBTyxJQUFJLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQztJQUN6QixDQUFDO0lBRUQ7O09BRUc7SUFDSCxJQUFJLE1BQU07UUFDUixPQUFPLElBQUksQ0FBQyxHQUFHLENBQUMsVUFBVSxFQUFFLElBQUksRUFBRSxDQUFDO0lBQ3JDLENBQUM7SUFFRDs7T0FFRztJQUNILElBQUksT0FBTztRQUNULE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQztJQUNyQixDQUFDO0lBRUQsZUFBZSxDQUFVO0lBRXpCOztPQUVHO0lBQ0gsSUFBSSxNQUFNO1FBQ1IsT0FBTyxJQUFJLENBQUMsR0FBRyxDQUFDLFVBQVUsQ0FB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
@@ -1,11 +1,10 @@
1
1
  export type CustomError = Error & {
2
- headers?: object;
2
+ headers?: Record<string, string>;
3
3
  status?: number;
4
4
  statusCode?: number;
5
5
  code?: string;
6
6
  expose?: boolean;
7
7
  };
8
- export type ProtoImplClass<T = object> = new (...args: any[]) => T;
9
8
  export type AnyProto = {
10
9
  [key: string]: any;
11
10
  };
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiJ9
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiJ9
@@ -0,0 +1,122 @@
1
+ /// <reference types="http-errors" />
2
+ /// <reference types="node" resolution-mode="require"/>
3
+ /// <reference types="node" resolution-mode="require"/>
4
+ /// <reference types="node" resolution-mode="require"/>
5
+ /// <reference types="node" resolution-mode="require"/>
6
+ import Emitter from 'node:events';
7
+ import util from 'node:util';
8
+ import http from 'node:http';
9
+ import type { AsyncLocalStorage } from 'node:async_hooks';
10
+ import type { IncomingMessage, ServerResponse } from 'node:http';
11
+ import Context from './context.js';
12
+ import Request from './request.js';
13
+ import Response from './response.js';
14
+ import type { ContextDelegation } from './context.js';
15
+ import type { CustomError, AnyProto } from './types.js';
16
+ export type ProtoImplClass<T = object> = new (...args: any[]) => T;
17
+ export type Next = () => Promise<void>;
18
+ export type MiddlewareFunc = (ctx: ContextDelegation, next: Next) => Promise<void> | void;
19
+ export type { ContextDelegation as Context } from './context.js';
20
+ /**
21
+ * Expose `Application` class.
22
+ * Inherits from `Emitter.prototype`.
23
+ */
24
+ export default class Application extends Emitter {
25
+ #private;
26
+ /**
27
+ * Make HttpError available to consumers of the library so that consumers don't
28
+ * have a direct dependency upon `http-errors`
29
+ */
30
+ static HttpError: import("http-errors").HttpErrorConstructor<number>;
31
+ proxy: boolean;
32
+ subdomainOffset: number;
33
+ proxyIpHeader: string;
34
+ maxIpsCount: number;
35
+ env: string;
36
+ keys?: string[];
37
+ middleware: MiddlewareFunc[];
38
+ ctxStorage: AsyncLocalStorage<ContextDelegation>;
39
+ silent: boolean;
40
+ ContextClass: ProtoImplClass<Context>;
41
+ context: AnyProto;
42
+ RequestClass: ProtoImplClass<Request>;
43
+ request: AnyProto;
44
+ ResponseClass: ProtoImplClass<Response>;
45
+ response: AnyProto;
46
+ /**
47
+ * Initialize a new `Application`.
48
+ *
49
+ * @param {object} [options] Application options
50
+ * @param {string} [options.env='development'] Environment
51
+ * @param {string[]} [options.keys] Signed cookie keys
52
+ * @param {boolean} [options.proxy] Trust proxy headers
53
+ * @param {number} [options.subdomainOffset] Subdomain offset
54
+ * @param {string} [options.proxyIpHeader] Proxy IP header, defaults to X-Forwarded-For
55
+ * @param {number} [options.maxIpsCount] Max IPs read from proxy IP header, default to 0 (means infinity)
56
+ */
57
+ constructor(options?: {
58
+ proxy?: boolean;
59
+ subdomainOffset?: number;
60
+ proxyIpHeader?: string;
61
+ maxIpsCount?: number;
62
+ env?: string;
63
+ keys?: string[];
64
+ });
65
+ /**
66
+ * Shorthand for:
67
+ *
68
+ * http.createServer(app.callback()).listen(...)
69
+ */
70
+ listen(...args: any[]): http.Server<typeof http.IncomingMessage, typeof http.ServerResponse>;
71
+ /**
72
+ * Return JSON representation.
73
+ * We only bother showing settings.
74
+ */
75
+ toJSON(): {
76
+ subdomainOffset: number;
77
+ proxy: boolean;
78
+ env: string;
79
+ };
80
+ /**
81
+ * Inspect implementation.
82
+ */
83
+ inspect(): {
84
+ subdomainOffset: number;
85
+ proxy: boolean;
86
+ env: string;
87
+ };
88
+ [util.inspect.custom](): {
89
+ subdomainOffset: number;
90
+ proxy: boolean;
91
+ env: string;
92
+ };
93
+ /**
94
+ * Use the given middleware `fn`.
95
+ *
96
+ * Old-style middleware will be converted.
97
+ */
98
+ use(fn: MiddlewareFunc): this;
99
+ /**
100
+ * Return a request handler callback
101
+ * for node's native http server.
102
+ */
103
+ callback(): (req: IncomingMessage, res: ServerResponse) => Promise<void | http.ServerResponse<http.IncomingMessage>>;
104
+ /**
105
+ * return current context from async local storage
106
+ */
107
+ get currentContext(): ContextDelegation | undefined;
108
+ /**
109
+ * Initialize a new context.
110
+ * @private
111
+ */
112
+ protected createContext(req: IncomingMessage, res: ServerResponse): ContextDelegation;
113
+ /**
114
+ * Default error handler.
115
+ * @private
116
+ */
117
+ protected onerror(err: CustomError): void;
118
+ /**
119
+ * Response helper.
120
+ */
121
+ protected _respond(ctx: ContextDelegation): http.ServerResponse<http.IncomingMessage> | undefined;
122
+ }