Adding auto-layout to the access-null simulation.
Enhancing the scale test for flows and routes.
Change-Id: Ib91720b409872e44eaff4263cf229bffa2e292fc
diff --git a/apps/dhcp/app/src/main/java/org/onosproject/dhcp/impl/DhcpManager.java b/apps/dhcp/app/src/main/java/org/onosproject/dhcp/impl/DhcpManager.java
index 103c9d3..f18b861 100644
--- a/apps/dhcp/app/src/main/java/org/onosproject/dhcp/impl/DhcpManager.java
+++ b/apps/dhcp/app/src/main/java/org/onosproject/dhcp/impl/DhcpManager.java
@@ -186,6 +186,7 @@
@Deactivate
protected void deactivate() {
+ componentConfigService.unregisterProperties(getClass(), false);
cfgService.removeListener(cfgListener);
factories.forEach(cfgService::unregisterConfigFactory);
packetService.removeProcessor(processor);
diff --git a/apps/test/route-scale/src/main/java/org/onosproject/routescale/CreateFlows.java b/apps/test/route-scale/src/main/java/org/onosproject/routescale/CreateFlows.java
index 6515def..8ab8e8f 100644
--- a/apps/test/route-scale/src/main/java/org/onosproject/routescale/CreateFlows.java
+++ b/apps/test/route-scale/src/main/java/org/onosproject/routescale/CreateFlows.java
@@ -17,23 +17,25 @@
import org.apache.karaf.shell.commands.Argument;
import org.apache.karaf.shell.commands.Command;
+import org.onosproject.cfg.ComponentConfigService;
import org.onosproject.cli.AbstractShellCommand;
/**
* Creates the specified number of routes for scale testing.
*/
-@Command(scope = "onos", name = "scale-create-flows",
- description = "Creates the specified number of flows for scale testing")
+@Command(scope = "onos", name = "scale-flows",
+ description = "Sets the specified number of flows for scale testing")
public class CreateFlows extends AbstractShellCommand {
- @Argument(index = 0, name = "flowCount", description = "Number of flows to create",
+ @Argument(index = 0, name = "flowCount", description = "Number of flows to maintain",
required = true)
int flowCount;
@Override
protected void execute() {
- ScaleTestManager service = get(ScaleTestManager.class);
- service.createFlows(flowCount);
+ ComponentConfigService service = get(ComponentConfigService.class);
+ service.setProperty("org.onosproject.routescale.ScaleManager",
+ "flowCount", String.valueOf(flowCount));
}
}
diff --git a/apps/test/route-scale/src/main/java/org/onosproject/routescale/CreateRoutes.java b/apps/test/route-scale/src/main/java/org/onosproject/routescale/CreateRoutes.java
index b43d2bb..d738b74 100644
--- a/apps/test/route-scale/src/main/java/org/onosproject/routescale/CreateRoutes.java
+++ b/apps/test/route-scale/src/main/java/org/onosproject/routescale/CreateRoutes.java
@@ -17,23 +17,26 @@
import org.apache.karaf.shell.commands.Argument;
import org.apache.karaf.shell.commands.Command;
+import org.onosproject.cfg.ComponentConfigService;
import org.onosproject.cli.AbstractShellCommand;
/**
* Creates the specified number of routes for scale testing.
*/
-@Command(scope = "onos", name = "scale-create-routes",
- description = "Creates the specified number of routes for scale testing")
+@Command(scope = "onos", name = "scale-routes",
+ description = "Sets the specified number of routes for scale testing")
public class CreateRoutes extends AbstractShellCommand {
- @Argument(index = 0, name = "routeCount", description = "Number of routes to create",
+ @Argument(index = 0, name = "routeCount", description = "Number of routes to maintain",
required = true)
int routeCount;
@Override
protected void execute() {
- ScaleTestManager service = get(ScaleTestManager.class);
- service.createRoutes(routeCount);
+ ComponentConfigService service = get(ComponentConfigService.class);
+ service.setProperty("org.onosproject.routescale.ScaleManager",
+ "routeCount", String.valueOf(routeCount));
+
}
}
diff --git a/apps/test/route-scale/src/main/java/org/onosproject/routescale/ScaleTestManager.java b/apps/test/route-scale/src/main/java/org/onosproject/routescale/ScaleTestManager.java
index 65c1c7f..25550f8 100644
--- a/apps/test/route-scale/src/main/java/org/onosproject/routescale/ScaleTestManager.java
+++ b/apps/test/route-scale/src/main/java/org/onosproject/routescale/ScaleTestManager.java
@@ -21,6 +21,8 @@
import org.apache.felix.scr.annotations.Activate;
import org.apache.felix.scr.annotations.Component;
import org.apache.felix.scr.annotations.Deactivate;
+import org.apache.felix.scr.annotations.Modified;
+import org.apache.felix.scr.annotations.Property;
import org.apache.felix.scr.annotations.Reference;
import org.apache.felix.scr.annotations.ReferenceCardinality;
import org.apache.felix.scr.annotations.Service;
@@ -29,29 +31,39 @@
import org.onlab.packet.IpPrefix;
import org.onlab.packet.MacAddress;
import org.onosproject.app.ApplicationService;
+import org.onosproject.cfg.ComponentConfigService;
import org.onosproject.core.ApplicationId;
import org.onosproject.net.Device;
import org.onosproject.net.DeviceId;
import org.onosproject.net.Host;
+import org.onosproject.net.MastershipRole;
import org.onosproject.net.PortNumber;
import org.onosproject.net.device.DeviceService;
import org.onosproject.net.flow.DefaultFlowRule;
import org.onosproject.net.flow.DefaultTrafficSelector;
import org.onosproject.net.flow.DefaultTrafficTreatment;
+import org.onosproject.net.flow.FlowEntry;
import org.onosproject.net.flow.FlowRule;
import org.onosproject.net.flow.FlowRuleOperations;
import org.onosproject.net.flow.FlowRuleService;
import org.onosproject.net.flow.TrafficSelector;
import org.onosproject.net.flow.TrafficTreatment;
import org.onosproject.net.host.HostAdminService;
+import org.onosproject.net.link.LinkService;
import org.onosproject.routeservice.Route;
import org.onosproject.routeservice.RouteAdminService;
+import org.osgi.service.component.ComponentContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
+import java.util.Dictionary;
import java.util.List;
+import java.util.Objects;
import java.util.Random;
+import static com.google.common.base.Strings.isNullOrEmpty;
+import static org.onlab.util.Tools.get;
+
@Component(immediate = true)
@Service(value = ScaleTestManager.class)
public class ScaleTestManager {
@@ -68,11 +80,25 @@
protected HostAdminService hostAdminService;
@Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
+ protected LinkService linkService;
+
+ @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
protected FlowRuleService flowRuleService;
@Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
protected RouteAdminService routeAdminService;
+ @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
+ protected ComponentConfigService componentConfigService;
+
+ @Property(name = "flowCount", intValue = 0,
+ label = "Number of flows to be maintained in the system")
+ private int flowCount = 0;
+
+ @Property(name = "routeCount", intValue = 0,
+ label = "Number of routes to be maintained in the system")
+ private int routeCount = 0;
+
private final Random random = new Random(System.currentTimeMillis());
private ApplicationId appId;
@@ -80,38 +106,128 @@
@Activate
protected void activate() {
appId = applicationService.getId("org.onosproject.routescale");
+ componentConfigService.registerProperties(getClass());
log.info("Started");
}
@Deactivate
protected void deactivate() {
+ componentConfigService.unregisterProperties(getClass(), false);
log.info("Stopped");
}
- public void createFlows(int flowCount) {
- for (Device device : deviceService.getAvailableDevices()) {
- DeviceId id = device.id();
- FlowRuleOperations.Builder ops = FlowRuleOperations.builder();
+ @Modified
+ public void modified(ComponentContext context) {
+ if (context == null) {
+ return;
+ }
- for (int i = 0; i < flowCount; i++) {
- FlowRule.Builder frb = DefaultFlowRule.builder();
- frb.fromApp(appId).makePermanent().withPriority(1000 + i);
- TrafficSelector.Builder tsb = DefaultTrafficSelector.builder();
- TrafficTreatment.Builder ttb = DefaultTrafficTreatment.builder();
+ Dictionary<?, ?> properties = context.getProperties();
+ try {
+ String s = get(properties, "flowCount");
+ flowCount = isNullOrEmpty(s) ? flowCount : Integer.parseInt(s.trim());
- tsb.matchEthType(Ethernet.TYPE_IPV4);
- ttb.setEthDst(randomMac()).setEthSrc(randomMac());
- ttb.setOutput(PortNumber.portNumber(random.nextInt(512)));
- frb.withSelector(tsb.build()).withTreatment(ttb.build());
- ops.add(frb.forDevice(id).build());
- }
+ s = get(properties, "routeCount");
+ routeCount = isNullOrEmpty(s) ? routeCount : Integer.parseInt(s.trim());
- flowRuleService.apply(ops.build());
+ log.info("Reconfigured; flowCount={}; routeCount={}", flowCount, routeCount);
+ adjustFlows();
+ adjustRoutes();
+
+ } catch (NumberFormatException | ClassCastException e) {
+ log.warn("Misconfigured", e);
}
}
- public void createRoutes(int routeCount) {
+ private void adjustFlows() {
+ int deviceCount = deviceService.getAvailableDeviceCount();
+ if (deviceCount == 0) {
+ return;
+ }
+
+ int flowsPerDevice = flowCount / deviceCount;
+ for (Device device : deviceService.getAvailableDevices()) {
+ DeviceId id = device.id();
+ if (deviceService.getRole(id) != MastershipRole.MASTER ||
+ flowsPerDevice == 0) {
+ continue;
+ }
+
+ int currentFlowCount = flowRuleService.getFlowRuleCount(id);
+ if (flowsPerDevice > currentFlowCount) {
+ addMoreFlows(flowsPerDevice, device, id, currentFlowCount);
+
+ } else if (flowsPerDevice < currentFlowCount) {
+ removeExcessFlows(flowsPerDevice, id, currentFlowCount);
+ }
+ }
+ }
+
+ private void addMoreFlows(int flowsPerDevice, Device device, DeviceId id,
+ int currentFlowCount) {
+ int c = flowsPerDevice - currentFlowCount;
+ log.info("Adding {} flows for device {}", c, id);
+ List<PortNumber> ports = devicePorts(device);
+ FlowRuleOperations.Builder ops = FlowRuleOperations.builder();
+ for (int i = 0; i < c; i++) {
+ FlowRule.Builder frb = DefaultFlowRule.builder();
+ frb.fromApp(appId).makePermanent().withPriority(1000 + i);
+ TrafficSelector.Builder tsb = DefaultTrafficSelector.builder();
+ TrafficTreatment.Builder ttb = DefaultTrafficTreatment.builder();
+
+ tsb.matchEthType(Ethernet.TYPE_IPV4);
+ ttb.setEthDst(randomMac()).setEthSrc(randomMac());
+ ttb.setOutput(randomPort(ports));
+ frb.withSelector(tsb.build()).withTreatment(ttb.build());
+ ops.add(frb.forDevice(id).build());
+ }
+ flowRuleService.apply(ops.build());
+ }
+
+ private void removeExcessFlows(int flowsPerDevice, DeviceId id,
+ int currentFlowCount) {
+ FlowRuleOperations.Builder ops = FlowRuleOperations.builder();
+ int c = flowsPerDevice - currentFlowCount;
+ log.info("Removing {} flows from device {}", c, id);
+ for (FlowEntry e : flowRuleService.getFlowEntries(id)) {
+ if (Objects.equals(e.appId(), appId.id()) && c > 0) {
+ ops.remove(e);
+ c--;
+ }
+ }
+ flowRuleService.apply(ops.build());
+ }
+
+ private void adjustRoutes() {
+ int currentRouteCount =
+ routeAdminService.getRouteTables().parallelStream()
+ .mapToInt(t -> routeAdminService.getRoutes(t).size()).sum();
+ if (currentRouteCount < routeCount) {
+ createRoutes(routeCount - currentRouteCount);
+ } else if (currentRouteCount > routeCount) {
+ removeRoutes(currentRouteCount - routeCount);
+ }
+ }
+
+ // Returns a list of ports on the given device that have either links or
+ // hosts connected to them.
+ private List<PortNumber> devicePorts(Device device) {
+ DeviceId id = device.id();
+ ImmutableList.Builder<PortNumber> ports = ImmutableList.builder();
+ linkService.getDeviceEgressLinks(id).forEach(l -> ports.add(l.src().port()));
+ hostAdminService.getConnectedHosts(id)
+ .forEach(h -> h.locations().stream()
+ .filter(l -> Objects.equals(id, l.elementId()))
+ .findFirst()
+ .ifPresent(l -> ports.add(l.port())));
+ return ports.build();
+ }
+
+ // Creates the specified number of random routes. Such routes are generated
+ // using random IP prefices with next hop being an IP address of a randomly
+ // chosen hosts.
+ private void createRoutes(int routeCount) {
List<Host> hosts = ImmutableList.copyOf(hostAdminService.getHosts());
ImmutableSet.Builder<Route> routes = ImmutableSet.builder();
for (int i = 0; i < routeCount; i++) {
@@ -122,21 +238,34 @@
routeAdminService.update(routes.build());
}
+ // Removes the specified number of routes chosen at random.
+ private void removeRoutes(int routeCount) {
+ log.warn("Not implemented yet");
+ }
+
+ // Generates a random IP address.
private IpAddress randomIp() {
byte[] bytes = new byte[4];
random.nextBytes(bytes);
return IpAddress.valueOf(IpAddress.Version.INET, bytes, 0);
}
+ // Generates a random MAC address.
+ private MacAddress randomMac() {
+ byte[] bytes = new byte[6];
+ random.nextBytes(bytes);
+ return MacAddress.valueOf(bytes);
+ }
+
+ // Returns IP address of a host randomly chosen from the specified list.
private IpAddress randomIp(List<Host> hosts) {
Host host = hosts.get(random.nextInt(hosts.size()));
return host.ipAddresses().iterator().next();
}
- private MacAddress randomMac() {
- byte[] bytes = new byte[6];
- random.nextBytes(bytes);
- return MacAddress.valueOf(bytes);
+ // Returns port number randomly chosen from the given list of port numbers.
+ private PortNumber randomPort(List<PortNumber> ports) {
+ return ports.get(random.nextInt(ports.size()));
}
}
diff --git a/providers/null/src/main/java/org/onosproject/provider/nil/NullFlowRuleProvider.java b/providers/null/src/main/java/org/onosproject/provider/nil/NullFlowRuleProvider.java
index beaeb6a..f118f09 100644
--- a/providers/null/src/main/java/org/onosproject/provider/nil/NullFlowRuleProvider.java
+++ b/providers/null/src/main/java/org/onosproject/provider/nil/NullFlowRuleProvider.java
@@ -74,12 +74,18 @@
@Override
public void applyFlowRule(FlowRule... flowRules) {
- throw new UnsupportedOperationException("Cannot apply individual flow rules");
+ for (FlowRule rule : flowRules) {
+ flowTable.getOrDefault(rule.deviceId(), Sets.newConcurrentHashSet())
+ .add(new DefaultFlowEntry(rule));
+ }
}
@Override
public void removeFlowRule(FlowRule... flowRules) {
- throw new UnsupportedOperationException("Cannot remove individual flow rules");
+ for (FlowRule rule : flowRules) {
+ flowTable.getOrDefault(rule.deviceId(), Sets.newConcurrentHashSet())
+ .remove(new DefaultFlowEntry(rule));
+ }
}
@Override
diff --git a/providers/null/src/main/java/org/onosproject/provider/nil/cli/CreateNullHosts.java b/providers/null/src/main/java/org/onosproject/provider/nil/cli/CreateNullHosts.java
index 0495038..c9b9301 100644
--- a/providers/null/src/main/java/org/onosproject/provider/nil/cli/CreateNullHosts.java
+++ b/providers/null/src/main/java/org/onosproject/provider/nil/cli/CreateNullHosts.java
@@ -19,9 +19,12 @@
import org.apache.karaf.shell.commands.Argument;
import org.apache.karaf.shell.commands.Command;
import org.onlab.packet.IpAddress;
+import org.onlab.util.Tools;
import org.onosproject.net.ConnectPoint;
import org.onosproject.net.HostId;
import org.onosproject.net.HostLocation;
+import org.onosproject.net.config.NetworkConfigService;
+import org.onosproject.net.config.basics.BasicHostConfig;
import org.onosproject.provider.nil.CustomTopologySimulator;
import org.onosproject.provider.nil.NullProviders;
import org.onosproject.provider.nil.TopologySimulator;
@@ -35,6 +38,8 @@
description = "Adds a simulated end-station host to the custom topology simulation")
public class CreateNullHosts extends CreateNullEntity {
+ private static final double NONE = -99999999.9999;
+
@Argument(index = 0, name = "deviceName", description = "Name of device where hosts are attached",
required = true)
String deviceName = null;
@@ -47,9 +52,26 @@
required = true)
int hostCount = 0;
+ @Argument(index = 3, name = "gridY", description = "Grid y-coord for top of host block")
+ double gridY = NONE;
+
+ @Argument(index = 4, name = "gridX", description = "Grid X-coord for center of host block")
+ double gridX = NONE;
+
+ @Argument(index = 5, name = "hostsPerRow", description = "Number of hosts to render per row in block")
+ int hostsPerRow = 5;
+
+ @Argument(index = 6, name = "rowGap", description = "Y gap between rows")
+ double rowGap = 70;
+
+ @Argument(index = 7, name = "colGap", description = "X gap between rows")
+ double colGap = 50;
+
+
@Override
protected void execute() {
NullProviders service = get(NullProviders.class);
+ NetworkConfigService cfgService = get(NetworkConfigService.class);
TopologySimulator simulator = service.currentSimulator();
if (!validateSimulator(simulator)) {
@@ -60,10 +82,27 @@
List<ConnectPoint> points = findAvailablePorts(sim.deviceId(deviceName));
String pattern = hostIpPattern.replace("*", "%d");
+ double yStep = rowGap / hostsPerRow;
+ double y = gridY;
+ double x = gridX - (colGap * (hostsPerRow - 1)) / 2;
+
for (int h = 0; h < hostCount; h++) {
HostLocation location = new HostLocation(points.get(h), System.currentTimeMillis());
IpAddress ip = IpAddress.valueOf(String.format(pattern, h));
HostId id = sim.nextHostId();
+
+ if (gridY != NONE) {
+ BasicHostConfig cfg = cfgService.addConfig(id, BasicHostConfig.class);
+ setUiCoordinates(cfg, GRID, y, x);
+ if (((h + 1) % hostsPerRow) == 0) {
+ x = gridX - (colGap * (hostsPerRow - 1)) / 2;
+ } else {
+ x += colGap;
+ y += yStep;
+ }
+ }
+
+ Tools.delay(10);
sim.createHost(id, location, ip);
}
}
diff --git a/tools/package/bin/onos-service b/tools/package/bin/onos-service
index 799bb2f..db7b006 100755
--- a/tools/package/bin/onos-service
+++ b/tools/package/bin/onos-service
@@ -4,7 +4,7 @@
# -----------------------------------------------------------------------------
# uncomment the following line for performance testing
-#export JAVA_OPTS="${JAVA_OPTS:--Xms8G -Xmx8G -XX:+UseConcMarkSweepGC -XX:+CMSIncrementalMode -XX:+PrintGCDetails -XX:+PrintGCTimeStamps}"
+export JAVA_OPTS="${JAVA_OPTS:--Xms16G -Xmx16G -XX:+UseConcMarkSweepGC -XX:+CMSIncrementalMode -XX:+PrintGCDetails -XX:+PrintGCTimeStamps}"
# uncomment the following line for Netty TLS encryption
# Do modify the keystore location/password and truststore location/password accordingly
diff --git a/tools/test/topos/access-null b/tools/test/topos/access-null
index 3ecefc3..460fb6b 100755
--- a/tools/test/topos/access-null
+++ b/tools/test/topos/access-null
@@ -62,7 +62,7 @@
}
function y {
- let p="${3:-300} * ($1 - 1) - (${3:-300} * ($2 - 1)) / 2"
+ let p="${3:-400} * ($1 - 1) - (${3:-400} * ($2 - 1)) / 2 + ${4:-0}"
echo $p
}
@@ -88,8 +88,9 @@
# Create hosts for each leaf group; multi-homed to each leaf in the pair
[ $pair = A ] && pn=1 || pn=2
+ [ $pair = A ] && offset=-400 || offset=400
for host in $(seq 1 $serviceHosts); do
- sim "null-create-host Leaf-${pair}1,Leaf-${pair}2 10.${pn}.1.${host}"
+ sim "null-create-host Leaf-${pair}1,Leaf-${pair}2 10.${pn}.1.${host} -400 $(y $host $serviceHosts 60 $offset) grid"
done
done
@@ -104,7 +105,7 @@
done
# Create hosts for each access single leaf
- sim "null-create-hosts Access-${access} 10.1${access}.1.*" $accessHosts
+ sim "null-create-hosts Access-${access} 10.1${access}.1.*" $accessHosts 500 $(y $access $accessLeaves) 6
# sim "null-create-hosts Access-${access} 10.1${access}.2.*" $accessHosts
done