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.
Files changed (149) hide show
  1. package/README.md +7 -1
  2. package/bin/wrangler.js +111 -57
  3. package/miniflare-dist/index.mjs +9 -2
  4. package/package.json +156 -154
  5. package/src/__tests__/config-cache-without-cache-dir.test.ts +38 -0
  6. package/src/__tests__/config-cache.test.ts +30 -24
  7. package/src/__tests__/configuration.test.ts +3935 -3476
  8. package/src/__tests__/dev.test.tsx +1128 -979
  9. package/src/__tests__/guess-worker-format.test.ts +68 -68
  10. package/src/__tests__/helpers/cmd-shim.d.ts +6 -6
  11. package/src/__tests__/helpers/faye-websocket.d.ts +4 -4
  12. package/src/__tests__/helpers/mock-account-id.ts +24 -24
  13. package/src/__tests__/helpers/mock-bin.ts +20 -20
  14. package/src/__tests__/helpers/mock-cfetch.ts +92 -92
  15. package/src/__tests__/helpers/mock-console.ts +49 -39
  16. package/src/__tests__/helpers/mock-dialogs.ts +94 -71
  17. package/src/__tests__/helpers/mock-http-server.ts +30 -30
  18. package/src/__tests__/helpers/mock-istty.ts +65 -18
  19. package/src/__tests__/helpers/mock-kv.ts +26 -26
  20. package/src/__tests__/helpers/mock-oauth-flow.ts +223 -228
  21. package/src/__tests__/helpers/mock-process.ts +39 -0
  22. package/src/__tests__/helpers/mock-stdin.ts +82 -77
  23. package/src/__tests__/helpers/mock-web-socket.ts +21 -21
  24. package/src/__tests__/helpers/run-in-tmp.ts +27 -27
  25. package/src/__tests__/helpers/run-wrangler.ts +8 -8
  26. package/src/__tests__/helpers/write-worker-source.ts +16 -16
  27. package/src/__tests__/helpers/write-wrangler-toml.ts +9 -9
  28. package/src/__tests__/https-options.test.ts +104 -104
  29. package/src/__tests__/index.test.ts +239 -234
  30. package/src/__tests__/init.test.ts +1605 -1250
  31. package/src/__tests__/jest.setup.ts +63 -33
  32. package/src/__tests__/kv.test.ts +1128 -1011
  33. package/src/__tests__/logger.test.ts +100 -74
  34. package/src/__tests__/package-manager.test.ts +303 -303
  35. package/src/__tests__/pages.test.ts +1152 -652
  36. package/src/__tests__/parse.test.ts +252 -252
  37. package/src/__tests__/publish.test.ts +6371 -5622
  38. package/src/__tests__/pubsub.test.ts +367 -0
  39. package/src/__tests__/r2.test.ts +133 -133
  40. package/src/__tests__/route.test.ts +18 -18
  41. package/src/__tests__/secret.test.ts +382 -377
  42. package/src/__tests__/tail.test.ts +530 -530
  43. package/src/__tests__/user.test.ts +123 -111
  44. package/src/__tests__/whoami.test.tsx +198 -117
  45. package/src/__tests__/worker-namespace.test.ts +327 -0
  46. package/src/abort.d.ts +1 -1
  47. package/src/api/dev.ts +49 -0
  48. package/src/api/index.ts +1 -0
  49. package/src/bundle-reporter.tsx +29 -0
  50. package/src/bundle.ts +157 -149
  51. package/src/cfetch/index.ts +80 -80
  52. package/src/cfetch/internal.ts +90 -83
  53. package/src/cli.ts +21 -7
  54. package/src/config/config.ts +204 -195
  55. package/src/config/diagnostics.ts +61 -61
  56. package/src/config/environment.ts +390 -357
  57. package/src/config/index.ts +206 -193
  58. package/src/config/validation-helpers.ts +366 -366
  59. package/src/config/validation.ts +1573 -1376
  60. package/src/config-cache.ts +79 -41
  61. package/src/create-worker-preview.ts +206 -136
  62. package/src/create-worker-upload-form.ts +247 -238
  63. package/src/dev/dev-vars.ts +13 -13
  64. package/src/dev/dev.tsx +329 -307
  65. package/src/dev/local.tsx +304 -275
  66. package/src/dev/remote.tsx +366 -224
  67. package/src/dev/use-esbuild.ts +126 -91
  68. package/src/dev.tsx +538 -0
  69. package/src/dialogs.tsx +97 -97
  70. package/src/durable.ts +87 -87
  71. package/src/entry.ts +234 -228
  72. package/src/environment-variables.ts +23 -23
  73. package/src/errors.ts +6 -6
  74. package/src/generate.ts +33 -0
  75. package/src/git-client.ts +42 -0
  76. package/src/https-options.ts +79 -79
  77. package/src/index.tsx +1775 -2763
  78. package/src/init.ts +549 -0
  79. package/src/inspect.ts +593 -593
  80. package/src/intl-polyfill.d.ts +123 -123
  81. package/src/is-interactive.ts +12 -0
  82. package/src/kv.ts +277 -277
  83. package/src/logger.ts +46 -39
  84. package/src/miniflare-cli/enum-keys.ts +8 -8
  85. package/src/miniflare-cli/index.ts +42 -31
  86. package/src/miniflare-cli/request-context.ts +18 -18
  87. package/src/module-collection.ts +212 -212
  88. package/src/open-in-browser.ts +4 -6
  89. package/src/package-manager.ts +123 -123
  90. package/src/pages/build.tsx +202 -0
  91. package/src/pages/constants.ts +7 -0
  92. package/src/pages/deployments.tsx +101 -0
  93. package/src/pages/dev.tsx +964 -0
  94. package/src/pages/functions/buildPlugin.ts +105 -0
  95. package/src/pages/functions/buildWorker.ts +151 -0
  96. package/{pages → src/pages}/functions/filepath-routing.test.ts +113 -113
  97. package/src/pages/functions/filepath-routing.ts +189 -0
  98. package/src/pages/functions/identifiers.ts +78 -0
  99. package/src/pages/functions/routes.ts +151 -0
  100. package/src/pages/index.tsx +84 -0
  101. package/src/pages/projects.tsx +157 -0
  102. package/src/pages/publish.tsx +335 -0
  103. package/src/pages/types.ts +40 -0
  104. package/src/pages/upload.tsx +384 -0
  105. package/src/pages/utils.ts +12 -0
  106. package/src/parse.ts +202 -138
  107. package/src/paths.ts +6 -6
  108. package/src/preview.ts +31 -0
  109. package/src/proxy.ts +400 -402
  110. package/src/publish.ts +667 -621
  111. package/src/pubsub/index.ts +286 -0
  112. package/src/pubsub/pubsub-commands.tsx +577 -0
  113. package/src/r2.ts +19 -19
  114. package/src/selfsigned.d.ts +23 -23
  115. package/src/sites.tsx +271 -225
  116. package/src/tail/filters.ts +108 -108
  117. package/src/tail/index.ts +217 -217
  118. package/src/tail/printing.ts +45 -45
  119. package/src/update-check.ts +11 -11
  120. package/src/user/choose-account.tsx +60 -0
  121. package/src/user/env-vars.ts +46 -0
  122. package/src/user/generate-auth-url.ts +33 -0
  123. package/src/user/generate-random-state.ts +16 -0
  124. package/src/user/index.ts +3 -0
  125. package/src/user/user.tsx +1161 -0
  126. package/src/whoami.tsx +61 -42
  127. package/src/worker-namespace.ts +190 -0
  128. package/src/worker.ts +110 -100
  129. package/src/zones.ts +39 -36
  130. package/templates/checked-fetch.js +17 -0
  131. package/templates/new-worker-scheduled.js +3 -3
  132. package/templates/new-worker-scheduled.ts +15 -15
  133. package/templates/new-worker.js +3 -3
  134. package/templates/new-worker.ts +15 -15
  135. package/templates/no-op-worker.js +10 -0
  136. package/templates/pages-template-plugin.ts +155 -0
  137. package/templates/pages-template-worker.ts +161 -0
  138. package/templates/static-asset-facade.js +31 -31
  139. package/templates/tsconfig.json +95 -95
  140. package/wrangler-dist/cli.js +55383 -54138
  141. package/pages/functions/buildPlugin.ts +0 -105
  142. package/pages/functions/buildWorker.ts +0 -151
  143. package/pages/functions/filepath-routing.ts +0 -189
  144. package/pages/functions/identifiers.ts +0 -78
  145. package/pages/functions/routes.ts +0 -156
  146. package/pages/functions/template-plugin.ts +0 -147
  147. package/pages/functions/template-worker.ts +0 -143
  148. package/src/pages.tsx +0 -2093
  149. 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
- logSpy: jest.SpyInstance,
11
- errorSpy: jest.SpyInstance,
12
- warnSpy: jest.SpyInstance;
10
+ logSpy: jest.SpyInstance,
11
+ errorSpy: jest.SpyInstance,
12
+ warnSpy: jest.SpyInstance;
13
13
 
14
14
  const std = {
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
- },
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
- return normalizeErrorMarkers(
31
- stripTrailingWhitespace(normalizeSlashes(stripTimings(captureCalls(spy))))
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
- return spy.mock.calls
37
- .map((args: unknown[]) => util.format("%s", ...args))
38
- .join("\n");
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
- beforeEach(() => {
43
- logger.columns = 100;
44
- debugSpy = jest.spyOn(console, "debug").mockImplementation();
45
- logSpy = jest.spyOn(console, "log").mockImplementation();
46
- errorSpy = jest.spyOn(console, "error").mockImplementation();
47
- warnSpy = jest.spyOn(console, "warn").mockImplementation();
48
- });
49
- afterEach(() => {
50
- debugSpy.mockRestore();
51
- logSpy.mockRestore();
52
- errorSpy.mockRestore();
53
- warnSpy.mockRestore();
54
- });
55
- return std;
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
- return str.replaceAll("✘", "X");
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
- return str.replace(/\\/g, "/");
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
- return stdout.replace(/\(\d+\.\d+ sec\)/g, "(TIMINGS)");
84
+ return stdout.replace(/\(\d+\.\d+ sec\)/g, "(TIMINGS)");
83
85
  }
84
86
 
85
87
  export function stripTrailingWhitespace(str: string): string {
86
- return str.replace(/[^\S\n]+\n/g, "\n");
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
- /** 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;
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
- (confirm as jest.Mock).mockImplementation((text: string) => {
23
- for (const { text: expectedText, result } of expectations) {
24
- if (normalizeSlashes(text) === normalizeSlashes(expectedText)) {
25
- return Promise.resolve(result);
26
- }
27
- }
28
- throw new Error(`Unexpected confirmation message: ${text}`);
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
- (confirm as jest.Mock).mockReset();
34
- // Because confirm was originally a spy, calling mockReset will simply reset
35
- // it as a function with no return value (!), so we need to accitionally reset
36
- // the mock implementation to the one that throws (from jest.setup.js).
37
- (confirm as jest.Mock).mockImplementation((text: string) => {
38
- throw new Error(
39
- `Unexpected call to \`confirm("${text}")\`.\nYou should use \`mockConfirm()\` to mock calls to \`confirm()\` with expectations. Search the codebase for \`mockConfirm\` to learn more.`
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
- /** The text expected to be seen in the prompt dialog. */
49
- text: string;
50
- /** The type of the prompt. */
51
- type: "text" | "password";
52
- /** The mock response send back from the prompt dialog. */
53
- result: string;
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
- (prompt as jest.Mock).mockImplementation(
65
- (text: string, type: "text" | "password") => {
66
- for (const {
67
- text: expectedText,
68
- type: expectedType,
69
- result,
70
- } of expectations) {
71
- if (text === expectedText && type == expectedType) {
72
- return Promise.resolve(result);
73
- }
74
- }
75
- throw new Error(`Unexpected confirmation message: ${text}`);
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
- (prompt as jest.Mock).mockReset();
82
- // Because prompt was originally a spy, calling mockReset will simply reset
83
- // it as a function with no return value (!), so we need to accitionally reset
84
- // the mock implementation to the one that throws (from jest.setup.js).
85
- (prompt as jest.Mock).mockImplementation((text: string) => {
86
- throw new Error(
87
- `Unexpected call to \`prompt(${text}, ...)\`.\nYou should use \`mockPrompt()\` to mock calls to \`prompt()\` with expectations. Search the codebase for \`mockPrompt\` to learn more.`
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
- /** The text expected to be seen in the select dialog. */
97
- text: string;
98
- /** The mock response send back from the select dialog. */
99
- result: string;
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
- (select as jest.Mock).mockImplementation((text: string) => {
111
- for (const { text: expectedText, result } of expectations) {
112
- if (normalizeSlashes(text) === normalizeSlashes(expectedText)) {
113
- return Promise.resolve(result);
114
- }
115
- }
116
- throw new Error(`Unexpected select message: ${text}`);
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
- (select as jest.Mock).mockReset();
122
- // Because select was originally a spy, calling mockReset will simply reset
123
- // it as a function with no return value (!), so we need to additionally reset
124
- // the mock implementation to the one that throws (from jest.setup.js).
125
- (select as jest.Mock).mockImplementation((text: string) => {
126
- throw new Error(
127
- `Unexpected call to \`select("${text}")\`.\nYou should use \`mockSelect()\` to mock calls to \`select()\` with expectations. Search the codebase for \`mockSelect\` to learn more.`
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
- let listener: http.RequestListener;
10
+ let listener: http.RequestListener;
11
11
 
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
- });
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
- 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
- );
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
- // 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");
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
- // 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);
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
- // Now trigger the end callback.
41
- const endCallback = endSpy.mock.calls[0].pop();
42
- endCallback?.();
40
+ // Now trigger the end callback.
41
+ const endCallback = endSpy.mock.calls[0].pop();
42
+ endCallback?.();
43
43
 
44
- return resp;
45
- };
44
+ return resp;
45
+ };
46
46
  }
@@ -1,27 +1,74 @@
1
- const ORIGINAL_STDOUT_ISTTY = process.stdout.isTTY;
2
- const ORIGINAL_STDIN_ISTTY = process.stdin.isTTY;
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
- * Explicitly set `process.stdout.isTTY` to a given value
10
- */
11
- const setIsTTY = (isTTY: boolean) => {
12
- process.stdout.isTTY = isTTY;
13
- process.stdin.isTTY = isTTY;
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
- beforeEach(() => {
17
- process.stdout.isTTY = ORIGINAL_STDOUT_ISTTY;
18
- process.stdin.isTTY = ORIGINAL_STDIN_ISTTY;
19
- });
20
+ beforeEach(() => {
21
+ Object.defineProperty(process, "stdout", { value: ORIGINAL_STDOUT });
22
+ Object.defineProperty(process, "stdin", { value: ORIGINAL_STDIN });
23
+ });
20
24
 
21
- afterEach(() => {
22
- process.stdout.isTTY = ORIGINAL_STDOUT_ISTTY;
23
- process.stdin.isTTY = ORIGINAL_STDIN_ISTTY;
24
- });
25
+ afterEach(() => {
26
+ Object.defineProperty(process, "stdout", { value: ORIGINAL_STDOUT });
27
+ Object.defineProperty(process, "stdin", { value: ORIGINAL_STDIN });
28
+ });
25
29
 
26
- return { setIsTTY };
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
- expectedNamespaceId: string,
6
- expectedKeys: NamespaceKeyInfo[],
7
- keysPerRequest = 1000,
8
- blankCursorValue: "" | undefined | null = undefined
5
+ expectedNamespaceId: string,
6
+ expectedKeys: NamespaceKeyInfo[],
7
+ keysPerRequest = 1000,
8
+ blankCursorValue: "" | undefined | null = undefined
9
9
  ) {
10
- const requests = { count: 0 };
11
- // See https://api.cloudflare.com/#workers-kv-namespace-list-a-namespace-s-keys
10
+ const requests = { count: 0 };
11
+ // See https://api.cloudflare.com/#workers-kv-namespace-list-a-namespace-s-keys
12
12
 
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;
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
  }