blob: 9daee5f2d7fd5e4cf272e3f40161b5ee49d8c4de [file] [log] [blame]
/*
* Copyright (c) 2004, 2011, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*
* THIS FILE WAS MODIFIED BY SUN MICROSYSTEMS, INC.
*/
package com.sun.xml.internal.fastinfoset.sax;
import com.sun.xml.internal.fastinfoset.Encoder;
import com.sun.xml.internal.fastinfoset.EncodingConstants;
import com.sun.xml.internal.fastinfoset.QualifiedName;
import com.sun.xml.internal.org.jvnet.fastinfoset.sax.FastInfosetWriter;
import com.sun.xml.internal.fastinfoset.util.LocalNameQualifiedNamesMap;
import java.io.IOException;
import com.sun.xml.internal.org.jvnet.fastinfoset.EncodingAlgorithmIndexes;
import com.sun.xml.internal.org.jvnet.fastinfoset.FastInfosetException;
import com.sun.xml.internal.org.jvnet.fastinfoset.RestrictedAlphabet;
import com.sun.xml.internal.org.jvnet.fastinfoset.sax.EncodingAlgorithmAttributes;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import com.sun.xml.internal.fastinfoset.CommonResourceBundle;
/**
* The Fast Infoset SAX serializer.
* <p>
* Instantiate this serializer to serialize a fast infoset document in accordance
* with the SAX API.
* <p>
* This utilizes the SAX API in a reverse manner to that of parsing. It is the
* responsibility of the client to call the appropriate event methods on the
* SAX handlers, and to ensure that such a sequence of methods calls results
* in the production well-formed fast infoset documents. The
* SAXDocumentSerializer performs no well-formed checks.
*
* <p>
* More than one fast infoset document may be encoded to the
* {@link java.io.OutputStream}.
*/
public class SAXDocumentSerializer extends Encoder implements FastInfosetWriter {
protected boolean _elementHasNamespaces = false;
protected boolean _charactersAsCDATA = false;
protected SAXDocumentSerializer(boolean v) {
super(v);
}
public SAXDocumentSerializer() {
}
public void reset() {
super.reset();
_elementHasNamespaces = false;
_charactersAsCDATA = false;
}
// ContentHandler
public final void startDocument() throws SAXException {
try {
reset();
encodeHeader(false);
encodeInitialVocabulary();
} catch (IOException e) {
throw new SAXException("startDocument", e);
}
}
public final void endDocument() throws SAXException {
try {
encodeDocumentTermination();
} catch (IOException e) {
throw new SAXException("endDocument", e);
}
}
public void startPrefixMapping(String prefix, String uri) throws SAXException {
try {
if (_elementHasNamespaces == false) {
encodeTermination();
// Mark the current buffer position to flag attributes if necessary
mark();
_elementHasNamespaces = true;
// Write out Element byte with namespaces
write(EncodingConstants.ELEMENT | EncodingConstants.ELEMENT_NAMESPACES_FLAG);
}
encodeNamespaceAttribute(prefix, uri);
} catch (IOException e) {
throw new SAXException("startElement", e);
}
}
public final void startElement(String namespaceURI, String localName, String qName, Attributes atts) throws SAXException {
// TODO consider using buffer for encoding of attributes, then pre-counting is not necessary
final int attributeCount = (atts != null && atts.getLength() > 0)
? countAttributes(atts) : 0;
try {
if (_elementHasNamespaces) {
_elementHasNamespaces = false;
if (attributeCount > 0) {
// Flag the marked byte with attributes
_octetBuffer[_markIndex] |= EncodingConstants.ELEMENT_ATTRIBUTE_FLAG;
}
resetMark();
write(EncodingConstants.TERMINATOR);
_b = 0;
} else {
encodeTermination();
_b = EncodingConstants.ELEMENT;
if (attributeCount > 0) {
_b |= EncodingConstants.ELEMENT_ATTRIBUTE_FLAG;
}
}
encodeElement(namespaceURI, qName, localName);
if (attributeCount > 0) {
encodeAttributes(atts);
}
} catch (IOException e) {
throw new SAXException("startElement", e);
} catch (FastInfosetException e) {
throw new SAXException("startElement", e);
}
}
public final void endElement(String namespaceURI, String localName, String qName) throws SAXException {
try {
encodeElementTermination();
} catch (IOException e) {
throw new SAXException("endElement", e);
}
}
public final void characters(char[] ch, int start, int length) throws SAXException {
if (length <= 0) {
return;
}
if (getIgnoreWhiteSpaceTextContent() &&
isWhiteSpace(ch, start, length)) return;
try {
encodeTermination();
if (!_charactersAsCDATA) {
encodeCharacters(ch, start, length);
} else {
encodeCIIBuiltInAlgorithmDataAsCDATA(ch, start, length);
}
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
public final void ignorableWhitespace(char[] ch, int start, int length) throws SAXException {
if (getIgnoreWhiteSpaceTextContent()) return;
characters(ch, start, length);
}
public final void processingInstruction(String target, String data) throws SAXException {
try {
if (getIgnoreProcesingInstructions()) return;
if (target.length() == 0) {
throw new SAXException(CommonResourceBundle.getInstance().
getString("message.processingInstructionTargetIsEmpty"));
}
encodeTermination();
encodeProcessingInstruction(target, data);
} catch (IOException e) {
throw new SAXException("processingInstruction", e);
}
}
public final void setDocumentLocator(org.xml.sax.Locator locator) {
}
public final void skippedEntity(String name) throws SAXException {
}
// LexicalHandler
public final void comment(char[] ch, int start, int length) throws SAXException {
try {
if (getIgnoreComments()) return;
encodeTermination();
encodeComment(ch, start, length);
} catch (IOException e) {
throw new SAXException("startElement", e);
}
}
public final void startCDATA() throws SAXException {
_charactersAsCDATA = true;
}
public final void endCDATA() throws SAXException {
_charactersAsCDATA = false;
}
public final void startDTD(String name, String publicId, String systemId) throws SAXException {
if (getIgnoreDTD()) return;
try {
encodeTermination();
encodeDocumentTypeDeclaration(publicId, systemId);
encodeElementTermination();
} catch (IOException e) {
throw new SAXException("startDTD", e);
}
}
public final void endDTD() throws SAXException {
}
public final void startEntity(String name) throws SAXException {
}
public final void endEntity(String name) throws SAXException {
}
// EncodingAlgorithmContentHandler
public final void octets(String URI, int id, byte[] b, int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
encodeNonIdentifyingStringOnThirdBit(URI, id, b, start, length);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
public final void object(String URI, int id, Object data) throws SAXException {
try {
encodeTermination();
encodeNonIdentifyingStringOnThirdBit(URI, id, data);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
// PrimitiveTypeContentHandler
public final void bytes(byte[] b, int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
encodeCIIOctetAlgorithmData(EncodingAlgorithmIndexes.BASE64, b, start, length);
} catch (IOException e) {
throw new SAXException(e);
}
}
public final void shorts(short[] s, int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
encodeCIIBuiltInAlgorithmData(EncodingAlgorithmIndexes.SHORT, s, start, length);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
public final void ints(int[] i, int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
encodeCIIBuiltInAlgorithmData(EncodingAlgorithmIndexes.INT, i, start, length);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
public final void longs(long[] l, int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
encodeCIIBuiltInAlgorithmData(EncodingAlgorithmIndexes.LONG, l, start, length);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
public final void booleans(boolean[] b, int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
encodeCIIBuiltInAlgorithmData(EncodingAlgorithmIndexes.BOOLEAN, b, start, length);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
public final void floats(float[] f, int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
encodeCIIBuiltInAlgorithmData(EncodingAlgorithmIndexes.FLOAT, f, start, length);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
public final void doubles(double[] d, int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
encodeCIIBuiltInAlgorithmData(EncodingAlgorithmIndexes.DOUBLE, d, start, length);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
public void uuids(long[] msblsb, int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
encodeCIIBuiltInAlgorithmData(EncodingAlgorithmIndexes.UUID, msblsb, start, length);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
// RestrictedAlphabetContentHandler
public void numericCharacters(char ch[], int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
final boolean addToTable = isCharacterContentChunkLengthMatchesLimit(length);
encodeNumericFourBitCharacters(ch, start, length, addToTable);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
public void dateTimeCharacters(char ch[], int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
final boolean addToTable = isCharacterContentChunkLengthMatchesLimit(length);
encodeDateTimeFourBitCharacters(ch, start, length, addToTable);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
public void alphabetCharacters(String alphabet, char ch[], int start, int length) throws SAXException {
if (length <= 0) {
return;
}
try {
encodeTermination();
final boolean addToTable = isCharacterContentChunkLengthMatchesLimit(length);
encodeAlphabetCharacters(alphabet, ch, start, length, addToTable);
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
// ExtendedContentHandler
public void characters(char[] ch, int start, int length, boolean index) throws SAXException {
if (length <= 0) {
return;
}
if (getIgnoreWhiteSpaceTextContent() &&
isWhiteSpace(ch, start, length)) return;
try {
encodeTermination();
if (!_charactersAsCDATA) {
encodeNonIdentifyingStringOnThirdBit(ch, start, length, _v.characterContentChunk, index, true);
} else {
encodeCIIBuiltInAlgorithmDataAsCDATA(ch, start, length);
}
} catch (IOException e) {
throw new SAXException(e);
} catch (FastInfosetException e) {
throw new SAXException(e);
}
}
protected final int countAttributes(Attributes atts) {
// Count attributes ignoring any in the XMLNS namespace
// Note, such attributes may be produced when transforming from a DOM node
int count = 0;
for (int i = 0; i < atts.getLength(); i++) {
final String uri = atts.getURI(i);
if (uri == "http://www.w3.org/2000/xmlns/" || uri.equals("http://www.w3.org/2000/xmlns/")) {
continue;
}
count++;
}
return count;
}
protected void encodeAttributes(Attributes atts) throws IOException, FastInfosetException {
boolean addToTable;
boolean mustBeAddedToTable;
String value;
if (atts instanceof EncodingAlgorithmAttributes) {
final EncodingAlgorithmAttributes eAtts = (EncodingAlgorithmAttributes)atts;
Object data;
String alphabet;
for (int i = 0; i < eAtts.getLength(); i++) {
if (encodeAttribute(atts.getURI(i), atts.getQName(i), atts.getLocalName(i))) {
data = eAtts.getAlgorithmData(i);
// If data is null then there is no algorithm data
if (data == null) {
value = eAtts.getValue(i);
addToTable = isAttributeValueLengthMatchesLimit(value.length());
mustBeAddedToTable = eAtts.getToIndex(i);
alphabet = eAtts.getAlpababet(i);
if (alphabet == null) {
encodeNonIdentifyingStringOnFirstBit(value, _v.attributeValue, addToTable, mustBeAddedToTable);
} else if (alphabet == RestrictedAlphabet.DATE_TIME_CHARACTERS) {
encodeDateTimeNonIdentifyingStringOnFirstBit(
value, addToTable, mustBeAddedToTable);
} else if (alphabet == RestrictedAlphabet.NUMERIC_CHARACTERS) {
encodeNumericNonIdentifyingStringOnFirstBit(
value, addToTable, mustBeAddedToTable);
} else {
encodeNonIdentifyingStringOnFirstBit(value, _v.attributeValue, addToTable, mustBeAddedToTable);
}
} else {
encodeNonIdentifyingStringOnFirstBit(eAtts.getAlgorithmURI(i),
eAtts.getAlgorithmIndex(i), data);
}
}
}
} else {
for (int i = 0; i < atts.getLength(); i++) {
if (encodeAttribute(atts.getURI(i), atts.getQName(i), atts.getLocalName(i))) {
value = atts.getValue(i);
addToTable = isAttributeValueLengthMatchesLimit(value.length());
encodeNonIdentifyingStringOnFirstBit(value, _v.attributeValue, addToTable, false);
}
}
}
_b = EncodingConstants.TERMINATOR;
_terminate = true;
}
protected void encodeElement(String namespaceURI, String qName, String localName) throws IOException {
LocalNameQualifiedNamesMap.Entry entry = _v.elementName.obtainEntry(qName);
if (entry._valueIndex > 0) {
QualifiedName[] names = entry._value;
for (int i = 0; i < entry._valueIndex; i++) {
final QualifiedName n = names[i];
if ((namespaceURI == n.namespaceName || namespaceURI.equals(n.namespaceName))) {
encodeNonZeroIntegerOnThirdBit(names[i].index);
return;
}
}
}
encodeLiteralElementQualifiedNameOnThirdBit(namespaceURI, getPrefixFromQualifiedName(qName),
localName, entry);
}
protected boolean encodeAttribute(String namespaceURI, String qName, String localName) throws IOException {
LocalNameQualifiedNamesMap.Entry entry = _v.attributeName.obtainEntry(qName);
if (entry._valueIndex > 0) {
QualifiedName[] names = entry._value;
for (int i = 0; i < entry._valueIndex; i++) {
if ((namespaceURI == names[i].namespaceName || namespaceURI.equals(names[i].namespaceName))) {
encodeNonZeroIntegerOnSecondBitFirstBitZero(names[i].index);
return true;
}
}
}
return encodeLiteralAttributeQualifiedNameOnSecondBit(namespaceURI, getPrefixFromQualifiedName(qName),
localName, entry);
}
}