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
@@ -3,10 +3,11 @@ import os from "node:os";
3
3
  import path from "node:path";
4
4
  import fetchMock from "jest-fetch-mock";
5
5
  import {
6
- readAuthConfigFile,
7
- requireAuth,
8
- USER_AUTH_CONFIG_FILE,
9
- writeAuthConfigFile,
6
+ loginOrRefreshIfRequired,
7
+ readAuthConfigFile,
8
+ requireAuth,
9
+ USER_AUTH_CONFIG_FILE,
10
+ writeAuthConfigFile,
10
11
  } from "../user";
11
12
  import { mockConsoleMethods } from "./helpers/mock-console";
12
13
  import { useMockIsTTY } from "./helpers/mock-istty";
@@ -17,115 +18,126 @@ import type { Config } from "../config";
17
18
  import type { UserAuthConfig } from "../user";
18
19
 
19
20
  describe("User", () => {
20
- runInTempDir({ homedir: "./home" });
21
- const std = mockConsoleMethods();
22
- const {
23
- mockOAuthServerCallback,
24
- mockGrantAccessToken,
25
- mockGrantAuthorization,
26
- mockRevokeAuthorization,
27
- mockExchangeRefreshTokenForAccessToken,
28
- } = mockOAuthFlow();
29
-
30
- const { setIsTTY } = useMockIsTTY();
31
-
32
- describe("login", () => {
33
- it("should login a user when `wrangler login` is run", async () => {
34
- mockOAuthServerCallback();
35
- const accessTokenRequest = mockGrantAccessToken({ respondWith: "ok" });
36
- mockGrantAuthorization({ respondWith: "success" });
37
-
38
- await runWrangler("login");
39
-
40
- expect(accessTokenRequest.actual.url).toEqual(
41
- accessTokenRequest.expected.url
42
- );
43
- expect(accessTokenRequest.actual.method).toEqual(
44
- accessTokenRequest.expected.method
45
- );
46
-
47
- expect(std.out).toMatchInlineSnapshot(`
21
+ runInTempDir({ homedir: "./home" });
22
+ const std = mockConsoleMethods();
23
+ const {
24
+ mockOAuthServerCallback,
25
+ mockGrantAccessToken,
26
+ mockGrantAuthorization,
27
+ mockRevokeAuthorization,
28
+ mockExchangeRefreshTokenForAccessToken,
29
+ } = mockOAuthFlow();
30
+
31
+ const { setIsTTY } = useMockIsTTY();
32
+
33
+ describe("login", () => {
34
+ it("should login a user when `wrangler login` is run", async () => {
35
+ mockOAuthServerCallback();
36
+ const accessTokenRequest = mockGrantAccessToken({ respondWith: "ok" });
37
+ mockGrantAuthorization({ respondWith: "success" });
38
+
39
+ await runWrangler("login");
40
+
41
+ expect(accessTokenRequest.actual.url).toEqual(
42
+ accessTokenRequest.expected.url
43
+ );
44
+ expect(accessTokenRequest.actual.method).toEqual(
45
+ accessTokenRequest.expected.method
46
+ );
47
+
48
+ expect(std.out).toMatchInlineSnapshot(`
48
49
  "Attempting to login via OAuth...
50
+ Opening a link in your default browser: https://dash.cloudflare.com/oauth2/auth?response_type=code&client_id=54d11594-84e4-41aa-b438-e81b8fa78ee7&redirect_uri=http%3A%2F%2Flocalhost%3A8976%2Foauth%2Fcallback&scope=account%3Aread%20user%3Aread%20workers%3Awrite%20workers_kv%3Awrite%20workers_routes%3Awrite%20workers_scripts%3Awrite%20workers_tail%3Aread%20pages%3Awrite%20zone%3Aread%20offline_access&state=MOCK_STATE_PARAM&code_challenge=MOCK_CODE_CHALLENGE&code_challenge_method=S256
49
51
  Successfully logged in."
50
52
  `);
51
53
 
52
- expect(readAuthConfigFile()).toEqual<UserAuthConfig>({
53
- api_token: undefined,
54
- oauth_token: "test-access-token",
55
- refresh_token: "test-refresh-token",
56
- expiration_time: expect.any(String),
57
- });
58
- });
59
- });
60
-
61
- describe("logout", () => {
62
- it("should exit with a message stating the user is not logged in", async () => {
63
- await runWrangler("logout");
64
- expect(std.out).toMatchInlineSnapshot(`"Not logged in, exiting..."`);
65
- });
66
-
67
- it("should logout user that has been properly logged in", async () => {
68
- writeAuthConfigFile({
69
- oauth_token: "some-oauth-tok",
70
- refresh_token: "some-refresh-tok",
71
- });
72
- const outcome = mockRevokeAuthorization();
73
-
74
- await runWrangler("logout");
75
-
76
- expect(outcome.actual.url).toEqual(
77
- "https://dash.cloudflare.com/oauth2/revoke"
78
- );
79
- expect(outcome.actual.method).toEqual("POST");
80
-
81
- expect(std.out).toMatchInlineSnapshot(`"Successfully logged out."`);
82
-
83
- // Make sure that we made the request to logout.
84
- expect(fetchMock).toHaveBeenCalledTimes(1);
85
-
86
- // Make sure that logout removed the config file containing the auth tokens.
87
- const config = path.join(os.homedir(), USER_AUTH_CONFIG_FILE);
88
- expect(fs.existsSync(config)).toBeFalsy();
89
- });
90
- });
91
-
92
- // TODO: Improve OAuth mocking to handle `/token` endpoints from different calls
93
- it("should handle errors for failed token refresh", async () => {
94
- setIsTTY(false);
95
- mockOAuthServerCallback();
96
- writeAuthConfigFile({
97
- oauth_token: "hunter2",
98
- refresh_token: "Order 66",
99
- });
100
- mockGrantAuthorization({ respondWith: "success" });
101
-
102
- mockExchangeRefreshTokenForAccessToken({
103
- respondWith: "badResponse",
104
- });
105
-
106
- // Handles the requireAuth error throw from failed login that is unhandled due to directly calling it here
107
- await expect(
108
- requireAuth({} as Config)
109
- ).rejects.toThrowErrorMatchingInlineSnapshot(
110
- `"Did not login, quitting..."`
111
- );
112
- });
113
-
114
- it("should confirm no error message when refresh is successful", async () => {
115
- setIsTTY(false);
116
- mockOAuthServerCallback();
117
- writeAuthConfigFile({
118
- oauth_token: "hunter2",
119
- refresh_token: "Order 66",
120
- });
121
- mockGrantAuthorization({ respondWith: "success" });
122
-
123
- mockExchangeRefreshTokenForAccessToken({
124
- respondWith: "refreshSuccess",
125
- });
126
-
127
- // Handles the requireAuth error throw from failed login that is unhandled due to directly calling it here
128
- await expect(requireAuth({} as Config)).rejects.toThrowError();
129
- expect(std.err).toContain("");
130
- });
54
+ expect(readAuthConfigFile()).toEqual<UserAuthConfig>({
55
+ api_token: undefined,
56
+ oauth_token: "test-access-token",
57
+ refresh_token: "test-refresh-token",
58
+ expiration_time: expect.any(String),
59
+ });
60
+ });
61
+ });
62
+
63
+ describe("logout", () => {
64
+ it("should exit with a message stating the user is not logged in", async () => {
65
+ await runWrangler("logout");
66
+ expect(std.out).toMatchInlineSnapshot(`"Not logged in, exiting..."`);
67
+ });
68
+
69
+ it("should logout user that has been properly logged in", async () => {
70
+ writeAuthConfigFile({
71
+ oauth_token: "some-oauth-tok",
72
+ refresh_token: "some-refresh-tok",
73
+ });
74
+ const outcome = mockRevokeAuthorization();
75
+
76
+ await runWrangler("logout");
77
+
78
+ expect(outcome.actual.url).toEqual(
79
+ "https://dash.cloudflare.com/oauth2/revoke"
80
+ );
81
+ expect(outcome.actual.method).toEqual("POST");
82
+
83
+ expect(std.out).toMatchInlineSnapshot(`"Successfully logged out."`);
84
+
85
+ // Make sure that we made the request to logout.
86
+ expect(fetchMock).toHaveBeenCalledTimes(1);
87
+
88
+ // Make sure that logout removed the config file containing the auth tokens.
89
+ const config = path.join(os.homedir(), USER_AUTH_CONFIG_FILE);
90
+ expect(fs.existsSync(config)).toBeFalsy();
91
+ });
92
+ });
93
+
94
+ // TODO: Improve OAuth mocking to handle `/token` endpoints from different calls
95
+ it("should handle errors for failed token refresh", async () => {
96
+ setIsTTY(false);
97
+ mockOAuthServerCallback();
98
+ writeAuthConfigFile({
99
+ oauth_token: "hunter2",
100
+ refresh_token: "Order 66",
101
+ });
102
+ mockGrantAuthorization({ respondWith: "success" });
103
+
104
+ mockExchangeRefreshTokenForAccessToken({
105
+ respondWith: "badResponse",
106
+ });
107
+
108
+ // Handles the requireAuth error throw from failed login that is unhandled due to directly calling it here
109
+ await expect(
110
+ requireAuth({} as Config)
111
+ ).rejects.toThrowErrorMatchingInlineSnapshot(
112
+ `"Did not login, quitting..."`
113
+ );
114
+ });
115
+
116
+ it("should confirm no error message when refresh is successful", async () => {
117
+ setIsTTY(false);
118
+ mockOAuthServerCallback();
119
+ writeAuthConfigFile({
120
+ oauth_token: "hunter2",
121
+ refresh_token: "Order 66",
122
+ });
123
+ mockGrantAuthorization({ respondWith: "success" });
124
+
125
+ mockExchangeRefreshTokenForAccessToken({
126
+ respondWith: "refreshSuccess",
127
+ });
128
+
129
+ // Handles the requireAuth error throw from failed login that is unhandled due to directly calling it here
130
+ await expect(requireAuth({} as Config)).rejects.toThrowError();
131
+ expect(std.err).toContain("");
132
+ });
133
+
134
+ it("should revert to non-interactive mode if isTTY throws an error", async () => {
135
+ setIsTTY({
136
+ stdin() {
137
+ throw new Error("stdin is not tty");
138
+ },
139
+ stdout: true,
140
+ });
141
+ await expect(loginOrRefreshIfRequired()).resolves.toEqual(false);
142
+ });
131
143
  });
@@ -2,98 +2,179 @@ import { render } from "ink-testing-library";
2
2
  import React from "react";
3
3
  import { writeAuthConfigFile } from "../user";
4
4
  import { getUserInfo, WhoAmI } from "../whoami";
5
- import { setMockResponse } from "./helpers/mock-cfetch";
5
+ import {
6
+ createFetchResult,
7
+ setMockRawResponse,
8
+ setMockResponse,
9
+ unsetAllMocks,
10
+ } from "./helpers/mock-cfetch";
6
11
  import { mockConsoleMethods } from "./helpers/mock-console";
7
12
  import { useMockIsTTY } from "./helpers/mock-istty";
8
13
  import { runInTempDir } from "./helpers/run-in-tmp";
9
14
  import type { UserInfo } from "../whoami";
10
15
 
11
16
  describe("getUserInfo()", () => {
12
- const ENV_COPY = process.env;
13
-
14
- runInTempDir({ homedir: "./home" });
15
- const std = mockConsoleMethods();
16
- const { setIsTTY } = useMockIsTTY();
17
-
18
- beforeEach(() => {
19
- setIsTTY(true);
20
- });
21
-
22
- afterEach(() => {
23
- process.env = ENV_COPY;
24
- });
25
-
26
- it("should return undefined if there is no config file", async () => {
27
- const userInfo = await getUserInfo();
28
- expect(userInfo).toBeUndefined();
29
- });
30
-
31
- it("should return undefined if there is an empty config file", async () => {
32
- writeAuthConfigFile({});
33
- const userInfo = await getUserInfo();
34
- expect(userInfo).toBeUndefined();
35
- });
36
-
37
- it("should say it's using an API token when one is set", async () => {
38
- process.env = {
39
- CLOUDFLARE_API_TOKEN: "123456789",
40
- };
41
- setMockResponse("/user", () => {
42
- return { email: "user@example.com" };
43
- });
44
- setMockResponse("/accounts", () => {
45
- return [
46
- { name: "Account One", id: "account-1" },
47
- { name: "Account Two", id: "account-2" },
48
- { name: "Account Three", id: "account-3" },
49
- ];
50
- });
51
-
52
- const userInfo = await getUserInfo();
53
- expect(userInfo).toEqual({
54
- authType: "API",
55
- apiToken: "123456789",
56
- email: "user@example.com",
57
- accounts: [
58
- { name: "Account One", id: "account-1" },
59
- { name: "Account Two", id: "account-2" },
60
- { name: "Account Three", id: "account-3" },
61
- ],
62
- });
63
- });
64
-
65
- it("should return the user's email and accounts if authenticated via config token", async () => {
66
- writeAuthConfigFile({ oauth_token: "some-oauth-token" });
67
-
68
- setMockResponse("/user", () => {
69
- return { email: "user@example.com" };
70
- });
71
- setMockResponse("/accounts", () => {
72
- return [
73
- { name: "Account One", id: "account-1" },
74
- { name: "Account Two", id: "account-2" },
75
- { name: "Account Three", id: "account-3" },
76
- ];
77
- });
78
-
79
- const userInfo = await getUserInfo();
80
-
81
- expect(userInfo).toEqual({
82
- authType: "OAuth",
83
- apiToken: "some-oauth-token",
84
- email: "user@example.com",
85
- accounts: [
86
- { name: "Account One", id: "account-1" },
87
- { name: "Account Two", id: "account-2" },
88
- { name: "Account Three", id: "account-3" },
89
- ],
90
- });
91
- });
92
-
93
- it("should display a warning message if the config file contains a legacy api_token field", async () => {
94
- writeAuthConfigFile({ api_token: "API_TOKEN" });
95
- await getUserInfo();
96
- expect(std.warn).toMatchInlineSnapshot(`
17
+ const ENV_COPY = process.env;
18
+
19
+ runInTempDir({ homedir: "./home" });
20
+ const std = mockConsoleMethods();
21
+ const { setIsTTY } = useMockIsTTY();
22
+
23
+ beforeEach(() => {
24
+ setIsTTY(true);
25
+ });
26
+
27
+ afterEach(() => {
28
+ process.env = ENV_COPY;
29
+ unsetAllMocks();
30
+ });
31
+
32
+ it("should return undefined if there is no config file", async () => {
33
+ const userInfo = await getUserInfo();
34
+ expect(userInfo).toBeUndefined();
35
+ });
36
+
37
+ it("should return undefined if there is an empty config file", async () => {
38
+ writeAuthConfigFile({});
39
+ const userInfo = await getUserInfo();
40
+ expect(userInfo).toBeUndefined();
41
+ });
42
+
43
+ it("should return undefined for email if the user settings API request fails with 9109", async () => {
44
+ process.env = {
45
+ CLOUDFLARE_API_TOKEN: "123456789",
46
+ };
47
+ setMockRawResponse("/user", () =>
48
+ createFetchResult(undefined, false, [
49
+ { code: 9109, message: "Uauthorized to access requested resource" },
50
+ ])
51
+ );
52
+ setMockResponse("/accounts", () => []);
53
+ const userInfo = await getUserInfo();
54
+ expect(userInfo?.email).toBeUndefined();
55
+ });
56
+
57
+ it("should say it's using an API token when one is set", async () => {
58
+ process.env = {
59
+ CLOUDFLARE_API_TOKEN: "123456789",
60
+ };
61
+ setMockResponse("/user", () => {
62
+ return { email: "user@example.com" };
63
+ });
64
+ setMockResponse("/accounts", () => {
65
+ return [
66
+ { name: "Account One", id: "account-1" },
67
+ { name: "Account Two", id: "account-2" },
68
+ { name: "Account Three", id: "account-3" },
69
+ ];
70
+ });
71
+
72
+ const userInfo = await getUserInfo();
73
+ expect(userInfo).toEqual({
74
+ authType: "API Token",
75
+ apiToken: "123456789",
76
+ email: "user@example.com",
77
+ accounts: [
78
+ { name: "Account One", id: "account-1" },
79
+ { name: "Account Two", id: "account-2" },
80
+ { name: "Account Three", id: "account-3" },
81
+ ],
82
+ });
83
+ });
84
+
85
+ it("should say it's using a Global API Key when one is set", async () => {
86
+ process.env = {
87
+ CLOUDFLARE_API_KEY: "123456789",
88
+ CLOUDFLARE_EMAIL: "user@example.com",
89
+ };
90
+ setMockResponse("/accounts", () => {
91
+ return [
92
+ { name: "Account One", id: "account-1" },
93
+ { name: "Account Two", id: "account-2" },
94
+ { name: "Account Three", id: "account-3" },
95
+ ];
96
+ });
97
+
98
+ const userInfo = await getUserInfo();
99
+ expect(userInfo).toEqual({
100
+ authType: "Global API Key",
101
+ apiToken: "123456789",
102
+ email: "user@example.com",
103
+ accounts: [
104
+ { name: "Account One", id: "account-1" },
105
+ { name: "Account Two", id: "account-2" },
106
+ { name: "Account Three", id: "account-3" },
107
+ ],
108
+ });
109
+ });
110
+
111
+ it("should use a Global API Key in preference to an API token", async () => {
112
+ process.env = {
113
+ CLOUDFLARE_API_KEY: "123456789",
114
+ CLOUDFLARE_EMAIL: "user@example.com",
115
+ CLOUDFLARE_API_TOKEN: "123456789",
116
+ };
117
+ setMockResponse("/accounts", () => {
118
+ return [
119
+ { name: "Account One", id: "account-1" },
120
+ { name: "Account Two", id: "account-2" },
121
+ { name: "Account Three", id: "account-3" },
122
+ ];
123
+ });
124
+
125
+ const userInfo = await getUserInfo();
126
+ expect(userInfo).toEqual({
127
+ authType: "Global API Key",
128
+ apiToken: "123456789",
129
+ email: "user@example.com",
130
+ accounts: [
131
+ { name: "Account One", id: "account-1" },
132
+ { name: "Account Two", id: "account-2" },
133
+ { name: "Account Three", id: "account-3" },
134
+ ],
135
+ });
136
+ });
137
+
138
+ it("should return undefined only a Global API Key, but not Email, is set", async () => {
139
+ process.env = {
140
+ CLOUDFLARE_API_KEY: "123456789",
141
+ };
142
+ const userInfo = await getUserInfo();
143
+ expect(userInfo).toEqual(undefined);
144
+ });
145
+
146
+ it("should return the user's email and accounts if authenticated via config token", async () => {
147
+ writeAuthConfigFile({ oauth_token: "some-oauth-token" });
148
+
149
+ setMockResponse("/user", () => {
150
+ return { email: "user@example.com" };
151
+ });
152
+ setMockResponse("/accounts", () => {
153
+ return [
154
+ { name: "Account One", id: "account-1" },
155
+ { name: "Account Two", id: "account-2" },
156
+ { name: "Account Three", id: "account-3" },
157
+ ];
158
+ });
159
+
160
+ const userInfo = await getUserInfo();
161
+
162
+ expect(userInfo).toEqual({
163
+ authType: "OAuth Token",
164
+ apiToken: "some-oauth-token",
165
+ email: "user@example.com",
166
+ accounts: [
167
+ { name: "Account One", id: "account-1" },
168
+ { name: "Account Two", id: "account-2" },
169
+ { name: "Account Three", id: "account-3" },
170
+ ],
171
+ });
172
+ });
173
+
174
+ it("should display a warning message if the config file contains a legacy api_token field", async () => {
175
+ writeAuthConfigFile({ api_token: "API_TOKEN" });
176
+ await getUserInfo();
177
+ expect(std.warn).toMatchInlineSnapshot(`
97
178
  "â–² [WARNING] It looks like you have used Wrangler 1's \`config\` command to login with an API token.
98
179
 
99
180
  This is no longer supported in the current version of Wrangler.
@@ -102,38 +183,38 @@ describe("getUserInfo()", () => {
102
183
 
103
184
  "
104
185
  `);
105
- });
186
+ });
106
187
  });
107
188
 
108
189
  describe("WhoAmI component", () => {
109
- it("should return undefined if there is no user", async () => {
110
- const { lastFrame } = render(<WhoAmI user={undefined}></WhoAmI>);
111
-
112
- expect(lastFrame()).toMatchInlineSnapshot(
113
- `"You are not authenticated. Please run \`wrangler login\`."`
114
- );
115
- });
116
-
117
- it("should display the user's email and accounts", async () => {
118
- const user: UserInfo = {
119
- authType: "OAuth",
120
- apiToken: "some-oauth-token",
121
- email: "user@example.com",
122
- accounts: [
123
- { name: "Account One", id: "account-1" },
124
- { name: "Account Two", id: "account-2" },
125
- { name: "Account Three", id: "account-3" },
126
- ],
127
- };
128
-
129
- const { lastFrame } = render(<WhoAmI user={user}></WhoAmI>);
130
-
131
- expect(lastFrame()).toContain(
132
- "You are logged in with an OAuth Token, associated with the email 'user@example.com'!"
133
- );
134
- expect(lastFrame()).toMatch(/Account Name .+ Account ID/);
135
- expect(lastFrame()).toMatch(/Account One .+ account-1/);
136
- expect(lastFrame()).toMatch(/Account Two .+ account-2/);
137
- expect(lastFrame()).toMatch(/Account Three .+ account-3/);
138
- });
190
+ it("should return undefined if there is no user", async () => {
191
+ const { lastFrame } = render(<WhoAmI user={undefined}></WhoAmI>);
192
+
193
+ expect(lastFrame()).toMatchInlineSnapshot(
194
+ `"You are not authenticated. Please run \`wrangler login\`."`
195
+ );
196
+ });
197
+
198
+ it("should display the user's email and accounts", async () => {
199
+ const user: UserInfo = {
200
+ authType: "OAuth Token",
201
+ apiToken: "some-oauth-token",
202
+ email: "user@example.com",
203
+ accounts: [
204
+ { name: "Account One", id: "account-1" },
205
+ { name: "Account Two", id: "account-2" },
206
+ { name: "Account Three", id: "account-3" },
207
+ ],
208
+ };
209
+
210
+ const { lastFrame } = render(<WhoAmI user={user}></WhoAmI>);
211
+
212
+ expect(lastFrame()).toContain(
213
+ "You are logged in with an OAuth Token, associated with the email 'user@example.com'!"
214
+ );
215
+ expect(lastFrame()).toMatch(/Account Name .+ Account ID/);
216
+ expect(lastFrame()).toMatch(/Account One .+ account-1/);
217
+ expect(lastFrame()).toMatch(/Account Two .+ account-2/);
218
+ expect(lastFrame()).toMatch(/Account Three .+ account-3/);
219
+ });
139
220
  });