blob: ac750896bc24fa6abe1de8c5782b911c639ad3bd [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.axiom.ts.soap12.envelope;
import java.io.PipedInputStream;
import java.io.PipedOutputStream;
import java.util.Iterator;
import org.apache.axiom.blob.Blob;
import org.apache.axiom.mime.MultipartBody;
import org.apache.axiom.mime.activation.PartDataHandlerBlobFactory;
import org.apache.axiom.mime.activation.PartDataHandler;
import org.apache.axiom.om.OMElement;
import org.apache.axiom.om.OMMetaFactory;
import org.apache.axiom.om.OMOutputFormat;
import org.apache.axiom.om.OMText;
import org.apache.axiom.om.OMXMLBuilderFactory;
import org.apache.axiom.soap.SOAPBody;
import org.apache.axiom.soap.SOAPEnvelope;
import org.apache.axiom.soap.SOAPFactory;
import org.apache.axiom.testutils.blob.TestBlob;
import org.apache.axiom.testutils.io.IOTestUtils;
import org.apache.axiom.ts.AxiomTestCase;
import org.apache.axiom.util.activation.DataHandlerUtils;
/**
* Tests that attachments are streamed (i.e. not read entirely into memory) if the original message
* was read from an input stream and {@link OMElement#serializeAndConsume(java.io.OutputStream,
* OMOutputFormat)} is used to serialize the message. This feature is important because it allows
* projects such as Synapse to forward MTOM messages very efficiently.
*/
public class TestMTOMForwardStreaming extends AxiomTestCase {
private final boolean buildSOAPPart;
public TestMTOMForwardStreaming(OMMetaFactory metaFactory, boolean buildSOAPPart) {
super(metaFactory);
addTestParameter("buildSOAPPart", buildSOAPPart);
this.buildSOAPPart = buildSOAPPart;
}
@Override
protected void runTest() throws Throwable {
Blob blob1 = new TestBlob('A', Runtime.getRuntime().maxMemory());
Blob blob2 = new TestBlob('B', Runtime.getRuntime().maxMemory());
// Programmatically create the original message
SOAPFactory factory = metaFactory.getSOAP12Factory();
final SOAPEnvelope orgEnvelope = factory.createSOAPEnvelope();
SOAPBody orgBody = factory.createSOAPBody(orgEnvelope);
OMElement orgBodyElement =
factory.createOMElement(
"test", factory.createOMNamespace("urn:test", "p"), orgBody);
OMElement orgData1 = factory.createOMElement("data", null, orgBodyElement);
orgData1.addChild(factory.createOMText(blob1, true));
OMElement orgData2 = factory.createOMElement("data", null, orgBodyElement);
orgData2.addChild(factory.createOMText(blob2, true));
OMOutputFormat format = new OMOutputFormat();
format.setDoOptimize(true);
format.setSOAP11(false);
String contentType = format.getContentType();
PipedOutputStream pipe1Out = new PipedOutputStream();
PipedInputStream pipe1In = new PipedInputStream(pipe1Out);
// Create the producer thread (simulating the client sending the MTOM message)
Thread producerThread =
new Thread(
() -> {
try {
try {
orgEnvelope.serialize(pipe1Out, format);
} finally {
pipe1Out.close();
}
} catch (Exception ex) {
ex.printStackTrace();
}
});
producerThread.start();
PipedOutputStream pipe2Out = new PipedOutputStream();
PipedInputStream pipe2In = new PipedInputStream(pipe2Out);
// Create the forwarder thread (simulating the mediation engine that forwards the message)
Thread forwarderThread =
new Thread(
() -> {
try {
try {
MultipartBody mb =
MultipartBody.builder()
.setInputStream(pipe1In)
.setContentType(contentType)
.setPartBlobFactory(
PartDataHandlerBlobFactory.DEFAULT)
.build();
SOAPEnvelope envelope =
OMXMLBuilderFactory.createSOAPModelBuilder(
metaFactory, mb)
.getSOAPEnvelope();
// The code path executed by serializeAndConsume is
// significantly different if
// the element is built. Therefore we need two different
// test executions.
if (buildSOAPPart) {
envelope.build();
}
// Usage of serializeAndConsume should enable streaming
envelope.serializeAndConsume(pipe2Out, format);
} finally {
pipe2Out.close();
}
} catch (Exception ex) {
ex.printStackTrace();
}
});
forwarderThread.start();
try {
MultipartBody mb =
MultipartBody.builder()
.setInputStream(pipe2In)
.setContentType(contentType)
.setPartBlobFactory(PartDataHandlerBlobFactory.DEFAULT)
.build();
SOAPEnvelope envelope =
OMXMLBuilderFactory.createSOAPModelBuilder(metaFactory, mb).getSOAPEnvelope();
OMElement bodyElement = envelope.getBody().getFirstElement();
Iterator<OMElement> it = bodyElement.getChildElements();
OMElement data1 = it.next();
OMElement data2 = it.next();
IOTestUtils.compareStreams(
blob1.getInputStream(),
((PartDataHandler)
DataHandlerUtils.toDataHandler(
((OMText) data1.getFirstOMChild()).getBlob()))
.getPart()
.getInputStream(false));
IOTestUtils.compareStreams(
blob2.getInputStream(),
((PartDataHandler)
DataHandlerUtils.toDataHandler(
((OMText) data2.getFirstOMChild()).getBlob()))
.getPart()
.getInputStream(false));
} finally {
pipe2In.close();
}
}
}