wrangler 0.0.0-e6733a3 → 0.0.0-e6ada079
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.
Potentially problematic release.
This version of wrangler might be problematic. Click here for more details.
- package/README.md +47 -16
- package/bin/wrangler.js +94 -31
- package/config-schema.json +3100 -0
- package/kv-asset-handler.js +1 -0
- package/package.json +154 -82
- package/templates/__tests__/pages-dev-util.test.ts +128 -0
- package/templates/__tests__/tsconfig-sanity.ts +12 -0
- package/templates/__tests__/tsconfig.json +8 -0
- package/templates/checked-fetch.js +30 -0
- package/templates/facade.d.ts +19 -0
- package/templates/gitignore +170 -0
- package/templates/init-tests/test-jest-new-worker.js +23 -0
- package/templates/init-tests/test-vitest-new-worker.js +24 -0
- package/templates/init-tests/test-vitest-new-worker.ts +25 -0
- package/templates/middleware/common.ts +67 -0
- package/templates/middleware/loader-modules.ts +134 -0
- package/templates/middleware/loader-sw.ts +229 -0
- package/templates/middleware/middleware-ensure-req-body-drained.ts +18 -0
- package/templates/middleware/middleware-miniflare3-json-error.ts +32 -0
- package/templates/middleware/middleware-pretty-error.ts +40 -0
- package/templates/middleware/middleware-scheduled.ts +15 -0
- package/templates/middleware/middleware-serve-static-assets.d.ts +6 -0
- package/templates/middleware/middleware-serve-static-assets.ts +56 -0
- package/templates/modules-watch-stub.js +4 -0
- package/templates/new-worker-scheduled.js +17 -0
- package/templates/new-worker-scheduled.ts +32 -0
- package/templates/new-worker.js +15 -0
- package/templates/new-worker.ts +33 -0
- package/templates/no-op-worker.js +10 -0
- package/templates/pages-dev-pipeline.ts +32 -0
- package/templates/pages-dev-util.ts +55 -0
- package/templates/pages-shim.ts +9 -0
- package/templates/pages-template-plugin.ts +190 -0
- package/templates/pages-template-worker.ts +198 -0
- package/templates/startDevWorker/InspectorProxyWorker.ts +664 -0
- package/templates/startDevWorker/ProxyWorker.ts +334 -0
- package/templates/tsconfig-sanity.ts +11 -0
- package/templates/tsconfig.init.json +22 -0
- package/templates/tsconfig.json +8 -0
- package/wrangler-dist/InspectorProxyWorker.js +464 -0
- package/wrangler-dist/InspectorProxyWorker.js.map +6 -0
- package/wrangler-dist/ProxyWorker.js +240 -0
- package/wrangler-dist/ProxyWorker.js.map +6 -0
- package/wrangler-dist/cli.d.ts +26391 -0
- package/wrangler-dist/cli.js +204293 -116652
- package/wrangler-dist/wasm-sync.wasm +0 -0
- package/import_meta_url.js +0 -3
- package/miniflare-config-stubs/.env.empty +0 -0
- package/miniflare-config-stubs/package.empty.json +0 -1
- package/miniflare-config-stubs/wrangler.empty.toml +0 -0
- package/pages/functions/buildWorker.ts +0 -62
- package/pages/functions/filepath-routing.test.ts +0 -39
- package/pages/functions/filepath-routing.ts +0 -221
- package/pages/functions/identifiers.ts +0 -78
- package/pages/functions/routes.ts +0 -158
- package/pages/functions/template-worker.ts +0 -144
- package/src/__tests__/clipboardy-mock.js +0 -4
- package/src/__tests__/dev.test.tsx +0 -66
- package/src/__tests__/index.test.ts +0 -287
- package/src/__tests__/jest.setup.ts +0 -22
- package/src/__tests__/kv.test.ts +0 -1098
- package/src/__tests__/mock-cfetch.ts +0 -171
- package/src/__tests__/mock-dialogs.ts +0 -65
- package/src/__tests__/run-in-tmp.ts +0 -19
- package/src/__tests__/run-wrangler.ts +0 -32
- package/src/api/form_data.ts +0 -131
- package/src/api/preview.ts +0 -128
- package/src/api/worker.ts +0 -155
- package/src/cfetch/index.ts +0 -102
- package/src/cfetch/internal.ts +0 -69
- package/src/cli.ts +0 -9
- package/src/config.ts +0 -487
- package/src/dev.tsx +0 -771
- package/src/dialogs.tsx +0 -77
- package/src/index.tsx +0 -1974
- package/src/inspect.ts +0 -524
- package/src/kv.tsx +0 -267
- package/src/module-collection.ts +0 -64
- package/src/pages.tsx +0 -1031
- package/src/proxy.ts +0 -294
- package/src/publish.ts +0 -358
- package/src/sites.tsx +0 -114
- package/src/tail.tsx +0 -73
- package/src/user.tsx +0 -1025
- package/static-asset-facade.js +0 -47
- package/vendor/@cloudflare/kv-asset-handler/CHANGELOG.md +0 -332
- package/vendor/@cloudflare/kv-asset-handler/LICENSE_APACHE +0 -176
- package/vendor/@cloudflare/kv-asset-handler/LICENSE_MIT +0 -25
- package/vendor/@cloudflare/kv-asset-handler/README.md +0 -245
- package/vendor/@cloudflare/kv-asset-handler/dist/index.d.ts +0 -32
- package/vendor/@cloudflare/kv-asset-handler/dist/index.js +0 -354
- package/vendor/@cloudflare/kv-asset-handler/dist/mocks.d.ts +0 -13
- package/vendor/@cloudflare/kv-asset-handler/dist/mocks.js +0 -148
- package/vendor/@cloudflare/kv-asset-handler/dist/test/getAssetFromKV.d.ts +0 -1
- package/vendor/@cloudflare/kv-asset-handler/dist/test/getAssetFromKV.js +0 -436
- package/vendor/@cloudflare/kv-asset-handler/dist/test/mapRequestToAsset.d.ts +0 -1
- package/vendor/@cloudflare/kv-asset-handler/dist/test/mapRequestToAsset.js +0 -40
- package/vendor/@cloudflare/kv-asset-handler/dist/test/serveSinglePageApp.d.ts +0 -1
- package/vendor/@cloudflare/kv-asset-handler/dist/test/serveSinglePageApp.js +0 -42
- package/vendor/@cloudflare/kv-asset-handler/dist/types.d.ts +0 -26
- package/vendor/@cloudflare/kv-asset-handler/dist/types.js +0 -31
- package/vendor/@cloudflare/kv-asset-handler/package.json +0 -52
- package/vendor/@cloudflare/kv-asset-handler/src/index.ts +0 -296
- package/vendor/@cloudflare/kv-asset-handler/src/mocks.ts +0 -136
- package/vendor/@cloudflare/kv-asset-handler/src/test/getAssetFromKV.ts +0 -464
- package/vendor/@cloudflare/kv-asset-handler/src/test/mapRequestToAsset.ts +0 -33
- package/vendor/@cloudflare/kv-asset-handler/src/test/serveSinglePageApp.ts +0 -42
- package/vendor/@cloudflare/kv-asset-handler/src/types.ts +0 -39
- package/vendor/wrangler-mime/CHANGELOG.md +0 -289
- package/vendor/wrangler-mime/LICENSE +0 -21
- package/vendor/wrangler-mime/Mime.js +0 -97
- package/vendor/wrangler-mime/README.md +0 -187
- package/vendor/wrangler-mime/cli.js +0 -46
- package/vendor/wrangler-mime/index.js +0 -4
- package/vendor/wrangler-mime/lite.js +0 -4
- package/vendor/wrangler-mime/package.json +0 -52
- package/vendor/wrangler-mime/types/other.js +0 -1
- package/vendor/wrangler-mime/types/standard.js +0 -1
- package/wrangler-dist/cli.js.map +0 -7
@@ -1,464 +0,0 @@
|
|
1
|
-
import test from 'ava'
|
2
|
-
import { mockGlobal, getEvent, sleep, mockKV, mockManifest } from '../mocks'
|
3
|
-
import { getAssetFromKV, mapRequestToAsset } from '../index'
|
4
|
-
import { KVError } from '../types'
|
5
|
-
|
6
|
-
test('getAssetFromKV return correct val from KV and default caching', async (t) => {
|
7
|
-
mockGlobal()
|
8
|
-
const event = getEvent(new Request('https://blah.com/key1.txt'))
|
9
|
-
const res = await getAssetFromKV(event)
|
10
|
-
|
11
|
-
if (res) {
|
12
|
-
t.is(res.headers.get('cache-control'), null)
|
13
|
-
t.is(res.headers.get('cf-cache-status'), 'MISS')
|
14
|
-
t.is(await res.text(), 'val1')
|
15
|
-
t.true(res.headers.get('content-type').includes('text'))
|
16
|
-
} else {
|
17
|
-
t.fail('Response was undefined')
|
18
|
-
}
|
19
|
-
})
|
20
|
-
test('getAssetFromKV evaluated the file matching the extensionless path first /client/ -> client', async (t) => {
|
21
|
-
mockGlobal()
|
22
|
-
const event = getEvent(new Request(`https://foo.com/client/`))
|
23
|
-
const res = await getAssetFromKV(event)
|
24
|
-
t.is(await res.text(), 'important file')
|
25
|
-
t.true(res.headers.get('content-type').includes('text'))
|
26
|
-
})
|
27
|
-
test('getAssetFromKV evaluated the file matching the extensionless path first /client -> client', async (t) => {
|
28
|
-
mockGlobal()
|
29
|
-
const event = getEvent(new Request(`https://foo.com/client`))
|
30
|
-
const res = await getAssetFromKV(event)
|
31
|
-
t.is(await res.text(), 'important file')
|
32
|
-
t.true(res.headers.get('content-type').includes('text'))
|
33
|
-
})
|
34
|
-
|
35
|
-
test('getAssetFromKV if not in asset manifest still returns nohash.txt', async (t) => {
|
36
|
-
mockGlobal()
|
37
|
-
const event = getEvent(new Request('https://blah.com/nohash.txt'))
|
38
|
-
const res = await getAssetFromKV(event)
|
39
|
-
|
40
|
-
if (res) {
|
41
|
-
t.is(await res.text(), 'no hash but still got some result')
|
42
|
-
t.true(res.headers.get('content-type').includes('text'))
|
43
|
-
} else {
|
44
|
-
t.fail('Response was undefined')
|
45
|
-
}
|
46
|
-
})
|
47
|
-
|
48
|
-
test('getAssetFromKV if no asset manifest /client -> client fails', async (t) => {
|
49
|
-
mockGlobal()
|
50
|
-
const event = getEvent(new Request(`https://foo.com/client`))
|
51
|
-
const error: KVError = await t.throwsAsync(getAssetFromKV(event, { ASSET_MANIFEST: {} }))
|
52
|
-
t.is(error.status, 404)
|
53
|
-
})
|
54
|
-
|
55
|
-
test('getAssetFromKV if sub/ -> sub/index.html served', async (t) => {
|
56
|
-
mockGlobal()
|
57
|
-
const event = getEvent(new Request(`https://foo.com/sub`))
|
58
|
-
const res = await getAssetFromKV(event)
|
59
|
-
if (res) {
|
60
|
-
t.is(await res.text(), 'picturedis')
|
61
|
-
} else {
|
62
|
-
t.fail('Response was undefined')
|
63
|
-
}
|
64
|
-
})
|
65
|
-
|
66
|
-
test('getAssetFromKV gets index.html by default for / requests', async (t) => {
|
67
|
-
mockGlobal()
|
68
|
-
const event = getEvent(new Request('https://blah.com/'))
|
69
|
-
const res = await getAssetFromKV(event)
|
70
|
-
|
71
|
-
if (res) {
|
72
|
-
t.is(await res.text(), 'index.html')
|
73
|
-
t.true(res.headers.get('content-type').includes('html'))
|
74
|
-
} else {
|
75
|
-
t.fail('Response was undefined')
|
76
|
-
}
|
77
|
-
})
|
78
|
-
|
79
|
-
test('getAssetFromKV non ASCII path support', async (t) => {
|
80
|
-
mockGlobal()
|
81
|
-
const event = getEvent(new Request('https://blah.com/测试.html'))
|
82
|
-
const res = await getAssetFromKV(event)
|
83
|
-
|
84
|
-
if (res) {
|
85
|
-
t.is(await res.text(), 'My filename is non-ascii')
|
86
|
-
} else {
|
87
|
-
t.fail('Response was undefined')
|
88
|
-
}
|
89
|
-
})
|
90
|
-
|
91
|
-
test('getAssetFromKV supports browser percent encoded URLs', async (t) => {
|
92
|
-
mockGlobal()
|
93
|
-
const event = getEvent(new Request('https://example.com/%not-really-percent-encoded.html'))
|
94
|
-
const res = await getAssetFromKV(event)
|
95
|
-
|
96
|
-
if (res) {
|
97
|
-
t.is(await res.text(), 'browser percent encoded')
|
98
|
-
} else {
|
99
|
-
t.fail('Response was undefined')
|
100
|
-
}
|
101
|
-
})
|
102
|
-
|
103
|
-
test('getAssetFromKV supports user percent encoded URLs', async (t) => {
|
104
|
-
mockGlobal()
|
105
|
-
const event = getEvent(new Request('https://blah.com/%2F.html'))
|
106
|
-
const res = await getAssetFromKV(event)
|
107
|
-
|
108
|
-
if (res) {
|
109
|
-
t.is(await res.text(), 'user percent encoded')
|
110
|
-
} else {
|
111
|
-
t.fail('Response was undefined')
|
112
|
-
}
|
113
|
-
})
|
114
|
-
|
115
|
-
test('getAssetFromKV only decode URL when necessary', async (t) => {
|
116
|
-
mockGlobal()
|
117
|
-
const event1 = getEvent(new Request('https://blah.com/%E4%BD%A0%E5%A5%BD.html'))
|
118
|
-
const event2 = getEvent(new Request('https://blah.com/你好.html'))
|
119
|
-
const res1 = await getAssetFromKV(event1)
|
120
|
-
const res2 = await getAssetFromKV(event2)
|
121
|
-
|
122
|
-
if (res1 && res2) {
|
123
|
-
t.is(await res1.text(), 'Im important')
|
124
|
-
t.is(await res2.text(), 'Im important')
|
125
|
-
} else {
|
126
|
-
t.fail('Response was undefined')
|
127
|
-
}
|
128
|
-
})
|
129
|
-
|
130
|
-
test('getAssetFromKV Support for user decode url path', async (t) => {
|
131
|
-
mockGlobal()
|
132
|
-
const event1 = getEvent(new Request('https://blah.com/%E4%BD%A0%E5%A5%BD/'))
|
133
|
-
const event2 = getEvent(new Request('https://blah.com/你好/'))
|
134
|
-
const res1 = await getAssetFromKV(event1)
|
135
|
-
const res2 = await getAssetFromKV(event2)
|
136
|
-
|
137
|
-
if (res1 && res2) {
|
138
|
-
t.is(await res1.text(), 'My path is non-ascii')
|
139
|
-
t.is(await res2.text(), 'My path is non-ascii')
|
140
|
-
} else {
|
141
|
-
t.fail('Response was undefined')
|
142
|
-
}
|
143
|
-
})
|
144
|
-
|
145
|
-
test('getAssetFromKV custom key modifier', async (t) => {
|
146
|
-
mockGlobal()
|
147
|
-
const event = getEvent(new Request('https://blah.com/docs/sub/blah.png'))
|
148
|
-
|
149
|
-
const customRequestMapper = (request: Request) => {
|
150
|
-
let defaultModifiedRequest = mapRequestToAsset(request)
|
151
|
-
|
152
|
-
let url = new URL(defaultModifiedRequest.url)
|
153
|
-
url.pathname = url.pathname.replace('/docs', '')
|
154
|
-
return new Request(url.toString(), request)
|
155
|
-
}
|
156
|
-
|
157
|
-
const res = await getAssetFromKV(event, { mapRequestToAsset: customRequestMapper })
|
158
|
-
|
159
|
-
if (res) {
|
160
|
-
t.is(await res.text(), 'picturedis')
|
161
|
-
} else {
|
162
|
-
t.fail('Response was undefined')
|
163
|
-
}
|
164
|
-
})
|
165
|
-
|
166
|
-
test('getAssetFromKV when setting browser caching', async (t) => {
|
167
|
-
mockGlobal()
|
168
|
-
const event = getEvent(new Request('https://blah.com/'))
|
169
|
-
|
170
|
-
const res = await getAssetFromKV(event, { cacheControl: { browserTTL: 22 } })
|
171
|
-
|
172
|
-
if (res) {
|
173
|
-
t.is(res.headers.get('cache-control'), 'max-age=22')
|
174
|
-
} else {
|
175
|
-
t.fail('Response was undefined')
|
176
|
-
}
|
177
|
-
})
|
178
|
-
|
179
|
-
test('getAssetFromKV when setting custom cache setting', async (t) => {
|
180
|
-
mockGlobal()
|
181
|
-
const event1 = getEvent(new Request('https://blah.com/'))
|
182
|
-
const event2 = getEvent(new Request('https://blah.com/key1.png?blah=34'))
|
183
|
-
const cacheOnlyPngs = (req: Request) => {
|
184
|
-
if (new URL(req.url).pathname.endsWith('.png'))
|
185
|
-
return {
|
186
|
-
browserTTL: 720,
|
187
|
-
edgeTTL: 720,
|
188
|
-
}
|
189
|
-
else
|
190
|
-
return {
|
191
|
-
bypassCache: true,
|
192
|
-
}
|
193
|
-
}
|
194
|
-
|
195
|
-
const res1 = await getAssetFromKV(event1, { cacheControl: cacheOnlyPngs })
|
196
|
-
const res2 = await getAssetFromKV(event2, { cacheControl: cacheOnlyPngs })
|
197
|
-
|
198
|
-
if (res1 && res2) {
|
199
|
-
t.is(res1.headers.get('cache-control'), null)
|
200
|
-
t.true(res2.headers.get('content-type').includes('png'))
|
201
|
-
t.is(res2.headers.get('cache-control'), 'max-age=720')
|
202
|
-
t.is(res2.headers.get('cf-cache-status'), 'MISS')
|
203
|
-
} else {
|
204
|
-
t.fail('Response was undefined')
|
205
|
-
}
|
206
|
-
})
|
207
|
-
test('getAssetFromKV caches on two sequential requests', async (t) => {
|
208
|
-
mockGlobal()
|
209
|
-
const resourceKey = 'cache.html'
|
210
|
-
const resourceVersion = JSON.parse(mockManifest())[resourceKey]
|
211
|
-
const event1 = getEvent(new Request(`https://blah.com/${resourceKey}`))
|
212
|
-
const event2 = getEvent(
|
213
|
-
new Request(`https://blah.com/${resourceKey}`, {
|
214
|
-
headers: {
|
215
|
-
'if-none-match': `"${resourceVersion}"`,
|
216
|
-
},
|
217
|
-
}),
|
218
|
-
)
|
219
|
-
|
220
|
-
const res1 = await getAssetFromKV(event1, { cacheControl: { edgeTTL: 720, browserTTL: 720 } })
|
221
|
-
await sleep(1)
|
222
|
-
const res2 = await getAssetFromKV(event2)
|
223
|
-
|
224
|
-
if (res1 && res2) {
|
225
|
-
t.is(res1.headers.get('cf-cache-status'), 'MISS')
|
226
|
-
t.is(res1.headers.get('cache-control'), 'max-age=720')
|
227
|
-
t.is(res2.headers.get('cf-cache-status'), 'REVALIDATED')
|
228
|
-
} else {
|
229
|
-
t.fail('Response was undefined')
|
230
|
-
}
|
231
|
-
})
|
232
|
-
test('getAssetFromKV does not store max-age on two sequential requests', async (t) => {
|
233
|
-
mockGlobal()
|
234
|
-
const resourceKey = 'cache.html'
|
235
|
-
const resourceVersion = JSON.parse(mockManifest())[resourceKey]
|
236
|
-
const event1 = getEvent(new Request(`https://blah.com/${resourceKey}`))
|
237
|
-
const event2 = getEvent(
|
238
|
-
new Request(`https://blah.com/${resourceKey}`, {
|
239
|
-
headers: {
|
240
|
-
'if-none-match': `"${resourceVersion}"`,
|
241
|
-
},
|
242
|
-
}),
|
243
|
-
)
|
244
|
-
|
245
|
-
const res1 = await getAssetFromKV(event1, { cacheControl: { edgeTTL: 720 } })
|
246
|
-
await sleep(100)
|
247
|
-
const res2 = await getAssetFromKV(event2)
|
248
|
-
|
249
|
-
if (res1 && res2) {
|
250
|
-
t.is(res1.headers.get('cf-cache-status'), 'MISS')
|
251
|
-
t.is(res1.headers.get('cache-control'), null)
|
252
|
-
t.is(res2.headers.get('cf-cache-status'), 'REVALIDATED')
|
253
|
-
t.is(res2.headers.get('cache-control'), null)
|
254
|
-
} else {
|
255
|
-
t.fail('Response was undefined')
|
256
|
-
}
|
257
|
-
})
|
258
|
-
|
259
|
-
test('getAssetFromKV does not cache on Cloudflare when bypass cache set', async (t) => {
|
260
|
-
mockGlobal()
|
261
|
-
const event = getEvent(new Request('https://blah.com/'))
|
262
|
-
|
263
|
-
const res = await getAssetFromKV(event, { cacheControl: { bypassCache: true } })
|
264
|
-
|
265
|
-
if (res) {
|
266
|
-
t.is(res.headers.get('cache-control'), null)
|
267
|
-
t.is(res.headers.get('cf-cache-status'), null)
|
268
|
-
} else {
|
269
|
-
t.fail('Response was undefined')
|
270
|
-
}
|
271
|
-
})
|
272
|
-
|
273
|
-
test('getAssetFromKV with no trailing slash on root', async (t) => {
|
274
|
-
mockGlobal()
|
275
|
-
const event = getEvent(new Request('https://blah.com'))
|
276
|
-
const res = await getAssetFromKV(event)
|
277
|
-
if (res) {
|
278
|
-
t.is(await res.text(), 'index.html')
|
279
|
-
} else {
|
280
|
-
t.fail('Response was undefined')
|
281
|
-
}
|
282
|
-
})
|
283
|
-
|
284
|
-
test('getAssetFromKV with no trailing slash on a subdirectory', async (t) => {
|
285
|
-
mockGlobal()
|
286
|
-
const event = getEvent(new Request('https://blah.com/sub/blah.png'))
|
287
|
-
const res = await getAssetFromKV(event)
|
288
|
-
if (res) {
|
289
|
-
t.is(await res.text(), 'picturedis')
|
290
|
-
} else {
|
291
|
-
t.fail('Response was undefined')
|
292
|
-
}
|
293
|
-
})
|
294
|
-
|
295
|
-
test('getAssetFromKV no result throws an error', async (t) => {
|
296
|
-
mockGlobal()
|
297
|
-
const event = getEvent(new Request('https://blah.com/random'))
|
298
|
-
const error: KVError = await t.throwsAsync(getAssetFromKV(event))
|
299
|
-
t.is(error.status, 404)
|
300
|
-
})
|
301
|
-
test('getAssetFromKV TTls set to null should not cache on browser or edge', async (t) => {
|
302
|
-
mockGlobal()
|
303
|
-
const event = getEvent(new Request('https://blah.com/'))
|
304
|
-
|
305
|
-
const res1 = await getAssetFromKV(event, { cacheControl: { browserTTL: null, edgeTTL: null } })
|
306
|
-
await sleep(100)
|
307
|
-
const res2 = await getAssetFromKV(event, { cacheControl: { browserTTL: null, edgeTTL: null } })
|
308
|
-
|
309
|
-
if (res1 && res2) {
|
310
|
-
t.is(res1.headers.get('cf-cache-status'), null)
|
311
|
-
t.is(res1.headers.get('cache-control'), null)
|
312
|
-
t.is(res2.headers.get('cf-cache-status'), null)
|
313
|
-
t.is(res2.headers.get('cache-control'), null)
|
314
|
-
} else {
|
315
|
-
t.fail('Response was undefined')
|
316
|
-
}
|
317
|
-
})
|
318
|
-
test('getAssetFromKV passing in a custom NAMESPACE serves correct asset', async (t) => {
|
319
|
-
mockGlobal()
|
320
|
-
let CUSTOM_NAMESPACE = mockKV({
|
321
|
-
'key1.123HASHBROWN.txt': 'val1',
|
322
|
-
})
|
323
|
-
Object.assign(global, { CUSTOM_NAMESPACE })
|
324
|
-
const event = getEvent(new Request('https://blah.com/'))
|
325
|
-
const res = await getAssetFromKV(event)
|
326
|
-
if (res) {
|
327
|
-
t.is(await res.text(), 'index.html')
|
328
|
-
t.true(res.headers.get('content-type').includes('html'))
|
329
|
-
} else {
|
330
|
-
t.fail('Response was undefined')
|
331
|
-
}
|
332
|
-
})
|
333
|
-
test('getAssetFromKV when custom namespace without the asset should fail', async (t) => {
|
334
|
-
mockGlobal()
|
335
|
-
let CUSTOM_NAMESPACE = mockKV({
|
336
|
-
'key5.123HASHBROWN.txt': 'customvalu',
|
337
|
-
})
|
338
|
-
|
339
|
-
const event = getEvent(new Request('https://blah.com'))
|
340
|
-
const error: KVError = await t.throwsAsync(
|
341
|
-
getAssetFromKV(event, { ASSET_NAMESPACE: CUSTOM_NAMESPACE }),
|
342
|
-
)
|
343
|
-
t.is(error.status, 404)
|
344
|
-
})
|
345
|
-
test('getAssetFromKV when namespace not bound fails', async (t) => {
|
346
|
-
mockGlobal()
|
347
|
-
var MY_CUSTOM_NAMESPACE = undefined
|
348
|
-
Object.assign(global, { MY_CUSTOM_NAMESPACE })
|
349
|
-
|
350
|
-
const event = getEvent(new Request('https://blah.com/'))
|
351
|
-
const error: KVError = await t.throwsAsync(
|
352
|
-
getAssetFromKV(event, { ASSET_NAMESPACE: MY_CUSTOM_NAMESPACE }),
|
353
|
-
)
|
354
|
-
t.is(error.status, 500)
|
355
|
-
})
|
356
|
-
|
357
|
-
test('getAssetFromKV when if-none-match === active resource version, should revalidate', async (t) => {
|
358
|
-
mockGlobal()
|
359
|
-
const resourceKey = 'key1.png'
|
360
|
-
const resourceVersion = JSON.parse(mockManifest())[resourceKey]
|
361
|
-
const event1 = getEvent(new Request(`https://blah.com/${resourceKey}`))
|
362
|
-
const event2 = getEvent(
|
363
|
-
new Request(`https://blah.com/${resourceKey}`, {
|
364
|
-
headers: {
|
365
|
-
'if-none-match': `W/"${resourceVersion}"`,
|
366
|
-
},
|
367
|
-
}),
|
368
|
-
)
|
369
|
-
|
370
|
-
const res1 = await getAssetFromKV(event1, { cacheControl: { edgeTTL: 720 } })
|
371
|
-
await sleep(100)
|
372
|
-
const res2 = await getAssetFromKV(event2)
|
373
|
-
|
374
|
-
if (res1 && res2) {
|
375
|
-
t.is(res1.headers.get('cf-cache-status'), 'MISS')
|
376
|
-
t.is(res2.headers.get('cf-cache-status'), 'REVALIDATED')
|
377
|
-
} else {
|
378
|
-
t.fail('Response was undefined')
|
379
|
-
}
|
380
|
-
})
|
381
|
-
|
382
|
-
test('getAssetFromKV when if-none-match equals etag of stale resource then should bypass cache', async (t) => {
|
383
|
-
mockGlobal()
|
384
|
-
const resourceKey = 'key1.png'
|
385
|
-
const resourceVersion = JSON.parse(mockManifest())[resourceKey]
|
386
|
-
const req1 = new Request(`https://blah.com/${resourceKey}`, {
|
387
|
-
headers: {
|
388
|
-
'if-none-match': `"${resourceVersion}"`,
|
389
|
-
},
|
390
|
-
})
|
391
|
-
const req2 = new Request(`https://blah.com/${resourceKey}`, {
|
392
|
-
headers: {
|
393
|
-
'if-none-match': `"${resourceVersion}-another-version"`,
|
394
|
-
},
|
395
|
-
})
|
396
|
-
const event = getEvent(req1)
|
397
|
-
const event2 = getEvent(req2)
|
398
|
-
const res1 = await getAssetFromKV(event, { cacheControl: { edgeTTL: 720 } })
|
399
|
-
const res2 = await getAssetFromKV(event)
|
400
|
-
const res3 = await getAssetFromKV(event2)
|
401
|
-
if (res1 && res2 && res3) {
|
402
|
-
t.is(res1.headers.get('cf-cache-status'), 'MISS')
|
403
|
-
t.is(res2.headers.get('etag'), `W/${req1.headers.get('if-none-match')}`)
|
404
|
-
t.is(res2.headers.get('cf-cache-status'), 'REVALIDATED')
|
405
|
-
t.not(res3.headers.get('etag'), req2.headers.get('if-none-match'))
|
406
|
-
t.is(res3.headers.get('cf-cache-status'), 'MISS')
|
407
|
-
} else {
|
408
|
-
t.fail('Response was undefined')
|
409
|
-
}
|
410
|
-
})
|
411
|
-
test('getAssetFromKV when resource in cache, etag should be weakened before returned to eyeball', async (t) => {
|
412
|
-
mockGlobal()
|
413
|
-
const resourceKey = 'key1.png'
|
414
|
-
const resourceVersion = JSON.parse(mockManifest())[resourceKey]
|
415
|
-
const req1 = new Request(`https://blah.com/${resourceKey}`, {
|
416
|
-
headers: {
|
417
|
-
'if-none-match': `"${resourceVersion}"`,
|
418
|
-
},
|
419
|
-
})
|
420
|
-
const event = getEvent(req1)
|
421
|
-
const res1 = await getAssetFromKV(event, { cacheControl: { edgeTTL: 720 } })
|
422
|
-
const res2 = await getAssetFromKV(event)
|
423
|
-
if (res1 && res2) {
|
424
|
-
t.is(res1.headers.get('cf-cache-status'), 'MISS')
|
425
|
-
t.is(res2.headers.get('etag'), `W/${req1.headers.get('if-none-match')}`)
|
426
|
-
} else {
|
427
|
-
t.fail('Response was undefined')
|
428
|
-
}
|
429
|
-
})
|
430
|
-
|
431
|
-
test('getAssetFromKV if-none-match not sent but resource in cache, should return cache hit 200 OK', async (t) => {
|
432
|
-
const resourceKey = 'cache.html'
|
433
|
-
const event = getEvent(new Request(`https://blah.com/${resourceKey}`))
|
434
|
-
const res1 = await getAssetFromKV(event, { cacheControl: { edgeTTL: 720 } })
|
435
|
-
await sleep(1)
|
436
|
-
const res2 = await getAssetFromKV(event)
|
437
|
-
if (res1 && res2) {
|
438
|
-
t.is(res1.headers.get('cf-cache-status'), 'MISS')
|
439
|
-
t.is(res1.headers.get('cache-control'), null)
|
440
|
-
t.is(res2.status, 200)
|
441
|
-
t.is(res2.headers.get('cf-cache-status'), 'HIT')
|
442
|
-
} else {
|
443
|
-
t.fail('Response was undefined')
|
444
|
-
}
|
445
|
-
})
|
446
|
-
|
447
|
-
test('getAssetFromKV if range request submitted and resource in cache, request fulfilled', async (t) => {
|
448
|
-
const resourceKey = 'cache.html'
|
449
|
-
const event1 = getEvent(new Request(`https://blah.com/${resourceKey}`))
|
450
|
-
const event2 = getEvent(
|
451
|
-
new Request(`https://blah.com/${resourceKey}`, { headers: { range: 'bytes=0-10' } }),
|
452
|
-
)
|
453
|
-
const res1 = getAssetFromKV(event1, { cacheControl: { edgeTTL: 720 } })
|
454
|
-
await res1
|
455
|
-
await sleep(2)
|
456
|
-
const res2 = await getAssetFromKV(event2)
|
457
|
-
if (res2.headers.has('content-range')) {
|
458
|
-
t.is(res2.status, 206)
|
459
|
-
} else {
|
460
|
-
t.fail('Response was undefined')
|
461
|
-
}
|
462
|
-
})
|
463
|
-
|
464
|
-
test.todo('getAssetFromKV when body not empty, should invoke .cancel()')
|
@@ -1,33 +0,0 @@
|
|
1
|
-
import test from 'ava'
|
2
|
-
import { mockGlobal } from '../mocks'
|
3
|
-
import { mapRequestToAsset } from '../index'
|
4
|
-
|
5
|
-
test('mapRequestToAsset() correctly changes /about -> /about/index.html', async (t) => {
|
6
|
-
mockGlobal()
|
7
|
-
let path = '/about'
|
8
|
-
let request = new Request(`https://foo.com${path}`)
|
9
|
-
let newRequest = mapRequestToAsset(request)
|
10
|
-
t.is(newRequest.url, request.url + '/index.html')
|
11
|
-
})
|
12
|
-
|
13
|
-
test('mapRequestToAsset() correctly changes /about/ -> /about/index.html', async (t) => {
|
14
|
-
let path = '/about/'
|
15
|
-
let request = new Request(`https://foo.com${path}`)
|
16
|
-
let newRequest = mapRequestToAsset(request)
|
17
|
-
t.is(newRequest.url, request.url + 'index.html')
|
18
|
-
})
|
19
|
-
|
20
|
-
test('mapRequestToAsset() correctly changes /about.me/ -> /about.me/index.html', async (t) => {
|
21
|
-
let path = '/about.me/'
|
22
|
-
let request = new Request(`https://foo.com${path}`)
|
23
|
-
let newRequest = mapRequestToAsset(request)
|
24
|
-
t.is(newRequest.url, request.url + 'index.html')
|
25
|
-
})
|
26
|
-
|
27
|
-
test('mapRequestToAsset() correctly changes /about -> /about/default.html', async (t) => {
|
28
|
-
mockGlobal()
|
29
|
-
let path = '/about'
|
30
|
-
let request = new Request(`https://foo.com${path}`)
|
31
|
-
let newRequest = mapRequestToAsset(request, { defaultDocument: 'default.html' })
|
32
|
-
t.is(newRequest.url, request.url + '/default.html')
|
33
|
-
})
|
@@ -1,42 +0,0 @@
|
|
1
|
-
import test from 'ava'
|
2
|
-
import { mockGlobal } from '../mocks'
|
3
|
-
import { serveSinglePageApp } from '../index'
|
4
|
-
|
5
|
-
function testRequest(path: string) {
|
6
|
-
mockGlobal()
|
7
|
-
let url = new URL('https://example.com')
|
8
|
-
url.pathname = path
|
9
|
-
let request = new Request(url.toString())
|
10
|
-
|
11
|
-
return request
|
12
|
-
}
|
13
|
-
|
14
|
-
test('serveSinglePageApp returns root asset path when request path ends in .html', async (t) => {
|
15
|
-
let path = '/foo/thing.html'
|
16
|
-
let request = testRequest(path)
|
17
|
-
|
18
|
-
let expected_request = testRequest('/index.html')
|
19
|
-
let actual_request = serveSinglePageApp(request)
|
20
|
-
|
21
|
-
t.deepEqual(expected_request, actual_request)
|
22
|
-
})
|
23
|
-
|
24
|
-
test('serveSinglePageApp returns root asset path when request path does not have extension', async (t) => {
|
25
|
-
let path = '/foo/thing'
|
26
|
-
let request = testRequest(path)
|
27
|
-
|
28
|
-
let expected_request = testRequest('/index.html')
|
29
|
-
let actual_request = serveSinglePageApp(request)
|
30
|
-
|
31
|
-
t.deepEqual(expected_request, actual_request)
|
32
|
-
})
|
33
|
-
|
34
|
-
test('serveSinglePageApp returns requested asset when request path has non-html extension', async (t) => {
|
35
|
-
let path = '/foo/thing.js'
|
36
|
-
let request = testRequest(path)
|
37
|
-
|
38
|
-
let expected_request = request
|
39
|
-
let actual_request = serveSinglePageApp(request)
|
40
|
-
|
41
|
-
t.deepEqual(expected_request, actual_request)
|
42
|
-
})
|
@@ -1,39 +0,0 @@
|
|
1
|
-
export type CacheControl = {
|
2
|
-
browserTTL: number
|
3
|
-
edgeTTL: number
|
4
|
-
bypassCache: boolean
|
5
|
-
}
|
6
|
-
export type Options = {
|
7
|
-
cacheControl: ((req: Request) => Partial<CacheControl>) | Partial<CacheControl>
|
8
|
-
ASSET_NAMESPACE: any
|
9
|
-
ASSET_MANIFEST: Object | string
|
10
|
-
mapRequestToAsset?: (req: Request, options?: Partial<Options>) => Request
|
11
|
-
defaultMimeType: string
|
12
|
-
defaultDocument: string
|
13
|
-
}
|
14
|
-
|
15
|
-
export class KVError extends Error {
|
16
|
-
constructor(message?: string, status: number = 500) {
|
17
|
-
super(message)
|
18
|
-
// see: typescriptlang.org/docs/handbook/release-notes/typescript-2-2.html
|
19
|
-
Object.setPrototypeOf(this, new.target.prototype) // restore prototype chain
|
20
|
-
this.name = KVError.name // stack traces display correctly now
|
21
|
-
this.status = status
|
22
|
-
}
|
23
|
-
status: number
|
24
|
-
}
|
25
|
-
export class MethodNotAllowedError extends KVError {
|
26
|
-
constructor(message: string = `Not a valid request method`, status: number = 405) {
|
27
|
-
super(message, status)
|
28
|
-
}
|
29
|
-
}
|
30
|
-
export class NotFoundError extends KVError {
|
31
|
-
constructor(message: string = `Not Found`, status: number = 404) {
|
32
|
-
super(message, status)
|
33
|
-
}
|
34
|
-
}
|
35
|
-
export class InternalError extends KVError {
|
36
|
-
constructor(message: string = `Internal Error in KV Asset Handler`, status: number = 500) {
|
37
|
-
super(message, status)
|
38
|
-
}
|
39
|
-
}
|