com.digitalpetri.opcua.stack.client.fsm.states.Connecting Maven / Gradle / Ivy
/*
* Copyright 2015 Kevin Herron
*
* Licensed 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 com.digitalpetri.opcua.stack.client.fsm.states;
import java.util.Optional;
import java.util.concurrent.CompletableFuture;
import com.digitalpetri.opcua.stack.client.UaTcpStackClient;
import com.digitalpetri.opcua.stack.client.fsm.ConnectionEvent;
import com.digitalpetri.opcua.stack.client.fsm.ConnectionState;
import com.digitalpetri.opcua.stack.client.fsm.ConnectionStateFsm;
import com.digitalpetri.opcua.stack.core.StatusCodes;
import com.digitalpetri.opcua.stack.core.UaException;
import com.digitalpetri.opcua.stack.core.channel.ClientSecureChannel;
import com.digitalpetri.opcua.stack.core.types.builtin.StatusCode;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class Connecting implements ConnectionState {
private final Logger logger = LoggerFactory.getLogger(getClass());
private volatile ClientSecureChannel secureChannel;
private final CompletableFuture channelFuture;
public Connecting(CompletableFuture channelFuture) {
this.channelFuture = channelFuture;
}
@Override
public CompletableFuture activate(ConnectionEvent event, ConnectionStateFsm fsm) {
CompletableFuture future = new CompletableFuture<>();
connect(fsm, true, new CompletableFuture<>()).whenComplete((sc, ex) -> {
if (sc != null) {
secureChannel = sc;
fsm.handleEvent(ConnectionEvent.ConnectSucceeded);
} else {
channelFuture.completeExceptionally(ex);
fsm.handleEvent(ConnectionEvent.ConnectFailed);
}
future.complete(null);
});
return future;
}
private CompletableFuture connect(ConnectionStateFsm fsm,
boolean initialAttempt,
CompletableFuture future) {
UaTcpStackClient.bootstrap(fsm.getClient(), Optional.empty()).whenComplete((sc, ex) -> {
if (sc != null) {
logger.debug("Channel bootstrap succeeded: localAddress={}, remoteAddress={}",
sc.getChannel().localAddress(), sc.getChannel().remoteAddress());
future.complete(sc);
} else {
logger.debug("Channel bootstrap failed: {}", ex.getMessage(), ex);
StatusCode statusCode = UaException.extract(ex)
.map(UaException::getStatusCode)
.orElse(StatusCode.BAD);
boolean secureChannelError =
statusCode.getValue() == StatusCodes.Bad_TcpSecureChannelUnknown ||
statusCode.getValue() == StatusCodes.Bad_SecureChannelIdInvalid;
if (initialAttempt && secureChannelError) {
// Try again if bootstrapping failed because we couldn't re-open the previous channel.
logger.debug("Previous channel unusable, retrying...");
connect(fsm, false, future);
} else {
future.completeExceptionally(ex);
}
}
});
return future;
}
@Override
public CompletableFuture deactivate(ConnectionEvent event, ConnectionStateFsm fsm) {
return CF_VOID_COMPLETED;
}
@Override
public ConnectionState transition(ConnectionEvent event, ConnectionStateFsm fsm) {
switch (event) {
case ConnectSucceeded:
return new Connected(secureChannel, channelFuture);
case ConnectFailed:
return new Idle();
}
return this;
}
@Override
public CompletableFuture getSecureChannel() {
return channelFuture;
}
@Override
public String toString() {
return "Connecting{}";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy