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
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
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
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
|
-
|
|
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
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
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
|
-
|
|
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
|
-
|
|
63
|
-
|
|
64
|
-
|
|
62
|
+
hasher: XXHashAPI,
|
|
63
|
+
filePath: string,
|
|
64
|
+
content: string
|
|
65
65
|
): string {
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
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
|
-
|
|
75
|
-
|
|
74
|
+
title: string,
|
|
75
|
+
accountId: string
|
|
76
76
|
) {
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
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
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
107
|
+
accountId: string | undefined,
|
|
108
|
+
scriptName: string,
|
|
109
|
+
siteAssets: AssetPaths | undefined,
|
|
110
|
+
preview: boolean,
|
|
111
|
+
dryRun: boolean | undefined
|
|
112
112
|
): Promise<{
|
|
113
|
-
|
|
114
|
-
|
|
113
|
+
manifest: { [filePath: string]: string } | undefined;
|
|
114
|
+
namespace: string | undefined;
|
|
115
115
|
}> {
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
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
|
-
|
|
232
|
-
|
|
231
|
+
patterns: string[],
|
|
232
|
+
exclude: boolean
|
|
233
233
|
): (filePath: string) => boolean {
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
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
|
-
|
|
250
|
-
|
|
249
|
+
absFilePath: string,
|
|
250
|
+
relativeFilePath: string
|
|
251
251
|
): Promise<void> {
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
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
|
-
|
|
262
|
-
|
|
263
|
-
|
|
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
|
-
|
|
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
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
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
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
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
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
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
|
}
|