blob: a0e3de21f571f4d235bc9c1c74ae6abf86d29196 [file] [log] [blame]
YuanyouZhange3f396f2015-08-05 18:38:09 +08001/*
2 * Copyright 2015 Open Networking Laboratory
3 *
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
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
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.
15 */
16package org.onosproject.ovsdb.controller.impl;
17
18import static com.google.common.base.Preconditions.checkNotNull;
19
20import java.math.BigInteger;
21import java.util.HashSet;
22import java.util.List;
23import java.util.Map;
24import java.util.Set;
25import java.util.concurrent.ConcurrentHashMap;
26import java.util.concurrent.CopyOnWriteArraySet;
27import java.util.concurrent.ExecutionException;
28
29import org.apache.felix.scr.annotations.Activate;
30import org.apache.felix.scr.annotations.Component;
31import org.apache.felix.scr.annotations.Deactivate;
32import org.apache.felix.scr.annotations.Service;
33import org.onlab.packet.IpAddress;
34import org.onlab.packet.MacAddress;
35import org.onosproject.ovsdb.controller.DefaultEventSubject;
36import org.onosproject.ovsdb.controller.EventSubject;
37import org.onosproject.ovsdb.controller.OvsdbClientService;
38import org.onosproject.ovsdb.controller.OvsdbConstant;
39import org.onosproject.ovsdb.controller.OvsdbController;
40import org.onosproject.ovsdb.controller.OvsdbDatapathId;
41import org.onosproject.ovsdb.controller.OvsdbEvent;
42import org.onosproject.ovsdb.controller.OvsdbEvent.Type;
43import org.onosproject.ovsdb.controller.OvsdbEventListener;
44import org.onosproject.ovsdb.controller.OvsdbIfaceId;
45import org.onosproject.ovsdb.controller.OvsdbNodeId;
46import org.onosproject.ovsdb.controller.OvsdbNodeListener;
47import org.onosproject.ovsdb.controller.OvsdbPortName;
48import org.onosproject.ovsdb.controller.OvsdbPortNumber;
49import org.onosproject.ovsdb.controller.OvsdbPortType;
50import org.onosproject.ovsdb.controller.driver.OvsdbAgent;
51import org.onosproject.ovsdb.rfc.jsonrpc.Callback;
52import org.onosproject.ovsdb.rfc.message.TableUpdate;
53import org.onosproject.ovsdb.rfc.message.TableUpdates;
54import org.onosproject.ovsdb.rfc.message.UpdateNotification;
55import org.onosproject.ovsdb.rfc.notation.OvsdbMap;
56import org.onosproject.ovsdb.rfc.notation.OvsdbSet;
57import org.onosproject.ovsdb.rfc.notation.Row;
58import org.onosproject.ovsdb.rfc.notation.UUID;
59import org.onosproject.ovsdb.rfc.schema.DatabaseSchema;
60import org.onosproject.ovsdb.rfc.table.Bridge;
61import org.onosproject.ovsdb.rfc.table.Interface;
62import org.onosproject.ovsdb.rfc.table.OvsdbTable;
63import org.onosproject.ovsdb.rfc.table.Port;
64import org.onosproject.ovsdb.rfc.table.TableGenerator;
65import org.onosproject.ovsdb.rfc.utils.FromJsonUtil;
66import org.osgi.service.component.ComponentContext;
67import org.slf4j.Logger;
68import org.slf4j.LoggerFactory;
69
70import com.fasterxml.jackson.databind.JsonNode;
71
72/**
73 * The implementation of OvsdbController.
74 */
75@Component(immediate = true)
76@Service
77public class OvsdbControllerImpl implements OvsdbController {
78
79 public static final Logger log = LoggerFactory
80 .getLogger(OvsdbControllerImpl.class);
81
82 protected ConcurrentHashMap<OvsdbNodeId, OvsdbClientService> ovsdbClients =
83 new ConcurrentHashMap<OvsdbNodeId, OvsdbClientService>();
84
85 protected OvsdbAgent agent = new InternalOvsdbNodeAgent();
86 protected InternalMonitorCallBack updateCallback = new InternalMonitorCallBack();
87
88 protected Set<OvsdbNodeListener> ovsdbNodeListener = new CopyOnWriteArraySet<>();
89 protected Set<OvsdbEventListener> ovsdbEventListener = new CopyOnWriteArraySet<>();
90
91 protected ConcurrentHashMap<String, OvsdbClientService> requestNotification =
92 new ConcurrentHashMap<String, OvsdbClientService>();
93
94 protected ConcurrentHashMap<String, String> requestDbName = new ConcurrentHashMap<String, String>();
95
96 private final Controller controller = new Controller();
97
98 @Activate
99 public void activate(ComponentContext context) {
100 controller.start(agent, updateCallback);
101 log.info("Started");
102 }
103
104 @Deactivate
105 public void deactivate() {
106 controller.stop();
107 log.info("Stoped");
108 }
109
110 @Override
111 public void addNodeListener(OvsdbNodeListener listener) {
112 if (!ovsdbNodeListener.contains(listener)) {
113 this.ovsdbNodeListener.add(listener);
114 }
115 }
116
117 @Override
118 public void removeNodeListener(OvsdbNodeListener listener) {
119 this.ovsdbNodeListener.remove(listener);
120 }
121
122 @Override
123 public void addOvsdbEventListener(OvsdbEventListener listener) {
124 if (!ovsdbEventListener.contains(listener)) {
125 this.ovsdbEventListener.add(listener);
126 }
127 }
128
129 @Override
130 public void removeOvsdbEventListener(OvsdbEventListener listener) {
131 this.ovsdbEventListener.remove(listener);
132 }
133
134 @Override
135 public List<OvsdbNodeId> getNodeIds() {
136 // TODO Auto-generated method stub
137 return null;
138 }
139
140 @Override
141 public OvsdbClientService getOvsdbClient(OvsdbNodeId nodeId) {
142 return ovsdbClients.get(nodeId);
143 }
144
145 /**
146 * Implementation of an Ovsdb Agent which is responsible for keeping track
147 * of connected node and the state in which they are.
148 */
149 private class InternalOvsdbNodeAgent implements OvsdbAgent {
150 @Override
151 public void addConnectedNode(OvsdbNodeId nodeId,
152 OvsdbClientService ovsdbClient) {
153
154 if (ovsdbClients.get(nodeId) != null) {
155 return;
156 } else {
157 ovsdbClients.put(nodeId, ovsdbClient);
158
159 try {
160 List<String> dbNames = ovsdbClient.listDbs().get();
161 for (String dbName : dbNames) {
162 DatabaseSchema dbSchema;
163 dbSchema = ovsdbClient.getOvsdbSchema(dbName).get();
164
165 log.debug("Begin to monitor tables");
166 String id = java.util.UUID.randomUUID().toString();
167 TableUpdates updates = ovsdbClient
168 .monitorTables(dbName, id).get();
169
170 requestDbName.put(id, dbName);
171 requestNotification.put(id, ovsdbClient);
172
173 if (updates != null) {
174 processTableUpdates(ovsdbClient, updates,
175 dbSchema.name());
176 }
177 }
178 } catch (InterruptedException e) {
179 log.warn("Interrupted while waiting to get message from ovsdb");
180 Thread.currentThread().interrupt();
181 } catch (ExecutionException e) {
182 log.error("Exception thrown while to get message from ovsdb");
183 }
184
185 log.debug("Add node to north");
186 for (OvsdbNodeListener l : ovsdbNodeListener) {
187 l.nodeAdded(nodeId);
188 }
189 return;
190 }
191 }
192
193 @Override
194 public void removeConnectedNode(OvsdbNodeId nodeId) {
195 ovsdbClients.remove(nodeId);
196 log.debug("Node connection is removed");
197 for (OvsdbNodeListener l : ovsdbNodeListener) {
198 l.nodeRemoved(nodeId);
199 }
200 }
201 }
202
203 /**
204 * Processes table updates.
205 *
206 * @param clientService OvsdbClientService instance
207 * @param updates TableUpdates instance
208 * @param dbName ovsdb database name
209 */
210 private void processTableUpdates(OvsdbClientService clientService,
211 TableUpdates updates, String dbName)
212 throws InterruptedException {
213 checkNotNull(clientService, "OvsdbClientService is not null");
214
215 DatabaseSchema dbSchema = clientService.getDatabaseSchema(dbName);
216
217 for (String tableName : updates.result().keySet()) {
218 TableUpdate update = updates.result().get(tableName);
219 for (UUID uuid : (Set<UUID>) update.rows().keySet()) {
220 log.debug("Begin to process table updates uuid: {}, databaseName: {}, tableName: {}",
221 uuid.value(), dbName, tableName);
222
223 Row row = clientService.getRow(dbName, tableName, uuid.value());
224 clientService.updateOvsdbStore(dbName, tableName, uuid.value(),
225 update.getNew(uuid));
226 if (update.getNew(uuid) != null) {
227 boolean isNewRow = (row == null) ? true : false;
228 if (isNewRow) {
229 if (OvsdbConstant.PORT.equals(tableName)) {
230 dispatchEvent(clientService, update.getNew(uuid),
231 null, OvsdbEvent.Type.PORT_ADDED,
232 dbSchema);
233 }
234 }
235 } else if (update.getOld(uuid) != null) {
236 clientService.removeRow(dbName, tableName, uuid.toString());
237 if (update.getOld(uuid) != null) {
238 if (OvsdbConstant.PORT.equals(tableName)) {
239 dispatchEvent(clientService, null,
240 update.getOld(uuid),
241 OvsdbEvent.Type.PORT_REMOVED,
242 dbSchema);
243 }
244 }
245 }
246 }
247 }
248 }
249
250 /**
251 * Dispatches event to the north.
252 *
253 * @param clientService OvsdbClientService instance
254 * @param newRow a new row
255 * @param oldRow an old row
256 * @param eventType type of event
257 * @param dbSchema ovsdb database schema
258 */
259 private void dispatchEvent(OvsdbClientService clientService, Row newRow,
260 Row oldRow, Type eventType,
261 DatabaseSchema dbSchema) {
262 Port port = null;
263 if (OvsdbEvent.Type.PORT_ADDED.equals(eventType)) {
264 port = (Port) TableGenerator.getTable(dbSchema, newRow,
265 OvsdbTable.PORT);
266 } else if (OvsdbEvent.Type.PORT_REMOVED.equals(eventType)) {
267 port = (Port) TableGenerator.getTable(dbSchema, oldRow,
268 OvsdbTable.PORT);
269 }
270 if (port == null) {
271 return;
272 }
273
274 long dpid = getDataPathid(clientService, dbSchema);
275 OvsdbSet intfUuidSet = (OvsdbSet) port.getInterfacesColumn().data();
276 @SuppressWarnings({ "unchecked" })
277 Set<UUID> intfUuids = intfUuidSet.set();
278 for (UUID intfUuid : intfUuids) {
279 Row intfRow = clientService
280 .getRow(OvsdbConstant.DATABASENAME, "Interface",
281 intfUuid.toString());
282 if (intfRow == null) {
283 continue;
284 }
285 Interface intf = (Interface) TableGenerator
286 .getTable(dbSchema, intfRow, OvsdbTable.INTERFACE);
287
288 String portType = (String) intf.getTypeColumn().data();
289 long localPort = getOfPort(intf);
290 String[] macAndIfaceId = getMacAndIfaceid(intf);
291 if (macAndIfaceId == null) {
292 return;
293 }
294 EventSubject eventSubject = new DefaultEventSubject(
295 MacAddress
296 .valueOf(macAndIfaceId[0]),
297 new HashSet<IpAddress>(),
298 new OvsdbPortName(port.getName()),
299 new OvsdbPortNumber(localPort),
300 new OvsdbDatapathId(Long.toString(dpid)),
301 new OvsdbPortType(portType),
302 new OvsdbIfaceId(macAndIfaceId[1]));
303 for (OvsdbEventListener listener : ovsdbEventListener) {
304 listener.handle(new OvsdbEvent<EventSubject>(eventType,
305 eventSubject));
306 }
307
308 }
309
310 }
311
312 /**
313 * Gets mac and iface from the table Interface.
314 *
315 * @param intf Interface instance
316 * @return attachedMac, ifaceid
317 */
318 private String[] getMacAndIfaceid(Interface intf) {
319 OvsdbMap ovsdbMap = (OvsdbMap) intf.getExternalIdsColumn().data();
320 @SuppressWarnings("unchecked")
321 Map<String, String> externalIds = ovsdbMap.map();
322 if (externalIds == null) {
323 log.warn("The external_ids is null");
324 return null;
325 }
326
327 String attachedMac = externalIds.get(OvsdbConstant.EXTERNAL_ID_VM_MAC);
328 if (attachedMac == null) {
329 log.warn("The attachedMac is null");
330 return null;
331 }
332 String ifaceid = externalIds
333 .get(OvsdbConstant.EXTERNAL_ID_INTERFACE_ID);
334 if (ifaceid == null) {
335 log.warn("The ifaceid is null");
336 return null;
337 }
338 return new String[] {attachedMac, ifaceid};
339 }
340
341 /**
342 * Gets ofPorts number from table Interface.
343 *
344 * @param intf Interface instance
345 * @return ofport the ofport number
346 */
347 private long getOfPort(Interface intf) {
348 OvsdbSet ovsdbSet = (OvsdbSet) intf.getOpenFlowPortColumn().data();
349 @SuppressWarnings("unchecked")
350 Set<Long> ofPorts = ovsdbSet.set();
351 while (ofPorts == null || ofPorts.size() <= 0) {
352 log.debug("The ofport is null in {}", intf.getName());
353 return 0;
354 }
355 return (long) ofPorts.toArray()[0];
356 }
357
358 /**
359 * Gets datapathid from table bridge.
360 *
361 * @param clientService OvsdbClientService instance
362 * @param dbSchema ovsdb database schema
363 * @return datapathid the bridge datapathid
364 */
365 private long getDataPathid(OvsdbClientService clientService,
366 DatabaseSchema dbSchema) {
367 String bridgeUuid = clientService
368 .getBridgeUuid(OvsdbConstant.INTEGRATION_BRIDGE);
369 if (bridgeUuid == null) {
370 log.debug("Unable to spot bridge uuid for {} in {}",
371 OvsdbConstant.INTEGRATION_BRIDGE, clientService);
372 return 0;
373 }
374
375 Row bridgeRow = clientService.getRow(OvsdbConstant.DATABASENAME,
376 "Bridge", bridgeUuid);
377 Bridge bridge = (Bridge) TableGenerator.getTable(dbSchema, bridgeRow,
378 OvsdbTable.BRIDGE);
379 OvsdbSet ovsdbSet = (OvsdbSet) bridge.getDatapathIdColumn().data();
380 @SuppressWarnings("unchecked")
381 Set<String> dpids = ovsdbSet.set();
382 if (dpids == null || dpids.size() == 0) {
383 return 0;
384 }
385 return stringToLong((String) dpids.toArray()[0]);
386 }
387
388 private long stringToLong(String values) {
389 long value = (new BigInteger(values.replaceAll(":", ""), 16))
390 .longValue();
391 return value;
392 }
393
394 /**
395 * Implementation of an Callback which is responsible for receiving request
396 * infomation from ovsdb.
397 */
398 private class InternalMonitorCallBack implements Callback {
399 @Override
Brian O'Connor25be4912015-08-11 16:15:09 -0700400 public void update(UpdateNotification updateNotification) {
401 Object key = updateNotification.jsonValue();
YuanyouZhange3f396f2015-08-05 18:38:09 +0800402 OvsdbClientService ovsdbClient = requestNotification.get(key);
403
404 String dbName = requestDbName.get(key);
Brian O'Connor25be4912015-08-11 16:15:09 -0700405 JsonNode updatesJson = updateNotification.tbUpdatesJsonNode();
YuanyouZhange3f396f2015-08-05 18:38:09 +0800406 DatabaseSchema dbSchema = ovsdbClient.getDatabaseSchema(dbName);
407 TableUpdates updates = FromJsonUtil
408 .jsonNodeToTableUpdates(updatesJson, dbSchema);
409 try {
410 processTableUpdates(ovsdbClient, updates, dbName);
411 } catch (InterruptedException e) {
412 log.warn("Interrupted while processing table updates");
413 Thread.currentThread().interrupt();
414 }
415 }
416
417 @Override
418 public void locked(List<String> ids) {
419 // TODO Auto-generated method stub
420 }
421
422 @Override
423 public void stolen(List<String> ids) {
424 // TODO Auto-generated method stub
425 }
426
427 }
428
429}