Fix XMLToXMLString to cope with a tag name in a default namespace that is not the declared default namespace for the tag (277779, r=shaver).

This commit is contained in:
brendan%mozilla.org 2005-01-20 01:02:49 +00:00
parent 4d147dcc76
commit af45dbe52f

View File

@ -2431,6 +2431,17 @@ GeneratePrefix(JSContext *cx, JSString *uri, JSXMLArray *decls)
return prefix;
}
static JSBool
namespace_match(const void *a, const void *b)
{
const JSXMLNamespace *nsa = (const JSXMLNamespace *) a;
const JSXMLNamespace *nsb = (const JSXMLNamespace *) b;
if (nsb->prefix)
return nsa->prefix && !js_CompareStrings(nsa->prefix, nsb->prefix);
return !js_CompareStrings(nsa->uri, nsb->uri);
}
/* ECMA-357 10.2.1 and 10.2.2 */
static JSString *
XMLToXMLString(JSContext *cx, JSXML *xml, const JSXMLArray *ancestorNSes,
@ -2616,7 +2627,7 @@ XMLToXMLString(JSContext *cx, JSXML *xml, const JSXMLArray *ancestorNSes,
* Assign the new prefix to a copy of ns. Flag this namespace as if
* it were declared, for assertion-testing's sake later below.
*
* Erratum: ns->prefix and xml->name are both null (*undefined* in
* Erratum: if ns->prefix and xml->name are both null (*undefined* in
* ECMA-357), we know that xml was named using the default namespace
* (proof: see GetNamespace and the Namespace constructor called with
* two arguments). So we ought not generate a new prefix here, when
@ -2636,6 +2647,25 @@ XMLToXMLString(JSContext *cx, JSXML *xml, const JSXMLArray *ancestorNSes,
if (!ns)
goto out;
/*
* If the xml->name was unprefixed, we must remove any declared default
* namespace from decls before appending ns. How can you get a default
* namespace in decls that doesn't match the one from name? Apparently
* by calling x.setNamespace(ns) where ns has no prefix. The other way
* to fix this is to update x's in-scope namespaces when setNamespace
* is called, but that's not specified by ECMA-357.
*
* Likely Erratum here, depending on whether the lack of update to x's
* in-scope namespace in XML.prototype.setNamespace (13.4.4.36) is an
* erratum or not. Note that changing setNamespace to update the list
* of in-scope namespaces will change x.namespaceDeclarations().
*/
if (IS_EMPTY(prefix)) {
i = XMLArrayFindMember(&decls, ns, namespace_match);
if (i != XML_NOT_FOUND)
XMLArrayDelete(cx, &decls, i, JS_TRUE);
}
/*
* In the spec, ancdecls has no name, but is always written out as
* (AncestorNamespaces U namespaceDeclarations). Since we compute
@ -5942,17 +5972,6 @@ xml_namespace(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
return JS_TRUE;
}
static JSBool
namespace_match(const void *a, const void *b)
{
const JSXMLNamespace *nsa = (const JSXMLNamespace *) a;
const JSXMLNamespace *nsb = (const JSXMLNamespace *) b;
if (nsb->prefix)
return nsa->prefix && !js_CompareStrings(nsa->prefix, nsb->prefix);
return !js_CompareStrings(nsa->uri, nsb->uri);
}
static JSBool
xml_namespaceDeclarations(JSContext *cx, JSObject *obj, uintN argc, jsval *argv,
jsval *rval)