问题描述
您好,我想知道如何(正确)处理 OPC UA milo 的重新订阅。目前我使用此代码来创建订阅:
private CompletableFuture<DataValue> subscribe(NodeId nodetoSubscribe,double samplingInterval,Consumer<DataValue> onChangeDo){
CompletableFuture<DataValue> result = new CompletableFuture<>();
try{
//sets parameters of subscription
ManagedSubscription subscription = ManagedSubscription.create(client,samplingInterval);
subscription.setDefaultSamplingInterval(samplingInterval);
subscription.setDefaultQueueSize(uint(10));
//adds "onChange" action
ManagedDataItem managedDataItem = subscription.createDataItem(nodetoSubscribe,item -> item.addDataValueListener(onChangeDo));
//wait till first value was read
ManagedDataItem.DataValueListener listener = managedDataItem.addDataValueListener(result::complete);
result.whenComplete((v,e) -> managedDataItem.removeDataValueListener(listener));
} catch (UaException e){
result.completeExceptionally(e);
}
return result;
}
但是每当服务器重新启动或客户端和服务器之间的连接中断时,订阅就不起作用。
我可以添加会话活动侦听器,并在连接中断时清除这样的订阅管理器(星星之间的代码)。
addSessionActivityListener(new SessionActivityListener(){
@Override
public void onSessionActive(UaSession session) {
log.info("Connecting PLC with IP address {}",ipAddress);
setConnectionStatus(ConnectionStatus.CONNECTED);
**subscribeAll();**
}
@Override
public void onSessionInactive(UaSession session) {
log.info("disconnecting PLC with IP address {}",ipAddress);
setConnectionStatus(ConnectionStatus.disCONNECTED);
**uaClient.getSubscriptionManager().clearSubscriptions();**
}
});
但也许有更好的方法来处理这个问题。我所有的代码都可以在这里找到GitHub
更新: 根据凯文哈伦的回复,我可以看到它应该自动完成,但每次服务器断开连接时我都会收到此信息(我的订阅日志和状态代码):
ON SUBSCRIPTION TRANSFER Failed: subscription org.eclipse.milo.opcua.sdk.client.subscriptions.OpcuaSubscription@3f344a79,status code StatusCode{name=Bad_ServiceUnsupported,value=0x800B0000,quality=bad}
如您所见,可能存在问题,该服务器不支持订阅转移。 Kevin 建议我应该使用 SubscriptionListener::onSubscriptionTransferFailed
来处理,所以我是这样实现的:
uaClient.getSubscriptionManager().addSubscriptionListener(new UaSubscriptionManager.SubscriptionListener() {
@Override
public void onSubscriptionTransferFailed(UaSubscription subscription,StatusCode statusCode) {
uaClient.getSubscriptionManager().clearSubscriptions();
subscribeAll();
}
它接缝工作,但我想知道这是最好的方法还是我错过了什么。谢谢大家的回答。
解决方法
正确的方法大多是什么都不做 - 重新连接和重新订阅的所有细节都会自动处理。
您需要处理的唯一情况是在创建新会话后无法转移订阅。实现 SubscriptionListener::onSubscriptionTransferFailed
以重新创建此回调指示传输失败的任何订阅。
实际上没有任何其他情况需要您手动清除和重新创建订阅。 onPublishFailure() 回调主要提供信息,不需要操作。 onSessionActive() 和 onSessionInactive() 仅提供信息,不需要您执行任何操作。