org.apache.cxf.transport.local.LocalDestination Maven / Gradle / Ivy
/**
* 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.cxf.transport.local;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PipedInputStream;
import java.io.PipedOutputStream;
import java.util.concurrent.Executor;
import java.util.logging.Logger;
import org.apache.cxf.common.logging.LogUtils;
import org.apache.cxf.io.AbstractWrappedOutputStream;
import org.apache.cxf.io.CachedOutputStream;
import org.apache.cxf.message.Exchange;
import org.apache.cxf.message.Message;
import org.apache.cxf.message.MessageImpl;
import org.apache.cxf.message.MessageUtils;
import org.apache.cxf.service.model.EndpointInfo;
import org.apache.cxf.transport.AbstractConduit;
import org.apache.cxf.transport.AbstractDestination;
import org.apache.cxf.transport.Conduit;
import org.apache.cxf.workqueue.SynchronousExecutor;
import org.apache.cxf.ws.addressing.EndpointReferenceType;
public class LocalDestination extends AbstractDestination {
private static final Logger LOG = LogUtils.getL7dLogger(LocalDestination.class);
private LocalTransportFactory localDestinationFactory;
public LocalDestination(LocalTransportFactory localDestinationFactory, EndpointReferenceType epr,
EndpointInfo ei) {
super(epr, ei);
this.localDestinationFactory = localDestinationFactory;
}
public void shutdown() {
localDestinationFactory.remove(this);
}
protected Logger getLogger() {
return LOG;
}
@Override
protected Conduit getInbuiltBackChannel(Message inMessage) {
Conduit conduit = (Conduit)inMessage.get(LocalConduit.IN_CONDUIT);
if (conduit instanceof LocalConduit) {
return new SynchronousConduit((LocalConduit)conduit);
}
return null;
}
class SynchronousConduit extends AbstractConduit {
private final class LocalDestinationOutputStream extends AbstractWrappedOutputStream {
private final Exchange exchange;
private final Message message;
private LocalDestinationOutputStream(Exchange exchange, Message message) {
this.exchange = exchange;
this.message = message;
}
public void close() throws IOException {
if (!written) {
dispatchToClient(true);
}
super.close();
}
protected void onFirstWrite() throws IOException {
dispatchToClient(false);
}
protected void dispatchToClient(boolean empty) throws IOException {
final MessageImpl m = new MessageImpl();
localDestinationFactory.copy(message, m);
if (!empty) {
final PipedInputStream stream = new PipedInputStream();
wrappedStream = new PipedOutputStream(stream);
m.setContent(InputStream.class, stream);
}
final Runnable receiver = new Runnable() {
public void run() {
if (exchange != null) {
exchange.setInMessage(m);
}
conduit.getMessageObserver().onMessage(m);
}
};
Executor ex = message.getExchange() != null
? message.getExchange().get(Executor.class) : null;
// Need to avoid to get the SynchronousExecutor
if (ex == null || SynchronousExecutor.isA(ex)) {
ex = localDestinationFactory.getExecutor();
if (ex != null) {
ex.execute(receiver);
} else {
new Thread(receiver).start();
}
} else {
ex.execute(receiver);
}
}
}
private LocalConduit conduit;
public SynchronousConduit(LocalConduit conduit) {
super(null);
this.conduit = conduit;
}
public void prepare(final Message message) throws IOException {
if (!Boolean.TRUE.equals(message.getExchange().get(LocalConduit.DIRECT_DISPATCH))) {
final Exchange exchange = (Exchange)message.getExchange().get(LocalConduit.IN_EXCHANGE);
AbstractWrappedOutputStream cout
= new LocalDestinationOutputStream(exchange, message);
message.setContent(OutputStream.class, cout);
} else {
CachedOutputStream stream = new CachedOutputStream();
message.setContent(OutputStream.class, stream);
message.setContent(CachedOutputStream.class, stream);
stream.holdTempFile();
}
}
@Override
public void close(Message message) throws IOException {
// set the pseudo status code if not set (REVISIT add this method in MessageUtils to be reused elsewhere?)
Integer i = (Integer)message.get(Message.RESPONSE_CODE);
if (i == null) {
int code = ((message.getExchange().isOneWay() && !MessageUtils.isPartialResponse(message))
|| MessageUtils.isEmptyPartialResponse(message)) ? 202 : 200;
message.put(Message.RESPONSE_CODE, code);
}
if (Boolean.TRUE.equals(message.getExchange().get(LocalConduit.DIRECT_DISPATCH))) {
final Exchange exchange = (Exchange)message.getExchange().get(LocalConduit.IN_EXCHANGE);
MessageImpl copy = new MessageImpl();
copy.putAll(message);
message.getContent(OutputStream.class).close();
CachedOutputStream stream = message.getContent(CachedOutputStream.class);
message.setContent(OutputStream.class, stream);
MessageImpl.copyContent(message, copy);
copy.setContent(InputStream.class, stream.getInputStream());
stream.releaseTempFileHold();
if (exchange != null && exchange.getInMessage() == null) {
exchange.setInMessage(copy);
}
conduit.getMessageObserver().onMessage(copy);
return;
}
super.close(message);
}
protected Logger getLogger() {
return LOG;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy