/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * 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 org.apache.tuscany.sca.databinding.json.axiom; import java.io.IOException; import java.io.OutputStream; import java.io.Writer; import java.util.HashMap; import java.util.Map; import javax.xml.stream.XMLStreamConstants; import javax.xml.stream.XMLStreamReader; import org.apache.axiom.om.OMDataSource; import org.apache.axiom.om.OMException; import org.apache.axiom.om.OMOutputFormat; import org.codehaus.jettison.json.JSONObject; import org.codehaus.jettison.json.JSONTokener; import org.codehaus.jettison.mapped.MappedXMLInputFactory; /** * JSONDataSource keeps the JSON String inside and consumes it when needed. This is to be kept in the * OMSourcedElementImpl and can be used either to expand the tree or get the JSON String directly without expanding. * This uses the "Mapped" JSON convention. * * @version $Rev$ $Date$ */ public class JSONDataSource implements OMDataSource { protected JSONObject json; public JSONDataSource(JSONObject json) { this.json = json; } /** * Writes JSON into the output stream. As this should write JSON, it directly gets the JSON string and writes it * without expanding the tree. * * @param outputStream the stream to be written into * @param omOutputFormat format of the message, this is ignored. * @throws javax.xml.stream.XMLStreamException if there is an error while writing the message in to the output * stream. */ public void serialize(OutputStream outputStream, OMOutputFormat omOutputFormat) throws javax.xml.stream.XMLStreamException { try { String encoding = omOutputFormat == null ? "UTF-8" : omOutputFormat.getCharSetEncoding(); outputStream.write(getJSONString().getBytes(encoding)); } catch (IOException e) { throw new OMException(); } } /** * Writes JSON through the writer. As this should write JSON, it directly gets the JSON string and writes it without * expanding the tree. * * @param writer Writer to be written into * @param omOutputFormat format of the message, this is ignored. * @throws javax.xml.stream.XMLStreamException if there is an error while writing the message through the writer. */ public void serialize(Writer writer, OMOutputFormat omOutputFormat) throws javax.xml.stream.XMLStreamException { try { writer.write(getJSONString()); } catch (IOException e) { throw new OMException(); } } /** * Writes XML through the XMLStreamWriter. As the input data source is JSON, this method needs to get a StAX reader * from that JSON String. Therefore this uses the getReader() method to get the StAX reader writes the events into * the XMLStreamWriter. * * @param xmlStreamWriter StAX writer to be written into * @throws javax.xml.stream.XMLStreamException if there is an error while writing the message through the StAX * writer. */ public void serialize(javax.xml.stream.XMLStreamWriter xmlStreamWriter) throws javax.xml.stream.XMLStreamException { XMLStreamReader reader = getReader(); xmlStreamWriter.writeStartDocument(); while (reader.hasNext()) { int x = reader.next(); switch (x) { case XMLStreamConstants.START_ELEMENT: xmlStreamWriter.writeStartElement(reader.getPrefix(), reader.getLocalName(), reader .getNamespaceURI()); int namespaceCount = reader.getNamespaceCount(); for (int i = namespaceCount - 1; i >= 0; i--) { xmlStreamWriter.writeNamespace(reader.getNamespacePrefix(i), reader.getNamespaceURI(i)); } int attributeCount = reader.getAttributeCount(); for (int i = 0; i < attributeCount; i++) { xmlStreamWriter.writeAttribute(reader.getAttributePrefix(i), reader.getAttributeNamespace(i), reader.getAttributeLocalName(i), reader.getAttributeValue(i)); } break; case XMLStreamConstants.START_DOCUMENT: break; case XMLStreamConstants.CHARACTERS: xmlStreamWriter.writeCharacters(reader.getText()); break; case XMLStreamConstants.CDATA: xmlStreamWriter.writeCData(reader.getText()); break; case XMLStreamConstants.END_ELEMENT: xmlStreamWriter.writeEndElement(); break; case XMLStreamConstants.END_DOCUMENT: xmlStreamWriter.writeEndDocument(); break; case XMLStreamConstants.SPACE: break; case XMLStreamConstants.COMMENT: xmlStreamWriter.writeComment(reader.getText()); break; case XMLStreamConstants.DTD: xmlStreamWriter.writeDTD(reader.getText()); break; case XMLStreamConstants.PROCESSING_INSTRUCTION: xmlStreamWriter.writeProcessingInstruction(reader.getPITarget(), reader.getPIData()); break; case XMLStreamConstants.ENTITY_REFERENCE: xmlStreamWriter.writeEntityRef(reader.getLocalName()); break; default: throw new OMException(); } } xmlStreamWriter.writeEndDocument(); } /** * Gives the StAX reader using the "Mapped" formatted input JSON String. * * @return The XMLStreamReader according to the JSON String. * @throws javax.xml.stream.XMLStreamException if there is an error while making the StAX reader. */ public javax.xml.stream.XMLStreamReader getReader() throws javax.xml.stream.XMLStreamException { Map<String, String> nsMap = new HashMap<String, String>(); nsMap.put("", ""); // input factory for "Mapped" convention MappedXMLInputFactory inputFactory = new MappedXMLInputFactory(nsMap); String jsonString = this.getJSONString(); return inputFactory.createXMLStreamReader(new JSONTokener(jsonString)); } // returns the json string by consuming the JSON input stream. protected String getJSONString() { return json.toString(); } }