blob: ec08832aaad8153f3ad5aac50e31ab1d83f7123f [file] [log] [blame]
/*
* 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.axis2.json;
import org.apache.axiom.om.OMDataSource;
import org.apache.axiom.om.OMElement;
import org.apache.axiom.om.OMOutputFormat;
import org.apache.axiom.om.OMSourcedElement;
import org.apache.axiom.soap.SOAPFault;
import org.apache.axis2.AxisFault;
import org.apache.axis2.Constants;
import org.apache.axis2.context.MessageContext;
import org.apache.axis2.description.WSDL2Constants;
import org.apache.axis2.transport.MessageFormatter;
import org.apache.axis2.transport.http.util.URIEncoderDecoder;
import javax.xml.stream.XMLStreamException;
import javax.xml.stream.XMLStreamWriter;
import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.StringWriter;
import java.io.UnsupportedEncodingException;
import java.io.Writer;
import java.net.MalformedURLException;
import java.net.URL;
/**
* Base class for JSON message formatters.
*/
public abstract class AbstractJSONMessageFormatter implements MessageFormatter {
private final Class<? extends AbstractJSONDataSource> dataSourceClass;
/**
* Constructor.
*
* @param dataSourceClass
* the {@link OMDataSource} class corresponding to the JSON format used by this
* message formatter; this information is used for optimization (pass through)
*/
public AbstractJSONMessageFormatter(Class<? extends AbstractJSONDataSource> dataSourceClass) {
this.dataSourceClass = dataSourceClass;
}
public String getContentType(MessageContext msgCtxt, OMOutputFormat format,
String soapActionString) {
String contentType = (String)msgCtxt.getProperty(Constants.Configuration.CONTENT_TYPE);
String encoding = format.getCharSetEncoding();
if (contentType == null) {
contentType = (String)msgCtxt.getProperty(Constants.Configuration.MESSAGE_TYPE);
}
if (encoding != null) {
contentType += "; charset=" + encoding;
}
return contentType;
}
public String formatSOAPAction(MessageContext msgCtxt, OMOutputFormat format,
String soapActionString) {
return null;
}
private XMLStreamWriter getJSONWriter(OutputStream outStream, OMOutputFormat format, MessageContext messageContext)
throws AxisFault, XMLStreamException {
try {
return getJSONWriter(new OutputStreamWriter(outStream, format.getCharSetEncoding()), messageContext);
} catch (UnsupportedEncodingException ex) {
throw AxisFault.makeFault(ex);
}
}
//returns the "Mapped" JSON writer
protected abstract XMLStreamWriter getJSONWriter(Writer writer, MessageContext messageContext) throws XMLStreamException;
/**
* Get the original JSON string from the given element if it is available and if the element has
* not been modified.
*
* @param element
* the element
* @return the JSON string to write, or <code>null</code> if it is not available
*/
private String getStringToWrite(OMElement element) {
if (element instanceof OMSourcedElement) {
return (String)((OMSourcedElement)element).getObject(dataSourceClass);
} else {
return null;
}
}
/**
* Writes the JSON message to the output stream with the correct convention. If the payload is
* an OMSourcedElement and it contains a JSONDataSource with a correctly formatted JSON
* String, gets it directly from the DataSource and writes to the output stream. If not, the OM
* tree is expanded and it is serialized into the output stream. *
*
* @param msgCtxt Message context which contains the soap envelope to be written
* @param format format of the message, this is ignored
* @param out output stream to be written in to
* @param preserve ignored
* @throws AxisFault if there is an error in writing the message using StAX writer or IF THE
* USER TRIES TO SEND A JSON MESSAGE WITH NAMESPACES USING THE "MAPPED"
* CONVENTION.
*/
public void writeTo(MessageContext msgCtxt, OMOutputFormat format,
OutputStream out, boolean preserve) throws AxisFault {
OMElement element = msgCtxt.getEnvelope().getBody().getFirstElement();
try {
//Mapped format cannot handle element with namespaces.. So cannot handle Faults
if (element instanceof SOAPFault) {
SOAPFault fault = (SOAPFault)element;
OMElement element2 = element.getOMFactory().createOMElement("Fault", null);
element2.setText(fault.toString());
element = element2;
}
String jsonToWrite = getStringToWrite(element);
if (jsonToWrite != null) {
out.write(jsonToWrite.getBytes());
} else {
XMLStreamWriter jsonWriter = getJSONWriter(out, format, msgCtxt);
// Jettison v1.2+ relies on writeStartDocument being called (AXIS2-5044)
jsonWriter.writeStartDocument();
element.serializeAndConsume(jsonWriter);
jsonWriter.writeEndDocument();
}
} catch (IOException e) {
throw AxisFault.makeFault(e);
} catch (XMLStreamException e) {
throw AxisFault.makeFault(e);
}
}
public URL getTargetAddress(MessageContext msgCtxt, OMOutputFormat format, URL targetURL)
throws AxisFault {
String httpMethod =
(String)msgCtxt.getProperty(Constants.Configuration.HTTP_METHOD);
OMElement dataOut = msgCtxt.getEnvelope().getBody().getFirstElement();
//if the http method is GET, send the json string as a parameter
if (dataOut != null && (httpMethod != null)
&& Constants.Configuration.HTTP_METHOD_GET.equalsIgnoreCase(httpMethod)) {
try {
String jsonString = getStringToWrite(dataOut);
if (jsonString == null) {
StringWriter out = new StringWriter();
XMLStreamWriter jsonWriter = getJSONWriter(out, msgCtxt);
// Jettison v1.2+ relies on writeStartDocument being called (AXIS2-5044)
jsonWriter.writeStartDocument();
dataOut.serializeAndConsume(jsonWriter);
jsonWriter.writeEndDocument();
jsonString = out.toString();
}
jsonString = URIEncoderDecoder.quoteIllegal(jsonString,
WSDL2Constants.LEGAL_CHARACTERS_IN_URL);
String param = "query=" + jsonString;
String returnURLFile = targetURL.getFile() + "?" + param;
return new URL(targetURL.getProtocol(), targetURL.getHost(),
targetURL.getPort(), returnURLFile);
} catch (MalformedURLException e) {
throw AxisFault.makeFault(e);
} catch (XMLStreamException e) {
throw AxisFault.makeFault(e);
} catch (UnsupportedEncodingException e) {
throw AxisFault.makeFault(e);
}
} else {
return targetURL;
}
}
}