/* * The Apache Software License, Version 1.1 * * * Copyright (c) 1999 The Apache Software Foundation. All rights * reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * 3. The end-user documentation included with the redistribution, * if any, must include the following acknowledgment: * "This product includes software developed by the * Apache Software Foundation (http://www.apache.org/)." * Alternately, this acknowledgment may appear in the software itself, * if and wherever such third-party acknowledgments normally appear. * * 4. The names "Xalan" and "Apache Software Foundation" must * not be used to endorse or promote products derived from this * software without prior written permission. For written * permission, please contact apache@apache.org. * * 5. Products derived from this software may not be called "Apache", * nor may "Apache" appear in their name, without prior written * permission of the Apache Software Foundation. * * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * ==================================================================== * * This software consists of voluntary contributions made by many * individuals on behalf of the Apache Software Foundation and was * originally based on software copyright (c) 1999, Lotus * Development Corporation., http://www.lotus.com. For more * information on the Apache Software Foundation, please see * <http://www.apache.org/>. */ package org.apache.xml.utils; import java.util.Stack; import java.util.StringTokenizer; import org.w3c.dom.Element; import org.apache.xpath.res.XPATHErrorResources; import org.apache.xalan.res.XSLMessages; /** * <meta name="usage" content="general"/> * Class to represent a qualified name: "The name of an internal XSLT object, * specifically a named template (see [7 Named Templates]), a mode (see [6.7 Modes]), * an attribute set (see [8.1.4 Named Attribute Sets]), a key (see [14.2 Keys]), * a locale (see [14.3 Number Formatting]), a variable or a parameter (see * [12 Variables and Parameters]) is specified as a QName. If it has a prefix, * then the prefix is expanded into a URI reference using the namespace declarations * in effect on the attribute in which the name occurs. The expanded name * consisting of the local part of the name and the possibly null URI reference * is used as the name of the object. The default namespace is not used for * unprefixed names." */ public class QName implements java.io.Serializable { /** * The local name. * @serial */ protected String _localName; /** * The namespace URI. * @serial */ protected String _namespaceURI; /** * The namespace prefix. * @serial */ protected String _prefix; /** * The XML namespace. */ public static final String S_XMLNAMESPACEURI = "http://www.w3.org/XML/1998/namespace"; /** * The cached hashcode, which is calculated at construction time. * @serial */ private int m_hashCode; /** * Constructs an empty QName. * 20001019: Try making this public, to support Serializable? -- JKESS */ public QName(){} /** * Constructs a new QName with the specified namespace URI and * local name. * * @param namespaceURI The namespace URI if known, or null * @param localName The local name */ public QName(String namespaceURI, String localName) { if (localName == null) throw new IllegalArgumentException("Argument 'localName' is null"); _namespaceURI = namespaceURI; _localName = localName; m_hashCode = toString().hashCode(); } /** * Constructs a new QName with the specified namespace URI, prefix * and local name. * * @param namespaceURI The namespace URI if known, or null * @param prefix The namespace prefix is known, or null * @param localName The local name */ public QName(String namespaceURI, String prefix, String localName) { if (localName == null) throw new IllegalArgumentException("Argument 'localName' is null"); _namespaceURI = namespaceURI; _prefix = prefix; _localName = localName; m_hashCode = toString().hashCode(); } /** * Construct a QName from a string, without namespace resolution. Good * for a few odd cases. * * @param localName Local part of qualified name */ public QName(String localName) { if (localName == null) throw new IllegalArgumentException("Argument 'localName' is null"); _namespaceURI = null; _localName = localName; m_hashCode = toString().hashCode(); } /** * Construct a QName from a string, resolving the prefix * using the given namespace stack. The default namespace is * not resolved. * * @param qname Qualified name to resolve * @param namespaces Namespace stack to use to resolve namespace */ public QName(String qname, Stack namespaces) { String namespace = null; String prefix = null; int indexOfNSSep = qname.indexOf(':'); if (indexOfNSSep > 0) { prefix = qname.substring(0, indexOfNSSep); if (prefix.equals("xml")) { namespace = S_XMLNAMESPACEURI; } else if (prefix.equals("xmlns")) { return; } else { int depth = namespaces.size(); for (int i = depth - 1; i >= 0; i--) { NameSpace ns = (NameSpace) namespaces.elementAt(i); while (null != ns) { if ((null != ns.m_prefix) && prefix.equals(ns.m_prefix)) { namespace = ns.m_uri; i = -1; break; } ns = ns.m_next; } } } if (null == namespace) { throw new RuntimeException( XSLMessages.createXPATHMessage( XPATHErrorResources.ER_PREFIX_MUST_RESOLVE, new Object[]{ prefix })); //"Prefix must resolve to a namespace: "+prefix); } } _localName = (indexOfNSSep < 0) ? qname : qname.substring(indexOfNSSep + 1); _namespaceURI = namespace; _prefix = prefix; m_hashCode = toString().hashCode(); } /** * Construct a QName from a string, resolving the prefix * using the given namespace context and prefix resolver. * The default namespace is not resolved. * * @param qname Qualified name to resolve * @param namespaceContext Namespace Context to use * @param resolver Prefix resolver for this context */ public QName(String qname, Element namespaceContext, PrefixResolver resolver) { _namespaceURI = null; int indexOfNSSep = qname.indexOf(':'); if (indexOfNSSep > 0) { if (null != namespaceContext) { String prefix = qname.substring(0, indexOfNSSep); _prefix = prefix; if (prefix.equals("xml")) { _namespaceURI = S_XMLNAMESPACEURI; } else { _namespaceURI = resolver.getNamespaceForPrefix(prefix, namespaceContext); } if (null == _namespaceURI) { throw new RuntimeException( XSLMessages.createXPATHMessage( XPATHErrorResources.ER_PREFIX_MUST_RESOLVE, new Object[]{ prefix })); //"Prefix must resolve to a namespace: "+prefix); } } else { // TODO: error or warning... } } _localName = (indexOfNSSep < 0) ? qname : qname.substring(indexOfNSSep + 1); m_hashCode = toString().hashCode(); } /** * Construct a QName from a string, resolving the prefix * using the given namespace stack. The default namespace is * not resolved. * * @param qname Qualified name to resolve * @param resolver Prefix resolver for this context */ public QName(String qname, PrefixResolver resolver) { _namespaceURI = null; int indexOfNSSep = qname.indexOf(':'); if (indexOfNSSep > 0) { String prefix = qname.substring(0, indexOfNSSep); if (prefix.equals("xml")) { _namespaceURI = S_XMLNAMESPACEURI; } else { _namespaceURI = resolver.getNamespaceForPrefix(prefix); } if (null == _namespaceURI) { throw new RuntimeException( XSLMessages.createXPATHMessage( XPATHErrorResources.ER_PREFIX_MUST_RESOLVE, new Object[]{ prefix })); //"Prefix must resolve to a namespace: "+prefix); } } _localName = (indexOfNSSep < 0) ? qname : qname.substring(indexOfNSSep + 1); m_hashCode = toString().hashCode(); } /** * Returns the namespace URI. Returns null if the namespace URI * is not known. * * @return The namespace URI, or null */ public String getNamespaceURI() { return _namespaceURI; } /** * Returns the namespace prefix. Returns null if the namespace * prefix is not known. * * @return The namespace prefix, or null */ public String getPrefix() { return _prefix; } /** * Returns the local part of the qualified name. * * @return The local part of the qualified name */ public String getLocalName() { return _localName; } /** * Return the string representation of the qualified name, using the * prefix if available, or the '{ns}foo' notation if not. Performs * string concatenation, so beware of performance issues. * * @return the string representation of the namespace */ public String toString() { return _prefix != null ? (_prefix + ":" + _localName) : (_namespaceURI != null ? ("{"+_namespaceURI + "}" + _localName) : _localName); } /** * Return the string representation of the qualified name using the * the '{ns}foo' notation. Performs * string concatenation, so beware of performance issues. * * @return the string representation of the namespace */ public String toNamespacedString() { return (_namespaceURI != null ? ("{"+_namespaceURI + "}" + _localName) : _localName); } /** * Get the namespace of the qualified name. * * @return the namespace URI of the qualified name */ public String getNamespace() { return getNamespaceURI(); } /** * Get the local part of the qualified name. * * @return the local part of the qualified name */ public String getLocalPart() { return getLocalName(); } /** * Return the cached hashcode of the qualified name. * * @return the cached hashcode of the qualified name */ public int hashCode() { return m_hashCode; } /** * Override equals and agree that we're equal if * the passed object is a string and it matches * the name of the arg. * * @param ns Namespace URI to compare to * @param localPart Local part of qualified name to compare to * * @return True if the local name and uri match */ public boolean equals(String ns, String localPart) { String thisnamespace = getNamespaceURI(); return getLocalName().equals(localPart) && (((null != thisnamespace) && (null != ns)) ? thisnamespace.equals(ns) : ((null == thisnamespace) && (null == ns))); } /** * Override equals and agree that we're equal if * the passed object is a QName and it matches * the name of the arg. * * @param qname Qualified name to compare to * * @return True if the qualified names are equal */ public boolean equals(Object object) { if (object == this) return true; if (object instanceof QName) { QName qname = (QName) object; String thisnamespace = getNamespaceURI(); String thatnamespace = qname.getNamespaceURI(); return getLocalName().equals(qname.getLocalName()) && (((null != thisnamespace) && (null != thatnamespace)) ? thisnamespace.equals(thatnamespace) : ((null == thisnamespace) && (null == thatnamespace))); } else return false; } /** * Given a string, create and return a QName object * * * @param name String to use to create QName * * @return a QName object */ public static QName getQNameFromString(String name) { StringTokenizer tokenizer = new StringTokenizer(name, "{}", false); QName qname; String s1 = tokenizer.nextToken(); String s2 = tokenizer.hasMoreTokens() ? tokenizer.nextToken() : null; if (null == s2) qname = new QName(null, s1); else qname = new QName(s1, s2); return qname; } /** * This function tells if a raw attribute name is a * xmlns attribute. * * @param attRawName Raw name of attribute * * @return True if the attribute starts with or is equal to xmlns */ public static boolean isXMLNSDecl(String attRawName) { return (attRawName.startsWith("xmlns") && (attRawName.equals("xmlns") || attRawName.startsWith("xmlns:"))); } /** * This function tells if a raw attribute name is a * xmlns attribute. * * @param attRawName Raw name of attribute * * @return Prefix of attribute */ public static String getPrefixFromXMLNSDecl(String attRawName) { int index = attRawName.indexOf(':'); return (index >= 0) ? attRawName.substring(index + 1) : ""; } /** * Returns the local name of the given node. * * @param qname Input name * * @return Local part of the name if prefixed, or the given name if not */ public static String getLocalPart(String qname) { int index = qname.indexOf(':'); return (index < 0) ? qname : qname.substring(index + 1); } /** * Returns the local name of the given node. * * @param qname Input name * * @return Prefix of name or empty string if none there */ public static String getPrefixPart(String qname) { int index = qname.indexOf(':'); return (index >= 0) ? qname.substring(0, index) : ""; } }