com.sun.xml.ws.rx.mc.runtime.McClientTube Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of webservices-rt Show documentation
Show all versions of webservices-rt Show documentation
This module contains the Metro runtime code.
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 1997-2017 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://oss.oracle.com/licenses/CDDL+GPL-1.1
* or LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
package com.sun.xml.ws.rx.mc.runtime;
import com.sun.xml.ws.rx.mc.dev.WsmcRuntimeProvider;
import com.sun.xml.ws.api.addressing.AddressingVersion;
import com.sun.xml.ws.api.addressing.WSEndpointReference;
import com.sun.xml.ws.api.message.Header;
import com.sun.xml.ws.api.message.MessageHeaders;
import com.sun.xml.ws.api.message.Message;
import com.sun.xml.ws.api.message.Packet;
import com.sun.xml.ws.api.pipe.Fiber;
import com.sun.xml.ws.api.pipe.NextAction;
import com.sun.xml.ws.api.pipe.Tube;
import com.sun.xml.ws.api.pipe.TubeCloner;
import com.sun.xml.ws.api.pipe.helper.AbstractFilterTubeImpl;
import com.sun.xml.ws.api.pipe.helper.AbstractTubeImpl;
import com.sun.xml.ws.api.server.Container;
import com.sun.istack.logging.Logger;
import com.sun.xml.ws.rx.RxRuntimeException;
import com.sun.xml.ws.rx.mc.localization.LocalizationMessages;
import com.sun.xml.ws.rx.mc.dev.ProtocolMessageHandler;
import com.sun.xml.ws.rx.util.Communicator;
import com.sun.xml.ws.rx.util.SuspendedFiberStorage;
import java.util.UUID;
import javax.xml.stream.XMLStreamException;
/**
*
* @author Marek Potociar
*/
public class McClientTube extends AbstractFilterTubeImpl implements WsmcRuntimeProvider {
//
private static final Logger LOGGER = Logger.getLogger(McClientTube.class);
//
private final McConfiguration configuration;
private final Header wsmcAnnonymousReplyToHeader;
private final Header wsmcAnnonymousFaultToHeader;
private final Communicator communicator;
private final SuspendedFiberStorage suspendedFiberStorage;
private final MakeConnectionSenderTask mcSenderTask;
private final WSEndpointReference wsmcAnonymousEndpointReference;
McClientTube(McConfiguration configuration, Tube tubelineHead, Container container) throws RxRuntimeException {
super(tubelineHead);
this.configuration = configuration;
this.communicator = Communicator.builder("mc-client-tube-communicator")
.tubelineHead(super.next)
.addressingVersion(configuration.getAddressingVersion())
.soapVersion(configuration.getSoapVersion())
.jaxbContext(configuration.getRuntimeVersion().getJaxbContext(configuration.getAddressingVersion()))
.container(container)
.build();
final String wsmcAnonymousAddress = configuration.getRuntimeVersion().getAnonymousAddress(UUID.randomUUID().toString());
this.wsmcAnonymousEndpointReference = new WSEndpointReference(wsmcAnonymousAddress, configuration.getAddressingVersion());
this.wsmcAnnonymousReplyToHeader = wsmcAnonymousEndpointReference.createHeader(configuration.getAddressingVersion().replyToTag);
this.wsmcAnnonymousFaultToHeader = wsmcAnonymousEndpointReference.createHeader(configuration.getAddressingVersion().faultToTag);
this.suspendedFiberStorage = new SuspendedFiberStorage();
this.mcSenderTask = new MakeConnectionSenderTask(
communicator,
suspendedFiberStorage,
wsmcAnonymousAddress,
wsmcAnnonymousReplyToHeader,
wsmcAnnonymousFaultToHeader,
configuration);
}
McClientTube(McClientTube original, TubeCloner cloner) {
super(original, cloner);
this.configuration = original.configuration;
this.wsmcAnnonymousReplyToHeader = original.wsmcAnnonymousReplyToHeader;
this.wsmcAnnonymousFaultToHeader = original.wsmcAnnonymousFaultToHeader;
this.communicator = original.communicator;
this.suspendedFiberStorage = original.suspendedFiberStorage;
this.mcSenderTask = original.mcSenderTask;
this.wsmcAnonymousEndpointReference = original.wsmcAnonymousEndpointReference;
}
@Override
public AbstractTubeImpl copy(TubeCloner cloner) {
LOGGER.entering();
try {
return new McClientTube(this, cloner);
} finally {
LOGGER.exiting();
}
}
@Override
public NextAction processRequest(Packet request) {
if (!mcSenderTask.isRunning() && !mcSenderTask.wasShutdown()) { // first packet
communicator.setDestinationAddressFrom(request); // setting actual destination endpoint
mcSenderTask.start(); // starting the McSenderTask
}
final Message message = request.getMessage();
if (!message.hasHeaders()) {
throw LOGGER.logSevereException(new RxRuntimeException(LocalizationMessages.WSMC_0102_NO_SOAP_HEADERS()));
}
if (needToSetWsmcAnnonymousHeaders(request)) { // annonymous request
setMcAnnonymousHeaders(
message.getHeaders(),
configuration.getAddressingVersion(),
wsmcAnnonymousReplyToHeader,
wsmcAnnonymousFaultToHeader);
String correlationId = message.getID(configuration.getAddressingVersion(), configuration.getSoapVersion());
Fiber.CompletionCallback responseHandler;
if (request.expectReply != null && request.expectReply) { // most likely Req-Resp MEP
responseHandler = new RequestResponseMepHandler(configuration, mcSenderTask, suspendedFiberStorage, correlationId);
} else { // most likely One-Way MEP
responseHandler = new OneWayMepHandler(configuration, mcSenderTask, suspendedFiberStorage, correlationId);
}
suspendedFiberStorage.register(correlationId, Fiber.current());
communicator.sendAsync(request, responseHandler);
return super.doSuspend();
} else { // not annonymous request - don't care
return super.processRequest(request);
}
}
@Override
public NextAction processResponse(Packet response) {
return super.processResponse(response);
}
@Override
public NextAction processException(Throwable t) {
return super.processException(t);
}
@Override
public void preDestroy() {
mcSenderTask.shutdown();
communicator.close();
super.preDestroy();
}
/**
* @see WsmcRuntimeProvider#getWsmcAnonymousEndpointReference()
*/
public final WSEndpointReference getWsmcAnonymousEndpointReference() {
return wsmcAnonymousEndpointReference;
}
/**
* @see WsmcRuntimeProvider#registerProtocolMessageHandler(com.sun.xml.ws.rx.mc.dev.ProtocolMessageHandler)
*/
public final void registerProtocolMessageHandler(ProtocolMessageHandler handler) {
mcSenderTask.register(handler);
}
/**
* Method check if the WS-A {@code ReplyTo} header is present or not. If it is not present, or if it is present but is annonymous,
* method returns true. If the WS-A {@code ReplyTo} header is present and non-annonymous, mehod returns false.
*/
private boolean needToSetWsmcAnnonymousHeaders(final Packet request) {
Header replyToHeader = request.getMessage().getHeaders().get(configuration.getAddressingVersion().replyToTag, false);
if (replyToHeader != null) {
try {
return replyToHeader.readAsEPR(configuration.getAddressingVersion()).isAnonymous();
} catch (XMLStreamException ex) {
throw LOGGER.logSevereException(new RxRuntimeException(LocalizationMessages.WSMC_0103_ERROR_RETRIEVING_WSA_REPLYTO_CONTENT(), ex));
}
}
// this request seems to be one-way, need to check if there are is an RM AckRequest set on it.
// FIXME: this should be made in a RM-agnostic way
return isBooleanFlagSet(request, McConfiguration.ACK_REQUESTED_HEADER_SET);
}
static void setMcAnnonymousHeaders(final MessageHeaders headers, AddressingVersion av, Header wsmcReplyToHeader, Header wsmcFaultToHeader) {
headers.remove(av.replyToTag);
headers.add(wsmcReplyToHeader);
if (headers.remove(av.faultToTag) != null) {
headers.add(wsmcFaultToHeader);
}
}
private Boolean isBooleanFlagSet(Packet packet, String flag) {
Boolean value = Boolean.class.cast(packet.invocationProperties.get(flag));
return value != null && value.booleanValue();
}
}