blob: fbac581d287f3222fd7dc1895e5427eaefe74113 [file] [log] [blame]
Hyunsun Mooncb799442016-01-15 20:03:18 -08001/*
Brian O'Connor8e57fd52016-04-09 01:19:45 -07002 * Copyright 2016-present Open Networking Laboratory
Hyunsun Mooncb799442016-01-15 20:03:18 -08003 *
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 */
alshabibb4d31712016-06-01 18:51:03 -070016package org.opencord.cordvtn.impl;
Hyunsun Mooncb799442016-01-15 20:03:18 -080017
Hyunsun Moon0592c3d2016-06-23 14:47:52 -070018import com.google.common.collect.Sets;
Hyunsun Moon126171d2016-02-09 01:55:48 -080019import com.jcraft.jsch.Session;
Hyunsun Mooncb799442016-01-15 20:03:18 -080020import org.apache.felix.scr.annotations.Activate;
21import org.apache.felix.scr.annotations.Component;
22import org.apache.felix.scr.annotations.Deactivate;
23import org.apache.felix.scr.annotations.Reference;
24import org.apache.felix.scr.annotations.ReferenceCardinality;
25import org.apache.felix.scr.annotations.Service;
Hyunsun Moon126171d2016-02-09 01:55:48 -080026import org.onlab.packet.IpAddress;
Hyunsun Moon0592c3d2016-06-23 14:47:52 -070027import org.onlab.util.ItemNotFoundException;
Hyunsun Mooncb799442016-01-15 20:03:18 -080028import org.onlab.util.KryoNamespace;
29import org.onosproject.cluster.ClusterService;
Hyunsun Moon7004fcf2016-03-08 04:36:02 -080030import org.onosproject.cluster.LeadershipService;
31import org.onosproject.cluster.NodeId;
alshabibb4d31712016-06-01 18:51:03 -070032import org.opencord.cordvtn.api.ConnectionHandler;
33import org.opencord.cordvtn.api.CordVtnConfig;
34import org.opencord.cordvtn.api.CordVtnNode;
35import org.opencord.cordvtn.api.CordVtnNodeState;
Hyunsun Moon5401aaa2016-06-12 17:40:34 -070036import org.opencord.cordvtn.api.InstanceService;
alshabibb4d31712016-06-01 18:51:03 -070037import org.opencord.cordvtn.api.NetworkAddress;
38import org.opencord.cordvtn.api.SshAccessInfo;
Hyunsun Mooncb799442016-01-15 20:03:18 -080039import org.onosproject.core.ApplicationId;
40import org.onosproject.core.CoreService;
Hyunsun Mooncb799442016-01-15 20:03:18 -080041import org.onosproject.net.ConnectPoint;
Hyunsun Moon0592c3d2016-06-23 14:47:52 -070042import org.onosproject.net.DefaultAnnotations;
Hyunsun Mooncb799442016-01-15 20:03:18 -080043import org.onosproject.net.Device;
44import org.onosproject.net.DeviceId;
Hyunsun Moon0592c3d2016-06-23 14:47:52 -070045import org.onosproject.net.Host;
Hyunsun Mooncb799442016-01-15 20:03:18 -080046import org.onosproject.net.Port;
Hyunsun Moone7e4bb32016-05-16 04:32:45 -070047import org.onosproject.net.PortNumber;
Hyunsun Mooncb799442016-01-15 20:03:18 -080048import org.onosproject.net.behaviour.BridgeConfig;
49import org.onosproject.net.behaviour.BridgeName;
50import org.onosproject.net.behaviour.ControllerInfo;
51import org.onosproject.net.behaviour.DefaultTunnelDescription;
Hyunsun Moon0592c3d2016-06-23 14:47:52 -070052import org.onosproject.net.behaviour.TunnelConfig;
Hyunsun Mooncb799442016-01-15 20:03:18 -080053import org.onosproject.net.behaviour.TunnelDescription;
Hyunsun Moon0592c3d2016-06-23 14:47:52 -070054import org.onosproject.net.behaviour.TunnelName;
Hyunsun Mooncb799442016-01-15 20:03:18 -080055import org.onosproject.net.config.NetworkConfigEvent;
56import org.onosproject.net.config.NetworkConfigListener;
57import org.onosproject.net.config.NetworkConfigRegistry;
58import org.onosproject.net.config.NetworkConfigService;
Hyunsun Mooncb799442016-01-15 20:03:18 -080059import org.onosproject.net.device.DeviceAdminService;
60import org.onosproject.net.device.DeviceEvent;
61import org.onosproject.net.device.DeviceListener;
62import org.onosproject.net.device.DeviceService;
Hyunsun Mooncb799442016-01-15 20:03:18 -080063import org.onosproject.net.host.HostService;
64import org.onosproject.ovsdb.controller.OvsdbClientService;
65import org.onosproject.ovsdb.controller.OvsdbController;
66import org.onosproject.ovsdb.controller.OvsdbNodeId;
67import org.onosproject.store.serializers.KryoNamespaces;
68import org.onosproject.store.service.ConsistentMap;
Hyunsun Moonff55e812016-03-10 12:40:16 -080069import org.onosproject.store.service.MapEvent;
70import org.onosproject.store.service.MapEventListener;
Hyunsun Mooncb799442016-01-15 20:03:18 -080071import org.onosproject.store.service.Serializer;
72import org.onosproject.store.service.StorageService;
Hyunsun Moond05b32e2016-03-02 19:27:26 -080073import org.onosproject.store.service.Versioned;
Hyunsun Mooncb799442016-01-15 20:03:18 -080074import org.slf4j.Logger;
75
Hyunsun Moon0592c3d2016-06-23 14:47:52 -070076import java.util.ArrayList;
77import java.util.HashMap;
Hyunsun Mooncb799442016-01-15 20:03:18 -080078import java.util.List;
Hyunsun Moon0592c3d2016-06-23 14:47:52 -070079import java.util.Map;
Hyunsun Moon7004fcf2016-03-08 04:36:02 -080080import java.util.Objects;
Hyunsun Moon126171d2016-02-09 01:55:48 -080081import java.util.Set;
Hyunsun Mooncb799442016-01-15 20:03:18 -080082import java.util.concurrent.ExecutorService;
Hyunsun Moon58ddbdc2016-03-07 16:37:17 -080083import java.util.stream.Collectors;
Hyunsun Mooncb799442016-01-15 20:03:18 -080084
85import static com.google.common.base.Preconditions.checkNotNull;
86import static java.util.concurrent.Executors.newSingleThreadScheduledExecutor;
87import static org.onlab.util.Tools.groupedThreads;
88import static org.onosproject.net.Device.Type.SWITCH;
89import static org.onosproject.net.behaviour.TunnelDescription.Type.VXLAN;
Hyunsun Moon5401aaa2016-06-12 17:40:34 -070090import static org.opencord.cordvtn.api.Constants.*;
alshabibb4d31712016-06-01 18:51:03 -070091import static org.opencord.cordvtn.impl.RemoteIpCommandUtil.*;
Hyunsun Mooncb799442016-01-15 20:03:18 -080092import static org.slf4j.LoggerFactory.getLogger;
93
94/**
95 * Reads node information from the network config file and handles the config
96 * update events.
97 * Only a leader controller performs the node addition or deletion.
98 */
99@Component(immediate = true)
100@Service(value = CordVtnNodeManager.class)
101public class CordVtnNodeManager {
102
103 protected final Logger log = getLogger(getClass());
104
105 private static final KryoNamespace.Builder NODE_SERIALIZER = KryoNamespace.newBuilder()
106 .register(KryoNamespaces.API)
107 .register(CordVtnNode.class)
Hyunsun Moon126171d2016-02-09 01:55:48 -0800108 .register(NodeState.class)
109 .register(SshAccessInfo.class)
110 .register(NetworkAddress.class);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800111
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700112 private static final Map<String, String> DEFAULT_TUNNEL_OPTIONS = new HashMap<String, String>() {
113 {
114 put("key", "flow");
115 put("remote_ip", "flow");
116 }
117 };
Hyunsun Mooncb799442016-01-15 20:03:18 -0800118 private static final int DPID_BEGIN = 3;
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700119 private static final int OFPORT = 6653;
Hyunsun Mooncb799442016-01-15 20:03:18 -0800120
121 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
122 protected CoreService coreService;
123
124 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
125 protected NetworkConfigRegistry configRegistry;
126
127 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
128 protected NetworkConfigService configService;
129
130 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
131 protected StorageService storageService;
132
133 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
134 protected DeviceAdminService adminService;
135
136 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700137 protected OvsdbController controller;
Hyunsun Mooncb799442016-01-15 20:03:18 -0800138
139 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
140 protected ClusterService clusterService;
141
142 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Hyunsun Mooncb799442016-01-15 20:03:18 -0800143 protected DeviceService deviceService;
144
145 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
146 protected HostService hostService;
147
148 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Hyunsun Moon7004fcf2016-03-08 04:36:02 -0800149 protected LeadershipService leadershipService;
Hyunsun Mooncb799442016-01-15 20:03:18 -0800150
151 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Hyunsun Moon5401aaa2016-06-12 17:40:34 -0700152 protected InstanceService instanceService;
Hyunsun Mooncb799442016-01-15 20:03:18 -0800153
154 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700155 protected CordVtnPipeline pipeline;
Hyunsun Mooncb799442016-01-15 20:03:18 -0800156
Hyunsun Mooncb799442016-01-15 20:03:18 -0800157 private final ExecutorService eventExecutor =
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700158 newSingleThreadScheduledExecutor(groupedThreads("onos/cordvtn-node", "event-handler"));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800159
160 private final NetworkConfigListener configListener = new InternalConfigListener();
161 private final DeviceListener deviceListener = new InternalDeviceListener();
Hyunsun Moonff55e812016-03-10 12:40:16 -0800162 private final MapEventListener<String, CordVtnNode> nodeStoreListener = new InternalMapListener();
Hyunsun Mooncb799442016-01-15 20:03:18 -0800163
164 private final OvsdbHandler ovsdbHandler = new OvsdbHandler();
165 private final BridgeHandler bridgeHandler = new BridgeHandler();
166
Hyunsun Moon58ddbdc2016-03-07 16:37:17 -0800167 private ConsistentMap<String, CordVtnNode> nodeStore;
Hyunsun Mooncb799442016-01-15 20:03:18 -0800168 private ApplicationId appId;
Hyunsun Moon7004fcf2016-03-08 04:36:02 -0800169 private NodeId localNodeId;
Hyunsun Mooncb799442016-01-15 20:03:18 -0800170
Hyunsun Moon58ddbdc2016-03-07 16:37:17 -0800171 private enum NodeState implements CordVtnNodeState {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800172
173 INIT {
174 @Override
175 public void process(CordVtnNodeManager nodeManager, CordVtnNode node) {
Hyunsun Moon126171d2016-02-09 01:55:48 -0800176 if (!nodeManager.isOvsdbConnected(node)) {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800177 nodeManager.connectOvsdb(node);
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700178 } else {
179 nodeManager.createIntegrationBridge(node);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800180 }
181 }
182 },
183 BRIDGE_CREATED {
184 @Override
185 public void process(CordVtnNodeManager nodeManager, CordVtnNode node) {
Hyunsun Moon126171d2016-02-09 01:55:48 -0800186 if (!nodeManager.isOvsdbConnected(node)) {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800187 nodeManager.connectOvsdb(node);
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700188 } else {
189 nodeManager.createTunnelInterface(node);
190 nodeManager.addDataPlaneInterface(node);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800191 }
192 }
193 },
Hyunsun Moon126171d2016-02-09 01:55:48 -0800194 PORTS_ADDED {
Hyunsun Moonbd572c12016-01-21 00:54:52 -0800195 @Override
196 public void process(CordVtnNodeManager nodeManager, CordVtnNode node) {
Hyunsun Moon126171d2016-02-09 01:55:48 -0800197 nodeManager.setIpAddress(node);
Hyunsun Moonbd572c12016-01-21 00:54:52 -0800198 }
Hyunsun Moonbd572c12016-01-21 00:54:52 -0800199 },
Hyunsun Mooncb799442016-01-15 20:03:18 -0800200 COMPLETE {
201 @Override
202 public void process(CordVtnNodeManager nodeManager, CordVtnNode node) {
203 nodeManager.postInit(node);
204 }
205 },
206 INCOMPLETE {
207 @Override
208 public void process(CordVtnNodeManager nodeManager, CordVtnNode node) {
209 }
210 };
211
Hyunsun Mooncb799442016-01-15 20:03:18 -0800212 public abstract void process(CordVtnNodeManager nodeManager, CordVtnNode node);
213 }
214
215 @Activate
Hyunsun Moon479b7752016-05-06 20:13:28 -0700216 protected void activate() {
Hyunsun Moon5401aaa2016-06-12 17:40:34 -0700217 appId = coreService.getAppId(CORDVTN_APP_ID);
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700218
Hyunsun Moon7004fcf2016-03-08 04:36:02 -0800219 localNodeId = clusterService.getLocalNode().id();
220 leadershipService.runForLeadership(appId.name());
221
Hyunsun Moon58ddbdc2016-03-07 16:37:17 -0800222 nodeStore = storageService.<String, CordVtnNode>consistentMapBuilder()
Hyunsun Mooncb799442016-01-15 20:03:18 -0800223 .withSerializer(Serializer.using(NODE_SERIALIZER.build()))
224 .withName("cordvtn-nodestore")
225 .withApplicationId(appId)
226 .build();
227
Hyunsun Moonff55e812016-03-10 12:40:16 -0800228 nodeStore.addListener(nodeStoreListener);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800229 deviceService.addListener(deviceListener);
230 configService.addListener(configListener);
Hyunsun Moon479b7752016-05-06 20:13:28 -0700231
232 log.info("Started");
Hyunsun Mooncb799442016-01-15 20:03:18 -0800233 }
234
235 @Deactivate
236 protected void deactivate() {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800237 configService.removeListener(configListener);
238 deviceService.removeListener(deviceListener);
Hyunsun Moonff55e812016-03-10 12:40:16 -0800239 nodeStore.removeListener(nodeStoreListener);
Hyunsun Moonff55e812016-03-10 12:40:16 -0800240
Hyunsun Moon7004fcf2016-03-08 04:36:02 -0800241 leadershipService.withdraw(appId.name());
Hyunsun Moonff55e812016-03-10 12:40:16 -0800242 eventExecutor.shutdown();
Hyunsun Moon479b7752016-05-06 20:13:28 -0700243
244 log.info("Stopped");
Hyunsun Mooncb799442016-01-15 20:03:18 -0800245 }
246
247 /**
Hyunsun Moonff55e812016-03-10 12:40:16 -0800248 * Adds or updates a new node to the service.
Hyunsun Mooncb799442016-01-15 20:03:18 -0800249 *
250 * @param node cordvtn node
251 */
Hyunsun Moonff55e812016-03-10 12:40:16 -0800252 public void addOrUpdateNode(CordVtnNode node) {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800253 checkNotNull(node);
Hyunsun Moon58ddbdc2016-03-07 16:37:17 -0800254 nodeStore.put(node.hostname(), CordVtnNode.getUpdatedNode(node, getNodeState(node)));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800255 }
256
257 /**
258 * Deletes a node from the service.
259 *
260 * @param node cordvtn node
261 */
262 public void deleteNode(CordVtnNode node) {
263 checkNotNull(node);
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700264
265 if (isOvsdbConnected(node)) {
266 disconnectOvsdb(node);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800267 }
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700268
Hyunsun Moon58ddbdc2016-03-07 16:37:17 -0800269 nodeStore.remove(node.hostname());
Hyunsun Mooncb799442016-01-15 20:03:18 -0800270 }
271
272 /**
Hyunsun Mooncb799442016-01-15 20:03:18 -0800273 * Returns node initialization state.
274 *
275 * @param node cordvtn node
276 * @return true if initial node setup is completed, otherwise false
277 */
Hyunsun Moon126171d2016-02-09 01:55:48 -0800278 public boolean isNodeInitComplete(CordVtnNode node) {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800279 checkNotNull(node);
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700280 return nodeStore.containsKey(node.hostname()) && getNodeState(node).equals(NodeState.COMPLETE);
281 }
282
283 /**
284 * Returns detailed node initialization state.
285 *
286 * @param node cordvtn node
287 * @return string including detailed node init state
288 */
289 public String checkNodeInitState(CordVtnNode node) {
290 checkNotNull(node);
291
292 if (!nodeStore.containsKey(node.hostname())) {
293 log.warn("Node {} does not exist, add node first", node.hostname());
294 return null;
295 }
296
297 Session session = connect(node.sshInfo());
298 if (session == null) {
299 log.debug("Failed to SSH to {}", node.hostname());
300 return null;
301 }
302
303 Set<IpAddress> intBrIps = getCurrentIps(session, DEFAULT_BRIDGE);
304 String result = String.format(
305 "Current state : %s%n" +
306 "br-int created and connected to ONOS : %s (%s)%n" +
307 "VXLAN interface added to br-int : %s%n" +
308 "Data plane interface is added to br-int and enabled : %s (%s)%n" +
309 "IP flushed from data plane interface : %s (%s)%n" +
310 "Data plane IP added to br-int : %s (%s)%n" +
311 "Local management IP added to br-int : %s (%s)",
312 node.state(),
313 isBrIntCreated(node) ? MSG_OK : MSG_NO, node.intBrId(),
314 isTunnelIntfCreated(node) ? MSG_OK : MSG_NO,
315 isDataPlaneIntfAdded(node) ? MSG_OK : MSG_NO, node.dpIntf(),
316 isInterfaceUp(session, node.dpIntf()) &&
317 getCurrentIps(session, node.dpIntf()).isEmpty() ? MSG_OK : MSG_NO, node.dpIntf(),
318 intBrIps.contains(node.dpIp().ip()) ? MSG_OK : MSG_NO, node.dpIp().cidr(),
319 intBrIps.contains(node.localMgmtIp().ip()) ? MSG_OK : MSG_NO, node.localMgmtIp().cidr());
320
321 disconnect(session);
322
323 return result;
Hyunsun Mooncb799442016-01-15 20:03:18 -0800324 }
325
326 /**
327 * Returns the number of the nodes known to the service.
328 *
329 * @return number of nodes
330 */
331 public int getNodeCount() {
332 return nodeStore.size();
333 }
334
335 /**
336 * Returns all nodes known to the service.
337 *
338 * @return list of nodes
339 */
340 public List<CordVtnNode> getNodes() {
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700341 return nodeStore.values().stream().map(Versioned::value).collect(Collectors.toList());
Hyunsun Mooncb799442016-01-15 20:03:18 -0800342 }
343
344 /**
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700345 * Returns all nodes in complete state.
Hyunsun Mooncb799442016-01-15 20:03:18 -0800346 *
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700347 * @return set of nodes
Hyunsun Mooncb799442016-01-15 20:03:18 -0800348 */
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700349 public Set<CordVtnNode> completeNodes() {
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700350 return getNodes().stream().filter(this::isNodeInitComplete).collect(Collectors.toSet());
Hyunsun Mooncb799442016-01-15 20:03:18 -0800351 }
352
353 /**
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700354 * Returns physical data plane port number of a given device.
Hyunsun Mooncb799442016-01-15 20:03:18 -0800355 *
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700356 * @param deviceId integration bridge device id
357 * @return port number; null otherwise
Hyunsun Mooncb799442016-01-15 20:03:18 -0800358 */
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700359 public PortNumber dpPort(DeviceId deviceId) {
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700360 CordVtnNode node = nodeByBridgeId(deviceId);
361 if (node == null) {
362 log.warn("Failed to get node for {}", deviceId);
363 return null;
364 }
Hyunsun Moon315b9a62016-06-23 14:48:04 -0700365 Port port = deviceService.getPorts(deviceId).stream()
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700366 .filter(p -> portName(p).contains(node.dpIntf()) &&
Hyunsun Moon315b9a62016-06-23 14:48:04 -0700367 p.isEnabled())
368 .findFirst().orElse(null);
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700369
Hyunsun Moon315b9a62016-06-23 14:48:04 -0700370 return port == null ? null : port.number();
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700371 }
372
373 /**
374 * Returns physical data plane IP address of a given device.
375 *
376 * @param deviceId integration bridge device id
377 * @return ip address; null otherwise
378 */
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700379 public IpAddress dpIp(DeviceId deviceId) {
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700380 CordVtnNode node = nodeByBridgeId(deviceId);
381 if (node == null) {
382 log.warn("Failed to get node for {}", deviceId);
383 return null;
384 }
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700385 return node.dpIp().ip();
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700386 }
387
388 /**
389 * Returns tunnel port number of a given device.
390 *
391 * @param deviceId integration bridge device id
392 * @return port number
393 */
394 public PortNumber tunnelPort(DeviceId deviceId) {
Hyunsun Moon315b9a62016-06-23 14:48:04 -0700395 Port port = deviceService.getPorts(deviceId).stream()
396 .filter(p -> portName(p).contains(DEFAULT_TUNNEL))
397 .findFirst().orElse(null);
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700398
Hyunsun Moon315b9a62016-06-23 14:48:04 -0700399 return port == null ? null : port.number();
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700400 }
401
402 /**
403 * Returns if current node state saved in nodeStore is COMPLETE or not.
404 *
405 * @param node cordvtn node
406 * @return true if it's complete state, otherwise false
407 */
408 private boolean isNodeStateComplete(CordVtnNode node) {
409 checkNotNull(node);
410
411 // the state saved in nodeStore can be wrong if IP address settings are changed
412 // after the node init has been completed since there's no way to detect it
413 // getNodeState and checkNodeInitState always return correct answer but can be slow
414 Versioned<CordVtnNode> versionedNode = nodeStore.get(node.hostname());
415 CordVtnNodeState state = versionedNode.value().state();
416 return state != null && state.equals(NodeState.COMPLETE);
417 }
418
419 /**
420 * Initiates node to serve virtual tenant network.
421 *
422 * @param node cordvtn node
423 */
424 private void initNode(CordVtnNode node) {
425 checkNotNull(node);
426
427 NodeState state = (NodeState) node.state();
428 log.debug("Processing node: {} state: {}", node.hostname(), state);
429
430 state.process(this, node);
431 }
432
433 /**
434 * Performs tasks after node initialization.
435 * It disconnects unnecessary OVSDB connection and installs initial flow
436 * rules on the device.
437 *
438 * @param node cordvtn node
439 */
440 private void postInit(CordVtnNode node) {
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700441 disconnectOvsdb(node);
442 pipeline.initPipeline(node, dpPort(node.intBrId()), tunnelPort(node.intBrId()));
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700443
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700444 deviceService.getPorts(node.intBrId()).stream()
Hyunsun Moonfb417942016-06-23 14:48:20 -0700445 .filter(port -> portName(port).startsWith(VPORT_PREFIX) &&
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700446 port.isEnabled())
Hyunsun Moon5401aaa2016-06-12 17:40:34 -0700447 .forEach(port -> instanceService.addInstance(connectPoint(port)));
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700448
449 hostService.getHosts().forEach(host -> {
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700450 if (deviceService.getPort(host.location().deviceId(),
451 host.location().port()) == null) {
452 instanceService.removeInstance(connectPoint(host));
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700453 }
454 });
455
456 log.info("Finished init {}", node.hostname());
Hyunsun Mooncb799442016-01-15 20:03:18 -0800457 }
458
459 /**
Hyunsun Mooncb799442016-01-15 20:03:18 -0800460 * Sets a new state for a given cordvtn node.
461 *
462 * @param node cordvtn node
463 * @param newState new node state
464 */
465 private void setNodeState(CordVtnNode node, NodeState newState) {
466 checkNotNull(node);
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700467
Hyunsun Moonff55e812016-03-10 12:40:16 -0800468 log.debug("Changed {} state: {}", node.hostname(), newState);
Hyunsun Moon58ddbdc2016-03-07 16:37:17 -0800469 nodeStore.put(node.hostname(), CordVtnNode.getUpdatedNode(node, newState));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800470 }
471
472 /**
473 * Checks current state of a given cordvtn node and returns it.
474 *
475 * @param node cordvtn node
476 * @return node state
477 */
Hyunsun Moon126171d2016-02-09 01:55:48 -0800478 private NodeState getNodeState(CordVtnNode node) {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800479 checkNotNull(node);
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700480
481 if (isBrIntCreated(node) && isTunnelIntfCreated(node) &&
482 isDataPlaneIntfAdded(node) && isIpAddressSet(node)) {
483 return NodeState.COMPLETE;
484 } else if (isDataPlaneIntfAdded(node) && isTunnelIntfCreated(node)) {
485 return NodeState.PORTS_ADDED;
486 } else if (isBrIntCreated(node)) {
487 return NodeState.BRIDGE_CREATED;
488 } else {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800489 return NodeState.INIT;
490 }
491 }
492
493 /**
Hyunsun Mooncb799442016-01-15 20:03:18 -0800494 * Returns connection state of OVSDB server for a given node.
495 *
496 * @param node cordvtn node
497 * @return true if it is connected, false otherwise
498 */
Hyunsun Moon126171d2016-02-09 01:55:48 -0800499 private boolean isOvsdbConnected(CordVtnNode node) {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800500 checkNotNull(node);
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700501
Hyunsun Mooncb799442016-01-15 20:03:18 -0800502 OvsdbClientService ovsdbClient = getOvsdbClient(node);
503 return deviceService.isAvailable(node.ovsdbId()) &&
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700504 ovsdbClient != null && ovsdbClient.isConnected();
Hyunsun Mooncb799442016-01-15 20:03:18 -0800505 }
506
507 /**
508 * Connects to OVSDB server for a given node.
509 *
510 * @param node cordvtn node
511 */
512 private void connectOvsdb(CordVtnNode node) {
513 checkNotNull(node);
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700514
515 if (!nodeStore.containsKey(node.hostname())) {
516 log.warn("Node {} does not exist", node.hostname());
517 return;
518 }
519
520 if (!isOvsdbConnected(node)) {
521 controller.connect(node.hostMgmtIp().ip(), node.ovsdbPort());
522 }
523 }
524
525 /**
526 * Disconnects OVSDB server for a given node.
527 *
528 * @param node cordvtn node
529 */
530 private void disconnectOvsdb(CordVtnNode node) {
531 checkNotNull(node);
532
533 if (!nodeStore.containsKey(node.hostname())) {
534 log.warn("Node {} does not exist", node.hostname());
535 return;
536 }
537
538 if (isOvsdbConnected(node)) {
539 OvsdbClientService ovsdbClient = getOvsdbClient(node);
540 ovsdbClient.disconnect();
541 }
Hyunsun Mooncb799442016-01-15 20:03:18 -0800542 }
543
544 /**
545 * Returns OVSDB client for a given node.
546 *
547 * @param node cordvtn node
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700548 * @return OVSDB client, or null if it fails to get OVSDB client
Hyunsun Mooncb799442016-01-15 20:03:18 -0800549 */
550 private OvsdbClientService getOvsdbClient(CordVtnNode node) {
551 checkNotNull(node);
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700552
553 OvsdbClientService ovsdbClient = controller.getOvsdbClient(
554 new OvsdbNodeId(node.hostMgmtIp().ip(), node.ovsdbPort().toInt()));
555 if (ovsdbClient == null) {
556 log.trace("Couldn't find OVSDB client for {}", node.hostname());
557 }
558 return ovsdbClient;
Hyunsun Mooncb799442016-01-15 20:03:18 -0800559 }
560
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700561 /**
562 * Creates an integration bridge for a given node.
563 *
564 * @param node cordvtn node
565 */
Hyunsun Mooncb799442016-01-15 20:03:18 -0800566 private void createIntegrationBridge(CordVtnNode node) {
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700567 if (isBrIntCreated(node)) {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800568 return;
569 }
570
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700571 List<ControllerInfo> controllers = new ArrayList<>();
572 Sets.newHashSet(clusterService.getNodes()).stream()
573 .forEach(controller -> {
574 ControllerInfo ctrlInfo = new ControllerInfo(controller.ip(), OFPORT, "tcp");
575 controllers.add(ctrlInfo);
576 });
Hyunsun Mooncb799442016-01-15 20:03:18 -0800577
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700578 String dpid = node.intBrId().toString().substring(DPID_BEGIN);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800579
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700580 try {
581 Device device = deviceService.getDevice(node.ovsdbId());
582 if (device.is(BridgeConfig.class)) {
583 BridgeConfig bridgeConfig = device.as(BridgeConfig.class);
584 bridgeConfig.addBridge(BridgeName.bridgeName(DEFAULT_BRIDGE), dpid, controllers);
585 } else {
586 log.warn("The bridging behaviour is not supported in device {}", device.id());
587 }
588 } catch (ItemNotFoundException e) {
589 log.warn("Failed to create integration bridge on {}", node.hostname());
590 }
Hyunsun Mooncb799442016-01-15 20:03:18 -0800591 }
592
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700593 /**
594 * Creates tunnel interface to the integration bridge for a given node.
595 *
596 * @param node cordvtn node
597 */
Hyunsun Mooncb799442016-01-15 20:03:18 -0800598 private void createTunnelInterface(CordVtnNode node) {
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700599 if (isTunnelIntfCreated(node)) {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800600 return;
601 }
602
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700603 DefaultAnnotations.Builder optionBuilder = DefaultAnnotations.builder();
604 for (String key : DEFAULT_TUNNEL_OPTIONS.keySet()) {
605 optionBuilder.set(key, DEFAULT_TUNNEL_OPTIONS.get(key));
606 }
Hyunsun Mooncb799442016-01-15 20:03:18 -0800607
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700608 TunnelDescription description = new DefaultTunnelDescription(
609 null, null, VXLAN, TunnelName.tunnelName(DEFAULT_TUNNEL),
610 optionBuilder.build());
611
612 try {
613 Device device = deviceService.getDevice(node.ovsdbId());
614 if (device.is(TunnelConfig.class)) {
615 TunnelConfig tunnelConfig = device.as(TunnelConfig.class);
616 tunnelConfig.createTunnelInterface(BridgeName.bridgeName(DEFAULT_BRIDGE), description);
617 } else {
618 log.warn("The tunneling behaviour is not supported in device {}", device.id());
619 }
620 } catch (ItemNotFoundException e) {
621 log.warn("Failed to create tunnel interface on {}", node.hostname());
622 }
Hyunsun Moonbd572c12016-01-21 00:54:52 -0800623 }
624
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700625 /**
626 * Adds data plane interface to a given node.
627 *
628 * @param node cordvtn node
629 */
630 private void addDataPlaneInterface(CordVtnNode node) {
631 if (isDataPlaneIntfAdded(node)) {
632 return;
633 }
634
Hyunsun Moon479b7752016-05-06 20:13:28 -0700635 Session session = connect(node.sshInfo());
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700636 if (session == null) {
637 log.debug("Failed to SSH to {}", node.hostname());
Hyunsun Moon479b7752016-05-06 20:13:28 -0700638 return;
639 }
640
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700641 if (!isInterfaceUp(session, node.dpIntf())) {
642 log.warn("Interface {} is not available", node.dpIntf());
643 return;
644 }
645 disconnect(session);
646
647 try {
648 Device device = deviceService.getDevice(node.ovsdbId());
649 if (device.is(BridgeConfig.class)) {
650 BridgeConfig bridgeConfig = device.as(BridgeConfig.class);
651 bridgeConfig.addPort(BridgeName.bridgeName(DEFAULT_BRIDGE), node.dpIntf());
652 } else {
653 log.warn("The bridging behaviour is not supported in device {}", device.id());
654 }
655 } catch (ItemNotFoundException e) {
656 log.warn("Failed to add {} on {}", node.dpIntf(), node.hostname());
657 }
Hyunsun Moon126171d2016-02-09 01:55:48 -0800658 }
659
660 /**
661 * Flushes IP address from data plane interface and adds data plane IP address
662 * to integration bridge.
663 *
664 * @param node cordvtn node
665 */
666 private void setIpAddress(CordVtnNode node) {
Hyunsun Moon479b7752016-05-06 20:13:28 -0700667 Session session = connect(node.sshInfo());
Hyunsun Moon126171d2016-02-09 01:55:48 -0800668 if (session == null) {
669 log.debug("Failed to SSH to {}", node.hostname());
670 return;
671 }
Hyunsun Moonf2760522016-03-04 19:24:08 -0800672
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700673 getCurrentIps(session, DEFAULT_BRIDGE).stream()
674 .filter(ip -> !ip.equals(node.localMgmtIp().ip()))
675 .filter(ip -> !ip.equals(node.dpIp().ip()))
676 .forEach(ip -> deleteIp(session, ip, DEFAULT_BRIDGE));
677
678 boolean result = flushIp(session, node.dpIntf()) &&
679 setInterfaceUp(session, node.dpIntf()) &&
680 addIp(session, node.dpIp(), DEFAULT_BRIDGE) &&
681 addIp(session, node.localMgmtIp(), DEFAULT_BRIDGE) &&
682 setInterfaceUp(session, DEFAULT_BRIDGE);
Hyunsun Moon126171d2016-02-09 01:55:48 -0800683
Hyunsun Moon479b7752016-05-06 20:13:28 -0700684 disconnect(session);
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700685
Hyunsun Moon126171d2016-02-09 01:55:48 -0800686 if (result) {
687 setNodeState(node, NodeState.COMPLETE);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800688 }
689 }
690
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700691 /**
692 * Checks if integration bridge exists and available.
693 *
694 * @param node cordvtn node
695 * @return true if the bridge is available, false otherwise
696 */
697 private boolean isBrIntCreated(CordVtnNode node) {
698 return (deviceService.getDevice(node.intBrId()) != null
699 && deviceService.isAvailable(node.intBrId()));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800700 }
701
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700702 /**
703 * Checks if tunnel interface exists.
704 *
705 * @param node cordvtn node
706 * @return true if the interface exists, false otherwise
707 */
708 private boolean isTunnelIntfCreated(CordVtnNode node) {
709 return deviceService.getPorts(node.intBrId())
710 .stream()
711 .filter(p -> portName(p).contains(DEFAULT_TUNNEL) &&
712 p.isEnabled())
713 .findAny().isPresent();
714 }
715
716 /**
717 * Checks if data plane interface exists.
718 *
719 * @param node cordvtn node
720 * @return true if the interface exists, false otherwise
721 */
722 private boolean isDataPlaneIntfAdded(CordVtnNode node) {
723 return deviceService.getPorts(node.intBrId())
724 .stream()
725 .filter(p -> portName(p).contains(node.dpIntf()) &&
726 p.isEnabled())
727 .findAny().isPresent();
Hyunsun Mooncb799442016-01-15 20:03:18 -0800728 }
729
730 /**
Hyunsun Moon126171d2016-02-09 01:55:48 -0800731 * Checks if the IP addresses are correctly set.
732 *
733 * @param node cordvtn node
734 * @return true if the IP is set, false otherwise
735 */
736 private boolean isIpAddressSet(CordVtnNode node) {
Hyunsun Moon479b7752016-05-06 20:13:28 -0700737 Session session = connect(node.sshInfo());
Hyunsun Moon126171d2016-02-09 01:55:48 -0800738 if (session == null) {
739 log.debug("Failed to SSH to {}", node.hostname());
740 return false;
741 }
742
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700743 Set<IpAddress> intBrIps = getCurrentIps(session, DEFAULT_BRIDGE);
744 boolean result = getCurrentIps(session, node.dpIntf()).isEmpty() &&
745 isInterfaceUp(session, node.dpIntf()) &&
746 intBrIps.contains(node.dpIp().ip()) &&
Hyunsun Moon126171d2016-02-09 01:55:48 -0800747 intBrIps.contains(node.localMgmtIp().ip()) &&
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700748 isInterfaceUp(session, DEFAULT_BRIDGE);
Hyunsun Moon126171d2016-02-09 01:55:48 -0800749
Hyunsun Moon479b7752016-05-06 20:13:28 -0700750 disconnect(session);
Hyunsun Moon126171d2016-02-09 01:55:48 -0800751 return result;
752 }
753
754 /**
Hyunsun Mooncb799442016-01-15 20:03:18 -0800755 * Returns connect point of a given port.
756 *
757 * @param port port
758 * @return connect point
759 */
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700760 private ConnectPoint connectPoint(Port port) {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800761 return new ConnectPoint(port.element().id(), port.number());
762 }
763
764 /**
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700765 * Returns connect point of a given host.
766 *
767 * @param host host
768 * @return connect point
769 */
770 private ConnectPoint connectPoint(Host host) {
771 return new ConnectPoint(host.location().deviceId(), host.location().port());
772 }
773
774 /**
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700775 * Returns cordvtn node associated with a given OVSDB device.
776 *
777 * @param ovsdbId OVSDB device id
778 * @return cordvtn node, null if it fails to find the node
779 */
780 private CordVtnNode nodeByOvsdbId(DeviceId ovsdbId) {
781 return getNodes().stream()
782 .filter(node -> node.ovsdbId().equals(ovsdbId))
783 .findFirst().orElse(null);
784 }
785
786 /**
787 * Returns cordvtn node associated with a given integration bridge.
788 *
789 * @param bridgeId device id of integration bridge
790 * @return cordvtn node, null if it fails to find the node
791 */
792 private CordVtnNode nodeByBridgeId(DeviceId bridgeId) {
793 return getNodes().stream()
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700794 .filter(node -> node.intBrId().equals(bridgeId))
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700795 .findFirst().orElse(null);
796 }
797
798 /**
799 * Returns port name.
800 *
801 * @param port port
802 * @return port name
803 */
804 private String portName(Port port) {
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700805 return port.annotations().value("portName");
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700806 }
807
Hyunsun Mooncb799442016-01-15 20:03:18 -0800808 private class OvsdbHandler implements ConnectionHandler<Device> {
809
810 @Override
811 public void connected(Device device) {
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700812 CordVtnNode node = nodeByOvsdbId(device.id());
Hyunsun Mooncb799442016-01-15 20:03:18 -0800813 if (node != null) {
Hyunsun Moon126171d2016-02-09 01:55:48 -0800814 setNodeState(node, getNodeState(node));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800815 } else {
816 log.debug("{} is detected on unregistered node, ignore it.", device.id());
817 }
818 }
819
820 @Override
821 public void disconnected(Device device) {
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700822 if (!deviceService.isAvailable(device.id())) {
823 log.debug("Device {} is disconnected", device.id());
824 adminService.removeDevice(device.id());
825 }
Hyunsun Mooncb799442016-01-15 20:03:18 -0800826 }
827 }
828
829 private class BridgeHandler implements ConnectionHandler<Device> {
830
831 @Override
832 public void connected(Device device) {
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700833 CordVtnNode node = nodeByBridgeId(device.id());
Hyunsun Mooncb799442016-01-15 20:03:18 -0800834 if (node != null) {
Hyunsun Moon126171d2016-02-09 01:55:48 -0800835 setNodeState(node, getNodeState(node));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800836 } else {
837 log.debug("{} is detected on unregistered node, ignore it.", device.id());
838 }
839 }
840
841 @Override
842 public void disconnected(Device device) {
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700843 CordVtnNode node = nodeByBridgeId(device.id());
Hyunsun Mooncb799442016-01-15 20:03:18 -0800844 if (node != null) {
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700845 log.debug("Integration Bridge is disconnected from {}", node.hostname());
Hyunsun Mooncb799442016-01-15 20:03:18 -0800846 setNodeState(node, NodeState.INCOMPLETE);
847 }
848 }
849
850 /**
851 * Handles port added situation.
Hyunsun Moon126171d2016-02-09 01:55:48 -0800852 * If the added port is tunnel or data plane interface, proceed to the remaining
853 * node initialization. Otherwise, do nothing.
Hyunsun Mooncb799442016-01-15 20:03:18 -0800854 *
855 * @param port port
856 */
857 public void portAdded(Port port) {
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700858 CordVtnNode node = nodeByBridgeId((DeviceId) port.element().id());
859 String portName = portName(port);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800860
861 if (node == null) {
862 log.debug("{} is added to unregistered node, ignore it.", portName);
863 return;
864 }
865
Hyunsun Moonff55e812016-03-10 12:40:16 -0800866 log.info("Port {} is added to {}", portName, node.hostname());
Hyunsun Mooncb799442016-01-15 20:03:18 -0800867
Hyunsun Moonfb417942016-06-23 14:48:20 -0700868 if (portName.startsWith(VPORT_PREFIX)) {
869 if (isNodeStateComplete(node)) {
870 instanceService.addInstance(connectPoint(port));
871 } else {
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700872 log.debug("VM is detected on incomplete node, ignore it.", portName);
Hyunsun Moonfb417942016-06-23 14:48:20 -0700873 }
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700874 } else if (portName.contains(DEFAULT_TUNNEL) || portName.equals(node.dpIntf())) {
Hyunsun Moon126171d2016-02-09 01:55:48 -0800875 setNodeState(node, getNodeState(node));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800876 }
877 }
878
879 /**
880 * Handles port removed situation.
Hyunsun Moon126171d2016-02-09 01:55:48 -0800881 * If the removed port is tunnel or data plane interface, proceed to the remaining
882 * node initialization.Others, do nothing.
Hyunsun Mooncb799442016-01-15 20:03:18 -0800883 *
884 * @param port port
885 */
886 public void portRemoved(Port port) {
Hyunsun Moone7e4bb32016-05-16 04:32:45 -0700887 CordVtnNode node = nodeByBridgeId((DeviceId) port.element().id());
888 String portName = portName(port);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800889
890 if (node == null) {
891 return;
892 }
893
Hyunsun Moonff55e812016-03-10 12:40:16 -0800894 log.info("Port {} is removed from {}", portName, node.hostname());
Hyunsun Mooncb799442016-01-15 20:03:18 -0800895
Hyunsun Moonfb417942016-06-23 14:48:20 -0700896 if (portName.startsWith(VPORT_PREFIX)) {
897 if (isNodeStateComplete(node)) {
898 instanceService.removeInstance(connectPoint(port));
899 } else {
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700900 log.debug("VM is vanished from incomplete node, ignore it.", portName);
Hyunsun Moonfb417942016-06-23 14:48:20 -0700901 }
Hyunsun Moon0592c3d2016-06-23 14:47:52 -0700902 } else if (portName.contains(DEFAULT_TUNNEL) || portName.equals(node.dpIntf())) {
Hyunsun Mooncb799442016-01-15 20:03:18 -0800903 setNodeState(node, NodeState.INCOMPLETE);
904 }
905 }
906 }
907
908 private class InternalDeviceListener implements DeviceListener {
909
910 @Override
911 public void event(DeviceEvent event) {
912
Hyunsun Moon7004fcf2016-03-08 04:36:02 -0800913 NodeId leaderNodeId = leadershipService.getLeader(appId.name());
914 if (!Objects.equals(localNodeId, leaderNodeId)) {
Hyunsun Moonff55e812016-03-10 12:40:16 -0800915 // do not allow to proceed without leadership
Hyunsun Moon7004fcf2016-03-08 04:36:02 -0800916 return;
917 }
918
Hyunsun Mooncb799442016-01-15 20:03:18 -0800919 Device device = event.subject();
920 ConnectionHandler<Device> handler =
921 (device.type().equals(SWITCH) ? bridgeHandler : ovsdbHandler);
922
923 switch (event.type()) {
924 case PORT_ADDED:
Hyunsun Moonff55e812016-03-10 12:40:16 -0800925 eventExecutor.execute(() -> bridgeHandler.portAdded(event.port()));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800926 break;
927 case PORT_UPDATED:
928 if (!event.port().isEnabled()) {
Hyunsun Moonff55e812016-03-10 12:40:16 -0800929 eventExecutor.execute(() -> bridgeHandler.portRemoved(event.port()));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800930 }
931 break;
932 case DEVICE_ADDED:
933 case DEVICE_AVAILABILITY_CHANGED:
934 if (deviceService.isAvailable(device.id())) {
Hyunsun Moonff55e812016-03-10 12:40:16 -0800935 eventExecutor.execute(() -> handler.connected(device));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800936 } else {
Hyunsun Moonff55e812016-03-10 12:40:16 -0800937 eventExecutor.execute(() -> handler.disconnected(device));
Hyunsun Mooncb799442016-01-15 20:03:18 -0800938 }
939 break;
940 default:
941 break;
942 }
943 }
944 }
945
946 /**
Hyunsun Moon3fc17f72016-01-24 21:47:06 -0800947 * Reads cordvtn nodes from config file.
Hyunsun Mooncb799442016-01-15 20:03:18 -0800948 */
949 private void readConfiguration() {
950 CordVtnConfig config = configRegistry.getConfig(appId, CordVtnConfig.class);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800951 if (config == null) {
Hyunsun Moon3fc17f72016-01-24 21:47:06 -0800952 log.debug("No configuration found");
Hyunsun Mooncb799442016-01-15 20:03:18 -0800953 return;
954 }
Hyunsun Moonff55e812016-03-10 12:40:16 -0800955 config.cordVtnNodes().forEach(this::addOrUpdateNode);
Hyunsun Mooncb799442016-01-15 20:03:18 -0800956 }
957
958 private class InternalConfigListener implements NetworkConfigListener {
959
960 @Override
961 public void event(NetworkConfigEvent event) {
Hyunsun Moonff55e812016-03-10 12:40:16 -0800962 NodeId leaderNodeId = leadershipService.getLeader(appId.name());
963 if (!Objects.equals(localNodeId, leaderNodeId)) {
964 // do not allow to proceed without leadership
965 return;
966 }
967
Hyunsun Mooncb799442016-01-15 20:03:18 -0800968 if (!event.configClass().equals(CordVtnConfig.class)) {
969 return;
970 }
971
972 switch (event.type()) {
973 case CONFIG_ADDED:
Hyunsun Mooncb799442016-01-15 20:03:18 -0800974 case CONFIG_UPDATED:
Hyunsun Mooncb799442016-01-15 20:03:18 -0800975 eventExecutor.execute(CordVtnNodeManager.this::readConfiguration);
976 break;
977 default:
978 break;
979 }
980 }
981 }
Hyunsun Moonff55e812016-03-10 12:40:16 -0800982
983 private class InternalMapListener implements MapEventListener<String, CordVtnNode> {
984
985 @Override
986 public void event(MapEvent<String, CordVtnNode> event) {
987 NodeId leaderNodeId = leadershipService.getLeader(appId.name());
988 if (!Objects.equals(localNodeId, leaderNodeId)) {
989 // do not allow to proceed without leadership
990 return;
991 }
992
993 CordVtnNode oldNode;
994 CordVtnNode newNode;
995
996 switch (event.type()) {
997 case UPDATE:
998 oldNode = event.oldValue().value();
999 newNode = event.newValue().value();
1000
Hyunsun Moon479b7752016-05-06 20:13:28 -07001001 log.info("Reloaded {}", newNode.hostname());
Hyunsun Moonff55e812016-03-10 12:40:16 -08001002 if (!newNode.equals(oldNode)) {
Hyunsun Moonff55e812016-03-10 12:40:16 -08001003 log.debug("New node: {}", newNode);
1004 }
Hyunsun Moon479b7752016-05-06 20:13:28 -07001005 // performs init procedure even if the node is not changed
1006 // for robustness since it's no harm to run init procedure
1007 // multiple times
Hyunsun Moonff55e812016-03-10 12:40:16 -08001008 eventExecutor.execute(() -> initNode(newNode));
1009 break;
1010 case INSERT:
1011 newNode = event.newValue().value();
1012 log.info("Added {}", newNode.hostname());
1013 eventExecutor.execute(() -> initNode(newNode));
1014 break;
1015 case REMOVE:
1016 oldNode = event.oldValue().value();
Hyunsun Moon479b7752016-05-06 20:13:28 -07001017 log.info("Removed {}", oldNode.hostname());
Hyunsun Moonff55e812016-03-10 12:40:16 -08001018 break;
1019 default:
1020 break;
1021 }
1022 }
1023 }
Hyunsun Mooncb799442016-01-15 20:03:18 -08001024}