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
package/src/sites.tsx CHANGED
@@ -4,11 +4,11 @@ import * as path from "node:path";
4
4
  import ignore from "ignore";
5
5
  import xxhash from "xxhash-wasm";
6
6
  import {
7
- createKVNamespace,
8
- listKVNamespaceKeys,
9
- listKVNamespaces,
10
- putKVBulkKeyValue,
11
- deleteKVBulkKeyValue,
7
+ createKVNamespace,
8
+ listKVNamespaceKeys,
9
+ listKVNamespaces,
10
+ putKVBulkKeyValue,
11
+ deleteKVBulkKeyValue,
12
12
  } from "./kv";
13
13
  import { logger } from "./logger";
14
14
  import type { Config } from "./config";
@@ -18,27 +18,27 @@ import type { XXHashAPI } from "xxhash-wasm";
18
18
  /** Paths to always ignore. */
19
19
  const ALWAYS_IGNORE = new Set(["node_modules"]);
20
20
  const HIDDEN_FILES_TO_INCLUDE = new Set([
21
- ".well-known", // See https://datatracker.ietf.org/doc/html/rfc8615
21
+ ".well-known", // See https://datatracker.ietf.org/doc/html/rfc8615
22
22
  ]);
23
23
 
24
24
  async function* getFilesInFolder(dirPath: string): AsyncIterable<string> {
25
- const files = await readdir(dirPath, { withFileTypes: true });
26
- for (const file of files) {
27
- // Skip files that we never want to process.
28
- if (ALWAYS_IGNORE.has(file.name)) {
29
- continue;
30
- }
31
- // Skip hidden files (starting with .) except for some special ones
32
- if (file.name.startsWith(".") && !HIDDEN_FILES_TO_INCLUDE.has(file.name)) {
33
- continue;
34
- }
35
- // TODO: follow symlinks??
36
- if (file.isDirectory()) {
37
- yield* await getFilesInFolder(path.join(dirPath, file.name));
38
- } else {
39
- yield path.join(dirPath, file.name);
40
- }
41
- }
25
+ const files = await readdir(dirPath, { withFileTypes: true });
26
+ for (const file of files) {
27
+ // Skip files that we never want to process.
28
+ if (ALWAYS_IGNORE.has(file.name)) {
29
+ continue;
30
+ }
31
+ // Skip hidden files (starting with .) except for some special ones
32
+ if (file.name.startsWith(".") && !HIDDEN_FILES_TO_INCLUDE.has(file.name)) {
33
+ continue;
34
+ }
35
+ // TODO: follow symlinks??
36
+ if (file.isDirectory()) {
37
+ yield* await getFilesInFolder(path.join(dirPath, file.name));
38
+ } else {
39
+ yield path.join(dirPath, file.name);
40
+ }
41
+ }
42
42
  }
43
43
 
44
44
  /**
@@ -49,7 +49,7 @@ async function* getFilesInFolder(dirPath: string): AsyncIterable<string> {
49
49
  * it's impossible to serve two different files with the same name
50
50
  */
51
51
  function hashFileContent(hasher: XXHashAPI, content: string): string {
52
- return hasher.h64ToString(content).substring(0, 10);
52
+ return hasher.h64ToString(content).substring(0, 10);
53
53
  }
54
54
 
55
55
  /**
@@ -59,37 +59,37 @@ function hashFileContent(hasher: XXHashAPI, content: string): string {
59
59
  * The algorithm used here matches that of Wrangler 1.
60
60
  */
61
61
  function hashAsset(
62
- hasher: XXHashAPI,
63
- filePath: string,
64
- content: string
62
+ hasher: XXHashAPI,
63
+ filePath: string,
64
+ content: string
65
65
  ): string {
66
- const extName = path.extname(filePath) || "";
67
- const baseName = path.basename(filePath, extName);
68
- const directory = path.dirname(filePath);
69
- const hash = hashFileContent(hasher, content);
70
- return urlSafe(path.join(directory, `${baseName}.${hash}${extName}`));
66
+ const extName = path.extname(filePath) || "";
67
+ const baseName = path.basename(filePath, extName);
68
+ const directory = path.dirname(filePath);
69
+ const hash = hashFileContent(hasher, content);
70
+ return urlSafe(path.join(directory, `${baseName}.${hash}${extName}`));
71
71
  }
72
72
 
73
73
  async function createKVNamespaceIfNotAlreadyExisting(
74
- title: string,
75
- accountId: string
74
+ title: string,
75
+ accountId: string
76
76
  ) {
77
- // check if it already exists
78
- // TODO: this is super inefficient, should be made better
79
- const namespaces = await listKVNamespaces(accountId);
80
- const found = namespaces.find((x) => x.title === title);
81
- if (found) {
82
- return { created: false, id: found.id };
83
- }
84
-
85
- // else we make the namespace
86
- const id = await createKVNamespace(accountId, title);
87
- logger.log(`🌀 Created namespace for Workers Site "${title}"`);
88
-
89
- return {
90
- created: true,
91
- id,
92
- };
77
+ // check if it already exists
78
+ // TODO: this is super inefficient, should be made better
79
+ const namespaces = await listKVNamespaces(accountId);
80
+ const found = namespaces.find((x) => x.title === title);
81
+ if (found) {
82
+ return { created: false, id: found.id };
83
+ }
84
+
85
+ // else we make the namespace
86
+ const id = await createKVNamespace(accountId, title);
87
+ logger.log(`🌀 Created namespace for Workers Site "${title}"`);
88
+
89
+ return {
90
+ created: true,
91
+ id,
92
+ };
93
93
  }
94
94
 
95
95
  /**
@@ -104,139 +104,139 @@ async function createKVNamespaceIfNotAlreadyExisting(
104
104
  * asset in the KV namespace.
105
105
  */
106
106
  export async function syncAssets(
107
- accountId: string | undefined,
108
- scriptName: string,
109
- siteAssets: AssetPaths | undefined,
110
- preview: boolean,
111
- dryRun: boolean | undefined
107
+ accountId: string | undefined,
108
+ scriptName: string,
109
+ siteAssets: AssetPaths | undefined,
110
+ preview: boolean,
111
+ dryRun: boolean | undefined
112
112
  ): Promise<{
113
- manifest: { [filePath: string]: string } | undefined;
114
- namespace: string | undefined;
113
+ manifest: { [filePath: string]: string } | undefined;
114
+ namespace: string | undefined;
115
115
  }> {
116
- if (siteAssets === undefined) {
117
- return { manifest: undefined, namespace: undefined };
118
- }
119
-
120
- if (dryRun) {
121
- logger.log("(Note: doing a dry run, not uploading or deleting anything.)");
122
- return { manifest: undefined, namespace: undefined };
123
- }
124
- assert(accountId, "Missing accountId");
125
-
126
- const title = `__${scriptName}-workers_sites_assets${
127
- preview ? "_preview" : ""
128
- }`;
129
-
130
- const { id: namespace } = await createKVNamespaceIfNotAlreadyExisting(
131
- title,
132
- accountId
133
- );
134
-
135
- // let's get all the keys in this namespace
136
- const namespaceKeysResponse = await listKVNamespaceKeys(accountId, namespace);
137
- const namespaceKeys = new Set(namespaceKeysResponse.map((x) => x.name));
138
-
139
- const manifest: Record<string, string> = {};
140
-
141
- // A batch of uploads where each bucket has to be less than 100mb
142
- const uploadBuckets: KeyValue[][] = [];
143
- // The "live" bucket that we'll keep filling until it's just below 100mb
144
- let uploadBucket: KeyValue[] = [];
145
- // A size counter for the live bucket
146
- let uploadBucketSize = 0;
147
-
148
- const include = createPatternMatcher(siteAssets.includePatterns, false);
149
- const exclude = createPatternMatcher(siteAssets.excludePatterns, true);
150
- const hasher = await xxhash();
151
-
152
- const assetDirectory = path.join(
153
- siteAssets.baseDirectory,
154
- siteAssets.assetDirectory
155
- );
156
- for await (const absAssetFile of getFilesInFolder(assetDirectory)) {
157
- const assetFile = path.relative(assetDirectory, absAssetFile);
158
- if (!include(assetFile)) {
159
- continue;
160
- }
161
- if (exclude(assetFile)) {
162
- continue;
163
- }
164
-
165
- logger.log(`Reading ${assetFile}...`);
166
- const content = await readFile(absAssetFile, "base64");
167
- await validateAssetSize(absAssetFile, assetFile);
168
- // while KV accepts files that are 25 MiB **before** b64 encoding
169
- // the overall bucket size must be below 100 MiB **after** b64 encoding
170
- const assetSize = Buffer.from(content).length;
171
- const assetKey = hashAsset(hasher, assetFile, content);
172
- validateAssetKey(assetKey);
173
-
174
- // now put each of the files into kv
175
- if (!namespaceKeys.has(assetKey)) {
176
- logger.log(`Uploading as ${assetKey}...`);
177
-
178
- // Check if adding this asset to the bucket would
179
- // push it over the 100 MiB limit KV bulk API limit
180
- if (uploadBucketSize + assetSize > 100 * 1024 * 1024) {
181
- // If so, move the current bucket into the batch,
182
- // and reset the counter/bucket
183
- uploadBuckets.push(uploadBucket);
184
- uploadBucketSize = 0;
185
- uploadBucket = [];
186
- }
187
-
188
- // Update the bucket and the size counter
189
- uploadBucketSize += assetSize;
190
- uploadBucket.push({
191
- key: assetKey,
192
- value: content,
193
- base64: true,
194
- });
195
- } else {
196
- logger.log(`Skipping - already uploaded.`);
197
- }
198
-
199
- // Remove the key from the set so we know what we've already uploaded
200
- namespaceKeys.delete(assetKey);
201
-
202
- // Prevent different manifest keys on windows
203
- const manifestKey = urlSafe(path.relative(assetDirectory, absAssetFile));
204
- manifest[manifestKey] = assetKey;
205
- }
206
-
207
- // Add the last (potentially only) bucket to the batch
208
- uploadBuckets.push(uploadBucket);
209
-
210
- // keys now contains all the files we're deleting
211
- for (const key of namespaceKeys) {
212
- logger.log(`Deleting ${key} from the asset store...`);
213
- }
214
-
215
- // upload each bucket in parallel
216
- const bucketsToPut = [];
217
- for (const bucket of uploadBuckets) {
218
- bucketsToPut.push(putKVBulkKeyValue(accountId, namespace, bucket));
219
- }
220
- await Promise.all(bucketsToPut);
221
-
222
- // then delete all the assets that aren't used anymore
223
- await deleteKVBulkKeyValue(accountId, namespace, Array.from(namespaceKeys));
224
-
225
- logger.log("↗️ Done syncing assets");
226
-
227
- return { manifest, namespace };
116
+ if (siteAssets === undefined) {
117
+ return { manifest: undefined, namespace: undefined };
118
+ }
119
+
120
+ if (dryRun) {
121
+ logger.log("(Note: doing a dry run, not uploading or deleting anything.)");
122
+ return { manifest: undefined, namespace: undefined };
123
+ }
124
+ assert(accountId, "Missing accountId");
125
+
126
+ const title = `__${scriptName}-workers_sites_assets${
127
+ preview ? "_preview" : ""
128
+ }`;
129
+
130
+ const { id: namespace } = await createKVNamespaceIfNotAlreadyExisting(
131
+ title,
132
+ accountId
133
+ );
134
+
135
+ // let's get all the keys in this namespace
136
+ const namespaceKeysResponse = await listKVNamespaceKeys(accountId, namespace);
137
+ const namespaceKeys = new Set(namespaceKeysResponse.map((x) => x.name));
138
+
139
+ const manifest: Record<string, string> = {};
140
+
141
+ // A batch of uploads where each bucket has to be less than 98mb
142
+ const uploadBuckets: KeyValue[][] = [];
143
+ // The "live" bucket that we'll keep filling until it's just below 98mb
144
+ let uploadBucket: KeyValue[] = [];
145
+ // A size counter for the live bucket
146
+ let uploadBucketSize = 0;
147
+
148
+ const include = createPatternMatcher(siteAssets.includePatterns, false);
149
+ const exclude = createPatternMatcher(siteAssets.excludePatterns, true);
150
+ const hasher = await xxhash();
151
+
152
+ const assetDirectory = path.join(
153
+ siteAssets.baseDirectory,
154
+ siteAssets.assetDirectory
155
+ );
156
+ for await (const absAssetFile of getFilesInFolder(assetDirectory)) {
157
+ const assetFile = path.relative(assetDirectory, absAssetFile);
158
+ if (!include(assetFile)) {
159
+ continue;
160
+ }
161
+ if (exclude(assetFile)) {
162
+ continue;
163
+ }
164
+
165
+ logger.log(`Reading ${assetFile}...`);
166
+ const content = await readFile(absAssetFile, "base64");
167
+ await validateAssetSize(absAssetFile, assetFile);
168
+ // while KV accepts files that are 25 MiB **before** b64 encoding
169
+ // the overall bucket size must be below 100 MB **after** b64 encoding
170
+ const assetSize = Buffer.from(content).length;
171
+ const assetKey = hashAsset(hasher, assetFile, content);
172
+ validateAssetKey(assetKey);
173
+
174
+ // now put each of the files into kv
175
+ if (!namespaceKeys.has(assetKey)) {
176
+ logger.log(`Uploading as ${assetKey}...`);
177
+
178
+ // Check if adding this asset to the bucket would
179
+ // push it over the 98 MiB limit KV bulk API limit
180
+ if (uploadBucketSize + assetSize > 98 * 1000 * 1000) {
181
+ // If so, move the current bucket into the batch,
182
+ // and reset the counter/bucket
183
+ uploadBuckets.push(uploadBucket);
184
+ uploadBucketSize = 0;
185
+ uploadBucket = [];
186
+ }
187
+
188
+ // Update the bucket and the size counter
189
+ uploadBucketSize += assetSize;
190
+ uploadBucket.push({
191
+ key: assetKey,
192
+ value: content,
193
+ base64: true,
194
+ });
195
+ } else {
196
+ logger.log(`Skipping - already uploaded.`);
197
+ }
198
+
199
+ // Remove the key from the set so we know what we've already uploaded
200
+ namespaceKeys.delete(assetKey);
201
+
202
+ // Prevent different manifest keys on windows
203
+ const manifestKey = urlSafe(path.relative(assetDirectory, absAssetFile));
204
+ manifest[manifestKey] = assetKey;
205
+ }
206
+
207
+ // Add the last (potentially only) bucket to the batch
208
+ uploadBuckets.push(uploadBucket);
209
+
210
+ // keys now contains all the files we're deleting
211
+ for (const key of namespaceKeys) {
212
+ logger.log(`Deleting ${key} from the asset store...`);
213
+ }
214
+
215
+ // upload each bucket in parallel
216
+ const bucketsToPut = [];
217
+ for (const bucket of uploadBuckets) {
218
+ bucketsToPut.push(putKVBulkKeyValue(accountId, namespace, bucket));
219
+ }
220
+ await Promise.all(bucketsToPut);
221
+
222
+ // then delete all the assets that aren't used anymore
223
+ await deleteKVBulkKeyValue(accountId, namespace, Array.from(namespaceKeys));
224
+
225
+ logger.log("↗️ Done syncing assets");
226
+
227
+ return { manifest, namespace };
228
228
  }
229
229
 
230
230
  function createPatternMatcher(
231
- patterns: string[],
232
- exclude: boolean
231
+ patterns: string[],
232
+ exclude: boolean
233
233
  ): (filePath: string) => boolean {
234
- if (patterns.length === 0) {
235
- return (_filePath) => !exclude;
236
- } else {
237
- const ignorer = ignore().add(patterns);
238
- return (filePath) => ignorer.test(filePath).ignored;
239
- }
234
+ if (patterns.length === 0) {
235
+ return (_filePath) => !exclude;
236
+ } else {
237
+ const ignorer = ignore().add(patterns);
238
+ return (filePath) => ignorer.test(filePath).ignored;
239
+ }
240
240
  }
241
241
 
242
242
  /**
@@ -246,23 +246,23 @@ function createPatternMatcher(
246
246
  * @param relativeFilePath
247
247
  */
248
248
  async function validateAssetSize(
249
- absFilePath: string,
250
- relativeFilePath: string
249
+ absFilePath: string,
250
+ relativeFilePath: string
251
251
  ): Promise<void> {
252
- const { size } = await stat(absFilePath);
253
- if (size > 25 * 1024 * 1024) {
254
- throw new Error(
255
- `File ${relativeFilePath} is too big, it should be under 25 MiB. See https://developers.cloudflare.com/workers/platform/limits#kv-limits`
256
- );
257
- }
252
+ const { size } = await stat(absFilePath);
253
+ if (size > 25 * 1024 * 1024) {
254
+ throw new Error(
255
+ `File ${relativeFilePath} is too big, it should be under 25 MiB. See https://developers.cloudflare.com/workers/platform/limits#kv-limits`
256
+ );
257
+ }
258
258
  }
259
259
 
260
260
  function validateAssetKey(assetKey: string) {
261
- if (assetKey.length > 512) {
262
- throw new Error(
263
- `The asset path key "${assetKey}" exceeds the maximum key size limit of 512. See https://developers.cloudflare.com/workers/platform/limits#kv-limits",`
264
- );
265
- }
261
+ if (assetKey.length > 512) {
262
+ throw new Error(
263
+ `The asset path key "${assetKey}" exceeds the maximum key size limit of 512. See https://developers.cloudflare.com/workers/platform/limits#kv-limits",`
264
+ );
265
+ }
266
266
  }
267
267
 
268
268
  /**
@@ -271,31 +271,31 @@ function validateAssetKey(assetKey: string) {
271
271
  * Primarily this involves converting Windows backslashes to forward slashes.
272
272
  */
273
273
  function urlSafe(filePath: string): string {
274
- return filePath.replace(/\\/g, "/");
274
+ return filePath.replace(/\\/g, "/");
275
275
  }
276
276
 
277
277
  /**
278
278
  * Information about the assets that should be uploaded
279
279
  */
280
280
  export interface AssetPaths {
281
- /**
282
- * Absolute path to the root of the project.
283
- *
284
- * This is the directory containing wrangler.toml or cwd if no config.
285
- */
286
- baseDirectory: string;
287
- /**
288
- * The path to the assets directory, relative to the `baseDirectory`.
289
- */
290
- assetDirectory: string;
291
- /**
292
- * An array of patterns that match files that should be uploaded.
293
- */
294
- includePatterns: string[];
295
- /**
296
- * An array of patterns that match files that should not be uploaded.
297
- */
298
- excludePatterns: string[];
281
+ /**
282
+ * Absolute path to the root of the project.
283
+ *
284
+ * This is the directory containing wrangler.toml or cwd if no config.
285
+ */
286
+ baseDirectory: string;
287
+ /**
288
+ * The path to the assets directory, relative to the `baseDirectory`.
289
+ */
290
+ assetDirectory: string;
291
+ /**
292
+ * An array of patterns that match files that should be uploaded.
293
+ */
294
+ includePatterns: string[];
295
+ /**
296
+ * An array of patterns that match files that should not be uploaded.
297
+ */
298
+ excludePatterns: string[];
299
299
  }
300
300
 
301
301
  /**
@@ -304,23 +304,69 @@ export interface AssetPaths {
304
304
  * Uses the args (passed from the command line) if available,
305
305
  * falling back to those defined in the config.
306
306
  *
307
+ * (This function corresponds to --assets/config.assets)
308
+ *
307
309
  */
308
310
  export function getAssetPaths(
309
- config: Config,
310
- assetDirectory = config.site?.bucket,
311
- includePatterns = config.site?.include ?? [],
312
- excludePatterns = config.site?.exclude ?? []
311
+ config: Config,
312
+ assetDirectory: string | undefined
313
+ ): AssetPaths | undefined {
314
+ const baseDirectory = assetDirectory
315
+ ? process.cwd()
316
+ : path.resolve(path.dirname(config.configPath ?? "wrangler.toml"));
317
+
318
+ assetDirectory ??=
319
+ typeof config.assets === "string"
320
+ ? config.assets
321
+ : config.assets !== undefined
322
+ ? config.assets.bucket
323
+ : undefined;
324
+
325
+ const includePatterns =
326
+ (typeof config.assets !== "string" && config.assets?.include) || [];
327
+
328
+ const excludePatterns =
329
+ (typeof config.assets !== "string" && config.assets?.exclude) || [];
330
+
331
+ return assetDirectory
332
+ ? {
333
+ baseDirectory,
334
+ assetDirectory,
335
+ includePatterns,
336
+ excludePatterns,
337
+ }
338
+ : undefined;
339
+ }
340
+
341
+ /**
342
+ * Get an object that describes what site assets to upload, if any.
343
+ *
344
+ * Uses the args (passed from the command line) if available,
345
+ * falling back to those defined in the config.
346
+ *
347
+ * (This function corresponds to --site/config.site)
348
+ *
349
+ */
350
+ export function getSiteAssetPaths(
351
+ config: Config,
352
+ assetDirectory?: string,
353
+ includePatterns = config.site?.include ?? [],
354
+ excludePatterns = config.site?.exclude ?? []
313
355
  ): AssetPaths | undefined {
314
- const baseDirectory = path.resolve(
315
- path.dirname(config.configPath ?? "wrangler.toml")
316
- );
317
-
318
- return assetDirectory
319
- ? {
320
- baseDirectory,
321
- assetDirectory,
322
- includePatterns,
323
- excludePatterns,
324
- }
325
- : undefined;
356
+ const baseDirectory = assetDirectory
357
+ ? process.cwd()
358
+ : path.resolve(path.dirname(config.configPath ?? "wrangler.toml"));
359
+
360
+ assetDirectory ??= config.site?.bucket;
361
+
362
+ if (assetDirectory) {
363
+ return {
364
+ baseDirectory,
365
+ assetDirectory,
366
+ includePatterns,
367
+ excludePatterns,
368
+ };
369
+ } else {
370
+ return undefined;
371
+ }
326
372
  }