All Downloads are FREE. Search and download functionalities are using the official Maven repository.

src.java.com.ctc.wstx.sr.NsInputElementStack Maven / Gradle / Ivy

There is a newer version: 4.0.6
Show newest version
/* Woodstox XML processor
 *
 * Copyright (c) 2004- Tatu Saloranta, [email protected]
 *
 * Licensed under the License specified in file LICENSE, included with
 * the source code.
 * You may not use this file except in compliance with the License.
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.ctc.wstx.sr;

import java.util.*;

import javax.xml.XMLConstants;
import javax.xml.namespace.NamespaceContext;
import javax.xml.namespace.QName;
import javax.xml.stream.Location;
import javax.xml.stream.XMLStreamException;

import org.codehaus.stax2.validation.XMLValidator;

import com.ctc.wstx.api.ReaderConfig;
import com.ctc.wstx.cfg.ErrorConsts;
import com.ctc.wstx.util.*;

/**
 * Sub-class of {@link InputElementStack} used when operating in
 * namespace-aware mode.
 *

* Implementation note: this class reuses {@link NamespaceContext} * instances, so that consequtive accesses just return the same instance, * as long as nothing in bindings change. As a result, only those instances * that explicitly add new bindings create distinct non-shareable context * instances. Although it would also be possible to share underlying String * array to further improve object sharing, it seems like marginal gain * with more complexity: as such the current simple scheme should work * just fine (and is measure to be very close to optimal for most common * namespace-heavey document types like Soap messages). */ public final class NsInputElementStack extends InputElementStack { /** * Top-most namespace URI assigned for root element, if not specifically * defined (default namespace unbound). * As per Stax specs (and related clarifying discussion on * the mailing list), null should consistently be used. */ final static String DEFAULT_NAMESPACE_URI = null; final static int IX_PREFIX = 0; final static int IX_LOCALNAME = 1; final static int IX_URI = 2; final static int IX_DEFAULT_NS = 3; final static int ENTRY_SIZE = 4; protected final static InternCache sInternCache = InternCache.getInstance(); /* ////////////////////////////////////////////////// // Configuration ////////////////////////////////////////////////// */ protected final NsAttributeCollector mAttrCollector; /** * Object that will need to be consulted about namespace bindings, * since it has some knowledge about default namespace declarations * (has default attribute value expansion). */ protected NsDefaultProvider mNsDefaultProvider; /* ////////////////////////////////////////////////// // Element stack state information ////////////////////////////////////////////////// */ /** * Vector that contains all currently active namespaces; one String for * prefix, another for matching URI. Does also include default name * spaces (at most one per level). */ protected final StringVector mNamespaces = new StringVector(64); /** * Array that contains path of open elements from root; for each there * are 4 Strings; prefix, localname, URI, and default name space URI. */ protected String[] mElements; /** * Number of Strings in {@link #mElements} that are valid (ie depth * multiplied by 4) */ protected int mSize; /** * Array that contains namespace offsets for each element; that is, * index of first 'local' name space entry, entry declared for * current element. Number of such local entries is * mCurrNsCount - mNsCounts[mSize-1] */ protected int[] mNsCounts; protected boolean mMayHaveNsDefaults = false; /* ////////////////////////////////////////////////// // Simple 1-slot QName cache; used for improving // efficiency of code that uses QNames extensively // (like StAX Event API implementation) ////////////////////////////////////////////////// */ protected String mLastLocalName = null; protected String mLastPrefix = null; protected String mLastNsURI = null; protected QName mLastName = null; /* ///////////////////////////////////////////////////// // Simple caching for non-transient NamespaceContext // instance - mostly for event API as well ///////////////////////////////////////////////////// */ /** * Last potentially shareable NamespaceContext created by * this stack. This reference is cleared each time bindings * change (either due to a start element with new bindings, or due * to the matching end element that closes scope of such binding(s)). */ protected BaseNsContext mLastNsContext = null; /* ////////////////////////////////////////////////// // Life-cycle (create, update state) ////////////////////////////////////////////////// */ public NsInputElementStack(int initialSize, ReaderConfig cfg) { super(cfg); mSize = 0; if (initialSize < 4) { initialSize = 4; } mElements = new String[initialSize << 2]; mNsCounts = new int[initialSize]; mAttrCollector = new NsAttributeCollector(cfg); } protected void setAutomaticDTDValidator(XMLValidator validator, NsDefaultProvider nsDefs) { mNsDefaultProvider = nsDefs; addValidator(validator); } public final void push(String prefix, String localName) { int index = mSize; if (index == mElements.length) { String[] old = mElements; mElements = new String[old.length + 64]; System.arraycopy(old, 0, mElements, 0, old.length); } mElements[index] = prefix; mElements[index+1] = localName; if (index == 0) { // root element mElements[IX_DEFAULT_NS] = DEFAULT_NAMESPACE_URI; } else { // Let's just duplicate parent's default NS URI as baseline: mElements[index + IX_DEFAULT_NS] = mElements[index - (ENTRY_SIZE - IX_DEFAULT_NS)]; } mSize = index+4; // Also need to update namespace stack: index >>= 2; if (index == mNsCounts.length) { int[] old = mNsCounts; mNsCounts = new int[old.length + 16]; System.arraycopy(old, 0, mNsCounts, 0, old.length); } mNsCounts[index] = mNamespaces.size(); mAttrCollector.reset(); /* 20-Feb-2006, TSa: Hmmh. Namespace default provider unfortunately * needs an advance warning... */ if (mNsDefaultProvider != null) { mMayHaveNsDefaults = mNsDefaultProvider.mayHaveNsDefaults(prefix, localName); } } public final void push(String fullName) { throw new Error("Internal error: push(fullName) shouldn't be called for namespace aware element stack."); } /** * Method called by the stream reader when encountering an end tag. * * @return Validation state that should be effective for the parent * element state */ public int pop() throws XMLStreamException { int index = mSize; if (index == 0) { throw new IllegalStateException("Popping from empty stack."); } int result; /* Can and should not shrink (or clear) the stack before calling * the validator, so let's just update the local count */ index -= 4; if (mValidator == null) { /* Let's allow GCing (not likely to matter, as Strings are very * likely interned... but it's a good habit */ result = XMLValidator.CONTENT_ALLOW_ANY_TEXT; } else { result = mValidator.validateElementEnd(mElements[index+IX_LOCALNAME], mElements[index+IX_URI], mElements[index+IX_PREFIX]); } // Now we can shrink the stack: mSize = index; mElements[index] = null; mElements[index+1] = null; mElements[index+2] = null; mElements[index+3] = null; // Need to purge namespaces? int nsCount = mNamespaces.size() - mNsCounts[index >> 2]; if (nsCount > 0) { // 2 entries for each NS mapping: mLastNsContext = null; // let's invalidate ns ctxt too, if we had one mNamespaces.removeLast(nsCount); } return result; } /** * Method called to update information about top of the stack, with * attribute information passed in. Will resolve namespace references, * and update namespace stack with information. * * @return Validation state that should be effective for the fully * resolved element context */ public int resolveAndValidateElement() throws XMLStreamException { if (mSize == 0) { // just a simple sanity check throw new IllegalStateException("Calling validate() on empty stack."); } NsAttributeCollector ac = mAttrCollector; // Any namespace declarations? { int nsCount = ac.getNsCount(); if (nsCount > 0) { /* let's first invalidate old (possibly) shared ns ctxt too, * if we had one; new one can be created at a later point */ mLastNsContext = null; String [] nsPrefixes = ac.getNsPrefixes(); TextBuilder nsURIs = ac.getNsURIs(); boolean internNsUris = mConfig.willInternNsURIs(); for (int i = 0; i < nsCount; ++i) { String nsUri = nsURIs.getEntry(i); if (internNsUris && nsUri.length() > 0) { nsUri = sInternCache.intern(nsUri); } // will have defaul ns's too; have null prefix String prefix = nsPrefixes[i]; /* 18-Jul-2004, TSa: Need to check that 'xml' and 'xmlns' * prefixes are not re-defined (and 'xmlns' not even * defined to its correct ns). */ if (prefix == "xmlns") { // xmlns can never be declared, even to its correct URI mReporter.throwParseError(ErrorConsts.ERR_NS_REDECL_XMLNS); } else if (prefix == "xml") { // whereas xml is ok, as long as it's same URI: if (!nsUri.equals(XMLConstants.XML_NS_URI)) { mReporter.throwParseError(ErrorConsts.ERR_NS_REDECL_XML, nsUri); } /* 09-Feb-2006, TSa: Hmmh. Now, should this explicit * xml declaration be visible to the app? SAX API * seem to ignore it. */ //mNamespaces.addStrings(prefix, nsUri); } else { // ok, valid prefix, so far /* 17-Mar-2006, TSa: Unbinding default NS needs to * result in null being added: */ if (nsUri == null || nsUri.length() == 0) { nsUri = DEFAULT_NAMESPACE_URI; } // The default ns binding needs special handling: if (prefix == null) { mElements[mSize-(ENTRY_SIZE - IX_DEFAULT_NS)] = nsUri; } /* But then let's ensure that URIs matching xml * and xmlns are not being bound to anything else */ if (internNsUris) { // identity comparison is ok: if (nsUri == XMLConstants.XML_NS_URI) { mReporter.throwParseError(ErrorConsts.ERR_NS_REDECL_XML_URI, prefix); } else if (nsUri == XMLConstants.XMLNS_ATTRIBUTE_NS_URI) { mReporter.throwParseError(ErrorConsts.ERR_NS_REDECL_XMLNS_URI); } } else { // need to check equals() if (nsUri.equals(XMLConstants.XML_NS_URI)) { mReporter.throwParseError(ErrorConsts.ERR_NS_REDECL_XML_URI, prefix); } else if (nsUri.equals(XMLConstants.XMLNS_ATTRIBUTE_NS_URI)) { mReporter.throwParseError(ErrorConsts.ERR_NS_REDECL_XMLNS_URI); } } /* and at any rate, binding needs to be added, to * be visible to the app (including def ns): */ mNamespaces.addStrings(prefix, nsUri); } } } } /* 20-Feb-2006, TSa: Any attribute defaults for namespace declaration * pseudo-attributes? */ if (mMayHaveNsDefaults) { mNsDefaultProvider.checkNsDefaults(this); } // Then, let's set element's namespace, if any: String prefix = mElements[mSize-(ENTRY_SIZE - IX_PREFIX)]; String ns; if (prefix == null || prefix.length() == 0) { // use default NS, if any ns = mElements[mSize-(ENTRY_SIZE - IX_DEFAULT_NS)]; } else if (prefix == "xml") { ns = XMLConstants.XML_NS_URI; } else { // Need to find namespace with the prefix: ns = mNamespaces.findLastFromMap(prefix); if (ns == null) { mReporter.throwParseError(ErrorConsts.ERR_NS_UNDECLARED, prefix); } } mElements[mSize-(ENTRY_SIZE - IX_URI)] = ns; // And finally, resolve attributes' namespaces too: int xmlidIx = ac.resolveNamespaces(mReporter, mNamespaces); mIdAttrIndex = xmlidIx; XMLValidator vld = mValidator; /* If we have no validator(s), nothing more to do, * except perhaps little bit of Xml:id handling: */ if (vld == null) { // no validator in use if (xmlidIx >= 0) { // need to normalize xml:id, still? normalizeXmlIdAttr(ac, xmlidIx); } return XMLValidator.CONTENT_ALLOW_ANY_TEXT; } // Otherwise need to call relevant validation methods. /* First, a call to check if the element itself may be acceptable * within structure: */ vld.validateElementStart (mElements[mSize-(ENTRY_SIZE - IX_LOCALNAME)], mElements[mSize-(ENTRY_SIZE - IX_URI)], prefix); // Then attributes, if any: StringVector attrNames = ac.getNameList(); int attrLen = ac.getCount(); if (attrLen > 0) { attrLen += attrLen; // 2 entries per name (prefix + ln) String[] attrURIs = ac.getAttrURIs(); String[] nameData = attrNames.getInternalArray(); TextBuilder attrBuilder = ac.getAttrBuilder(); char[] attrCB = attrBuilder.getCharBuffer(); for (int i = 0, nr = 0; i < attrLen; i += 2, ++nr) { prefix = nameData[i]; String ln = nameData[i+1]; String normValue = mValidator.validateAttribute (ln, attrURIs[nr], prefix, attrCB, attrBuilder.getOffset(nr), attrBuilder.getOffset(nr+1)); if (normValue != null) { ac.setNormalizedValue(nr, normValue); } } } /* And finally let's wrap things up to see what textual content * is allowed as child content, if any: */ return mValidator.validateElementAndAttributes(); } /* /////////////////////////////////////////////////// // Public methods /////////////////////////////////////////////////// */ public final boolean isNamespaceAware() { return true; } public final int getDepth() { return (mSize >> 2); } public final AttributeCollector getAttrCollector() { return mAttrCollector; } /** * Method called to construct a non-transient NamespaceContext instance; * generally needed when creating events to return from event-based * iterators. */ public final BaseNsContext createNonTransientNsContext(Location loc) { // Have an instance we can reuse? Great! if (mLastNsContext != null) { return mLastNsContext; } // No namespaces declared at this point? Easy, as well: int totalNsSize = mNamespaces.size(); if (totalNsSize < 1) { return (mLastNsContext = EmptyNamespaceContext.getInstance()); } // Otherwise, we need to create a new non-empty context: int localCount = getCurrentNsCount() << 1; BaseNsContext nsCtxt = new CompactNsContext (loc, getDefaultNsURI(), mNamespaces.asArray(), totalNsSize, totalNsSize - localCount); /* And it can be shared if there are no new ('local', ie. included * within this start element) bindings -- if there are, underlying * array might be shareable, but offsets wouldn't be) */ if (localCount == 0) { mLastNsContext = nsCtxt; } return nsCtxt; } /* /////////////////////////////////////////////////// // Implementation of NamespaceContext: /////////////////////////////////////////////////// */ public final String getNamespaceURI(String prefix) { if (prefix == null) { throw new IllegalArgumentException(ErrorConsts.ERR_NULL_ARG); } if (prefix.length() == 0) { if (mSize == 0) { // unexpected... but let's not err at this point return null; } return mElements[mSize-(ENTRY_SIZE - IX_DEFAULT_NS)]; } if (prefix.equals(XMLConstants.XML_NS_PREFIX)) { return XMLConstants.XML_NS_URI; } if (prefix.equals(XMLConstants.XMLNS_ATTRIBUTE)) { return XMLConstants.XMLNS_ATTRIBUTE_NS_URI; } /* Ok, need to find the match, if any; starting from end of the * list of active namespaces. Note that we can not count on prefix * being interned/canonicalized. */ return mNamespaces.findLastNonInterned(prefix); } public final String getPrefix(String nsURI) { if (nsURI == null || nsURI.length() == 0) { throw new IllegalArgumentException("Illegal to pass null/empty prefix as argument."); } if (nsURI.equals(XMLConstants.XML_NS_URI)) { return XMLConstants.XML_NS_PREFIX; } if (nsURI.equals(XMLConstants.XMLNS_ATTRIBUTE_NS_URI)) { return XMLConstants.XMLNS_ATTRIBUTE; } /* Ok, need to find the match, if any; starting from end of the * list of active namespaces. Note that we can not count on prefix * being interned/canonicalized. */ //String prefix = mNamespaces.findLastByValueNonInterned(nsURI); String prefix = null; /* 29-Sep-2004, TSa: Need to check for namespace masking, too... */ String[] strs = mNamespaces.getInternalArray(); int len = mNamespaces.size(); main_loop: for (int index = len-1; index > 0; index -= 2) { if (nsURI.equals(strs[index])) { // Ok, is prefix masked? prefix = strs[index-1]; for (int j = index+1; j < len; j += 2) { if (strs[j] == prefix) { // masked! prefix = null; continue main_loop; } } // nah, it's good // 17-Mar-2006, TSa: ... but default NS has prefix null... if (prefix == null) { prefix = ""; } break main_loop; } } return prefix; } public final Iterator getPrefixes(String nsURI) { if (nsURI == null || nsURI.length() == 0) { throw new IllegalArgumentException("Illegal to pass null/empty prefix as argument."); } if (nsURI.equals(XMLConstants.XML_NS_URI)) { return new SingletonIterator(XMLConstants.XML_NS_PREFIX); } if (nsURI.equals(XMLConstants.XMLNS_ATTRIBUTE_NS_URI)) { return new SingletonIterator(XMLConstants.XMLNS_ATTRIBUTE); } //return mNamespaces.findAllByValueNonInterned(nsURI); /* 29-Sep-2004, TSa: Need to check for namespace masking, too... */ String[] strs = mNamespaces.getInternalArray(); int len = mNamespaces.size(); ArrayList l = null; main_loop: for (int index = len-1; index > 0; index -= 2) { if (nsURI.equals(strs[index])) { // Ok, is prefix masked? String prefix = strs[index-1]; for (int j = index+1; j < len; j += 2) { if (strs[j] == prefix) { // masked! continue main_loop; } } // nah, it's good! if (l == null) { l = new ArrayList(); } l.add(prefix); } } return (l == null) ? EmptyIterator.getInstance() : l.iterator(); } /* /////////////////////////////////////////////////// // AttributeInfo methods (StAX2) /////////////////////////////////////////////////// */ public final int getAttributeCount() { return mAttrCollector.getCount(); } public final int findAttributeIndex(String nsURI, String localName) { return mAttrCollector.findIndex(nsURI, localName); } /* /////////////////////////////////////////////////// // ValidationContext methods /////////////////////////////////////////////////// */ public final QName getCurrentElementName() { if (mSize == 0) { return null; } String prefix = mElements[mSize-(ENTRY_SIZE - IX_PREFIX)]; /* 17-Mar-2006, TSa: We only map prefix to empty String because * some QName impls barf on nulls. Otherwise we will always * use null to indicate missing prefixes. */ if (prefix == null) { prefix = ""; } /* 03-Dec-2004, TSa: Maybe we can just reuse the last QName * object created, if we have same data? (happens if * state hasn't changed, or we got end element for a leaf * element, or repeating leaf elements) */ String nsURI = mElements[mSize-(ENTRY_SIZE - IX_URI)]; String ln = mElements[mSize-(ENTRY_SIZE - IX_LOCALNAME)]; /* Since we generally intern most Strings, can do identity * comparisons here: */ if (ln != mLastLocalName) { mLastLocalName = ln; mLastPrefix = prefix; mLastNsURI = nsURI; } else if (prefix != mLastPrefix) { mLastPrefix = prefix; mLastNsURI = nsURI; } else if (nsURI != mLastNsURI) { mLastNsURI = nsURI; } else { return mLastName; } QName n = new QName(nsURI, ln, prefix); mLastName = n; return n; } public int addDefaultAttribute(String localName, String uri, String prefix, String value) { return mAttrCollector.addDefaultAttribute(localName, uri, prefix, value); } /* /////////////////////////////////////////////////// // Support for NsDefaultProvider /////////////////////////////////////////////////// */ public boolean isPrefixLocallyDeclared(String internedPrefix) { if (internedPrefix != null && internedPrefix.length() == 0) { // default ns internedPrefix = null; } int offset = mNsCounts[(mSize-1) >> 2]; for (int len = mNamespaces.size(); offset < len; offset += 2) { // both interned, can use identity comparison String thisPrefix = mNamespaces.getString(offset); if (thisPrefix == internedPrefix) { return true; } } return false; } /** * Callback method called by the namespace default provider. At * this point we can trust it to only call this method with somewhat * valid arguments (no dups etc). */ public void addNsBinding(String prefix, String uri) { // Unbind? (xml 1.1...) if ((uri == null) || (uri.length() == 0)) { uri = null; } // Default ns declaration? if ((prefix == null) || (prefix.length() == 0)) { prefix = null; mElements[mSize-(ENTRY_SIZE - IX_DEFAULT_NS)] = uri; } mNamespaces.addStrings(prefix, uri); } /* /////////////////////////////////////////////////// // Accessors: /////////////////////////////////////////////////// */ // // // Generic stack information: public final boolean isEmpty() { return mSize == 0; } // // // Information about element at top of stack: public final String getDefaultNsURI() { if (mSize == 0) { throw new IllegalStateException("Illegal access, empty stack."); } return mElements[mSize-(ENTRY_SIZE - IX_DEFAULT_NS)]; } public final String getNsURI() { if (mSize == 0) { throw new IllegalStateException("Illegal access, empty stack."); } return mElements[mSize-(ENTRY_SIZE - IX_URI)]; } public final String getPrefix() { if (mSize == 0) { throw new IllegalStateException("Illegal access, empty stack."); } return mElements[mSize-(ENTRY_SIZE - IX_PREFIX)]; } public final String getLocalName() { if (mSize == 0) { throw new IllegalStateException("Illegal access, empty stack."); } return mElements[mSize-(ENTRY_SIZE - IX_LOCALNAME)]; } public final boolean matches(String prefix, String localName) { if (mSize == 0) { throw new IllegalStateException("Illegal access, empty stack."); } String thisPrefix = mElements[mSize-(ENTRY_SIZE - IX_PREFIX)]; if (prefix == null || prefix.length() == 0) { // no name space if (thisPrefix != null && thisPrefix.length() > 0) { return false; } } else { if (thisPrefix != prefix && !thisPrefix.equals(prefix)) { return false; } } String thisName = mElements[mSize-3]; return (thisName == localName) || thisName.equals(localName); } public final String getTopElementDesc() { if (mSize == 0) { throw new IllegalStateException("Illegal access, empty stack."); } String name = mElements[mSize-3]; String prefix = mElements[mSize-4]; if (prefix == null || prefix.length() == 0) { // no name space return name; } return prefix + ":" + name; } // // // Namespace information: /** * @return Number of active prefix/namespace mappings for current scope, * including mappings from enclosing elements. */ public final int getTotalNsCount() { return mNamespaces.size() >> 1; } /** * @return Number of active prefix/namespace mappings for current scope, * NOT including mappings from enclosing elements. */ public final int getCurrentNsCount() { // Need not check for empty stack; should return 0 properly return (mNamespaces.size() - mNsCounts[(mSize-1) >> 2]) >> 1; } public final String getLocalNsPrefix(int index) { int offset = mNsCounts[(mSize-1) >> 2]; int localCount = (mNamespaces.size() - offset); index <<= 1; // 2 entries, prefix/URI for each NS if (index < 0 || index >= localCount) { throwIllegalIndex(index >> 1, localCount >> 1); } return mNamespaces.getString(offset + index); } public final String getLocalNsURI(int index) { int offset = mNsCounts[(mSize-1) >> 2]; int localCount = (mNamespaces.size() - offset); index <<= 1; // 2 entries, prefix/URI for each NS if (index < 0 || index >= localCount) { throwIllegalIndex(index >> 1, localCount >> 1); } return mNamespaces.getString(offset + index + 1); } private void throwIllegalIndex(int index, int localCount) { throw new IllegalArgumentException("Illegal namespace index " +(index >> 1) +"; current scope only has " +(localCount >> 1) +" namespace declarations."); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy