Closed Bug 366770 Opened 18 years ago Closed 17 years ago

Use the file's first binding if no fragment identifier is present in a binding URI

Categories

(Core :: XBL, defect)

defect
Not set
normal

Tracking

()

RESOLVED FIXED
mozilla1.9alpha3

People

(Reporter: Gavin, Assigned: Gavin)

References

Details

(Keywords: dev-doc-complete, testcase)

Attachments

(3 files, 2 obsolete files)

Attached file testcase
Attached patch patch (obsolete) — Splinter Review
I added a new method on nsIXBLDocumentInfo called SetFirstPrototypeBinding to let it keep track of the document's first binding. To support data: URIs as binding references, I needed to have nsXBLPrototypeBinding keep it's ID separately from the URI (since data: URIs are not nsIURLs, so we can't use the ref).

I know that both of your review queues are pretty full already, but I don't know of anyone else who can review this. Suggestions welcome :)
Attachment #254551 - Flags: superreview?(jonas)
Attachment #254551 - Flags: review?(bzbarsky)
You could try bryner or hyatt, but I dunno how far that would get you... ;)

How urgent is this?  I'd really like to do enn's thing ASAP, but if you need this very soon I can make time for it.  If not, I might be a week or two, at worst.

Out of curiosity, what's the plan for data: URIs which have a '#' on the end, with this patch?

Also, you needed to save the ref for GetID()? That seems to be used exactly once, in a debug printf.  I say we just nix it in favor of the URI in that printf.  ;)
(In reply to comment #3)
> How urgent is this?  I'd really like to do enn's thing ASAP, but if you need
> this very soon I can make time for it.  If not, I might be a week or two, at
> worst.

It's not urgent at all. Two weeks wait is fine.

> Out of curiosity, what's the plan for data: URIs which have a '#' on the end,
> with this patch?

The "#" at the end is just treated as part of the data, right? nsSimpleURIs aren't nsIURLs, and that's what XBL uses to get the fragment.

> Also, you needed to save the ref for GetID()? That seems to be used exactly
> once, in a debug printf.  I say we just nix it in favor of the URI in that
> printf.  ;)

Ah, good point. I'll do that.
> The "#" at the end is just treated as part of the data, right?

Ok.  Sounds good!
Are there security issues that need to be considered?
I mean, this would allow javascript to be used inside css, javascript which is able to reach the whole document.
Attachment #254551 - Attachment is obsolete: true
Attachment #254551 - Flags: superreview?(jonas)
Attachment #254551 - Flags: review?(bzbarsky)
(In reply to comment #6)
> Are there security issues that need to be considered?
> I mean, this would allow javascript to be used inside css, javascript which is
> able to reach the whole document.

I don't think there are, though I'm not exactly 100% sure. nsXBLService::LoadBindings does security checks before ever calling nsXBLService::GetBinding, so the code I'm changing runs after all of that's been done. In this case I don't think the javascript: URIs are able to reach the document being loaded, in case you're worried of document mutations while loading bindings or something like that. I'm not exactly sure how javascript: URIs are handled in this case, though, so maybe I'm wrong. I suspect CSS url(javascript:...) security is handled at a different level than the code I'm changing (I certainly hope so, anyways).
Martijn, this wouldn't allow anything that's not already allowed -- you can already link to an http:// moz-binding, and you can already have javascript: URIs in CSS (e.g. background-image).
Attached patch patch (obsolete) — Splinter Review
Attachment #254843 - Flags: superreview?(jonas)
Attachment #254843 - Flags: review?(bzbarsky)
Comment on attachment 254843 [details] [diff] [review]
patch

>Index: content/xbl/src/nsXBLBinding.cpp
>-    nsCAutoString id;
>-    mPrototypeBinding->GetID(id);
>-    nsCAutoString message("An XBL Binding with an id of ");
>-    message += id;
>-    message += " and found in the file ";
>+    nsCAutoString message("An XBL Binding with URI ");
>     nsCAutoString uri;
>     mPrototypeBinding->DocURI()->GetSpec(uri);

How about using mPrototypeBinding->BindingURI() instead, so the id isn't lost if there is one?

>Index: content/xbl/src/nsXBLDocumentInfo.h
>+  nsXBLPrototypeBinding* mFirstBinding;

Worth documenting that this is a non-owning pointer and that the mFirstBinding is just owned by the table like all the other bindings.

>Index: content/xbl/src/nsXBLPrototypeBinding.cpp
>-  nsresult rv = NS_NewURI(getter_AddRefs(uri),

This could probably use a comment explaining why it's done this way.

And why not just Clone() directly into mBindingURI?

With those nits fixed, r+sr=bzbarsky
Attachment #254843 - Flags: superreview?(jonas)
Attachment #254843 - Flags: superreview+
Attachment #254843 - Flags: review?(bzbarsky)
Attachment #254843 - Flags: review+
(In reply to comment #10)
> (From update of attachment 254843 [details] [diff] [review])
> >Index: content/xbl/src/nsXBLBinding.cpp
> >-    nsCAutoString id;
> >-    mPrototypeBinding->GetID(id);
> >-    nsCAutoString message("An XBL Binding with an id of ");
> >-    message += id;
> >-    message += " and found in the file ";
> >+    nsCAutoString message("An XBL Binding with URI ");
> >     nsCAutoString uri;
> >     mPrototypeBinding->DocURI()->GetSpec(uri);
> 
> How about using mPrototypeBinding->BindingURI() instead, so the id isn't lost
> if there is one?

Good idea, made that change.

> >Index: content/xbl/src/nsXBLDocumentInfo.h
> >+  nsXBLPrototypeBinding* mFirstBinding;
> 
> Worth documenting that this is a non-owning pointer and that the mFirstBinding
> is just owned by the table like all the other bindings.

Done

> >Index: content/xbl/src/nsXBLPrototypeBinding.cpp
> >-  nsresult rv = NS_NewURI(getter_AddRefs(uri),
> 
> This could probably use a comment explaining why it's done this way.
> 
> And why not just Clone() directly into mBindingURI?

Made both changes, let me know if the comment is OK?
Attachment #254843 - Attachment is obsolete: true
Looks good!
mozilla/content/xbl/Makefile.in 	1.5
mozilla/content/xbl/public/nsIXBLDocumentInfo.h 	1.18
mozilla/content/xbl/src/nsXBLBinding.cpp 	1.226
mozilla/content/xbl/src/nsXBLContentSink.cpp 	1.77
mozilla/content/xbl/src/nsXBLContentSink.h 	1.28
mozilla/content/xbl/src/nsXBLDocumentInfo.cpp 	1.52
mozilla/content/xbl/src/nsXBLDocumentInfo.h 	1.15
mozilla/content/xbl/src/nsXBLPrototypeBinding.cpp 	1.125
mozilla/content/xbl/src/nsXBLPrototypeBinding.h 	1.46
mozilla/content/xbl/src/nsXBLService.cpp 	1.228
mozilla/content/xbl/src/nsXBLService.h 	1.46

I'd like to add a test for this, but I'm not sure how. I can't think of a way to test this with mochitest. A reftest with the testcase in the bug would probably be easiest, but since there are no reftests outside of layout I'm not sure how to proceed. How do the test boxes find the reftests? Is it just hardcoded to look at layout/reftests/reftest.list? Should I just add the test somewhere under layout/reftests, even though it doesn't have anything to do with layout?
Status: NEW → RESOLVED
Closed: 17 years ago
Flags: in-testsuite?
Resolution: --- → FIXED
Target Milestone: --- → mozilla1.9alpha3
> I can't think of a way to test this with mochitest.

Attach a binding via a data: URI.  Have the binding constructor touch some variable in the document scope?
Attached patch testSplinter Review
(In reply to comment #14)
> > I can't think of a way to test this with mochitest.
> 
> Attach a binding via a data: URI.  Have the binding constructor touch some
> variable in the document scope?

Ah, why didn't I think of that? :)
Attachment #256372 - Flags: review?(bzbarsky)
Comment on attachment 256372 [details] [diff] [review]
test

Maybe toss the contents of that data: URI into a comment somewhere so it's easy to tell what it's doing?
Attachment #256372 - Flags: review?(bzbarsky) → review+
Landed the test with the data: URI in a comment.

mozilla/content/xbl/Makefile.in 	1.7
mozilla/content/xbl/test/Makefile.in 	1.1
mozilla/content/xbl/test/test_bug366770.html 	1.1
Flags: in-testsuite? → in-testsuite+
I guess it would be nice if this new feature was mentioned somewhere on mdc.
Keywords: dev-doc-needed
I filed bug 397570 on this not working if the ID attribute is omitted.
Depends on: 397596
Can someone explain to me what this means and why it matters, so I can turn that into documentation?  :)
You don't need to add a fragment identifier in the url referencing the binding, in that case the first binding in the list with bindings in the binding file is used.
It also makes it possible to use data urls for bindings, which is very convenient for making testcases.
I will try. Gavin uses the following XBL binding in his testcase:

<bindings id="xbltestBindings" xmlns="http://www.mozilla.org/xbl">
  <binding id="xbltest"><content>PASS</content></binding>
</bindings>

Usually you would attach it to an element with a CSS rule like this one:

#element {
  -moz-binding: url(http://foo.com/binding.xml#xbltest);
}

But the problem here is that we don't want to have the binding as a separate file - data: URLs are good for specifying content inline. So we want to write:

#element {
  -moz-binding: url(data:text/xml,<binding%20id=%22xbltestBindings%22...);
}

Only that this way the "#xbltest" part will be missing (data: URLs cannot have fragment identifiers) and Gecko will not know which binding to use (more than one can be specified in one file). Until now Gecko would simply ignore such an URL as "invalid". Now it will take the first binding available instead. Note that the binding still needs an id attribute even though it isn't addressed by this ID, this identifier is required internally.

Pretty much every XBL testcase uses this feature already, it makes creating an additional XML file for each testcase unnecessary.
(Martijn and Wladimir mid-aired me and provided their own clear explanations, but I spent time typing this comment up so I figured I'd post it anyways :)

(In reply to comment #20)
> Can someone explain to me what this means and why it matters, so I can turn
> that into documentation?  :)

XBL used to require a fragment identifier to point to a specific binding ID in the XBL document referred to in -moz-binding rules (so "-moz-binding: url(http://foo.com/binding.xml#mybinding)" would point to the binding with ID "mybinding" in binding.xml). We've removed that requirement by having the code use the first binding defined in the XBL document if no fragment identifier is present. This makes it possible to use data: URIs in -moz-binding rules, which was previously impossible because data: URIs don't support fragment identifiers.

The test (attachment 256372 [details] [diff] [review]) shows an example of how this functionality can be used (notice the -moz-binding rule that uses a data: URI). It's mostly useful for people who are creating standalone testcases and want to avoid having to refer to an external file, or having their document include the XBL and self-reference. Bug 243917, and bug 243917 comment 6 in particular, give a bit more context.
Thanks for the awesome explanations!  I've added some information into this section of the XBL reference.  Let me know if there's anything sucky about it:

http://developer.mozilla.org/en/docs/XBL:XBL_1.0_Reference:Elements#binding

Marking this as doc-complete.
OK, now actually setting as doc complete.
This fix for bug 379959 made it so you can't use -moz-binding:url(data:...) from content for security reasons.
Depends on: 432813
Depends on: 444490
Depends on: 507991
You need to log in before you can comment on or make changes to this bug.

Attachment

General

Created:
Updated:
Size: