3

I need to create an XML xPath parser. All parsing has to happen on client side (using javascript). I created an javascript that does this, and everything looks OK until default namespaces come into play. I simply can't query XML that has default namespace.

I created an example code on fiddle. In xmlString is XML string received from server. In xPathString is query done on received XML.

Here are some scenarios:

  1. http://jsfiddle.net/BF34q/1/ - no namespaces - everything works OK
  2. http://jsfiddle.net/BF34q/2/ - ns namespace added. element has ns: prefix. xPath uses this prefix - OK
  3. http://jsfiddle.net/BF34q/3/ - default namespace used - not sure how to configure xPathString.

Note that others will use this parser, so I would really like to avoid solutions like

var xPathString = "//*[local-name()='book']";

and enable them to parse it using simple xPath expressions. I wonder if it is possible to assign default namespace prefix in javascript?

Note: The example provided on fiddle will not work in IE.

dugokontov
  • 4,402
  • 1
  • 25
  • 25

2 Answers2

9

I think there are three ways to do this:

  1. Use //*[local-name()='book'] syntax for accessing nodes
  2. Convert XML to string, remove default namespace using RegExp, convert it back to XML
  3. For XML files where you know namespaces in advance, you can create your own namespace resolver, which will allow you to use your own prefix for default namespace.

This can be achieved like this:

function nsResolver(prefix) {
    switch (prefix) {
        case 'xhtml':
            return 'http://www.w3.org/1999/xhtml';
        case 'mathml':
            return 'http://www.w3.org/1998/Math/MathML';
        default:
            return 'http://example.com/domain';
    }
}
xml.evaluate('//myPrefix:book', xml, nsResolver, XPathResult.ANY_TYPE, null);
dugokontov
  • 4,402
  • 1
  • 25
  • 25
0

I've got the impression that your understanding of the XPath processing doesn't match the implementation - unless, that is, the implementation you're dealing with is very different from the ones I'm familiar with.

Usually, the XPath processor has to have namespaces registered and prefixes mapped to them in order for the expression to be successfully evaluated. So the prefixes could be anything - the only thing that matters is what they're mapped to. See this answer by a known expert to get more information.

Community
  • 1
  • 1
Lumi
  • 14,775
  • 8
  • 59
  • 92
  • You can omit prefix for namespace, and then that namespace is considered default namespace. Here are some examples and clarifications: http://www.w3schools.com/xml/xml_namespaces.asp, http://www.developerfusion.com/article/3720/understanding-xml-namespaces/5/ – dugokontov Jul 21 '11 at 06:51
  • @dugokontov, I think you're confusing **no namespace** and **default namespace**. In an XML doc, you set the default namespace using `xmlns="something"`, so without a prefix. By default, the default namespace **is** the empty namespace (no namespace). The default namespace depends on the context; XPath makes no exception here. By not specifying any prefix with his XPath expressions and not registering any namespaces, the OP's expression very likely matches nodes in no namespace. – Lumi Jul 21 '11 at 08:26
  • you just sad the same thing as I did in my comment: _In an XML doc, you set the default namespace using xmlns="something", so without a prefix_ == _You can omit prefix for namespace, and then that namespace is considered default namespace_. Also, your scenario in your last sentence is described in my fist example provided: http://jsfiddle.net/BF34q/1/. So, I'm little confused what do you want to say to me? – dugokontov Jul 21 '11 at 09:48
  • @dugokontov, I can see why you're confused :-) I must have misread your statement, sorry for that. What I want to say to you is that you need to register the namespace `http://example.com/domain` with your XPath processor. Just map it to some prefix `x` and then use that prefix with your query. It'll work using libraries such as LibXML2. Not sure what the incantation for the processor that's embedded in the browser should be, though. Maybe try using the [Saxon library](http://markmail.org/message/ebhyr5hxn5iphlwc). – Lumi Jul 21 '11 at 12:57
  • don't get me wrong, but I have feeling that you haven't read my question at all. I know I have to assign some prefix to default namespace, and my question was _I wonder if it is possible to assign default namespace prefix in javascript?_. Plz read before answering... – dugokontov Jul 25 '11 at 10:57