2 * Copyright 2014-2015 Open Networking Laboratory
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
16 package org.onosproject.xosintegration;
18 import com.eclipsesource.json.JsonArray;
19 import com.eclipsesource.json.JsonObject;
20 import com.google.common.collect.Maps;
21 import com.sun.jersey.api.client.Client;
22 import com.sun.jersey.api.client.ClientHandlerException;
23 import com.sun.jersey.api.client.ClientResponse;
24 import com.sun.jersey.api.client.WebResource;
25 import com.sun.jersey.api.client.filter.HTTPBasicAuthFilter;
26 import org.apache.felix.scr.annotations.Activate;
27 import org.apache.felix.scr.annotations.Component;
28 import org.apache.felix.scr.annotations.Deactivate;
29 import org.apache.felix.scr.annotations.Modified;
30 import org.apache.felix.scr.annotations.Property;
31 import org.apache.felix.scr.annotations.Reference;
32 import org.apache.felix.scr.annotations.ReferenceCardinality;
33 import org.apache.felix.scr.annotations.Service;
34 import org.onlab.packet.VlanId;
35 import org.onlab.util.Tools;
36 import org.onosproject.cfg.ComponentConfigService;
37 import org.onosproject.core.ApplicationId;
38 import org.onosproject.core.CoreService;
39 import org.onosproject.net.ConnectPoint;
40 import org.onosproject.net.DeviceId;
41 import org.onosproject.net.PortNumber;
42 import org.onosproject.net.flow.DefaultTrafficSelector;
43 import org.onosproject.net.flow.DefaultTrafficTreatment;
44 import org.onosproject.net.flow.TrafficSelector;
45 import org.onosproject.net.flow.TrafficTreatment;
46 import org.onosproject.net.flowobjective.DefaultForwardingObjective;
47 import org.onosproject.net.flowobjective.FlowObjectiveService;
48 import org.onosproject.net.flowobjective.ForwardingObjective;
49 import org.osgi.service.component.ComponentContext;
50 import org.slf4j.Logger;
52 import java.util.Dictionary;
55 import java.util.stream.Collectors;
56 import java.util.stream.IntStream;
58 import static com.google.common.base.Strings.isNullOrEmpty;
59 import static com.google.common.net.MediaType.JSON_UTF_8;
60 import static java.net.HttpURLConnection.HTTP_CREATED;
61 import static java.net.HttpURLConnection.HTTP_NO_CONTENT;
62 import static java.net.HttpURLConnection.HTTP_OK;
63 import static org.slf4j.LoggerFactory.getLogger;
67 * XOS interface application.
69 @Component(immediate = true)
71 public class OnosXOSIntegrationManager implements VoltTenantService {
72 private static final String XOS_SERVER_ADDRESS_PROPERTY_NAME =
74 private static final String XOS_SERVER_PORT_PROPERTY_NAME =
76 private static final String XOS_PROVIDER_SERVICE_PROPERTY_NAME =
79 private static final String TEST_XOS_SERVER_ADDRESS = "10.254.1.22";
80 private static final int TEST_XOS_SERVER_PORT = 8000;
81 private static final String XOS_TENANT_BASE_URI = "/xoslib/volttenant/";
82 private static final int TEST_XOS_PROVIDER_SERVICE = 1;
84 private static final int PRIORITY = 50000;
85 private static final DeviceId FABRIC_DEVICE_ID = DeviceId.deviceId("of:5e3e486e73000187");
86 private static final PortNumber FABRIC_OLT_CONNECT_POINT = PortNumber.portNumber(2);
87 private static final PortNumber FABRIC_VCPE_CONNECT_POINT = PortNumber.portNumber(3);
88 private static final String FABRIC_CONTROLLER_ADDRESS = "10.0.3.136";
89 private static final int FABRIC_SERVER_PORT = 8181;
90 private static final String FABRIC_BASE_URI = "/onos/cordfabric/vlans/add";
92 private static final DeviceId OLT_DEVICE_ID = DeviceId.deviceId("of:90e2ba82f97791e9");
93 private static final int OLT_UPLINK_PORT = 129;
95 private static final ConnectPoint FABRIC_PORT = new ConnectPoint(
96 DeviceId.deviceId("of:000090e2ba82f974"),
97 PortNumber.portNumber(2));
99 private final Logger log = getLogger(getClass());
100 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
101 protected CoreService coreService;
102 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
103 protected ComponentConfigService cfgService;
105 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
106 protected FlowObjectiveService flowObjectiveService;
108 @Property(name = XOS_SERVER_ADDRESS_PROPERTY_NAME,
109 value = TEST_XOS_SERVER_ADDRESS,
110 label = "XOS Server address")
111 protected String xosServerAddress = TEST_XOS_SERVER_ADDRESS;
113 @Property(name = XOS_SERVER_PORT_PROPERTY_NAME,
114 intValue = TEST_XOS_SERVER_PORT,
115 label = "XOS Server port")
116 protected int xosServerPort = TEST_XOS_SERVER_PORT;
118 @Property(name = XOS_PROVIDER_SERVICE_PROPERTY_NAME,
119 intValue = TEST_XOS_PROVIDER_SERVICE,
120 label = "XOS Provider Service")
121 protected int xosProviderService = TEST_XOS_PROVIDER_SERVICE;
123 private ApplicationId appId;
124 private Map<String, ConnectPoint> nodeToPort;
125 private Map<Long, Short> portToVlan;
126 private Map<ConnectPoint, String> portToSsid;
129 public void activate(ComponentContext context) {
130 log.info("XOS app is starting");
131 cfgService.registerProperties(getClass());
132 appId = coreService.registerApplication("org.onosproject.xosintegration");
136 readComponentConfiguration(context);
138 log.info("XOS({}) started", appId.id());
142 public void deactivate() {
143 cfgService.unregisterProperties(getClass(), false);
144 log.info("XOS({}) stopped", appId.id());
148 public void modified(ComponentContext context) {
149 readComponentConfiguration(context);
152 private void setupMap() {
153 nodeToPort = Maps.newHashMap();
155 nodeToPort.put("cordcompute01.onlab.us", new ConnectPoint(FABRIC_DEVICE_ID,
156 PortNumber.portNumber(4)));
158 nodeToPort.put("cordcompute02.onlab.us", new ConnectPoint(FABRIC_DEVICE_ID,
159 PortNumber.portNumber(3)));
161 portToVlan = Maps.newHashMap();
162 portToVlan.putIfAbsent(1L, (short) 201);
163 portToVlan.putIfAbsent(6L, (short) 401);
165 portToSsid = Maps.newHashMap();
166 portToSsid.put(new ConnectPoint(OLT_DEVICE_ID, PortNumber.portNumber(1)), "0");
167 portToSsid.put(new ConnectPoint(FABRIC_DEVICE_ID, PortNumber.portNumber(6)), "1");
171 * Converts a JSON representation of a tenant into a tenant object.
173 * @param jsonTenant JSON object representing the tenant
174 * @return volt tenant object
176 private VoltTenant jsonToTenant(JsonObject jsonTenant) {
177 return VoltTenant.builder()
178 .withHumanReadableName(jsonTenant.get("humanReadableName").asString())
179 .withId(jsonTenant.get("id").asInt())
180 .withProviderService(jsonTenant.get("provider_service").asInt())
181 .withServiceSpecificId(jsonTenant.get("service_specific_id").asString())
182 .withVlanId(jsonTenant.get("vlan_id").asString())
187 * Converts a tenant object into a JSON string.
189 * @param tenant volt tenant object to convert
190 * @return JSON string for the tenant
192 private String tenantToJson(VoltTenant tenant) {
194 + "\"humanReadableName\": \"" + tenant.humanReadableName() + "\","
195 + "\"id\": \"" + tenant.id() + "\","
196 + "\"provider_service\": \"" + tenant.providerService() + "\","
197 + "\"service_specific_id\": \"" + tenant.serviceSpecificId() + "\","
198 + "\"vlan_id\": \"" + tenant.vlanId() + "\""
203 * Gets a client web resource builder for the base XOS REST API
204 * with no additional URI.
206 * @return web resource builder
207 * @deprecated in Cardinal Release
210 private WebResource.Builder getClientBuilder() {
211 return getClientBuilder("");
215 * Gets a client web resource builder for the base XOS REST API
216 * with an optional additional URI.
218 * @return web resource builder
219 * @deprecated in Cardinal Release
222 private WebResource.Builder getClientBuilder(String uri) {
223 String baseUrl = "http://" + xosServerAddress + ":"
224 + Integer.toString(xosServerPort);
225 Client client = Client.create();
226 client.addFilter(new HTTPBasicAuthFilter("padmin@vicci.org", "letmein"));
227 WebResource resource = client.resource(baseUrl
228 + XOS_TENANT_BASE_URI + uri);
229 return resource.accept(JSON_UTF_8.toString())
230 .type(JSON_UTF_8.toString());
234 * Performs a REST GET operation on the base XOS REST URI.
236 * @return JSON string fetched by the GET operation
237 * @deprecated in Cardinal Release
240 private String getRest() {
245 * Performs a REST GET operation on the base XOS REST URI with
246 * an optional additional URI.
248 * @return JSON string fetched by the GET operation
249 * @deprecated in Cardinal Release
252 private String getRest(String uri) {
253 WebResource.Builder builder = getClientBuilder(uri);
254 ClientResponse response = builder.get(ClientResponse.class);
256 if (response.getStatus() != HTTP_OK) {
257 log.info("REST GET request returned error code {}",
258 response.getStatus());
260 String jsonString = response.getEntity(String.class);
261 log.info("JSON read:\n{}", jsonString);
267 * Performs a REST POST operation of a json string on the base
268 * XOS REST URI with an optional additional URI.
270 * @param json JSON string to post
271 * @deprecated in Cardinal Release
274 private String postRest(String json) {
275 WebResource.Builder builder = getClientBuilder();
276 ClientResponse response;
279 response = builder.post(ClientResponse.class, json);
280 } catch (ClientHandlerException e) {
281 log.warn("Unable to contact REST server: {}", e.getMessage());
282 return "{ 'error' : 'oops no one home' }";
285 if (response.getStatus() != HTTP_CREATED) {
286 log.info("REST POST request returned error code {}",
287 response.getStatus());
289 return response.getEntity(String.class);
293 * Performs a REST DELETE operation on the base
294 * XOS REST URI with an optional additional URI.
296 * @param uri optional additional URI
297 * @deprecated in Cardinal Release
300 private void deleteRest(String uri) {
301 WebResource.Builder builder = getClientBuilder(uri);
302 ClientResponse response = builder.delete(ClientResponse.class);
304 if (response.getStatus() != HTTP_NO_CONTENT) {
305 log.info("REST DELETE request returned error code {}",
306 response.getStatus());
311 * Deletes the tenant with the given ID.
313 * @param tenantId ID of tenant to delete
315 private void deleteTenant(long tenantId) {
316 deleteRest(Long.toString(tenantId));
320 public Set<VoltTenant> getAllTenants() {
321 String jsonString = getRest();
323 JsonArray voltTenantItems = JsonArray.readFrom(jsonString);
325 return IntStream.range(0, voltTenantItems.size())
326 .mapToObj(index -> jsonToTenant(voltTenantItems.get(index).asObject()))
327 .collect(Collectors.toSet());
331 public void removeTenant(long id) {
336 public VoltTenant addTenant(VoltTenant newTenant) {
337 long providerServiceId = newTenant.providerService();
338 if (providerServiceId == -1) {
339 providerServiceId = xosProviderService;
342 PortNumber onuPort = newTenant.port().port();
343 VlanId subscriberVlan = VlanId.vlanId(portToVlan.get(onuPort.toLong()));
345 VoltTenant tenantToCreate = VoltTenant.builder()
346 .withProviderService(providerServiceId)
347 .withServiceSpecificId(portToSsid.get(newTenant.port()))
348 .withVlanId(String.valueOf(subscriberVlan.toShort()))
349 .withPort(newTenant.port())
351 String json = tenantToJson(tenantToCreate);
354 provisionVlanOnPort(OLT_DEVICE_ID, OLT_UPLINK_PORT, onuPort, subscriberVlan.toShort());
356 String retJson = postRest(json);
358 fetchCPELocation(tenantToCreate, retJson);
363 private void fetchCPELocation(VoltTenant newTenant, String jsonString) {
364 JsonObject json = JsonObject.readFrom(jsonString);
366 if (json.get("computeNodeName") != null) {
367 ConnectPoint point = nodeToPort.get(json.get("computeNodeName").asString());
368 //ConnectPoint fromPoint = newTenant.port();
369 ConnectPoint oltPort = new ConnectPoint(FABRIC_DEVICE_ID, FABRIC_OLT_CONNECT_POINT);
371 provisionFabric(VlanId.vlanId(Short.parseShort(newTenant.vlanId())),
378 public VoltTenant getTenant(long id) {
379 String jsonString = getRest(Long.toString(id));
380 JsonObject jsonTenant = JsonObject.readFrom(jsonString);
381 if (jsonTenant.get("id") != null) {
382 return jsonToTenant(jsonTenant);
388 private void provisionVlanOnPort(DeviceId deviceId, int uplinkPort, PortNumber p, short vlanId) {
390 TrafficSelector upstream = DefaultTrafficSelector.builder()
391 .matchVlanId(VlanId.ANY)
395 TrafficSelector downstream = DefaultTrafficSelector.builder()
396 .matchVlanId(VlanId.vlanId(vlanId))
397 .matchInPort(PortNumber.portNumber(uplinkPort))
400 TrafficTreatment upstreamTreatment = DefaultTrafficTreatment.builder()
401 .setVlanId(VlanId.vlanId(vlanId))
402 .setOutput(PortNumber.portNumber(uplinkPort))
405 TrafficTreatment downstreamTreatment = DefaultTrafficTreatment.builder()
411 ForwardingObjective upFwd = DefaultForwardingObjective.builder()
412 .withFlag(ForwardingObjective.Flag.VERSATILE)
415 .withSelector(upstream)
417 .withTreatment(upstreamTreatment)
420 ForwardingObjective downFwd = DefaultForwardingObjective.builder()
421 .withFlag(ForwardingObjective.Flag.VERSATILE)
424 .withSelector(downstream)
426 .withTreatment(downstreamTreatment)
429 flowObjectiveService.forward(deviceId, upFwd);
430 flowObjectiveService.forward(deviceId, downFwd);
434 private void provisionDataPlane(VoltTenant tenant) {
435 VlanId vlan = VlanId.vlanId(Short.parseShort(tenant.vlanId()));
437 TrafficSelector fromGateway = DefaultTrafficSelector.builder()
438 .matchInPhyPort(tenant.port().port())
441 TrafficSelector fromFabric = DefaultTrafficSelector.builder()
442 .matchInPhyPort(FABRIC_PORT.port())
446 TrafficTreatment toFabric = DefaultTrafficTreatment.builder()
449 .setOutput(FABRIC_PORT.port())
452 TrafficTreatment toGateway = DefaultTrafficTreatment.builder()
454 .setOutput(tenant.port().port())
457 ForwardingObjective forwardToFabric = DefaultForwardingObjective.builder()
458 .withFlag(ForwardingObjective.Flag.VERSATILE)
459 .withPriority(PRIORITY)
462 .withSelector(fromGateway)
463 .withTreatment(toFabric)
466 ForwardingObjective forwardToGateway = DefaultForwardingObjective.builder()
467 .withFlag(ForwardingObjective.Flag.VERSATILE)
468 .withPriority(PRIORITY)
471 .withSelector(fromFabric)
472 .withTreatment(toGateway)
475 flowObjectiveService.forward(FABRIC_PORT.deviceId(), forwardToFabric);
476 flowObjectiveService.forward(FABRIC_PORT.deviceId(), forwardToGateway);
479 private void provisionFabric(VlanId vlanId, ConnectPoint point, ConnectPoint fromPoint) {
481 long vlan = vlanId.toShort();
483 JsonObject node = new JsonObject();
484 node.add("vlan", vlan);
486 node.add("iptv", true);
488 node.add("iptv", false);
490 JsonArray array = new JsonArray();
491 JsonObject cp1 = new JsonObject();
492 JsonObject cp2 = new JsonObject();
493 cp1.add("device", point.deviceId().toString());
494 cp1.add("port", point.port().toLong());
495 cp2.add("device", fromPoint.deviceId().toString());
496 cp2.add("port", fromPoint.port().toLong());
499 node.add("ports", array);
502 String baseUrl = "http://" + FABRIC_CONTROLLER_ADDRESS + ":"
503 + Integer.toString(FABRIC_SERVER_PORT);
504 Client client = Client.create();
505 WebResource resource = client.resource(baseUrl + FABRIC_BASE_URI);
506 WebResource.Builder builder = resource.accept(JSON_UTF_8.toString())
507 .type(JSON_UTF_8.toString());
510 builder.post(ClientResponse.class, node.toString());
511 } catch (ClientHandlerException e) {
512 log.warn("Unable to contact fabric REST server: {}", e.getMessage());
518 * Extracts properties from the component configuration context.
520 * @param context the component context
522 private void readComponentConfiguration(ComponentContext context) {
523 Dictionary<?, ?> properties = context.getProperties();
525 String newXosServerAddress =
526 Tools.get(properties, XOS_SERVER_ADDRESS_PROPERTY_NAME);
527 if (!isNullOrEmpty(newXosServerAddress)) {
528 xosServerAddress = newXosServerAddress;
531 String newXosServerPortString =
532 Tools.get(properties, XOS_SERVER_PORT_PROPERTY_NAME);
533 if (!isNullOrEmpty(newXosServerPortString)) {
534 xosServerPort = Integer.parseInt(newXosServerPortString);
537 String newXosProviderServiceString =
538 Tools.get(properties, XOS_PROVIDER_SERVICE_PROPERTY_NAME);
539 if (!isNullOrEmpty(newXosProviderServiceString)) {
540 xosProviderService = Integer.parseInt(newXosProviderServiceString);