wrangler 2.0.12 → 2.0.16
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.
- package/README.md +7 -1
- package/bin/wrangler.js +111 -57
- package/miniflare-dist/index.mjs +9 -2
- package/package.json +156 -154
- package/src/__tests__/config-cache-without-cache-dir.test.ts +38 -0
- package/src/__tests__/config-cache.test.ts +30 -24
- package/src/__tests__/configuration.test.ts +3935 -3476
- package/src/__tests__/dev.test.tsx +1128 -979
- package/src/__tests__/guess-worker-format.test.ts +68 -68
- package/src/__tests__/helpers/cmd-shim.d.ts +6 -6
- package/src/__tests__/helpers/faye-websocket.d.ts +4 -4
- package/src/__tests__/helpers/mock-account-id.ts +24 -24
- package/src/__tests__/helpers/mock-bin.ts +20 -20
- package/src/__tests__/helpers/mock-cfetch.ts +92 -92
- package/src/__tests__/helpers/mock-console.ts +49 -39
- package/src/__tests__/helpers/mock-dialogs.ts +94 -71
- package/src/__tests__/helpers/mock-http-server.ts +30 -30
- package/src/__tests__/helpers/mock-istty.ts +65 -18
- package/src/__tests__/helpers/mock-kv.ts +26 -26
- package/src/__tests__/helpers/mock-oauth-flow.ts +223 -228
- package/src/__tests__/helpers/mock-process.ts +39 -0
- package/src/__tests__/helpers/mock-stdin.ts +82 -77
- package/src/__tests__/helpers/mock-web-socket.ts +21 -21
- package/src/__tests__/helpers/run-in-tmp.ts +27 -27
- package/src/__tests__/helpers/run-wrangler.ts +8 -8
- package/src/__tests__/helpers/write-worker-source.ts +16 -16
- package/src/__tests__/helpers/write-wrangler-toml.ts +9 -9
- package/src/__tests__/https-options.test.ts +104 -104
- package/src/__tests__/index.test.ts +239 -234
- package/src/__tests__/init.test.ts +1605 -1250
- package/src/__tests__/jest.setup.ts +63 -33
- package/src/__tests__/kv.test.ts +1128 -1011
- package/src/__tests__/logger.test.ts +100 -74
- package/src/__tests__/package-manager.test.ts +303 -303
- package/src/__tests__/pages.test.ts +1152 -652
- package/src/__tests__/parse.test.ts +252 -252
- package/src/__tests__/publish.test.ts +6371 -5622
- package/src/__tests__/pubsub.test.ts +367 -0
- package/src/__tests__/r2.test.ts +133 -133
- package/src/__tests__/route.test.ts +18 -18
- package/src/__tests__/secret.test.ts +382 -377
- package/src/__tests__/tail.test.ts +530 -530
- package/src/__tests__/user.test.ts +123 -111
- package/src/__tests__/whoami.test.tsx +198 -117
- package/src/__tests__/worker-namespace.test.ts +327 -0
- package/src/abort.d.ts +1 -1
- package/src/api/dev.ts +49 -0
- package/src/api/index.ts +1 -0
- package/src/bundle-reporter.tsx +29 -0
- package/src/bundle.ts +157 -149
- package/src/cfetch/index.ts +80 -80
- package/src/cfetch/internal.ts +90 -83
- package/src/cli.ts +21 -7
- package/src/config/config.ts +204 -195
- package/src/config/diagnostics.ts +61 -61
- package/src/config/environment.ts +390 -357
- package/src/config/index.ts +206 -193
- package/src/config/validation-helpers.ts +366 -366
- package/src/config/validation.ts +1573 -1376
- package/src/config-cache.ts +79 -41
- package/src/create-worker-preview.ts +206 -136
- package/src/create-worker-upload-form.ts +247 -238
- package/src/dev/dev-vars.ts +13 -13
- package/src/dev/dev.tsx +329 -307
- package/src/dev/local.tsx +304 -275
- package/src/dev/remote.tsx +366 -224
- package/src/dev/use-esbuild.ts +126 -91
- package/src/dev.tsx +538 -0
- package/src/dialogs.tsx +97 -97
- package/src/durable.ts +87 -87
- package/src/entry.ts +234 -228
- package/src/environment-variables.ts +23 -23
- package/src/errors.ts +6 -6
- package/src/generate.ts +33 -0
- package/src/git-client.ts +42 -0
- package/src/https-options.ts +79 -79
- package/src/index.tsx +1775 -2763
- package/src/init.ts +549 -0
- package/src/inspect.ts +593 -593
- package/src/intl-polyfill.d.ts +123 -123
- package/src/is-interactive.ts +12 -0
- package/src/kv.ts +277 -277
- package/src/logger.ts +46 -39
- package/src/miniflare-cli/enum-keys.ts +8 -8
- package/src/miniflare-cli/index.ts +42 -31
- package/src/miniflare-cli/request-context.ts +18 -18
- package/src/module-collection.ts +212 -212
- package/src/open-in-browser.ts +4 -6
- package/src/package-manager.ts +123 -123
- package/src/pages/build.tsx +202 -0
- package/src/pages/constants.ts +7 -0
- package/src/pages/deployments.tsx +101 -0
- package/src/pages/dev.tsx +964 -0
- package/src/pages/functions/buildPlugin.ts +105 -0
- package/src/pages/functions/buildWorker.ts +151 -0
- package/{pages → src/pages}/functions/filepath-routing.test.ts +113 -113
- package/src/pages/functions/filepath-routing.ts +189 -0
- package/src/pages/functions/identifiers.ts +78 -0
- package/src/pages/functions/routes.ts +151 -0
- package/src/pages/index.tsx +84 -0
- package/src/pages/projects.tsx +157 -0
- package/src/pages/publish.tsx +335 -0
- package/src/pages/types.ts +40 -0
- package/src/pages/upload.tsx +384 -0
- package/src/pages/utils.ts +12 -0
- package/src/parse.ts +202 -138
- package/src/paths.ts +6 -6
- package/src/preview.ts +31 -0
- package/src/proxy.ts +400 -402
- package/src/publish.ts +667 -621
- package/src/pubsub/index.ts +286 -0
- package/src/pubsub/pubsub-commands.tsx +577 -0
- package/src/r2.ts +19 -19
- package/src/selfsigned.d.ts +23 -23
- package/src/sites.tsx +271 -225
- package/src/tail/filters.ts +108 -108
- package/src/tail/index.ts +217 -217
- package/src/tail/printing.ts +45 -45
- package/src/update-check.ts +11 -11
- package/src/user/choose-account.tsx +60 -0
- package/src/user/env-vars.ts +46 -0
- package/src/user/generate-auth-url.ts +33 -0
- package/src/user/generate-random-state.ts +16 -0
- package/src/user/index.ts +3 -0
- package/src/user/user.tsx +1161 -0
- package/src/whoami.tsx +61 -42
- package/src/worker-namespace.ts +190 -0
- package/src/worker.ts +110 -100
- package/src/zones.ts +39 -36
- package/templates/checked-fetch.js +17 -0
- package/templates/new-worker-scheduled.js +3 -3
- package/templates/new-worker-scheduled.ts +15 -15
- package/templates/new-worker.js +3 -3
- package/templates/new-worker.ts +15 -15
- package/templates/no-op-worker.js +10 -0
- package/templates/pages-template-plugin.ts +155 -0
- package/templates/pages-template-worker.ts +161 -0
- package/templates/static-asset-facade.js +31 -31
- package/templates/tsconfig.json +95 -95
- package/wrangler-dist/cli.js +55383 -54138
- package/pages/functions/buildPlugin.ts +0 -105
- package/pages/functions/buildWorker.ts +0 -151
- package/pages/functions/filepath-routing.ts +0 -189
- package/pages/functions/identifiers.ts +0 -78
- package/pages/functions/routes.ts +0 -156
- package/pages/functions/template-plugin.ts +0 -147
- package/pages/functions/template-worker.ts +0 -143
- package/src/pages.tsx +0 -2093
- package/src/user.tsx +0 -1214
|
@@ -7,52 +7,54 @@ import { logger } from "../../logger";
|
|
|
7
7
|
*/
|
|
8
8
|
|
|
9
9
|
let debugSpy: jest.SpyInstance,
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
10
|
+
logSpy: jest.SpyInstance,
|
|
11
|
+
errorSpy: jest.SpyInstance,
|
|
12
|
+
warnSpy: jest.SpyInstance;
|
|
13
13
|
|
|
14
14
|
const std = {
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
15
|
+
get debug() {
|
|
16
|
+
return normalizeOutput(debugSpy);
|
|
17
|
+
},
|
|
18
|
+
get out() {
|
|
19
|
+
return normalizeOutput(logSpy);
|
|
20
|
+
},
|
|
21
|
+
get err() {
|
|
22
|
+
return normalizeOutput(errorSpy);
|
|
23
|
+
},
|
|
24
|
+
get warn() {
|
|
25
|
+
return normalizeOutput(warnSpy);
|
|
26
|
+
},
|
|
27
27
|
};
|
|
28
28
|
|
|
29
29
|
function normalizeOutput(spy: jest.SpyInstance): string {
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
30
|
+
return normalizeErrorMarkers(
|
|
31
|
+
replaceByte(
|
|
32
|
+
stripTrailingWhitespace(normalizeSlashes(stripTimings(captureCalls(spy))))
|
|
33
|
+
)
|
|
34
|
+
);
|
|
33
35
|
}
|
|
34
36
|
|
|
35
37
|
function captureCalls(spy: jest.SpyInstance): string {
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
38
|
+
return spy.mock.calls
|
|
39
|
+
.map((args: unknown[]) => util.format("%s", ...args))
|
|
40
|
+
.join("\n");
|
|
39
41
|
}
|
|
40
42
|
|
|
41
43
|
export function mockConsoleMethods() {
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
44
|
+
beforeEach(() => {
|
|
45
|
+
logger.columns = 100;
|
|
46
|
+
debugSpy = jest.spyOn(console, "debug").mockImplementation();
|
|
47
|
+
logSpy = jest.spyOn(console, "log").mockImplementation();
|
|
48
|
+
errorSpy = jest.spyOn(console, "error").mockImplementation();
|
|
49
|
+
warnSpy = jest.spyOn(console, "warn").mockImplementation();
|
|
50
|
+
});
|
|
51
|
+
afterEach(() => {
|
|
52
|
+
debugSpy.mockRestore();
|
|
53
|
+
logSpy.mockRestore();
|
|
54
|
+
errorSpy.mockRestore();
|
|
55
|
+
warnSpy.mockRestore();
|
|
56
|
+
});
|
|
57
|
+
return std;
|
|
56
58
|
}
|
|
57
59
|
|
|
58
60
|
/**
|
|
@@ -61,7 +63,7 @@ export function mockConsoleMethods() {
|
|
|
61
63
|
* Windows gets a different character.
|
|
62
64
|
*/
|
|
63
65
|
function normalizeErrorMarkers(str: string): string {
|
|
64
|
-
|
|
66
|
+
return str.replaceAll("✘", "X");
|
|
65
67
|
}
|
|
66
68
|
|
|
67
69
|
/**
|
|
@@ -70,7 +72,7 @@ function normalizeErrorMarkers(str: string): string {
|
|
|
70
72
|
* Use this in snapshot tests to be resilient to file-system differences.
|
|
71
73
|
*/
|
|
72
74
|
export function normalizeSlashes(str: string): string {
|
|
73
|
-
|
|
75
|
+
return str.replace(/\\/g, "/");
|
|
74
76
|
}
|
|
75
77
|
|
|
76
78
|
/**
|
|
@@ -79,9 +81,17 @@ export function normalizeSlashes(str: string): string {
|
|
|
79
81
|
* Use this in snapshot tests to be resilient to slight changes in timing of processing.
|
|
80
82
|
*/
|
|
81
83
|
export function stripTimings(stdout: string): string {
|
|
82
|
-
|
|
84
|
+
return stdout.replace(/\(\d+\.\d+ sec\)/g, "(TIMINGS)");
|
|
83
85
|
}
|
|
84
86
|
|
|
85
87
|
export function stripTrailingWhitespace(str: string): string {
|
|
86
|
-
|
|
88
|
+
return str.replace(/[^\S\n]+\n/g, "\n");
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
/**
|
|
92
|
+
* Removing leading kilobit (tenth of a byte) from test output due to
|
|
93
|
+
* variation causing every few tests the value to change by ± .01
|
|
94
|
+
*/
|
|
95
|
+
function replaceByte(stdout: string): string {
|
|
96
|
+
return stdout.replaceAll(/.[0-9][0-9] KiB/g, "xx KiB");
|
|
87
97
|
}
|
|
@@ -5,10 +5,10 @@ import { normalizeSlashes } from "./mock-console";
|
|
|
5
5
|
* The expected values for a confirmation request.
|
|
6
6
|
*/
|
|
7
7
|
export interface ConfirmExpectation {
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
8
|
+
/** The text expected to be seen in the confirmation dialog. */
|
|
9
|
+
text: string;
|
|
10
|
+
/** The mock response send back from the confirmation dialog. */
|
|
11
|
+
result: boolean;
|
|
12
12
|
}
|
|
13
13
|
|
|
14
14
|
/**
|
|
@@ -19,38 +19,47 @@ export interface ConfirmExpectation {
|
|
|
19
19
|
* then an error is thrown.
|
|
20
20
|
*/
|
|
21
21
|
export function mockConfirm(...expectations: ConfirmExpectation[]) {
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
22
|
+
(confirm as jest.Mock).mockImplementation((text: string) => {
|
|
23
|
+
for (const expectation of expectations) {
|
|
24
|
+
if (normalizeSlashes(text) === normalizeSlashes(expectation.text)) {
|
|
25
|
+
expectations = expectations.filter((e) => e !== expectation);
|
|
26
|
+
return Promise.resolve(expectation.result);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
throw new Error(`Unexpected confirmation message: ${text}`);
|
|
30
|
+
});
|
|
31
|
+
return () => {
|
|
32
|
+
if (expectations.length > 0) {
|
|
33
|
+
throw new Error(
|
|
34
|
+
"The following expected confirmation dialogs were not used:\n" +
|
|
35
|
+
expectations.map((e) => `- "${e.text}"`).join("\n")
|
|
36
|
+
);
|
|
37
|
+
}
|
|
38
|
+
};
|
|
30
39
|
}
|
|
31
40
|
|
|
32
41
|
export function clearConfirmMocks() {
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
+
(confirm as jest.Mock).mockReset();
|
|
43
|
+
// Because confirm was originally a spy, calling mockReset will simply reset
|
|
44
|
+
// it as a function with no return value (!), so we need to accitionally reset
|
|
45
|
+
// the mock implementation to the one that throws (from jest.setup.js).
|
|
46
|
+
(confirm as jest.Mock).mockImplementation((text: string) => {
|
|
47
|
+
throw new Error(
|
|
48
|
+
`Unexpected call to \`confirm("${text}")\`.\nYou should use \`mockConfirm()\` to mock calls to \`confirm()\` with expectations. Search the codebase for \`mockConfirm\` to learn more.`
|
|
49
|
+
);
|
|
50
|
+
});
|
|
42
51
|
}
|
|
43
52
|
|
|
44
53
|
/**
|
|
45
54
|
* The expected values for a prompt request.
|
|
46
55
|
*/
|
|
47
56
|
export interface PromptExpectation {
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
57
|
+
/** The text expected to be seen in the prompt dialog. */
|
|
58
|
+
text: string;
|
|
59
|
+
/** The type of the prompt. */
|
|
60
|
+
type: "text" | "password";
|
|
61
|
+
/** The mock response send back from the prompt dialog. */
|
|
62
|
+
result: string;
|
|
54
63
|
}
|
|
55
64
|
|
|
56
65
|
/**
|
|
@@ -61,42 +70,47 @@ export interface PromptExpectation {
|
|
|
61
70
|
* then an error is thrown.
|
|
62
71
|
*/
|
|
63
72
|
export function mockPrompt(...expectations: PromptExpectation[]) {
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
73
|
+
(prompt as jest.Mock).mockImplementation(
|
|
74
|
+
(text: string, type: "text" | "password") => {
|
|
75
|
+
for (const expectation of expectations) {
|
|
76
|
+
if (text === expectation.text && type == expectation.type) {
|
|
77
|
+
expectations = expectations.filter((e) => e !== expectation);
|
|
78
|
+
return Promise.resolve(expectation.result);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
throw new Error(`Unexpected confirmation message: ${text}`);
|
|
82
|
+
}
|
|
83
|
+
);
|
|
84
|
+
return () => {
|
|
85
|
+
if (expectations.length > 0) {
|
|
86
|
+
throw new Error(
|
|
87
|
+
"The following expected prompt dialogs were not used:\n" +
|
|
88
|
+
expectations.map((e) => `- "${e.text}"`).join("\n")
|
|
89
|
+
);
|
|
90
|
+
}
|
|
91
|
+
};
|
|
78
92
|
}
|
|
79
93
|
|
|
80
94
|
export function clearPromptMocks() {
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
95
|
+
(prompt as jest.Mock).mockReset();
|
|
96
|
+
// Because prompt was originally a spy, calling mockReset will simply reset
|
|
97
|
+
// it as a function with no return value (!), so we need to accitionally reset
|
|
98
|
+
// the mock implementation to the one that throws (from jest.setup.js).
|
|
99
|
+
(prompt as jest.Mock).mockImplementation((text: string) => {
|
|
100
|
+
throw new Error(
|
|
101
|
+
`Unexpected call to \`prompt(${text}, ...)\`.\nYou should use \`mockPrompt()\` to mock calls to \`prompt()\` with expectations. Search the codebase for \`mockPrompt\` to learn more.`
|
|
102
|
+
);
|
|
103
|
+
});
|
|
90
104
|
}
|
|
91
105
|
|
|
92
106
|
/**
|
|
93
107
|
* The expected values for a select request.
|
|
94
108
|
*/
|
|
95
109
|
export interface SelectExpectation {
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
110
|
+
/** The text expected to be seen in the select dialog. */
|
|
111
|
+
text: string;
|
|
112
|
+
/** The mock response send back from the select dialog. */
|
|
113
|
+
result: string;
|
|
100
114
|
}
|
|
101
115
|
|
|
102
116
|
/**
|
|
@@ -107,24 +121,33 @@ export interface SelectExpectation {
|
|
|
107
121
|
* then an error is thrown.
|
|
108
122
|
*/
|
|
109
123
|
export function mockSelect(...expectations: SelectExpectation[]) {
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
124
|
+
(select as jest.Mock).mockImplementation((text: string) => {
|
|
125
|
+
for (const expectation of expectations) {
|
|
126
|
+
if (normalizeSlashes(text) === normalizeSlashes(expectation.text)) {
|
|
127
|
+
expectations = expectations.filter((e) => e !== expectation);
|
|
128
|
+
return Promise.resolve(expectation.result);
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
throw new Error(`Unexpected select message: ${text}`);
|
|
132
|
+
});
|
|
133
|
+
return () => {
|
|
134
|
+
if (expectations.length > 0) {
|
|
135
|
+
throw new Error(
|
|
136
|
+
"The following expected select dialogs were not used:\n" +
|
|
137
|
+
expectations.map((e) => `- "${e.text}"`).join("\n")
|
|
138
|
+
);
|
|
139
|
+
}
|
|
140
|
+
};
|
|
118
141
|
}
|
|
119
142
|
|
|
120
143
|
export function clearSelectMocks() {
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
144
|
+
(select as jest.Mock).mockReset();
|
|
145
|
+
// Because select was originally a spy, calling mockReset will simply reset
|
|
146
|
+
// it as a function with no return value (!), so we need to additionally reset
|
|
147
|
+
// the mock implementation to the one that throws (from jest.setup.js).
|
|
148
|
+
(select as jest.Mock).mockImplementation((text: string) => {
|
|
149
|
+
throw new Error(
|
|
150
|
+
`Unexpected call to \`select("${text}")\`.\nYou should use \`mockSelect()\` to mock calls to \`select()\` with expectations. Search the codebase for \`mockSelect\` to learn more.`
|
|
151
|
+
);
|
|
152
|
+
});
|
|
130
153
|
}
|
|
@@ -7,40 +7,40 @@ import type { Request } from "undici";
|
|
|
7
7
|
* @returns a `fetch`-like function that will trigger the mock server to handle the request.
|
|
8
8
|
*/
|
|
9
9
|
export function mockHttpServer() {
|
|
10
|
-
|
|
10
|
+
let listener: http.RequestListener;
|
|
11
11
|
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
12
|
+
beforeEach(() => {
|
|
13
|
+
jest
|
|
14
|
+
.spyOn(http, "createServer")
|
|
15
|
+
.mockImplementation((...args: unknown[]) => {
|
|
16
|
+
listener = args.pop() as http.RequestListener;
|
|
17
|
+
return {
|
|
18
|
+
listen: jest.fn(),
|
|
19
|
+
close(callback?: (err?: Error) => void) {
|
|
20
|
+
callback?.();
|
|
21
|
+
return this;
|
|
22
|
+
},
|
|
23
|
+
} as unknown as http.Server;
|
|
24
|
+
});
|
|
25
|
+
});
|
|
26
26
|
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
27
|
+
return async (req: Request) => {
|
|
28
|
+
const resp = new http.ServerResponse(
|
|
29
|
+
// If you squint you can just about see that an `IncomingMessages` is like a `Request`!
|
|
30
|
+
req as unknown as http.IncomingMessage
|
|
31
|
+
);
|
|
32
32
|
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
33
|
+
// The listener will attache a callback to the response by calling `resp.end(callback)`.
|
|
34
|
+
// We want to capture that so that we can trigger it after the listener has completed its work.
|
|
35
|
+
const endSpy = jest.spyOn(resp, "end");
|
|
36
36
|
|
|
37
|
-
|
|
38
|
-
|
|
37
|
+
// The `await` here is important to allow the listener to complete its async work before we end the response.
|
|
38
|
+
await listener(req as unknown as http.IncomingMessage, resp);
|
|
39
39
|
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
40
|
+
// Now trigger the end callback.
|
|
41
|
+
const endCallback = endSpy.mock.calls[0].pop();
|
|
42
|
+
endCallback?.();
|
|
43
43
|
|
|
44
|
-
|
|
45
|
-
|
|
44
|
+
return resp;
|
|
45
|
+
};
|
|
46
46
|
}
|
|
@@ -1,27 +1,74 @@
|
|
|
1
|
-
const
|
|
2
|
-
const
|
|
1
|
+
const ORIGINAL_STDOUT = process.stdout;
|
|
2
|
+
const ORIGINAL_STDIN = process.stdin;
|
|
3
3
|
|
|
4
4
|
/**
|
|
5
5
|
* Mock `process.stdout.isTTY`
|
|
6
6
|
*/
|
|
7
7
|
export function useMockIsTTY() {
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
8
|
+
/**
|
|
9
|
+
* Explicitly set `process.stdout.isTTY` to a given value (or to a getter function).
|
|
10
|
+
*/
|
|
11
|
+
const setIsTTY = (
|
|
12
|
+
isTTY:
|
|
13
|
+
| boolean
|
|
14
|
+
| { stdin: boolean | (() => boolean); stdout: boolean | (() => boolean) }
|
|
15
|
+
) => {
|
|
16
|
+
mockStdStream("stdout", ORIGINAL_STDOUT, isTTY);
|
|
17
|
+
mockStdStream("stdin", ORIGINAL_STDIN, isTTY);
|
|
18
|
+
};
|
|
15
19
|
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
+
beforeEach(() => {
|
|
21
|
+
Object.defineProperty(process, "stdout", { value: ORIGINAL_STDOUT });
|
|
22
|
+
Object.defineProperty(process, "stdin", { value: ORIGINAL_STDIN });
|
|
23
|
+
});
|
|
20
24
|
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
+
afterEach(() => {
|
|
26
|
+
Object.defineProperty(process, "stdout", { value: ORIGINAL_STDOUT });
|
|
27
|
+
Object.defineProperty(process, "stdin", { value: ORIGINAL_STDIN });
|
|
28
|
+
});
|
|
25
29
|
|
|
26
|
-
|
|
30
|
+
return { setIsTTY };
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* Create a mock version of the specified stream which overrides `isTTY`
|
|
35
|
+
* with the given mock responses.
|
|
36
|
+
*
|
|
37
|
+
* @param streamName the property name on `process` for the stream to be mocked.
|
|
38
|
+
* @param originalStream the original stream object from the `process` object to be overridden.
|
|
39
|
+
* @param isTTY the mock behaviour for the `isTTY` property:
|
|
40
|
+
* - boolean or `{ [streamName]: boolean } - use this value for isTTY;
|
|
41
|
+
* - { [streamName]: () => boolean } - use this function as a getter for isTTY.
|
|
42
|
+
*/
|
|
43
|
+
function mockStdStream<T extends object>(
|
|
44
|
+
streamName: "stdout" | "stdin",
|
|
45
|
+
originalStream: T,
|
|
46
|
+
isTTY:
|
|
47
|
+
| boolean
|
|
48
|
+
| { stdin: boolean | (() => boolean); stdout: boolean | (() => boolean) }
|
|
49
|
+
) {
|
|
50
|
+
Object.defineProperty(process, streamName, {
|
|
51
|
+
value: createStdProxy(
|
|
52
|
+
originalStream,
|
|
53
|
+
typeof isTTY === "boolean" ? isTTY : isTTY[streamName]
|
|
54
|
+
),
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* Create a proxy wrapper around the given `stream` object that overrides the `isTTY` property.
|
|
60
|
+
*/
|
|
61
|
+
function createStdProxy<T extends object>(
|
|
62
|
+
stream: T,
|
|
63
|
+
isTTY: boolean | (() => boolean)
|
|
64
|
+
): T {
|
|
65
|
+
return new Proxy(stream, {
|
|
66
|
+
get(target, prop) {
|
|
67
|
+
return prop === "isTTY"
|
|
68
|
+
? typeof isTTY === "boolean"
|
|
69
|
+
? isTTY
|
|
70
|
+
: isTTY()
|
|
71
|
+
: target[prop as keyof typeof target];
|
|
72
|
+
},
|
|
73
|
+
});
|
|
27
74
|
}
|
|
@@ -2,32 +2,32 @@ import { createFetchResult, setMockRawResponse } from "./mock-cfetch";
|
|
|
2
2
|
import type { NamespaceKeyInfo } from "../../kv";
|
|
3
3
|
|
|
4
4
|
export function mockKeyListRequest(
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
5
|
+
expectedNamespaceId: string,
|
|
6
|
+
expectedKeys: NamespaceKeyInfo[],
|
|
7
|
+
keysPerRequest = 1000,
|
|
8
|
+
blankCursorValue: "" | undefined | null = undefined
|
|
9
9
|
) {
|
|
10
|
-
|
|
11
|
-
|
|
10
|
+
const requests = { count: 0 };
|
|
11
|
+
// See https://api.cloudflare.com/#workers-kv-namespace-list-a-namespace-s-keys
|
|
12
12
|
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
13
|
+
setMockRawResponse(
|
|
14
|
+
"/accounts/:accountId/storage/kv/namespaces/:namespaceId/keys",
|
|
15
|
+
"GET",
|
|
16
|
+
([_url, accountId, namespaceId], _init, query) => {
|
|
17
|
+
requests.count++;
|
|
18
|
+
expect(accountId).toEqual("some-account-id");
|
|
19
|
+
expect(namespaceId).toEqual(expectedNamespaceId);
|
|
20
|
+
if (expectedKeys.length <= keysPerRequest) {
|
|
21
|
+
return createFetchResult(expectedKeys);
|
|
22
|
+
} else {
|
|
23
|
+
const start = parseInt(query.get("cursor") ?? "0") || 0;
|
|
24
|
+
const end = start + keysPerRequest;
|
|
25
|
+
const cursor = end < expectedKeys.length ? end : blankCursorValue;
|
|
26
|
+
return createFetchResult(expectedKeys.slice(start, end), true, [], [], {
|
|
27
|
+
cursor,
|
|
28
|
+
});
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
);
|
|
32
|
+
return requests;
|
|
33
33
|
}
|