@eggjs/supertest 8.0.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,80 @@
1
+ import type { Server } from 'node:net';
2
+ import { Request, type Response } from 'superagent';
3
+ import { AssertError } from './error/AssertError.js';
4
+ export type TestApplication = Server | string;
5
+ export type AssertFunction = (res: Response) => AssertError | void;
6
+ export type CallbackFunction = (err: AssertError | Error | null, res: Response) => void;
7
+ export type ResponseError = Error & {
8
+ syscall?: string;
9
+ code?: string;
10
+ status?: number;
11
+ };
12
+ export interface ExpectHeader {
13
+ name: string;
14
+ value: string | number | RegExp;
15
+ }
16
+ export declare class Test extends Request {
17
+ app: TestApplication;
18
+ _server: Server;
19
+ _asserts: AssertFunction[];
20
+ /**
21
+ * Initialize a new `Test` with the given `app`,
22
+ * request `method` and `path`.
23
+ */
24
+ constructor(app: TestApplication, method: string, path: string);
25
+ /**
26
+ * Returns a URL, extracted from a server.
27
+ *
28
+ * @return {String} URL address
29
+ * @private
30
+ */
31
+ protected serverAddress(app: Server, path: string): string;
32
+ /**
33
+ * Expectations:
34
+ *
35
+ * ```js
36
+ * .expect(200)
37
+ * .expect(200, fn)
38
+ * .expect(200, body)
39
+ * .expect('Some body')
40
+ * .expect('Some body', fn)
41
+ * .expect(['json array body', { key: 'val' }])
42
+ * .expect('Content-Type', 'application/json')
43
+ * .expect('Content-Type', 'application/json', fn)
44
+ * .expect(fn)
45
+ * .expect([200, 404])
46
+ * ```
47
+ *
48
+ * @return {Test} The current Test instance for chaining.
49
+ */
50
+ expect(a: number | string | RegExp | object | AssertFunction, b?: string | number | RegExp | CallbackFunction, c?: CallbackFunction): Test;
51
+ /**
52
+ * Defer invoking superagent's `.end()` until
53
+ * the server is listening.
54
+ */
55
+ end(fn: CallbackFunction): this;
56
+ /**
57
+ * Perform assertions and invoke `fn(err, res)`.
58
+ */
59
+ assert(resError: ResponseError | null, res: Response, fn: CallbackFunction): void;
60
+ /**
61
+ * Perform assertions on a response body and return an Error upon failure.
62
+ */
63
+ _assertBody(body: RegExp | string | number | object | null | undefined, res: Response): AssertError | undefined;
64
+ /**
65
+ * Perform assertions on a response header and return an Error upon failure.
66
+ */
67
+ _assertHeader(header: ExpectHeader, res: Response): AssertError | undefined;
68
+ /**
69
+ * Perform assertions on the response status and return an Error upon failure.
70
+ */
71
+ _assertStatus(status: number, res: Response): AssertError | undefined;
72
+ /**
73
+ * Perform assertions on the response status and return an Error upon failure.
74
+ */
75
+ _assertStatusArray(statusArray: number[], res: Response): AssertError | undefined;
76
+ /**
77
+ * Performs an assertion by calling a function and return an Error upon failure.
78
+ */
79
+ _assertFunction(fn: AssertFunction, res: Response): Error | undefined;
80
+ }
@@ -0,0 +1,278 @@
1
+ import { inspect } from 'node:util';
2
+ import { STATUS_CODES } from 'node:http';
3
+ import { Server as HttpsServer } from 'node:tls';
4
+ import { deepStrictEqual } from 'node:assert';
5
+ import { Request } from 'superagent';
6
+ import { AssertError } from './error/AssertError.js';
7
+ export class Test extends Request {
8
+ app;
9
+ _server;
10
+ _asserts = [];
11
+ /**
12
+ * Initialize a new `Test` with the given `app`,
13
+ * request `method` and `path`.
14
+ */
15
+ constructor(app, method, path) {
16
+ super(method.toUpperCase(), path);
17
+ this.redirects(0);
18
+ this.buffer();
19
+ this.app = app;
20
+ this.url = typeof app === 'string'
21
+ ? app + path
22
+ : this.serverAddress(app, path);
23
+ }
24
+ /**
25
+ * Returns a URL, extracted from a server.
26
+ *
27
+ * @return {String} URL address
28
+ * @private
29
+ */
30
+ serverAddress(app, path) {
31
+ const addr = app.address();
32
+ if (!addr) {
33
+ this._server = app.listen(0);
34
+ }
35
+ const port = app.address().port;
36
+ const protocol = (app instanceof HttpsServer || this._server instanceof HttpsServer) ? 'https' : 'http';
37
+ return `${protocol}://127.0.0.1:${port}${path}`;
38
+ }
39
+ /**
40
+ * Expectations:
41
+ *
42
+ * ```js
43
+ * .expect(200)
44
+ * .expect(200, fn)
45
+ * .expect(200, body)
46
+ * .expect('Some body')
47
+ * .expect('Some body', fn)
48
+ * .expect(['json array body', { key: 'val' }])
49
+ * .expect('Content-Type', 'application/json')
50
+ * .expect('Content-Type', 'application/json', fn)
51
+ * .expect(fn)
52
+ * .expect([200, 404])
53
+ * ```
54
+ *
55
+ * @return {Test} The current Test instance for chaining.
56
+ */
57
+ expect(a, b, c) {
58
+ // callback
59
+ if (typeof a === 'function') {
60
+ // .expect(fn)
61
+ this._asserts.push(wrapAssertFn(a));
62
+ return this;
63
+ }
64
+ if (typeof b === 'function') {
65
+ // .expect('Some body', fn)
66
+ this.end(b);
67
+ }
68
+ if (typeof c === 'function') {
69
+ // .expect('Content-Type', 'application/json', fn)
70
+ this.end(c);
71
+ }
72
+ // status
73
+ if (typeof a === 'number') {
74
+ this._asserts.push(wrapAssertFn(this._assertStatus.bind(this, a)));
75
+ // body
76
+ if (typeof b !== 'function' && arguments.length > 1) {
77
+ // .expect(200, 'body')
78
+ // .expect(200, null)
79
+ // .expect(200, 9999999)
80
+ this._asserts.push(wrapAssertFn(this._assertBody.bind(this, b)));
81
+ }
82
+ return this;
83
+ }
84
+ // multiple statuses
85
+ if (Array.isArray(a) && a.length > 0 && a.every(val => typeof val === 'number')) {
86
+ // .expect([200, 300])
87
+ this._asserts.push(wrapAssertFn(this._assertStatusArray.bind(this, a)));
88
+ return this;
89
+ }
90
+ // header field
91
+ if (typeof b === 'string' || typeof b === 'number' || b instanceof RegExp) {
92
+ // .expect('Content-Type', 'application/json')
93
+ // .expect('Content-Type', /json/)
94
+ this._asserts.push(wrapAssertFn(this._assertHeader.bind(this, { name: String(a), value: b })));
95
+ return this;
96
+ }
97
+ // body
98
+ // .expect('body')
99
+ // .expect(['json array body', { key: 'val' }])
100
+ // .expect(/foo/)
101
+ this._asserts.push(wrapAssertFn(this._assertBody.bind(this, a)));
102
+ return this;
103
+ }
104
+ /**
105
+ * Defer invoking superagent's `.end()` until
106
+ * the server is listening.
107
+ */
108
+ end(fn) {
109
+ const server = this._server;
110
+ super.end((err, res) => {
111
+ const localAssert = () => {
112
+ this.assert(err, res, fn);
113
+ };
114
+ if (server && '_handle' in server && server._handle) {
115
+ return server.close(localAssert);
116
+ }
117
+ localAssert();
118
+ });
119
+ return this;
120
+ }
121
+ /**
122
+ * Perform assertions and invoke `fn(err, res)`.
123
+ */
124
+ assert(resError, res, fn) {
125
+ let errorObj;
126
+ // check for unexpected network errors or server not running/reachable errors
127
+ // when there is no response and superagent sends back a System Error
128
+ // do not check further for other asserts, if any, in such case
129
+ // https://nodejs.org/api/errors.html#errors_common_system_errors
130
+ const sysErrors = {
131
+ ECONNREFUSED: 'Connection refused',
132
+ ECONNRESET: 'Connection reset by peer',
133
+ EPIPE: 'Broken pipe',
134
+ ETIMEDOUT: 'Operation timed out',
135
+ };
136
+ if (!res && resError) {
137
+ if (resError instanceof Error
138
+ && resError.syscall === 'connect'
139
+ && resError.code
140
+ && sysErrors[resError.code]) {
141
+ errorObj = new Error(resError.code + ': ' + sysErrors[resError.code]);
142
+ }
143
+ else {
144
+ errorObj = resError;
145
+ }
146
+ }
147
+ // asserts
148
+ for (let i = 0; i < this._asserts.length && !errorObj; i += 1) {
149
+ errorObj = this._assertFunction(this._asserts[i], res);
150
+ }
151
+ // set unexpected superagent error if no other error has occurred.
152
+ if (!errorObj && resError instanceof Error && (!res || resError.status !== res.status)) {
153
+ errorObj = resError;
154
+ }
155
+ fn.call(this, errorObj || null, res);
156
+ }
157
+ /**
158
+ * Perform assertions on a response body and return an Error upon failure.
159
+ */
160
+ _assertBody(body, res) {
161
+ const isRegexp = body instanceof RegExp;
162
+ // parsed
163
+ if (typeof body === 'object' && !isRegexp) {
164
+ try {
165
+ deepStrictEqual(body, res.body);
166
+ }
167
+ catch (err) {
168
+ const a = inspect(body);
169
+ const b = inspect(res.body);
170
+ return new AssertError('expected ' + a + ' response body, got ' + b, body, res.body, { cause: err });
171
+ }
172
+ }
173
+ else if (body !== res.text) {
174
+ // string
175
+ const a = inspect(body);
176
+ const b = inspect(res.text);
177
+ // regexp
178
+ if (isRegexp) {
179
+ if (!body.test(res.text)) {
180
+ return new AssertError('expected body ' + b + ' to match ' + body, body, res.body);
181
+ }
182
+ }
183
+ else {
184
+ return new AssertError('expected ' + a + ' response body, got ' + b, body, res.body);
185
+ }
186
+ }
187
+ }
188
+ /**
189
+ * Perform assertions on a response header and return an Error upon failure.
190
+ */
191
+ _assertHeader(header, res) {
192
+ const field = header.name;
193
+ const actual = res.header[field.toLowerCase()];
194
+ const fieldExpected = header.value;
195
+ if (typeof actual === 'undefined') {
196
+ return new AssertError('expected "' + field + '" header field', header, actual);
197
+ }
198
+ // This check handles header values that may be a String or single element Array
199
+ if ((Array.isArray(actual) && actual.toString() === fieldExpected)
200
+ || fieldExpected === actual) {
201
+ return;
202
+ }
203
+ if (fieldExpected instanceof RegExp) {
204
+ if (!fieldExpected.test(actual)) {
205
+ return new AssertError('expected "' + field + '" matching '
206
+ + fieldExpected + ', got "' + actual + '"', header, actual);
207
+ }
208
+ }
209
+ else {
210
+ return new AssertError('expected "' + field + '" of "' + fieldExpected + '", got "' + actual + '"', header, actual);
211
+ }
212
+ }
213
+ /**
214
+ * Perform assertions on the response status and return an Error upon failure.
215
+ */
216
+ _assertStatus(status, res) {
217
+ if (res.status !== status) {
218
+ const a = STATUS_CODES[status];
219
+ const b = STATUS_CODES[res.status];
220
+ return new AssertError('expected ' + status + ' "' + a + '", got ' + res.status + ' "' + b + '"', status, res.status);
221
+ }
222
+ }
223
+ /**
224
+ * Perform assertions on the response status and return an Error upon failure.
225
+ */
226
+ _assertStatusArray(statusArray, res) {
227
+ if (!statusArray.includes(res.status)) {
228
+ const b = STATUS_CODES[res.status];
229
+ const expectedList = statusArray.join(', ');
230
+ return new AssertError('expected one of "' + expectedList + '", got ' + res.status + ' "' + b + '"', statusArray, res.status);
231
+ }
232
+ }
233
+ /**
234
+ * Performs an assertion by calling a function and return an Error upon failure.
235
+ */
236
+ _assertFunction(fn, res) {
237
+ let err;
238
+ try {
239
+ err = fn(res);
240
+ }
241
+ catch (e) {
242
+ err = e;
243
+ }
244
+ if (err instanceof Error) {
245
+ return err;
246
+ }
247
+ }
248
+ }
249
+ /**
250
+ * Wraps an assert function into another.
251
+ * The wrapper function edit the stack trace of any assertion error, prepending a more useful stack to it.
252
+ *
253
+ * @param {Function} assertFn
254
+ * @return {Function} wrapped assert function
255
+ */
256
+ function wrapAssertFn(assertFn) {
257
+ const savedStack = new Error().stack.split('\n').slice(3);
258
+ return (res) => {
259
+ let badStack;
260
+ let err;
261
+ try {
262
+ err = assertFn(res);
263
+ }
264
+ catch (e) {
265
+ err = e;
266
+ }
267
+ if (err instanceof Error && err.stack) {
268
+ badStack = err.stack.replace(err.message, '').split('\n').slice(1);
269
+ err.stack = [err.toString()]
270
+ .concat(savedStack)
271
+ .concat('----')
272
+ .concat(badStack)
273
+ .join('\n');
274
+ }
275
+ return err;
276
+ };
277
+ }
278
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxXQUFXLENBQUM7QUFDcEMsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLFdBQVcsQ0FBQztBQUN6QyxPQUFPLEVBQUUsTUFBTSxJQUFJLFdBQVcsRUFBRSxNQUFNLFVBQVUsQ0FBQztBQUVqRCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQzlDLE9BQU8sRUFBRSxPQUFPLEVBQWlCLE1BQU0sWUFBWSxDQUFDO0FBQ3BELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQVlyRCxNQUFNLE9BQU8sSUFBSyxTQUFRLE9BQU87SUFDL0IsR0FBRyxDQUFrQjtJQUNyQixPQUFPLENBQVM7SUFDaEIsUUFBUSxHQUFxQixFQUFFLENBQUM7SUFFaEM7OztPQUdHO0lBQ0gsWUFBWSxHQUFvQixFQUFFLE1BQWMsRUFBRSxJQUFZO1FBQzVELEtBQUssQ0FBQyxNQUFNLENBQUMsV0FBVyxFQUFFLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFFbEMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNsQixJQUFJLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDZCxJQUFJLENBQUMsR0FBRyxHQUFHLEdBQUcsQ0FBQztRQUNmLElBQUksQ0FBQyxHQUFHLEdBQUcsT0FBTyxHQUFHLEtBQUssUUFBUTtZQUNoQyxDQUFDLENBQUMsR0FBRyxHQUFHLElBQUk7WUFDWixDQUFDLENBQUMsSUFBSSxDQUFDLGFBQWEsQ0FBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDcEMsQ0FBQztJQUVEOzs7OztPQUtHO0lBQ08sYUFBYSxDQUFDLEdBQVcsRUFBRSxJQUFZO1FBQy9DLE1BQU0sSUFBSSxHQUFHLEdBQUcsQ0FBQyxPQUFPLEVBQUUsQ0FBQztRQUMzQixJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7WUFDVixJQUFJLENBQUMsT0FBTyxHQUFHLEdBQUcsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDL0IsQ0FBQztRQUNELE1BQU0sSUFBSSxHQUFJLEdBQUcsQ0FBQyxPQUFPLEVBQWtCLENBQUMsSUFBSSxDQUFDO1FBQ2pELE1BQU0sUUFBUSxHQUFHLENBQUMsR0FBRyxZQUFZLFdBQVcsSUFBSSxJQUFJLENBQUMsT0FBTyxZQUFZLFdBQVcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQztRQUN4RyxPQUFPLEdBQUcsUUFBUSxnQkFBZ0IsSUFBSSxHQUFHLElBQUksRUFBRSxDQUFDO0lBQ2xELENBQUM7SUFFRDs7Ozs7Ozs7Ozs7Ozs7Ozs7T0FpQkc7SUFDSCxNQUFNLENBQUMsQ0FBcUQsRUFBRSxDQUErQyxFQUFFLENBQW9CO1FBQ2pJLFdBQVc7UUFDWCxJQUFJLE9BQU8sQ0FBQyxLQUFLLFVBQVUsRUFBRSxDQUFDO1lBQzVCLGNBQWM7WUFDZCxJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsQ0FBbUIsQ0FBQyxDQUFDLENBQUM7WUFDdEQsT0FBTyxJQUFJLENBQUM7UUFDZCxDQUFDO1FBQ0QsSUFBSSxPQUFPLENBQUMsS0FBSyxVQUFVLEVBQUUsQ0FBQztZQUM1QiwyQkFBMkI7WUFDM0IsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNkLENBQUM7UUFDRCxJQUFJLE9BQU8sQ0FBQyxLQUFLLFVBQVUsRUFBRSxDQUFDO1lBQzVCLGtEQUFrRDtZQUNsRCxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2QsQ0FBQztRQUVELFNBQVM7UUFDVCxJQUFJLE9BQU8sQ0FBQyxLQUFLLFFBQVEsRUFBRSxDQUFDO1lBQzFCLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQ25FLE9BQU87WUFDUCxJQUFJLE9BQU8sQ0FBQyxLQUFLLFVBQVUsSUFBSSxTQUFTLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRSxDQUFDO2dCQUNwRCx1QkFBdUI7Z0JBQ3ZCLHFCQUFxQjtnQkFDckIsd0JBQXdCO2dCQUN4QixJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUNuRSxDQUFDO1lBQ0QsT0FBTyxJQUFJLENBQUM7UUFDZCxDQUFDO1FBRUQsb0JBQW9CO1FBQ3BCLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsTUFBTSxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsT0FBTyxHQUFHLEtBQUssUUFBUSxDQUFDLEVBQUUsQ0FBQztZQUNoRixzQkFBc0I7WUFDdEIsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUN4RSxPQUFPLElBQUksQ0FBQztRQUNkLENBQUM7UUFFRCxlQUFlO1FBQ2YsSUFBSSxPQUFPLENBQUMsS0FBSyxRQUFRLElBQUksT0FBTyxDQUFDLEtBQUssUUFBUSxJQUFJLENBQUMsWUFBWSxNQUFNLEVBQUUsQ0FBQztZQUMxRSw4Q0FBOEM7WUFDOUMsa0NBQWtDO1lBQ2xDLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsRUFBRSxJQUFJLEVBQUUsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLEtBQUssRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUMvRixPQUFPLElBQUksQ0FBQztRQUNkLENBQUM7UUFFRCxPQUFPO1FBQ1Asa0JBQWtCO1FBQ2xCLCtDQUErQztRQUMvQyxpQkFBaUI7UUFDakIsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLElBQUksRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFFakUsT0FBTyxJQUFJLENBQUM7SUFDZCxDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsR0FBRyxDQUFDLEVBQW9CO1FBQ3RCLE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUM7UUFFNUIsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLEdBQUcsRUFBRSxHQUFHLEVBQUUsRUFBRTtZQUNyQixNQUFNLFdBQVcsR0FBRyxHQUFHLEVBQUU7Z0JBQ3ZCLElBQUksQ0FBQyxNQUFNLENBQUMsR0FBRyxFQUFFLEdBQUcsRUFBRSxFQUFFLENBQUMsQ0FBQztZQUM1QixDQUFDLENBQUM7WUFFRixJQUFJLE1BQU0sSUFBSSxTQUFTLElBQUksTUFBTSxJQUFJLE1BQU0sQ0FBQyxPQUFPLEVBQUUsQ0FBQztnQkFDcEQsT0FBTyxNQUFNLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxDQUFDO1lBQ25DLENBQUM7WUFFRCxXQUFXLEVBQUUsQ0FBQztRQUNoQixDQUFDLENBQUMsQ0FBQztRQUVILE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQztJQUVEOztPQUVHO0lBQ0gsTUFBTSxDQUFDLFFBQThCLEVBQUUsR0FBYSxFQUFFLEVBQW9CO1FBQ3hFLElBQUksUUFBMkIsQ0FBQztRQUVoQyw2RUFBNkU7UUFDN0UscUVBQXFFO1FBQ3JFLCtEQUErRDtRQUMvRCxpRUFBaUU7UUFDakUsTUFBTSxTQUFTLEdBQTJCO1lBQ3hDLFlBQVksRUFBRSxvQkFBb0I7WUFDbEMsVUFBVSxFQUFFLDBCQUEwQjtZQUN0QyxLQUFLLEVBQUUsYUFBYTtZQUNwQixTQUFTLEVBQUUscUJBQXFCO1NBQ2pDLENBQUM7UUFFRixJQUFJLENBQUMsR0FBRyxJQUFJLFFBQVEsRUFBRSxDQUFDO1lBQ3JCLElBQUksUUFBUSxZQUFZLEtBQUs7bUJBQ3hCLFFBQVEsQ0FBQyxPQUFPLEtBQUssU0FBUzttQkFDOUIsUUFBUSxDQUFDLElBQUk7bUJBQ2IsU0FBUyxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO2dCQUM5QixRQUFRLEdBQUcsSUFBSSxLQUFLLENBQUMsUUFBUSxDQUFDLElBQUksR0FBRyxJQUFJLEdBQUcsU0FBUyxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO1lBQ3hFLENBQUM7aUJBQU0sQ0FBQztnQkFDTixRQUFRLEdBQUcsUUFBUSxDQUFDO1lBQ3RCLENBQUM7UUFDSCxDQUFDO1FBRUQsVUFBVTtRQUNWLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxJQUFJLENBQUMsUUFBUSxDQUFDLE1BQU0sSUFBSSxDQUFDLFFBQVEsRUFBRSxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUM7WUFDOUQsUUFBUSxHQUFHLElBQUksQ0FBQyxlQUFlLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztRQUN6RCxDQUFDO1FBRUQsa0VBQWtFO1FBQ2xFLElBQUksQ0FBQyxRQUFRLElBQUksUUFBUSxZQUFZLEtBQUssSUFBSSxDQUFDLENBQUMsR0FBRyxJQUFJLFFBQVEsQ0FBQyxNQUFNLEtBQUssR0FBRyxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUM7WUFDdkYsUUFBUSxHQUFHLFFBQVEsQ0FBQztRQUN0QixDQUFDO1FBRUQsRUFBRSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsUUFBUSxJQUFJLElBQUksRUFBRSxHQUFHLENBQUMsQ0FBQztJQUN2QyxDQUFDO0lBRUQ7O09BRUc7SUFDSCxXQUFXLENBQUMsSUFBMEQsRUFBRSxHQUFhO1FBQ25GLE1BQU0sUUFBUSxHQUFHLElBQUksWUFBWSxNQUFNLENBQUM7UUFFeEMsU0FBUztRQUNULElBQUksT0FBTyxJQUFJLEtBQUssUUFBUSxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUM7WUFDMUMsSUFBSSxDQUFDO2dCQUNILGVBQWUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDO1lBQ2xDLENBQUM7WUFBQyxPQUFPLEdBQUcsRUFBRSxDQUFDO2dCQUNiLE1BQU0sQ0FBQyxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQztnQkFDeEIsTUFBTSxDQUFDLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQztnQkFDNUIsT0FBTyxJQUFJLFdBQVcsQ0FBQyxXQUFXLEdBQUcsQ0FBQyxHQUFHLHNCQUFzQixHQUFHLENBQUMsRUFBRSxJQUFJLEVBQUUsR0FBRyxDQUFDLElBQUksRUFBRSxFQUFFLEtBQUssRUFBRSxHQUFHLEVBQUUsQ0FBQyxDQUFDO1lBQ3ZHLENBQUM7UUFDSCxDQUFDO2FBQU0sSUFBSSxJQUFJLEtBQUssR0FBRyxDQUFDLElBQUksRUFBRSxDQUFDO1lBQzdCLFNBQVM7WUFDVCxNQUFNLENBQUMsR0FBRyxPQUFPLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDeEIsTUFBTSxDQUFDLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQztZQUU1QixTQUFTO1lBQ1QsSUFBSSxRQUFRLEVBQUUsQ0FBQztnQkFDYixJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQztvQkFDekIsT0FBTyxJQUFJLFdBQVcsQ0FBQyxnQkFBZ0IsR0FBRyxDQUFDLEdBQUcsWUFBWSxHQUFHLElBQUksRUFBRSxJQUFJLEVBQUUsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDO2dCQUNyRixDQUFDO1lBQ0gsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLE9BQU8sSUFBSSxXQUFXLENBQUMsV0FBVyxHQUFHLENBQUMsR0FBRyxzQkFBc0IsR0FBRyxDQUFDLEVBQUUsSUFBSSxFQUFFLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQztZQUN2RixDQUFDO1FBQ0gsQ0FBQztJQUNILENBQUM7SUFFRDs7T0FFRztJQUNILGFBQWEsQ0FBQyxNQUFvQixFQUFFLEdBQWE7UUFDL0MsTUFBTSxLQUFLLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQztRQUMxQixNQUFNLE1BQU0sR0FBRyxHQUFHLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxXQUFXLEVBQUUsQ0FBQyxDQUFDO1FBQy9DLE1BQU0sYUFBYSxHQUFHLE1BQU0sQ0FBQyxLQUFLLENBQUM7UUFFbkMsSUFBSSxPQUFPLE1BQU0sS0FBSyxXQUFXLEVBQUUsQ0FBQztZQUNsQyxPQUFPLElBQUksV0FBVyxDQUFDLFlBQVksR0FBRyxLQUFLLEdBQUcsZ0JBQWdCLEVBQUUsTUFBTSxFQUFFLE1BQU0sQ0FBQyxDQUFDO1FBQ2xGLENBQUM7UUFDRCxnRkFBZ0Y7UUFDaEYsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLElBQUksTUFBTSxDQUFDLFFBQVEsRUFBRSxLQUFLLGFBQWEsQ0FBQztlQUM3RCxhQUFhLEtBQUssTUFBTSxFQUFFLENBQUM7WUFDOUIsT0FBTztRQUNULENBQUM7UUFDRCxJQUFJLGFBQWEsWUFBWSxNQUFNLEVBQUUsQ0FBQztZQUNwQyxJQUFJLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDO2dCQUNoQyxPQUFPLElBQUksV0FBVyxDQUFDLFlBQVksR0FBRyxLQUFLLEdBQUcsYUFBYTtzQkFDdkQsYUFBYSxHQUFHLFNBQVMsR0FBRyxNQUFNLEdBQUcsR0FBRyxFQUFFLE1BQU0sRUFBRSxNQUFNLENBQUMsQ0FBQztZQUNoRSxDQUFDO1FBQ0gsQ0FBQzthQUFNLENBQUM7WUFDTixPQUFPLElBQUksV0FBVyxDQUFDLFlBQVksR0FBRyxLQUFLLEdBQUcsUUFBUSxHQUFHLGFBQWEsR0FBRyxVQUFVLEdBQUcsTUFBTSxHQUFHLEdBQUcsRUFDaEcsTUFBTSxFQUFFLE1BQU0sQ0FDZixDQUFDO1FBQ0osQ0FBQztJQUNILENBQUM7SUFFRDs7T0FFRztJQUNILGFBQWEsQ0FBQyxNQUFjLEVBQUUsR0FBYTtRQUN6QyxJQUFJLEdBQUcsQ0FBQyxNQUFNLEtBQUssTUFBTSxFQUFFLENBQUM7WUFDMUIsTUFBTSxDQUFDLEdBQUcsWUFBWSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1lBQy9CLE1BQU0sQ0FBQyxHQUFHLFlBQVksQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLENBQUM7WUFDbkMsT0FBTyxJQUFJLFdBQVcsQ0FBQyxXQUFXLEdBQUcsTUFBTSxHQUFHLElBQUksR0FBRyxDQUFDLEdBQUcsU0FBUyxHQUFHLEdBQUcsQ0FBQyxNQUFNLEdBQUcsSUFBSSxHQUFHLENBQUMsR0FBRyxHQUFHLEVBQzlGLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTSxDQUNuQixDQUFDO1FBQ0osQ0FBQztJQUNILENBQUM7SUFFRDs7T0FFRztJQUNILGtCQUFrQixDQUFDLFdBQXFCLEVBQUUsR0FBYTtRQUNyRCxJQUFJLENBQUMsV0FBVyxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztZQUN0QyxNQUFNLENBQUMsR0FBRyxZQUFZLENBQUMsR0FBRyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1lBQ25DLE1BQU0sWUFBWSxHQUFHLFdBQVcsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDNUMsT0FBTyxJQUFJLFdBQVcsQ0FDcEIsbUJBQW1CLEdBQUcsWUFBWSxHQUFHLFNBQVMsR0FBRyxHQUFHLENBQUMsTUFBTSxHQUFHLElBQUksR0FBRyxDQUFDLEdBQUcsR0FBRyxFQUM1RSxXQUFXLEVBQUUsR0FBRyxDQUFDLE1BQU0sQ0FDeEIsQ0FBQztRQUNKLENBQUM7SUFDSCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxlQUFlLENBQUMsRUFBa0IsRUFBRSxHQUFhO1FBQy9DLElBQUksR0FBRyxDQUFDO1FBQ1IsSUFBSSxDQUFDO1lBQ0gsR0FBRyxHQUFHLEVBQUUsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUNoQixDQUFDO1FBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNYLEdBQUcsR0FBRyxDQUFDLENBQUM7UUFDVixDQUFDO1FBQ0QsSUFBSSxHQUFHLFlBQVksS0FBSyxFQUFFLENBQUM7WUFDekIsT0FBTyxHQUFHLENBQUM7UUFDYixDQUFDO0lBQ0gsQ0FBQztDQUNGO0FBRUQ7Ozs7OztHQU1HO0FBRUgsU0FBUyxZQUFZLENBQUMsUUFBd0I7SUFDNUMsTUFBTSxVQUFVLEdBQUcsSUFBSSxLQUFLLEVBQUUsQ0FBQyxLQUFNLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUUzRCxPQUFPLENBQUMsR0FBYSxFQUFFLEVBQUU7UUFDdkIsSUFBSSxRQUFRLENBQUM7UUFDYixJQUFJLEdBQUcsQ0FBQztRQUNSLElBQUksQ0FBQztZQUNILEdBQUcsR0FBRyxRQUFRLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDdEIsQ0FBQztRQUFDLE9BQU8sQ0FBTSxFQUFFLENBQUM7WUFDaEIsR0FBRyxHQUFHLENBQUMsQ0FBQztRQUNWLENBQUM7UUFDRCxJQUFJLEdBQUcsWUFBWSxLQUFLLElBQUksR0FBRyxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQ3RDLFFBQVEsR0FBRyxHQUFHLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDbkUsR0FBRyxDQUFDLEtBQUssR0FBRyxDQUFFLEdBQUcsQ0FBQyxRQUFRLEVBQUUsQ0FBRTtpQkFDM0IsTUFBTSxDQUFDLFVBQVUsQ0FBQztpQkFDbEIsTUFBTSxDQUFDLE1BQU0sQ0FBQztpQkFDZCxNQUFNLENBQUMsUUFBUSxDQUFDO2lCQUNoQixJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDaEIsQ0FBQztRQUNELE9BQU8sR0FBRyxDQUFDO0lBQ2IsQ0FBQyxDQUFDO0FBQ0osQ0FBQyJ9
@@ -0,0 +1,10 @@
1
+ import type { RequestListener } from 'node:http';
2
+ import type { Http2ServerRequest, Http2ServerResponse } from 'node:http2';
3
+ import type { Server } from 'node:net';
4
+ import type { AgentOptions as SAgentOptions } from 'superagent';
5
+ export type H2RequestListener = (request: Http2ServerRequest, response: Http2ServerResponse) => void;
6
+ export type H1RequestListener = RequestListener;
7
+ export type App = Server | H1RequestListener | H2RequestListener | string;
8
+ export interface AgentOptions extends SAgentOptions {
9
+ http2?: boolean;
10
+ }
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiJ9
@@ -0,0 +1,4 @@
1
+ {
2
+ "name": "@eggjs/supertest",
3
+ "version": "8.0.0"
4
+ }
package/package.json ADDED
@@ -0,0 +1,86 @@
1
+ {
2
+ "name": "@eggjs/supertest",
3
+ "description": "SuperAgent driven library for testing HTTP servers",
4
+ "version": "8.0.0",
5
+ "publishConfig": {
6
+ "access": "public"
7
+ },
8
+ "author": "TJ Holowaychuk",
9
+ "keywords": [
10
+ "bdd",
11
+ "http",
12
+ "request",
13
+ "superagent",
14
+ "tdd",
15
+ "test",
16
+ "testing"
17
+ ],
18
+ "license": "MIT",
19
+ "repository": {
20
+ "type": "git",
21
+ "url": "https://github.com/eggjs/supertest.git"
22
+ },
23
+ "engines": {
24
+ "node": ">= 18.19.0"
25
+ },
26
+ "dependencies": {
27
+ "superagent": "^9.0.1"
28
+ },
29
+ "devDependencies": {
30
+ "@arethetypeswrong/cli": "^0.17.1",
31
+ "@eggjs/tsconfig": "1",
32
+ "@types/body-parser": "^1.19.5",
33
+ "@types/cookie-parser": "^1.4.8",
34
+ "@types/express": "^5.0.0",
35
+ "@types/mocha": "10",
36
+ "@types/node": "22",
37
+ "@types/superagent": "^8.1.9",
38
+ "body-parser": "^1.20.3",
39
+ "cookie-parser": "^1.4.6",
40
+ "egg-bin": "6",
41
+ "eslint": "8",
42
+ "eslint-config-egg": "14",
43
+ "express": "^4.18.2",
44
+ "nock": "^13.3.0",
45
+ "nyc": "^15.1.0",
46
+ "should": "^13.2.3",
47
+ "tshy": "3",
48
+ "tshy-after": "1",
49
+ "typescript": "5"
50
+ },
51
+ "scripts": {
52
+ "lint": "eslint --cache src test --ext .ts",
53
+ "pretest": "npm run lint -- --fix && npm run prepublishOnly",
54
+ "test": "egg-bin test",
55
+ "preci": "npm run lint && npm run prepublishOnly && attw --pack",
56
+ "ci": "egg-bin cov",
57
+ "prepublishOnly": "tshy && tshy-after"
58
+ },
59
+ "type": "module",
60
+ "tshy": {
61
+ "exports": {
62
+ ".": "./src/index.ts",
63
+ "./package.json": "./package.json"
64
+ }
65
+ },
66
+ "exports": {
67
+ ".": {
68
+ "import": {
69
+ "types": "./dist/esm/index.d.ts",
70
+ "default": "./dist/esm/index.js"
71
+ },
72
+ "require": {
73
+ "types": "./dist/commonjs/index.d.ts",
74
+ "default": "./dist/commonjs/index.js"
75
+ }
76
+ },
77
+ "./package.json": "./package.json"
78
+ },
79
+ "files": [
80
+ "dist",
81
+ "src"
82
+ ],
83
+ "types": "./dist/commonjs/index.d.ts",
84
+ "main": "./dist/commonjs/index.js",
85
+ "module": "./dist/esm/index.js"
86
+ }
package/src/agent.ts ADDED
@@ -0,0 +1,93 @@
1
+ import http from 'node:http';
2
+ import http2 from 'node:http2';
3
+ import type { Server } from 'node:net';
4
+ import { agent as Agent } from 'superagent';
5
+ import { Test } from './test.js';
6
+ import type { AgentOptions, H1RequestListener, H2RequestListener, App } from './types.js';
7
+
8
+ /**
9
+ * Initialize a new `TestAgent`.
10
+ *
11
+ * @param {Function|Server} app
12
+ * @param {Object} options
13
+ */
14
+
15
+ export class TestAgent extends Agent {
16
+ app: Server | string;
17
+ _host: string;
18
+ #http2 = false;
19
+
20
+ constructor(appOrListener: App, options: AgentOptions = {}) {
21
+ super(options);
22
+ if (typeof appOrListener === 'function') {
23
+ if (options.http2) {
24
+ this.#http2 = true;
25
+ this.app = http2.createServer(appOrListener as H2RequestListener); // eslint-disable-line no-param-reassign
26
+ } else {
27
+ this.app = http.createServer(appOrListener as H1RequestListener); // eslint-disable-line no-param-reassign
28
+ }
29
+ } else {
30
+ this.app = appOrListener;
31
+ }
32
+ }
33
+
34
+ // set a host name
35
+ host(host: string) {
36
+ this._host = host;
37
+ return this;
38
+ }
39
+
40
+ // TestAgent.prototype.del = TestAgent.prototype.delete;
41
+
42
+ protected _testRequest(method: string, url: string) {
43
+ const req = new Test(this.app, method.toUpperCase(), url);
44
+ if (this.#http2) {
45
+ req.http2();
46
+ }
47
+
48
+ if (this._host) {
49
+ req.set('host', this._host);
50
+ }
51
+
52
+ const that = this as any;
53
+ // access not internal methods
54
+ req.on('response', that._saveCookies.bind(this));
55
+ req.on('redirect', that._saveCookies.bind(this));
56
+ req.on('redirect', that._attachCookies.bind(this, req));
57
+ that._setDefaults(req);
58
+ that._attachCookies(req);
59
+
60
+ return req;
61
+ }
62
+ delete(url: string) {
63
+ return this._testRequest('delete', url);
64
+ }
65
+ del(url: string) {
66
+ return this._testRequest('delete', url);
67
+ }
68
+ get(url: string) {
69
+ return this._testRequest('get', url);
70
+ }
71
+ head(url: string) {
72
+ return this._testRequest('head', url);
73
+ }
74
+ put(url: string) {
75
+ return this._testRequest('put', url);
76
+ }
77
+ post(url: string) {
78
+ return this._testRequest('post', url);
79
+ }
80
+ patch(url: string) {
81
+ return this._testRequest('patch', url);
82
+ }
83
+ options(url: string) {
84
+ return this._testRequest('options', url);
85
+ }
86
+ }
87
+
88
+ // allow keep use by `agent()`
89
+ export const proxyAgent = new Proxy(TestAgent, {
90
+ apply(target, _, argumentsList) {
91
+ return new target(argumentsList[0], argumentsList[1]);
92
+ },
93
+ });
@@ -0,0 +1,12 @@
1
+ export class AssertError extends Error {
2
+ expected: any;
3
+ actual: any;
4
+
5
+ constructor(message: string, expected: any, actual: any, options?: ErrorOptions) {
6
+ super(message, options);
7
+ this.name = this.constructor.name;
8
+ this.expected = expected;
9
+ this.actual = actual;
10
+ Error.captureStackTrace(this, this.constructor);
11
+ }
12
+ }
package/src/index.ts ADDED
@@ -0,0 +1,39 @@
1
+ import { Request, RequestOptions } from './request.js';
2
+ import { TestAgent, proxyAgent } from './agent.js';
3
+ import type { App, AgentOptions } from './types.js';
4
+
5
+ /**
6
+ * Test against the given `app`,
7
+ * returning a new `Test`.
8
+ */
9
+ export function request(app: App, options: RequestOptions = {}) {
10
+ return new Request(app, options);
11
+ }
12
+
13
+ export {
14
+ Request, RequestOptions,
15
+ TestAgent,
16
+ // import { agent } from '@eggjs/supertest';
17
+ // agent()
18
+ proxyAgent as agent,
19
+ };
20
+
21
+ export * from './test.js';
22
+
23
+ // import request from '@eggjs/supertest';
24
+ // request()
25
+ export default new Proxy(request, {
26
+ apply(target, _, argumentsList) {
27
+ return target(argumentsList[0], argumentsList[1]);
28
+ },
29
+ get(target, property, receiver) {
30
+ // import request from '@eggjs/supertest';
31
+ // request.agent()
32
+ if (property === 'agent') {
33
+ return proxyAgent;
34
+ }
35
+ return Reflect.get(target, property, receiver);
36
+ },
37
+ }) as unknown as ((app: App, options?: RequestOptions) => Request) & {
38
+ agent: (app: App, options?: AgentOptions) => TestAgent;
39
+ };