Skip to content

Commit aca2b85

Browse files
committed
Add tests for interim response handling
1 parent 4ef6cc6 commit aca2b85

File tree

7 files changed

+160
-4
lines changed

7 files changed

+160
-4
lines changed

docker/nginx/nginx.conf

+1
Original file line numberDiff line numberDiff line change
@@ -9,5 +9,6 @@ server {
99
proxy_pass http://localhost:8000;
1010
proxy_cache my-cache;
1111
proxy_cache_revalidate on;
12+
proxy_http_version 1.1;
1213
}
1314
}

package.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,8 @@
1111
"dependencies": {
1212
"liquidjs": "^10.9.2",
1313
"marked": "^15.0.0",
14-
"npm": "^11.0.0"
14+
"npm": "^11.0.0",
15+
"undici": "^7.4.0"
1516
},
1617
"scripts": {
1718
"server": "node test-engine/server/server.mjs",

test-engine/client/test.mjs

+42-2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import * as utils from '../lib/utils.mjs'
44
import * as config from './config.mjs'
55
import * as clientUtils from './utils.mjs'
66
import * as fetching from './fetching.mjs'
7+
import { getGlobalDispatcher } from 'undici'
78
const assert = utils.assert
89
const setupCheck = clientUtils.setupCheck
910

@@ -32,11 +33,19 @@ export async function makeTest (test) {
3233
controller.abort()
3334
}, config.requestTimeout * 1000)
3435
init.signal = controller.signal
36+
37+
const interimResponses = []
38+
if ('expected_interim_responses' in reqConfig) {
39+
const globalDispatcher = getGlobalDispatcher()
40+
const dispatcher = globalDispatcher.compose(clientUtils.interimResponsesCollectingInterceptor(interimResponses))
41+
init.dispatcher = dispatcher
42+
}
43+
3544
if (test.dump === true) clientUtils.logRequest(url, init, reqNum)
3645
return fetch(url, init)
3746
.then(response => {
3847
responses.push(response)
39-
return checkResponse(test, requests, idx, response)
48+
return checkResponse(test, requests, idx, response, interimResponses)
4049
})
4150
.finally(() => {
4251
clearTimeout(timeout)
@@ -84,7 +93,7 @@ export async function makeTest (test) {
8493
})
8594
}
8695

87-
function checkResponse (test, requests, idx, response) {
96+
function checkResponse (test, requests, idx, response, interimResponses) {
8897
const reqNum = idx + 1
8998
const reqConfig = requests[idx]
9099
const resNum = parseInt(response.headers.get('Server-Request-Count'))
@@ -185,6 +194,37 @@ function checkResponse (test, requests, idx, response) {
185194
}
186195
})
187196
}
197+
198+
// check interim responses
199+
if ('expected_interim_responses' in reqConfig) {
200+
const isSetup = setupCheck(reqConfig, 'expected_interim_responses')
201+
202+
reqConfig.expected_interim_responses.forEach(([statusCode, headers = []], idx) => {
203+
if (interimResponses[idx] == null) {
204+
assert(isSetup, false, `Interim response ${idx + 1} not received`)
205+
} else {
206+
assert(isSetup, interimResponses[idx][0] === statusCode, `Interim response ${idx + 1} status is ${interimResponses[idx][0]}, not ${statusCode}`)
207+
208+
const receivedHeaders = interimResponses[idx][1]
209+
headers.forEach(([header, value]) => {
210+
if (typeof header === 'string') {
211+
assert(isSetup, header in receivedHeaders,
212+
`Interim response ${idx + 1} ${header} header not present.`)
213+
} else if (header.length > 2) {
214+
assert(isSetup, header in receivedHeaders,
215+
`Interim response ${idx + 1} ${header} header not present.`)
216+
217+
const receivedValue = receivedHeaders[header]
218+
assert(isSetup, value === receivedValue,
219+
`Interim response ${idx + 1} header ${header} is ${receivedValue}, should ${value}`)
220+
} else {
221+
console.log('ERROR: Unknown header item in expected_interim_responses', header)
222+
}
223+
})
224+
}
225+
})
226+
}
227+
188228
return response.text().then(makeCheckResponseBody(test, reqConfig, response.status))
189229
}
190230

test-engine/client/utils.mjs

+43
Original file line numberDiff line numberDiff line change
@@ -80,3 +80,46 @@ export function logResponse (response, reqNum) {
8080
})
8181
console.log('')
8282
}
83+
84+
class InterimResponsesCollectingHandler {
85+
#handler
86+
#interimResponses
87+
88+
constructor (handler, interimResponses) {
89+
this.#handler = handler
90+
this.#interimResponses = interimResponses
91+
}
92+
93+
onRequestStart (controller, context) {
94+
this.#handler.onRequestStart?.(controller, context)
95+
}
96+
97+
onRequestUpgrade (controller, statusCode, headers, socket) {
98+
this.#handler.onRequestUpgrade?.(controller, statusCode, headers, socket)
99+
}
100+
101+
onResponseStart (controller, statusCode, headers, statusMessage) {
102+
if (statusCode < 200) this.#interimResponses.push([statusCode, headers])
103+
this.#handler.onResponseStart?.(controller, statusCode, headers, statusMessage)
104+
}
105+
106+
onResponseData (controller, data) {
107+
this.#handler.onResponseData?.(controller, data)
108+
}
109+
110+
onResponseEnd (controller, trailers) {
111+
this.#handler.onResponseEnd?.(controller, trailers)
112+
}
113+
114+
onResponseError (controller, err) {
115+
this.#handler.onResponseError?.(controller, err)
116+
}
117+
}
118+
119+
export function interimResponsesCollectingInterceptor (collectInto) {
120+
return (dispatch) => {
121+
return (opts, handler) => {
122+
return dispatch(opts, new InterimResponsesCollectingHandler(handler, collectInto))
123+
}
124+
}
125+
}

test-engine/server/handle-test.mjs

+11
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,17 @@ function continueHandleTest (uuid, request, response, requests, serverState) {
4343
const previousConfig = requests[reqNum - 2]
4444
const now = Date.now()
4545

46+
const interimResponses = reqConfig.interim_responses || []
47+
for (const [status, headers = []] of interimResponses) {
48+
if (status === 102) {
49+
response.writeProcessing()
50+
} else if (status === 103) {
51+
response.writeEarlyHints(Object.fromEntries(headers))
52+
} else {
53+
console.log(`ERROR: Sending ${status} is not yet supported`)
54+
}
55+
}
56+
4657
// Determine what the response status should be
4758
let httpStatus = reqConfig.response_status || [200, 'OK']
4859
if ('expected_type' in reqConfig && reqConfig.expected_type.endsWith('validated')) {

tests/index.mjs

+2-1
Original file line numberDiff line numberDiff line change
@@ -22,5 +22,6 @@ import partial from './partial.mjs'
2222
import auth from './authorization.mjs'
2323
import other from './other.mjs'
2424
import cdncc from './cdn-cache-control.mjs'
25+
import interim from './interim.mjs'
2526

26-
export default [ccFreshness, ccParse, ageParse, expires, expiresParse, ccResponse, stale, heuristic, methods, statuses, ccRequest, pragma, vary, varyParse, conditionalLm, conditionalEtag, headers, update304, updateHead, invalidation, partial, auth, other, cdncc]
27+
export default [ccFreshness, ccParse, ageParse, expires, expiresParse, ccResponse, stale, heuristic, methods, statuses, ccRequest, pragma, vary, varyParse, conditionalLm, conditionalEtag, headers, update304, updateHead, invalidation, partial, auth, other, cdncc, interim]

tests/interim.mjs

+59
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
export default
2+
3+
{
4+
name: 'Interim Response Handling',
5+
id: 'interim',
6+
description: 'These tests check how caches handle interim responses.',
7+
tests: [
8+
{
9+
name: 'An optimal HTTP cache passes a 102 response through and caches the final response',
10+
id: 'interim-102',
11+
browser_skip: true, // Fetch API in browsers don't expose interim responses
12+
kind: 'optimal',
13+
requests: [
14+
{
15+
interim_responses: [[102]],
16+
expected_interim_responses: [[102]],
17+
response_headers: [
18+
['Cache-Control', 'max-age=100000'],
19+
['Date', 0]
20+
],
21+
pause_after: true
22+
},
23+
{
24+
expected_type: 'cached'
25+
}
26+
]
27+
},
28+
{
29+
name: 'An optimal HTTP cache passes a 103 response through and caches the final response',
30+
id: 'interim-103',
31+
browser_skip: true,
32+
kind: 'optimal',
33+
requests: [
34+
{
35+
interim_responses: [
36+
[103, [
37+
['link', '</styles.css>; rel=preload; as=style'],
38+
['x-my-header', 'test']
39+
]]
40+
],
41+
expected_interim_responses: [
42+
[103, [
43+
['link', '</styles.css>; rel=preload; as=style'],
44+
['x-my-header', 'test']
45+
]]
46+
],
47+
response_headers: [
48+
['Cache-Control', 'max-age=100000'],
49+
['Date', 0]
50+
],
51+
pause_after: true
52+
},
53+
{
54+
expected_type: 'cached'
55+
}
56+
]
57+
}
58+
]
59+
}

0 commit comments

Comments
 (0)