blob: daf31e1d8c27ce638a214dfaf84ce157fd0a92b7 [file] [log] [blame]
/*
* Copyright 2018-present Open Networking Foundation
*
* 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 org.onosproject.odtn.internal;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.atomic.AtomicReference;
import org.onosproject.config.FailedException;
import org.onosproject.odtn.TapiResolver;
import org.onosproject.odtn.TapiConnectivityConfig;
import org.onosproject.odtn.TapiConnectivityService;
import org.onosproject.odtn.behaviour.OdtnDeviceDescriptionDiscovery;
import org.onosproject.odtn.utils.tapi.DcsBasedTapiObjectRefFactory;
import org.onosproject.odtn.utils.tapi.TapiCepPair;
import org.onosproject.odtn.utils.tapi.TapiConnection;
import org.onosproject.odtn.utils.tapi.TapiNepPair;
import org.onosproject.odtn.utils.tapi.TapiCepRefHandler;
import org.onosproject.odtn.utils.tapi.TapiConnectionHandler;
import org.onosproject.yang.gen.v1.tapiconnectivity.rev20181210.tapiconnectivity.connectivitycontext.DefaultConnection;
import org.onosproject.yang.gen.v1.tapiconnectivity.rev20181210.tapiconnectivity.connection.ConnectionEndPoint;
import org.onosproject.net.ConnectPoint;
import org.onosproject.odtn.utils.tapi.TapiNepRef;
import org.onosproject.odtn.utils.tapi.TapiRouteHandler;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import static org.onlab.osgi.DefaultServiceDirectory.getService;
/**
* DCS-dependent Tapi connection manager implementation.
*/
public final class DcsBasedTapiConnectionManager implements TapiConnectionManager {
private final Logger log = LoggerFactory.getLogger(getClass());
protected TapiPathComputer connectionController;
private TapiResolver resolver;
private TapiConnectivityService processor;
private List<DcsBasedTapiConnectionManager> connectionManagerList = new ArrayList<>();
private TapiConnection connection = null;
private TapiConnectionHandler connectionHandler = TapiConnectionHandler.create();
private Operation op = null;
enum Operation {
CREATE,
DELETE
}
private DcsBasedTapiConnectionManager() {
connectionController = DefaultTapiPathComputer.create();
resolver = getService(TapiResolver.class);
processor = getService(TapiConnectivityService.class);
}
public static DcsBasedTapiConnectionManager create() {
return new DcsBasedTapiConnectionManager();
}
@Override
public TapiConnectionHandler createConnection(TapiNepPair neps) {
// Calculate route
TapiConnection connection = connectionController.pathCompute(neps);
log.debug("Calculated path: {}", connection);
createConnectionRecursively(connection);
/*
* RCAS Create Intent: Assume that if the result of the pathCompute is flat
* and there are no lower connections, it has to be mapped to an intent
* It is a connection between line ports (OCh) with an OpticalConnectivity
* Intent.
*/
if (connection.getLowerConnections().isEmpty()) {
TapiNepRef left = neps.left();
TapiNepRef right = neps.right();
ConnectPoint leftConnectPoint = left.getConnectPoint();
ConnectPoint rightConnectPoint = right.getConnectPoint();
log.debug("Creating Intent from {} to {} {}",
leftConnectPoint,
rightConnectPoint,
connectionHandler.getId());
notifyTapiConnectivityChange(connectionHandler.getId().toString(),
leftConnectPoint,
rightConnectPoint,
true);
}
return connectionHandler;
}
@Override
public void deleteConnection(TapiConnectionHandler connectionHandler) {
// Retrieve the target to be deleted (right now we have the uuid)
connectionHandler.read();
// Remove Intent if exists
if (connectionHandler.getLowerConnections().isEmpty()) {
// Connection object
DefaultConnection connection = connectionHandler.getModelObject();
// These are two connection.ConnectionEndpoint (Actually Refs, mainly UUID)
ConnectionEndPoint cepLeft = connection.connectionEndPoint().get(0);
ConnectionEndPoint cepRight = connection.connectionEndPoint().get(1);
TapiNepRef left = TapiNepRef.create(
cepLeft.topologyUuid().toString(),
cepLeft.nodeUuid().toString(),
cepLeft.nodeEdgePointUuid().toString());
TapiNepRef right = TapiNepRef.create(
cepRight.topologyUuid().toString(),
cepRight.nodeUuid().toString(),
cepRight.nodeEdgePointUuid().toString());
// update with latest data in DCS
left = resolver.getNepRef(left);
right = resolver.getNepRef(right);
log.debug("Removing intent connection: {}", connection);
notifyTapiConnectivityChange(
connectionHandler.getId().toString(),
left.getConnectPoint(),
right.getConnectPoint(),
false);
}
deleteConnectionRecursively(connectionHandler);
}
@Override
public void apply() {
connectionManagerList.forEach(DcsBasedTapiConnectionManager::apply);
switch (op) {
case CREATE:
notifyDeviceConfigChange(true);
connectionHandler.add();
break;
case DELETE:
notifyDeviceConfigChange(false);
connectionHandler.remove();
break;
default:
throw new FailedException("Unknown operation type.");
}
}
/**
* Emit NetworkConfig event with parameters for device config,
* to notify configuration change to device drivers.
*/
private void notifyDeviceConfigChange(boolean enable) {
if (!this.connection.getCeps().isSameNode()) {
return;
}
TapiNepRef left = this.connection.getCeps().left().getNepRef();
TapiNepRef right = this.connection.getCeps().right().getNepRef();
// update with latest data in DCS
left = resolver.getNepRef(left);
right = resolver.getNepRef(right);
AtomicReference<TapiNepRef> line = new AtomicReference<>();
AtomicReference<TapiNepRef> client = new AtomicReference<>();
Arrays.asList(left, right).forEach(nep -> {
if (nep.getPortType() == OdtnDeviceDescriptionDiscovery.OdtnPortType.LINE) {
line.set(nep);
}
if (nep.getPortType() == OdtnDeviceDescriptionDiscovery.OdtnPortType.CLIENT) {
client.set(nep);
}
});
DeviceConfigEventEmitter eventEmitter = DeviceConfigEventEmitter.create();
eventEmitter.emit(line.get(), client.get(), enable);
}
/**
* Request Application to setup / release Intent.
*/
private void notifyTapiConnectivityChange(String uuid, ConnectPoint left, ConnectPoint right, boolean enable) {
TapiConnectivityConfig cfg = new TapiConnectivityConfig(uuid, left, right, enable);
processor.processTapiEvent(cfg);
}
/**
* Generate TAPI connection and its under connections recursively
* and add them to creation queue.
*
* @param connection connection to be created
*/
private void createConnectionRecursively(TapiConnection connection) {
op = Operation.CREATE;
connectionManagerList.clear();
this.connection = connection;
TapiRouteHandler routeBuilder = TapiRouteHandler.create();
// Create under connection, and set them into routeBuilder
connection.getLowerConnections().forEach(lowerConnection -> {
delegateConnectionCreation(lowerConnection);
routeBuilder.addCep(lowerConnection.getCeps().left());
routeBuilder.addCep(lowerConnection.getCeps().right());
});
connectionHandler.addRoute(routeBuilder.getModelObject());
connectionHandler.addCep(TapiCepRefHandler.create()
.setCep(connection.getCeps().left()).getModelObject());
connectionHandler.addCep(TapiCepRefHandler.create()
.setCep(connection.getCeps().right()).getModelObject());
connectionManagerList.forEach(manager ->
connectionHandler.addLowerConnection(manager.getConnectionHandler().getModelObject()));
}
/**
* Generate TAPI connection and its under connections recursively
* and add them to deletion queue.
*
* @param connectionHandler connectionHandler of connection to be deleted
*/
private void deleteConnectionRecursively(TapiConnectionHandler connectionHandler) {
op = Operation.DELETE;
connectionManagerList.clear();
log.info("model: {}", connectionHandler.getModelObject());
this.connection = TapiConnection.create(
TapiCepPair.create(
DcsBasedTapiObjectRefFactory.create(connectionHandler.getCeps().get(0)),
DcsBasedTapiObjectRefFactory.create(connectionHandler.getCeps().get(1)))
);
this.connectionHandler = connectionHandler;
this.connectionHandler.getLowerConnections().forEach(lowerConnectionHandler -> {
delegateConnectionDeletion(lowerConnectionHandler);
});
}
/**
* Delegate lower-connection creation to other corresponding TapiConnectionManager of each Nodes.
*
* @param connection connection to be created
*/
private void delegateConnectionCreation(TapiConnection connection) {
log.info("ceps: {}", connection.getCeps());
DcsBasedTapiConnectionManager manager = DcsBasedTapiConnectionManager.create();
manager.createConnectionRecursively(connection);
connectionManagerList.add(manager);
}
/**
* Delegate lower-connection deletion to other corresponding TapiConnectionManager of each Nodes.
*
* @param connectionHandler connectionHandler of connection to be deleted
*/
private void delegateConnectionDeletion(TapiConnectionHandler connectionHandler) {
log.info("model: {}", connectionHandler.getModelObject());
DcsBasedTapiConnectionManager manager = DcsBasedTapiConnectionManager.create();
manager.deleteConnectionRecursively(connectionHandler);
connectionManagerList.add(manager);
}
public TapiConnectionHandler getConnectionHandler() {
return connectionHandler;
}
}