<!doctype html>
<html>
<head>
<title>XMLHttpRequest: progress events and GZIP encoding</title>
<meta name="timeout" content="long">
<script src="/resources/testharness.js"></script>
<script src="/resources/testharnessreport.js"></script>
<link rel="help" href="https://xhr.spec.whatwg.org/#firing-events-using-the-progressevent-interface-for-http" data-tested-assertations="following::p[contains(text(),'content-encodings')]" />
<!-- TODO: find better spec reference when https://www.w3.org/Bugs/Public/show_bug.cgi?id=25587 is fixed -->
</head>
<body>
<div id="log"></div>
<script>
var test = async_test()
test.step(function() {
var client = new XMLHttpRequest()
/*
Two behaviours are considered acceptable, so there are two ways to
pass this test
a) Set data for the compressed resource:
* event.total reflects the Content-length of the gzipp'ed resource
* event.loaded how many gzipped bytes have arrived over the wire so far
* lengthComputable is true
or
b) If the implementation does not provide progress details for the compressed
resource, set
* lengthComputable to false
* event.total to 0
* event.loaded to the number of bytes available so far after gzip decoding
Implications of this are tested here as follows:
* If lengthComputable is true:
* Event.total must match Content-length header
* event.loaded must only ever increase in progress events
(and may never repeat its value).
* event.loaded must never exceed the Content-length.
* If lengthComputable is false:
* event.total should be 0
* event.loaded must only ever increase in progress events
(and may never repeat its value).
* event.loaded should be the length of the decompressed content, i.e.
bigger than Content-length header value when finished loading
*/
var lastTotal;
var lastLoaded = -1;
client.addEventListener('loadend', test.step_func(function(e){
var len = parseInt(client.getResponseHeader('content-length'), 10)
if(e.lengthComputable){
assert_equals(e.total, len, 'event.total is content-length')
assert_equals(e.loaded, len, 'event.loaded should be content-length at loadend')
}else{
assert_equals(e.total, 0, 'if implementation can\'t compute event.total for gzipped content it is 0')
assert_true(e.loaded >= len, 'event.loaded should be set even if total is not computable')
}
test.done();
}), false)
client.addEventListener('progress', test.step_func(function(e){
if(lastTotal === undefined){
lastTotal = e.total;
}
if(e.lengthComputable && e.total && e.loaded){
assert_equals(e.total, lastTotal, 'event.total should remain invariant')
assert_less_than_equal(e.loaded, lastTotal, 'event.loaded should not exceed content-length')
}else{
assert_equals(e.total, 0, 'event.total should be 0')
}
assert_greater_than(e.loaded, lastLoaded, 'event.loaded should only ever increase')
lastLoaded = e.loaded;
}), false)
// image.gif is 165375 bytes compressed. Sending 45000 bytes at a time with 1 second delay will load it in 4 seconds
client.open("GET", "resources/image.gif?pipe=gzip|trickle(45000:d1:r2)", true)
client.send()
})
</script>
</body>
</html>