-
Notifications
You must be signed in to change notification settings - Fork 315
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Clarify if Cache put()/add()/addAll() should mark Request body used #550
Comments
In F2F discussion, @slightlyoff says these operations should indeed consume the body. |
Will add "If request's body's length is not zero, set request's body's used flag." to those operations. |
If the Request has only gotten the headers, it may not reliably know the length of the body. @annevk can we determine at header resolution time if there is a body or not? If not, I would say just always mark the body used. |
|
Fetch can auto-clone since the redirect process is not exposed to content. It does not create any "magical" clones as far as content is concerned because the multiple requests are not directly exposed. If someone wants to use Cache.put() and then reuse the same Request for a new fetch, they need to manually clone() it. Note, you can use the Request in Cache.match() without draining the body because Cache doesn't need to look at it. |
Should put() reject if passed a Request that has an already used body? |
Based on the discussion so far, I believe so. |
Blink issue: http://crbug.com/430962 |
@inexorabletash @jakearchibald @annevk What do you think about Request objects that conclusively do not have a body? For example:
Should this set the body flag used or not? On the one hand, it would be nice to be consistent. On the other hand, I feel like this is just making this unnecessarily hard for the developer. |
IMO the bodyUsed flag should be set when the body is actually read from (you know, used). So if there is no body to read from it should never be set. Some earlier comment of mine explained how I would implement bodyUsed via the streams API and is consistent with this idea. |
Thanks @domenic. Sounds good to me. Also, just to clarify. This all goes for the Response object as well. I believe Cache.put() should check for Response.bodyUsed and set it true if it reads a Response body. |
I think this might need a fetch spec tweak. The default "no stream" value in the Body mixin is: "Let stream be an empty byte stream. " And there is nothing in the spec about not marking it consumed, as far as I can tell. Actually, the fetch spec even has:
@annevk, what do you think? Can we have a way to detect an empty/null stream without set the body used flag? |
So what kind of semantics do we want? Should the convenience methods reject if stream is null? |
It seems there are a few possibilities:
I think I like (1) or (3). It seems (3) would have the least impact on developers, but seem a bit magical. What do other people think? |
I think we should retain the "feature" that only one of these methods works and that it only works once. 3) doesn't do that. It kind of makes sense to me that reading from null results in an exception, but it's also rather cumbersome API-wise. Another internal flag could solve both, but is also not really elegant. I have no good ideas. |
Do we want developers to be aware of the semantic difference between "no body" and "empty body"? I think that such a distinction doesn't exist on the HTTP level, so maybe it should not exist on the conceptual level? |
I think it does exist on the HTTP level. It's the difference between not having |
Or
Not sure if I like that. |
No. We'll have a |
Well, I would be fine with just fixing this in the SW spec for Cache. Something like "determine if Request/Response body exists and mark used if true". Of course, this would leave content with the same problem of always marking the body used in order to determine if the body is empty, though. |
See w3c/ServiceWorker#550 The reason we do not do this for `text()` et al is because they are convenience methods and we do not want to end up with a situation where you can invoke them multiple times, but only if body is null.
Taking the basic cache-as-you-go-along code: self.onfetch = function(event) {
event.respondWith(
caches.match(event.request).then(function(response) {
return response || fetch(event.request.clone()).then(function(response) {
caches.open('whatever').then(function(cache) {
cache.put(event.request, response);
});
return response.clone();
});
})
);
}; If you're telling me (as a developer) that |
To be honest, when we were discussing explicit clone semantics before I was mainly thinking of the Responses (value objects in the cache). Of course now I see it effects the Requests (keys in the cache). A map-like API with one-time-use keys seems pretty clunky to me from a developer standpoint. :-( |
That code fails if request has a body, unless |
|
cache.match() does not touch the body. cache.put() only touches the body because it needs to reproduce it in cache.keys(). Maybe we can avoid having to reproduce request bodies, so we don't have to look at the request body at all? |
When you do
I came to a similar conclusion, "ignore request body when storing in the cache". That would be great now, but I think it'd come back to haunt us if we started allowing |
Eg, if we allowed self.onfetch = function(event) {
if (sendingAnEmail) {
event.respondWith(
fetch(event.request.clone()).catch(e => undefined).then(function(response) {
if (response && response.status == 200) {
return response;
}
// ok, email send looks like it failed
return caches.open('outbox')
.then(c => c.put(event.request, new Response('')))
.then(_ => new Repsonse("Safely in outbox"))
})
);
}
}; It's a bit clunky using the cache for this, probably should be idb, but don't think we should screw the possibility of the above. |
Yea, this wasn't really my concern. I think what I'm trying to say is that this:
Makes me think developers would want implicit clone() for fetch()/cache.put(). If the API essentially requires them to do something every time for safety, the API should just do it. |
And yes, I'm sorry for arguing for explicit clone here before. I see now, though, that it doesn't really buy us much in terms of memory improvement if developers are going to call clone() all the time anyway. :-( |
Summary of an IRC chat with @wanderview: The problem case is still piping a large response (HD video) to two places, most likely the browser and the cache. If a clone is sent to the browser, and the main response to the cache, we only get buffering if the browser/cache consumes slower than the other, and that should be small. In an auto-clone world, we'll still have the ability (while the credits are rolling) to read the response from the start, which means keeping the whole movie around. Although (as @annevk points out) the massive-cache may need a new feature that allows the SW to go away during the download and wake up to fire an event once succeeded or failed. An uncomfortable truth is that However, it's pretty easy for us to go from a manual-clone model to an auto-clone model in terms of API. |
Yes, @jakearchibald talked me off the auto-clone ledge. I think we should stick with explicit clone(). Sorry for my confusion/waffling here. Going back to the original issue, though. I still think we should avoid checking/marking bodyUsed if there is no body. We need something in the SW spec that says that. |
http://crbug.com/430962 Merged to blink's MVP release (M40) |
Addressed those points in put() and addAll(): c800aa4. |
Closing. |
Please clarify how the Cache should treat the Request body for the put()/add()/addAll() methods.
My interpretation is that the Cache must store the Request body in order to reproduce the Request in the keys() method. The Fetch spec does not provide a way to retrieve the body from the Request without marking the body used. So my position is that the current spec requires put()/add()/addAll() to mark the Request body used.
Also, I believe the discussion in #372 came down on the side of avoiding magical implicit clones in favor of explicit clones.
cc / @jakearchibald @inexorabletash
The text was updated successfully, but these errors were encountered: