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.

Files changed (119) hide show
  1. package/README.md +47 -16
  2. package/bin/wrangler.js +94 -31
  3. package/config-schema.json +3100 -0
  4. package/kv-asset-handler.js +1 -0
  5. package/package.json +154 -82
  6. package/templates/__tests__/pages-dev-util.test.ts +128 -0
  7. package/templates/__tests__/tsconfig-sanity.ts +12 -0
  8. package/templates/__tests__/tsconfig.json +8 -0
  9. package/templates/checked-fetch.js +30 -0
  10. package/templates/facade.d.ts +19 -0
  11. package/templates/gitignore +170 -0
  12. package/templates/init-tests/test-jest-new-worker.js +23 -0
  13. package/templates/init-tests/test-vitest-new-worker.js +24 -0
  14. package/templates/init-tests/test-vitest-new-worker.ts +25 -0
  15. package/templates/middleware/common.ts +67 -0
  16. package/templates/middleware/loader-modules.ts +134 -0
  17. package/templates/middleware/loader-sw.ts +229 -0
  18. package/templates/middleware/middleware-ensure-req-body-drained.ts +18 -0
  19. package/templates/middleware/middleware-miniflare3-json-error.ts +32 -0
  20. package/templates/middleware/middleware-pretty-error.ts +40 -0
  21. package/templates/middleware/middleware-scheduled.ts +15 -0
  22. package/templates/middleware/middleware-serve-static-assets.d.ts +6 -0
  23. package/templates/middleware/middleware-serve-static-assets.ts +56 -0
  24. package/templates/modules-watch-stub.js +4 -0
  25. package/templates/new-worker-scheduled.js +17 -0
  26. package/templates/new-worker-scheduled.ts +32 -0
  27. package/templates/new-worker.js +15 -0
  28. package/templates/new-worker.ts +33 -0
  29. package/templates/no-op-worker.js +10 -0
  30. package/templates/pages-dev-pipeline.ts +32 -0
  31. package/templates/pages-dev-util.ts +55 -0
  32. package/templates/pages-shim.ts +9 -0
  33. package/templates/pages-template-plugin.ts +190 -0
  34. package/templates/pages-template-worker.ts +198 -0
  35. package/templates/startDevWorker/InspectorProxyWorker.ts +664 -0
  36. package/templates/startDevWorker/ProxyWorker.ts +334 -0
  37. package/templates/tsconfig-sanity.ts +11 -0
  38. package/templates/tsconfig.init.json +22 -0
  39. package/templates/tsconfig.json +8 -0
  40. package/wrangler-dist/InspectorProxyWorker.js +464 -0
  41. package/wrangler-dist/InspectorProxyWorker.js.map +6 -0
  42. package/wrangler-dist/ProxyWorker.js +240 -0
  43. package/wrangler-dist/ProxyWorker.js.map +6 -0
  44. package/wrangler-dist/cli.d.ts +26391 -0
  45. package/wrangler-dist/cli.js +204293 -116652
  46. package/wrangler-dist/wasm-sync.wasm +0 -0
  47. package/import_meta_url.js +0 -3
  48. package/miniflare-config-stubs/.env.empty +0 -0
  49. package/miniflare-config-stubs/package.empty.json +0 -1
  50. package/miniflare-config-stubs/wrangler.empty.toml +0 -0
  51. package/pages/functions/buildWorker.ts +0 -62
  52. package/pages/functions/filepath-routing.test.ts +0 -39
  53. package/pages/functions/filepath-routing.ts +0 -221
  54. package/pages/functions/identifiers.ts +0 -78
  55. package/pages/functions/routes.ts +0 -158
  56. package/pages/functions/template-worker.ts +0 -144
  57. package/src/__tests__/clipboardy-mock.js +0 -4
  58. package/src/__tests__/dev.test.tsx +0 -66
  59. package/src/__tests__/index.test.ts +0 -287
  60. package/src/__tests__/jest.setup.ts +0 -22
  61. package/src/__tests__/kv.test.ts +0 -1098
  62. package/src/__tests__/mock-cfetch.ts +0 -171
  63. package/src/__tests__/mock-dialogs.ts +0 -65
  64. package/src/__tests__/run-in-tmp.ts +0 -19
  65. package/src/__tests__/run-wrangler.ts +0 -32
  66. package/src/api/form_data.ts +0 -131
  67. package/src/api/preview.ts +0 -128
  68. package/src/api/worker.ts +0 -155
  69. package/src/cfetch/index.ts +0 -102
  70. package/src/cfetch/internal.ts +0 -69
  71. package/src/cli.ts +0 -9
  72. package/src/config.ts +0 -487
  73. package/src/dev.tsx +0 -771
  74. package/src/dialogs.tsx +0 -77
  75. package/src/index.tsx +0 -1974
  76. package/src/inspect.ts +0 -524
  77. package/src/kv.tsx +0 -267
  78. package/src/module-collection.ts +0 -64
  79. package/src/pages.tsx +0 -1031
  80. package/src/proxy.ts +0 -294
  81. package/src/publish.ts +0 -358
  82. package/src/sites.tsx +0 -114
  83. package/src/tail.tsx +0 -73
  84. package/src/user.tsx +0 -1025
  85. package/static-asset-facade.js +0 -47
  86. package/vendor/@cloudflare/kv-asset-handler/CHANGELOG.md +0 -332
  87. package/vendor/@cloudflare/kv-asset-handler/LICENSE_APACHE +0 -176
  88. package/vendor/@cloudflare/kv-asset-handler/LICENSE_MIT +0 -25
  89. package/vendor/@cloudflare/kv-asset-handler/README.md +0 -245
  90. package/vendor/@cloudflare/kv-asset-handler/dist/index.d.ts +0 -32
  91. package/vendor/@cloudflare/kv-asset-handler/dist/index.js +0 -354
  92. package/vendor/@cloudflare/kv-asset-handler/dist/mocks.d.ts +0 -13
  93. package/vendor/@cloudflare/kv-asset-handler/dist/mocks.js +0 -148
  94. package/vendor/@cloudflare/kv-asset-handler/dist/test/getAssetFromKV.d.ts +0 -1
  95. package/vendor/@cloudflare/kv-asset-handler/dist/test/getAssetFromKV.js +0 -436
  96. package/vendor/@cloudflare/kv-asset-handler/dist/test/mapRequestToAsset.d.ts +0 -1
  97. package/vendor/@cloudflare/kv-asset-handler/dist/test/mapRequestToAsset.js +0 -40
  98. package/vendor/@cloudflare/kv-asset-handler/dist/test/serveSinglePageApp.d.ts +0 -1
  99. package/vendor/@cloudflare/kv-asset-handler/dist/test/serveSinglePageApp.js +0 -42
  100. package/vendor/@cloudflare/kv-asset-handler/dist/types.d.ts +0 -26
  101. package/vendor/@cloudflare/kv-asset-handler/dist/types.js +0 -31
  102. package/vendor/@cloudflare/kv-asset-handler/package.json +0 -52
  103. package/vendor/@cloudflare/kv-asset-handler/src/index.ts +0 -296
  104. package/vendor/@cloudflare/kv-asset-handler/src/mocks.ts +0 -136
  105. package/vendor/@cloudflare/kv-asset-handler/src/test/getAssetFromKV.ts +0 -464
  106. package/vendor/@cloudflare/kv-asset-handler/src/test/mapRequestToAsset.ts +0 -33
  107. package/vendor/@cloudflare/kv-asset-handler/src/test/serveSinglePageApp.ts +0 -42
  108. package/vendor/@cloudflare/kv-asset-handler/src/types.ts +0 -39
  109. package/vendor/wrangler-mime/CHANGELOG.md +0 -289
  110. package/vendor/wrangler-mime/LICENSE +0 -21
  111. package/vendor/wrangler-mime/Mime.js +0 -97
  112. package/vendor/wrangler-mime/README.md +0 -187
  113. package/vendor/wrangler-mime/cli.js +0 -46
  114. package/vendor/wrangler-mime/index.js +0 -4
  115. package/vendor/wrangler-mime/lite.js +0 -4
  116. package/vendor/wrangler-mime/package.json +0 -52
  117. package/vendor/wrangler-mime/types/other.js +0 -1
  118. package/vendor/wrangler-mime/types/standard.js +0 -1
  119. 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
- }