import java.util.List;
import java.util.Map;
import java.util.Objects;
-import java.util.concurrent.Callable;
-import java.util.concurrent.ExecutionException;
-import java.util.concurrent.ExecutorService;
-import java.util.concurrent.Executors;
-import java.util.concurrent.Future;
import java.util.stream.Collectors;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import org.opendaylight.controller.md.sal.binding.api.DataBroker;
-import org.opendaylight.controller.md.sal.binding.api.MountPoint;
import org.opendaylight.controller.md.sal.binding.api.MountPointService;
import org.opendaylight.controller.md.sal.binding.api.ReadOnlyTransaction;
import org.opendaylight.controller.md.sal.binding.api.ReadTransaction;
import org.opendaylight.groupbasedpolicy.renderer.vpp.lisp.info.container.HostRelatedInfoContainer;
import org.opendaylight.groupbasedpolicy.renderer.vpp.lisp.info.container.states.PhysicalInterfaces;
import org.opendaylight.groupbasedpolicy.renderer.vpp.nat.NatUtil;
+import org.opendaylight.groupbasedpolicy.renderer.vpp.util.MountedDataBrokerProvider;
import org.opendaylight.groupbasedpolicy.renderer.vpp.util.VppIidFactory;
import org.opendaylight.groupbasedpolicy.renderer.vpp.util.VppRendererProcessingException;
import org.opendaylight.groupbasedpolicy.util.DataStoreHelper;
import com.google.common.util.concurrent.FutureCallback;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
+import com.google.common.util.concurrent.MoreExecutors;
public class VppNodeManager {
- private static final short DURATION = 3000;
private static final TopologyId TOPOLOGY_ID = new TopologyId("topology-netconf");
private static final Logger LOG = LoggerFactory.getLogger(VppNodeManager.class);
private static final String V3PO_CAPABILITY = "(urn:opendaylight:params:xml:ns:yang:v3po?revision=2017-06-07)v3po";
private final List<String> requiredCapabilities;
private final MountPointService mountService;
private final HostRelatedInfoContainer hostRelatedInfoContainer = HostRelatedInfoContainer.getInstance();
+ private final MountedDataBrokerProvider mountProvider;
public VppNodeManager(@Nonnull final DataBroker dataBroker,
@Nonnull final BindingAwareBroker.ProviderContext session, @Nullable String physicalInterfaces) {
this.dataBroker = Preconditions.checkNotNull(dataBroker);
this.mountService = Preconditions.checkNotNull(session.getSALService(MountPointService.class));
+ this.mountProvider = new MountedDataBrokerProvider(mountService, dataBroker);
requiredCapabilities = initializeRequiredCapabilities();
if (!Strings.isNullOrEmpty(physicalInterfaces) && !Objects.equals(physicalInterfaces, NO_PUBLIC_INT_SPECIFIED)) {
loadPhysicalInterfaces(physicalInterfaces);
/**
* Synchronizes nodes to DataStore based on their modification state which results in
* create/update/remove of Node.
+ * @param dataAfter data after modification
+ * @param dataBefore data Before modification
*/
public void syncNodes(final Node dataAfter, final Node dataBefore) {
if (isControllerConfigNode(dataAfter, dataBefore)) {
public void onFailure(@Nonnull Throwable t) {
LOG.warn("Node synchronization failed. Data before: {} after {}", dataBefore, dataAfter);
}
- });
+ }, MoreExecutors.directExecutor());
}
private boolean isControllerConfigNode(final Node dataAfter, final Node dataBefore) {
final String message = String.format("Node %s is not connected", nodeId);
return Futures.immediateFuture(message);
}
- final DataBroker mountpoint = getNodeMountPoint(mountPointIid);
- if (mountpoint == null) {
+ final Optional<DataBroker> mountpoint = mountProvider.resolveDataBrokerForMountPoint(mountPointIid);
+ if (mountpoint.isPresent()) {
final String message = String.format("Mountpoint not available for node %s", nodeId);
return Futures.immediateFuture(message);
}
final boolean submit = DataStoreHelper.submitToDs(wTx);
if (submit) {
final String message = String.format("Node %s is capable and ready", nodeId);
- syncPhysicalInterfacesInLocalDs(mountpoint, mountPointIid);
- NatUtil.resolveOutboundNatInterface(mountpoint, mountPointIid, node.getNodeId(), extInterfaces);
+ syncPhysicalInterfacesInLocalDs(mountpoint.get(), mountPointIid);
+ NatUtil.resolveOutboundNatInterface(mountPointIid, node.getNodeId(), extInterfaces);
return Futures.immediateFuture(message);
} else {
final String message = String.format("Failed to resolve connected node %s", nodeId);
final RendererNode rendererNode = remapNode(mountPointIid);
final WriteTransaction wTx = dataBroker.newWriteOnlyTransaction();
wTx.delete(LogicalDatastoreType.OPERATIONAL, VppIidFactory.getRendererNodeIid(rendererNode));
- extInterfaces.remove(node.getNodeId());
final CheckedFuture<Void, TransactionCommitFailedException> checkedFuture = wTx.submit();
try {
checkedFuture.checkedGet();
}
}
- @Nullable
- private DataBroker getNodeMountPoint(final InstanceIdentifier<Node> mountPointIid) {
- final Future<Optional<MountPoint>> futureOptionalObject = getMountpointFromSal(mountPointIid);
- try {
- final Optional<MountPoint> optionalObject = futureOptionalObject.get();
- LOG.debug("Optional mountpoint object: {}", optionalObject);
- MountPoint mountPoint;
- if (optionalObject.isPresent()) {
- mountPoint = optionalObject.get();
- Optional<DataBroker> optionalDataBroker = mountPoint.getService(DataBroker.class);
- if (optionalDataBroker.isPresent()) {
- return optionalDataBroker.get();
- } else {
- LOG.warn("Cannot obtain data broker from mountpoint {}", mountPoint);
- }
- } else {
- LOG.warn("Cannot obtain mountpoint with IID {}", mountPointIid);
- }
- return null;
- } catch (ExecutionException | InterruptedException e) {
- LOG.warn("Unable to obtain mountpoint ... {}", e);
- return null;
- }
- }
-
private RendererNode remapNode(final InstanceIdentifier<Node> path) {
final RendererNodeBuilder rendererNodeBuilder = new RendererNodeBuilder();
rendererNodeBuilder.setKey(new RendererNodeKey(path)).setNodePath(path);
return Arrays.asList(capabilityEntries);
}
- // TODO bug 7699
- // This works as a workaround for mountpoint registration in cluster. If application is registered on different
- // node as netconf service, it obtains mountpoint registered by SlaveSalFacade (instead of MasterSalFacade). However
- // this service registers mountpoint a moment later then connectionStatus is set to "Connected". If NodeManager hits
- // state where device is connected but mountpoint is not yet available, try to get it again in a while
- private Future<Optional<MountPoint>> getMountpointFromSal(final InstanceIdentifier<Node> iid) {
- final ExecutorService executorService = Executors.newSingleThreadExecutor();
- final Callable<Optional<MountPoint>> task = () -> {
- byte attempt = 0;
- do {
- try {
- final Optional<MountPoint> optionalMountpoint = mountService.getMountPoint(iid);
- if (optionalMountpoint.isPresent()) {
- return optionalMountpoint;
- }
- LOG.warn("Mountpoint {} is not registered yet", iid);
- Thread.sleep(DURATION);
- } catch (InterruptedException e) {
- LOG.warn("Thread interrupted to ", e);
- }
- attempt++;
- } while (attempt <= 3);
- return Optional.absent();
- };
- return executorService.submit(task);
- }
-
private void syncPhysicalInterfacesInLocalDs(DataBroker mountPointDataBroker, InstanceIdentifier<Node> nodeIid) {
ReadWriteTransaction rwTx = dataBroker.newReadWriteTransaction();
ReadOnlyTransaction rTx = mountPointDataBroker.newReadOnlyTransaction();
java.util.Optional<PhysicalInterface> pubInt = rn.getAugmentation(VppInterfaceAugmentation.class)
.getPhysicalInterface()
.stream()
- .filter(phInt -> phInt.isExternal())
+ .filter(PhysicalInterface::isExternal)
.findFirst();
if (pubInt.isPresent()) {
nodes.put(rn.getNodePath().firstKeyOf(Node.class).getNodeId(), pubInt.get().getInterfaceName());