-
Notifications
You must be signed in to change notification settings - Fork 341
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
Deferred fetching #1647
base: main
Are you sure you want to change the base?
Deferred fetching #1647
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
- "inactive timeout" -> "deferred delay". At least that seems clearer to me.
- I think
fetchLater
was quite nice in that it sorts the same asfetch
.fetchDeferred
could also work, though is a bit harder to spell. I don't think we need request in the name. - It's not clear how the name fetch group states get activated. That needs some kind of additional PR against HTML I suppose?
- I think we should describe the API in the same section as the fetch method. Could be called "Fetch methods" then.
- Deferred fetching itself could then precede the "Fetch API" section. Maybe it could even be a subsection of "Fetching" though I don't mind a new top-level section.
But this is deferred delay specifically in the case of inactivity. "inactivity deferred delay"?
I was thinking about "what are we doing right now?" which is requesting/scheduling a deferred fetch for later. But
Yes, HTML would activate/deactivate in the BFCache code path. Need to prepare a PR for that but wanted to see that I'm on the right track first.
Will do
OK |
cc @mingyc @fergald @yoavweiss @clelland latest API shape proposal for PendingBeacon |
Another note about "origin" of a beacon request: there were some previous discussion about using 3P storage partitioning key (not origin, which is stricter) to decide whether pending beacon requests in a page are sendable or not in terms of privacy concern, see WICG/pending-beacon#30 (comment) and comments there below. I am not sure how this should be spec. |
OK, perhaps the 64kb constraint can be per network partition key rather than origin. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
PTAL, added some more comments.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
noamr@ PTAL. I've added some more questions and comments. Really thanks for your help!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you noamr@! I've added some last comments. And also added the missing features from the original proposal belows. Please let us know if they are suitable here.
- It is suggested to allow only secure requests for new API in Do we want to enforce HTTPS request? WICG/pending-beacon#27. Should we enforce HTTPS-only requests on
fetchLater()
? - Should this spec mention [Permission Policy]https://www.w3.org/TR/permissions-policy/? In [Fetch-Based API] Permissions Policy WICG/pending-beacon#77, the suggestion is to allow the API by default. But we might want to provide a way to manage 3rd party iframe's usage.
- Consider to support retry mechanism WICG/pending-beacon#40 Should this spec mention retry when
fetchLater()
fails to send/commit? - The original PendingBeacon proposal also includes Crash recovery WICG/pending-beacon#34, not sure how it can be incorporated into fetch spec.
Probably a good idea, from the point of view of enabling new features only for secure requests.
I don't think we should integrate with permission policy. But we should allow the user agent to deny a
Perhaps consider adding this later?
I don't think that changes anything in the spec. |
@noamr Following up on the sendable beacon discussion: As mentioned in WICG/pending-beacon#30 (comment), there were discussions around whether a beacon (or deferred request) should be sent when network changes. I tried to summarize them in [this PR](WICG/pending-beacon@feb3cf9, but basically to process a beacon request when BackgroundSync is off, we need to see if another open document (tab/frame/etc) with the same storage partitioning key as the current document's one, to avoid unexpected sending the request after network changes. Do you think the above makes sense to be integrated into Fetch spec? |
fetch.bs
Outdated
<p>To <dfn export>reserve deferred-fetch quota</dfn> for a <a>navigable container</a> | ||
<var>container</var> given an <a for=/>origin</a> <var>originToNavigateTo</var>: | ||
|
||
<p class=note>This is called when <var>container</var> and the document that initiated the |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is called when container and the document that initiated the navigation (the "source document") are same origin.
It potentially reserves either 64kb or 8kb of quota for the frame, if it is not same origin with its parent and the permissions policy allow.
Could you please verify the triggering condition? The first sentence mentions same-origin only, while the second sentence still brings up if it is not same origin with its parent
.
Even if the "not same origin" condition only applies to the 8kb case, when should it be triggered? Is it when different-origin container gets created or starts to navigate? But a different-origin container will get its initial policy copied from parent container, which it should not modifiy?
Given the following example
<!-- a.com/index.html -->
let iframe = document.createElement("iframe");
iframe.src = "b.com";
document.body.appendChild(iframe)
<!-- b.com/index.html -->
<script>
fetchLater("b.com");
</script>
- When
iframe
is created, it does not have any parent container. - When
iframe.src
is set tob.com
, it navigates to a cross-origin destination, with FramePolicy copied from parent document (?) - When should
iframe
triggers thereserve deferred policy quota
algorithm?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sorry, the informative note was a bit misleading.
This is called when the iframe is being navigated, provided that the sourceDocument
of the navigation is the container document.
In the above case, navigation would only take place when the iframe is connected to the page. This happens here, in the iframe's post-connection steps.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is called when the iframe is being navigated,
It might still be ambigous. This algorithm later relies on inherited policies, which is also only available on navigation.
Combined with my question in this commit, could you please clarify?
provided that the sourceDocument of the navigation is the container document.
What's the container document
here, is it iframe
or the main document
?
To reserve deferred-fetch quota for a navigable container container given an origin originToNavigateTo:
Trying to summarize the triggering condition and place. This algorithm:
- Is triggered by a navigable container
- Given the above case,
navigable container
=iframe
?
- Given the above case,
Should be triggered on navigation, when the source document of the navigation is the navigable's parent document.
- Is
source document of the navigation
=iframe's parent document
=main document
? - Is
navigable
=iframe
? - Is
navigable's parent document
=main document
?
- Is
- Combined the above, this algorithm should be run by
iframe
when its source document is main document and when it is navigating its content document?- But
inherited policies
of the content document will only be available during navigation... - If this is run by
content document
after it obtainsinherited policies
,content document
might not be able to get access toiframe
if it is cross-origin.
- But
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is called when the iframe is being navigated,
It might still be ambigous. This algorithm later relies on inherited policies, which is also only available on navigation.
Combined with my question in this commit, could you please clarify?
provided that the sourceDocument of the navigation is the container document.
What's the
container document
here, is itiframe
or themain document
?
Main document
To reserve deferred-fetch quota for a navigable container container given an origin originToNavigateTo:
Trying to summarize the triggering condition and place. This algorithm:
Is triggered by a navigable container
- Given the above case,
navigable container
=iframe
?
Yes, like the iframe element.
Should be triggered on navigation, when the source document of the navigation is the navigable's parent document.
- Is
source document of the navigation
=iframe's parent document
=main document
?
Yes, when the iframe is being navigated by its container, e.g. by setting the iframe's src.
- Is
navigable
=iframe
?
Sort of, It's like the iframe's contents.
- Is
navigable's parent document
=main document
?
Yes, in this case
- Combined the above, this algorithm should be run by
iframe
when its source document is main document and when it is navigating its content document?
Yes
- But
inherited policies
of the content document will only be available during navigation...
Not exactly. It should run the https://w3c.github.io/webappsec-permissions-policy/#algo-define-inherited-policy-in-container algorithm which only needs the container and the target origin.
- If this is run by
content document
after it obtainsinherited policies
,content document
might not be able to get access toiframe
if it is cross-origin.
See previous response
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for answering!
Getting back to the above example where a cross-origin iframe
is created first, and gets appended later.
- I tried
potentially free deferred-fetch quota
algorithm, which gets executed on Document creation. It looks like when running with the example,iframe
will clear its navigable container (which isiframe
?)deferred-fetch quota
because
if document’s node navigable’s container document is not null, and its origin is same origin with document,
SameOrigin(iframe, main document) = SameOrigin("about:blank", "a.com") = true
-
Later on beginning navigation to
b.com
,reserve deferred-fetch quota
is called,iframe
would theoretically getdeferred-fetch-minimal
=> is this correct? -
However, when calling
get the available deferred-fetch quota
for any requests made withiniframe
, it will always return 0 from a condition not listed in Step 5:
isTopLevel == false && deferredFetchAllowed == false && deferredFetchMinimalAllowed == false
(also asked here)
I suspects the policy doesn't get properly passed in-between process, i.e. the policy updated at (2) does not propagate to remote frame. Maybe this is not spec issue but I am not sure...
|
||
<li><p>If <var>quota</var> is equal or less than 0, then return 0. | ||
|
||
<li><p>If <var>quota</var> is less than <var>quotaForRequestOrigin</var>, then return |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If quota is less than quotaForOrigin, then return quota.
(Discussed offline in chat, but copying here)
Given the following examples:
// root (a.com) -> frame-1 (a.com) -> frame-2 (b.com)
// -> frame-3 (b.com) -> frame-4 (a.com)
I'd like to verify quotas for the above 5 documents:
root: 64kb? (the result I got, but isn't 8kb taken by frame-2?)
frame-1: 54kb
frame-2: 8kb
frame-3: 8kb
frame-4: 0kb?
The questionable part is root
and frame-1
, they can't get more than 64kb quota due to the quoted return step at the beginning.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Right, for a specific request they can never get more than 64kb at this point, even though their overall quota would be 512kb.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should frame-1
get 64kb
or 54kb
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
64kb
delegate quota to the navigated frame, and the reserved quota would only apply in that case, and | ||
would be ignored if it ends up being shared. If quota was reserved and the document ends up being | ||
<a>same origin</a> with its parent, the quota would be | ||
<a data-lt="potentially free deferred-fetch quota">freed</a>. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If quota was reserved and the document ends up being
same origin with its parent, the quota would be
freed
Could the triggering condition of free
be more explicit? Does the algorithm looks like the following?
- container document (iframe) navigates
- container document inherits policies
- container document reserves 64kb or 8kb
- document (iframe's content document) gets created
- document frees quota
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It will be explicit in the HTML spec. I'll prepare a PR.
What you wrote is correct.
set <var>document</var>'s <a>node navigable</a>'s <a>navigable container</a>'s | ||
<a>reserved deferred-fetch quota</a> to 0. | ||
|
||
<p class=note>This is called when a {{Document}} is created. It ensures that same-origin nested |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is called when a Document is created. It ensures that same-origin nested documents don’t reserve quota,
Is it true that cross-origin document will not have access to its navigable container's quota? If so, this algorithm automatically only works on same-origin content document?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Correct
The logic for deferred fetching (the `fetchLater` function), as defined in the fetch spec, specifies a "quota" which is shared with between a document and its direct same-origin descendants. For this logic to work in a secure way, the quota needs to be: - reserved when a frame-initiated navigation starts. This way, the container document can only reserve quota based on URLs it knows it navigates to. - freed if the document ends up being same origin with its container, upon document creation. This ensures quota is handled correctly in the case of redirects. This PR adds those two calls: - Call "reserve" on navigation, based on `sourceDocument`. - Call "potentially free" on document creation. Depends on whatwg/fetch#1647, where the quota logic itself is defined.
would be ignored if it ends up being shared. If quota was reserved and the document ends up being | ||
<a>same origin</a> with its parent, the quota would be | ||
the container and its navigable, if allowed permissions policy. It is not observable to the | ||
cotnainer document whether the reserved quota was used in practice. This algorithm assumes that the |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
cotnainer
typo
<a>same origin</a> with its parent, the quota would be | ||
the container and its navigable, if allowed permissions policy. It is not observable to the | ||
cotnainer document whether the reserved quota was used in practice. This algorithm assumes that the | ||
container's document might delegate quota to the navigated frame, and the reserved quota would only |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
might delegate quota to the navigated frame
what's frame
here? Is it container's navigable?
<a>allowed to use</a> the <a>policy-controlled feature</a> "{{PermissionsPolicy/deferred-fetch}}"; | ||
otherwise false. | ||
|
||
<li><p>Let <var>deferredFetchMinimalAllowed</var> be true if <var>controlDocument</var> is |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
deferredFetchMinimalAllowed is true
Is it possible to bypass all these 5 conditions without a value for quota
, i.e. isTopLevel == false
&& deferredFetchAllowed == false
&& deferredFetchMinimalAllowed == false
?
Add a JS-exposed function to request a deferred fetch, called
fetchLater
.A deferred fetch would be invoked in one of two scenarios:
A few constraints:
The quota algorithm is a bit intricate, but its default should be somewhat reasonable for all but advanced cases.
fetchLater
in sequence.deferred-fetch-minimal
) controls that, and the top-level document can disable that allocated quota by disabling that permissions policy.deferred-fetch
permissions policy.src
on an iframe. It is not guaranteed that the subframe would actually be able to use thatquota, as it might end up navigating to a same-origin URL or disable the feature in its own permissions policy. However, the container's document only cares about the initial reserved value for subframes it doesn't have direct access to.
See WICG/pending-beacon#70
(See WHATWG Working Mode: Changes for more details.)
Preview | Diff