blob: 538a5fd8a275475d938897b94e71087b46d6ffb1 [file] [log] [blame]
Hyunsun Moon44aac662017-02-18 02:07:01 +09001/*
Brian O'Connora09fe5b2017-08-03 21:12:30 -07002 * Copyright 2016-present Open Networking Foundation
Hyunsun Moon44aac662017-02-18 02:07:01 +09003 *
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.openstacknetworking.impl;
17
18import com.google.common.base.Strings;
19import org.apache.felix.scr.annotations.Activate;
20import org.apache.felix.scr.annotations.Component;
21import org.apache.felix.scr.annotations.Deactivate;
22import org.apache.felix.scr.annotations.Reference;
23import org.apache.felix.scr.annotations.ReferenceCardinality;
24import org.apache.felix.scr.annotations.Service;
25import org.onlab.packet.IpAddress;
26import org.onlab.packet.MacAddress;
27import org.onlab.packet.VlanId;
28import org.onlab.util.Tools;
29import org.onosproject.core.CoreService;
30import org.onosproject.mastership.MastershipService;
31import org.onosproject.net.ConnectPoint;
32import org.onosproject.net.DefaultAnnotations;
33import org.onosproject.net.Device;
34import org.onosproject.net.Host;
35import org.onosproject.net.HostId;
36import org.onosproject.net.HostLocation;
37import org.onosproject.net.Port;
38import org.onosproject.net.device.DeviceEvent;
39import org.onosproject.net.device.DeviceListener;
40import org.onosproject.net.device.DeviceService;
41import org.onosproject.net.host.DefaultHostDescription;
42import org.onosproject.net.host.HostDescription;
43import org.onosproject.net.host.HostProvider;
44import org.onosproject.net.host.HostProviderRegistry;
45import org.onosproject.net.host.HostProviderService;
46import org.onosproject.net.host.HostService;
47import org.onosproject.net.provider.AbstractProvider;
48import org.onosproject.net.provider.ProviderId;
49import org.onosproject.openstacknetworking.api.OpenstackNetworkService;
Hyunsun Moon0d457362017-06-27 17:19:41 +090050import org.onosproject.openstacknode.api.OpenstackNode;
51import org.onosproject.openstacknode.api.OpenstackNodeEvent;
52import org.onosproject.openstacknode.api.OpenstackNodeListener;
53import org.onosproject.openstacknode.api.OpenstackNodeService;
Hyunsun Moon44aac662017-02-18 02:07:01 +090054import org.openstack4j.model.network.Network;
daniel park796c2eb2018-03-22 17:01:51 +090055import org.openstack4j.model.network.NetworkType;
Hyunsun Moon44aac662017-02-18 02:07:01 +090056import org.slf4j.Logger;
57import org.slf4j.LoggerFactory;
58
59import java.util.Set;
60import java.util.concurrent.ExecutorService;
61import java.util.concurrent.Executors;
62import java.util.stream.Collectors;
63
64import static org.onlab.util.Tools.groupedThreads;
65import static org.onosproject.net.AnnotationKeys.PORT_NAME;
daniel park796c2eb2018-03-22 17:01:51 +090066import static org.onosproject.openstacknetworking.api.Constants.OPENSTACK_NETWORKING_APP_ID;
Daniel Parkc4d06402018-05-28 15:57:37 +090067import static org.onosproject.openstacknetworking.api.Constants.PORT_NAME_PREFIX_MAP;
Hyunsun Moon44aac662017-02-18 02:07:01 +090068import static org.onosproject.openstacknetworking.impl.HostBasedInstancePort.ANNOTATION_CREATE_TIME;
69import static org.onosproject.openstacknetworking.impl.HostBasedInstancePort.ANNOTATION_NETWORK_ID;
70import static org.onosproject.openstacknetworking.impl.HostBasedInstancePort.ANNOTATION_PORT_ID;
71
72@Service
73@Component(immediate = true)
74public final class OpenstackSwitchingHostProvider extends AbstractProvider implements HostProvider {
75
76 private final Logger log = LoggerFactory.getLogger(getClass());
77
78 private static final String PORT_NAME_PREFIX_VM = "tap";
Daniel Parkc4d06402018-05-28 15:57:37 +090079 private static final String PORT_NAME_PREFIX_CAVIUM = "enp";
Hyunsun Moon44aac662017-02-18 02:07:01 +090080 private static final String ERR_ADD_HOST = "Failed to add host: ";
daniel park5fe30792017-08-22 15:42:54 +090081 private static final String ANNOTATION_SEGMENT_ID = "segId";
Jian Li9d676fb2018-03-02 15:25:36 +090082 private static final String SONA_HOST_SCHEME = "sona";
Hyunsun Moon44aac662017-02-18 02:07:01 +090083
84 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jian Li9d676fb2018-03-02 15:25:36 +090085 protected CoreService coreService;
Hyunsun Moon44aac662017-02-18 02:07:01 +090086
87 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jian Li9d676fb2018-03-02 15:25:36 +090088 protected DeviceService deviceService;
Hyunsun Moon44aac662017-02-18 02:07:01 +090089
90 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jian Li9d676fb2018-03-02 15:25:36 +090091 protected HostProviderRegistry hostProviderRegistry;
Hyunsun Moon44aac662017-02-18 02:07:01 +090092
93 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jian Li9d676fb2018-03-02 15:25:36 +090094 protected HostService hostService;
Hyunsun Moon44aac662017-02-18 02:07:01 +090095
96 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jian Li9d676fb2018-03-02 15:25:36 +090097 protected MastershipService mastershipService;
Hyunsun Moon44aac662017-02-18 02:07:01 +090098
99 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jian Li9d676fb2018-03-02 15:25:36 +0900100 protected OpenstackNetworkService osNetworkService;
Hyunsun Moon44aac662017-02-18 02:07:01 +0900101
102 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jian Li9d676fb2018-03-02 15:25:36 +0900103 protected OpenstackNodeService osNodeService;
Hyunsun Moon44aac662017-02-18 02:07:01 +0900104
105 private final ExecutorService deviceEventExecutor =
106 Executors.newSingleThreadExecutor(groupedThreads("openstacknetworking", "device-event"));
Hyunsun Moon44aac662017-02-18 02:07:01 +0900107 private final InternalDeviceListener internalDeviceListener = new InternalDeviceListener();
108 private final InternalOpenstackNodeListener internalNodeListener = new InternalOpenstackNodeListener();
109
110 private HostProviderService hostProvider;
111
112 /**
113 * Creates OpenStack switching host provider.
114 */
115 public OpenstackSwitchingHostProvider() {
Jian Li9d676fb2018-03-02 15:25:36 +0900116 super(new ProviderId(SONA_HOST_SCHEME, OPENSTACK_NETWORKING_APP_ID));
Hyunsun Moon44aac662017-02-18 02:07:01 +0900117 }
118
119 @Activate
Ray Milkey9c9cde42018-01-12 14:22:06 -0800120 void activate() {
Hyunsun Moon44aac662017-02-18 02:07:01 +0900121 coreService.registerApplication(OPENSTACK_NETWORKING_APP_ID);
122 deviceService.addListener(internalDeviceListener);
123 osNodeService.addListener(internalNodeListener);
124 hostProvider = hostProviderRegistry.register(this);
125
126 log.info("Started");
127 }
128
129 @Deactivate
Ray Milkey9c9cde42018-01-12 14:22:06 -0800130 void deactivate() {
Hyunsun Moon44aac662017-02-18 02:07:01 +0900131 hostProviderRegistry.unregister(this);
132 osNodeService.removeListener(internalNodeListener);
133 deviceService.removeListener(internalDeviceListener);
134
135 deviceEventExecutor.shutdown();
Hyunsun Moon44aac662017-02-18 02:07:01 +0900136
137 log.info("Stopped");
138 }
139
140 @Override
141 public void triggerProbe(Host host) {
142 // no probe is required
143 }
144
Jian Li9d676fb2018-03-02 15:25:36 +0900145 /**
146 * Processes port addition event.
147 * Once a port addition event is detected, it tries to create a host instance
148 * with openstack augmented host information such as networkId, portId,
149 * createTime, segmentId and notifies to host provider.
150 *
151 * @param port port object used in ONOS
152 */
Hyunsun Moon44aac662017-02-18 02:07:01 +0900153 private void processPortAdded(Port port) {
154 // TODO check the node state is COMPLETE
155 org.openstack4j.model.network.Port osPort = osNetworkService.port(port);
156 if (osPort == null) {
157 log.warn(ERR_ADD_HOST + "OpenStack port for {} not found", port);
158 return;
159 }
160
161 Network osNet = osNetworkService.network(osPort.getNetworkId());
162 if (osNet == null) {
163 log.warn(ERR_ADD_HOST + "OpenStack network {} not found",
164 osPort.getNetworkId());
165 return;
166 }
167
168 if (osPort.getFixedIps().isEmpty()) {
169 log.warn(ERR_ADD_HOST + "no fixed IP for port {}", osPort.getId());
170 return;
171 }
172
173 MacAddress macAddr = MacAddress.valueOf(osPort.getMacAddress());
174 Set<IpAddress> fixedIps = osPort.getFixedIps().stream()
175 .map(ip -> IpAddress.valueOf(ip.getIpAddress()))
176 .collect(Collectors.toSet());
177 ConnectPoint connectPoint = new ConnectPoint(port.element().id(), port.number());
178
179 DefaultAnnotations.Builder annotations = DefaultAnnotations.builder()
180 .set(ANNOTATION_NETWORK_ID, osPort.getNetworkId())
181 .set(ANNOTATION_PORT_ID, osPort.getId())
daniel park796c2eb2018-03-22 17:01:51 +0900182 .set(ANNOTATION_CREATE_TIME, String.valueOf(System.currentTimeMillis()));
183
184 if (osNet.getNetworkType() != NetworkType.FLAT) {
185 annotations.set(ANNOTATION_SEGMENT_ID, osNet.getProviderSegID());
186
187 }
Hyunsun Moon44aac662017-02-18 02:07:01 +0900188
189 HostDescription hostDesc = new DefaultHostDescription(
190 macAddr,
191 VlanId.NONE,
192 new HostLocation(connectPoint, System.currentTimeMillis()),
193 fixedIps,
194 annotations.build());
195
196 HostId hostId = HostId.hostId(macAddr);
197 hostProvider.hostDetected(hostId, hostDesc, false);
198 }
199
Jian Li9d676fb2018-03-02 15:25:36 +0900200 /**
201 * Processes port removal event.
202 * Once a port removal event is detected, it tries to look for a host
203 * instance through host provider by giving connect point information,
204 * and vanishes it.
205 *
206 * @param port port object used in ONOS
207 */
Hyunsun Moon44aac662017-02-18 02:07:01 +0900208 private void processPortRemoved(Port port) {
209 ConnectPoint connectPoint = new ConnectPoint(port.element().id(), port.number());
Jian Li9d676fb2018-03-02 15:25:36 +0900210 hostService.getConnectedHosts(connectPoint)
211 .forEach(host -> hostProvider.hostVanished(host.id()));
Hyunsun Moon44aac662017-02-18 02:07:01 +0900212 }
213
Jian Li9d676fb2018-03-02 15:25:36 +0900214 /**
215 * An internal device listener which listens the port events generated from
216 * OVS integration bridge.
217 */
Hyunsun Moon44aac662017-02-18 02:07:01 +0900218 private class InternalDeviceListener implements DeviceListener {
219
220 @Override
221 public boolean isRelevant(DeviceEvent event) {
222 Device device = event.subject();
223 if (!mastershipService.isLocalMaster(device.id())) {
224 // do not allow to proceed without mastership
225 return false;
226 }
227 Port port = event.port();
228 if (port == null) {
229 return false;
230 }
231 String portName = port.annotations().value(PORT_NAME);
Jian Li9d676fb2018-03-02 15:25:36 +0900232
233 return !Strings.isNullOrEmpty(portName) &&
Daniel Parkc4d06402018-05-28 15:57:37 +0900234 (portName.startsWith(PORT_NAME_PREFIX_VM) || isDirectPort(portName));
235 }
236
237 private boolean isDirectPort(String portName) {
238 return PORT_NAME_PREFIX_MAP.values().stream().filter(p -> portName.startsWith(p)).findAny().isPresent();
Hyunsun Moon44aac662017-02-18 02:07:01 +0900239 }
240
241 @Override
242 public void event(DeviceEvent event) {
Daniel Parkc4d06402018-05-28 15:57:37 +0900243 log.info("Device event occurred with type {}", event.type());
Hyunsun Moon44aac662017-02-18 02:07:01 +0900244 switch (event.type()) {
245 case PORT_UPDATED:
246 if (!event.port().isEnabled()) {
Jian Li9d676fb2018-03-02 15:25:36 +0900247 portRemovedHelper(deviceEventExecutor, event);
Hyunsun Moon44aac662017-02-18 02:07:01 +0900248 } else if (event.port().isEnabled()) {
Jian Li9d676fb2018-03-02 15:25:36 +0900249 portAddedHelper(deviceEventExecutor, event);
Hyunsun Moon44aac662017-02-18 02:07:01 +0900250 }
251 break;
252 case PORT_ADDED:
Jian Li9d676fb2018-03-02 15:25:36 +0900253 portAddedHelper(deviceEventExecutor, event);
Hyunsun Moon44aac662017-02-18 02:07:01 +0900254 break;
Hyunsun Moonb7a9cd22017-02-24 11:12:53 +0900255 case PORT_REMOVED:
Jian Li9d676fb2018-03-02 15:25:36 +0900256 portRemovedHelper(deviceEventExecutor, event);
Ray Milkeyd6a67c32018-02-02 10:30:35 -0800257 break;
Hyunsun Moon44aac662017-02-18 02:07:01 +0900258 default:
259 break;
260 }
261 }
262 }
263
Jian Li9d676fb2018-03-02 15:25:36 +0900264 /**
265 * A helper method which logs the port addition event and performs port
266 * addition action.
267 *
268 * @param executor device executor service
269 * @param event device event
270 */
271 private void portAddedHelper(ExecutorService executor, DeviceEvent event) {
272 executor.execute(() -> {
273 log.debug("Instance port {} is detected from {}",
274 event.port().annotations().value(PORT_NAME),
275 event.subject().id());
276 processPortAdded(event.port());
277 });
278 }
279
280 /**
281 * A helper method which logs the port removal event and performs port
282 * removal action.
283 *
284 * @param executor device executor service
285 * @param event device event
286 */
287 private void portRemovedHelper(ExecutorService executor, DeviceEvent event) {
288 executor.execute(() -> {
289 log.debug("Instance port {} is removed from {}",
290 event.port().annotations().value(PORT_NAME),
291 event.subject().id());
292 processPortRemoved(event.port());
293 });
294 }
295
Hyunsun Moon44aac662017-02-18 02:07:01 +0900296 private class InternalOpenstackNodeListener implements OpenstackNodeListener {
297
298 @Override
Jian Li9d676fb2018-03-02 15:25:36 +0900299 public boolean isRelevant(OpenstackNodeEvent event) {
300 // do not allow to proceed without mastership
301 Device device = deviceService.getDevice(event.subject().intgBridge());
Jian Li7ddea782018-03-30 11:17:42 +0900302 if (device == null) {
303 return false;
304 }
Jian Li9d676fb2018-03-02 15:25:36 +0900305 return mastershipService.isLocalMaster(device.id());
306 }
307
308 @Override
Hyunsun Moon44aac662017-02-18 02:07:01 +0900309 public void event(OpenstackNodeEvent event) {
310 OpenstackNode osNode = event.subject();
Hyunsun Moon44aac662017-02-18 02:07:01 +0900311
312 switch (event.type()) {
Hyunsun Moon0d457362017-06-27 17:19:41 +0900313 case OPENSTACK_NODE_COMPLETE:
Hyunsun Moon44aac662017-02-18 02:07:01 +0900314 deviceEventExecutor.execute(() -> {
315 log.info("COMPLETE node {} is detected", osNode.hostname());
316 processCompleteNode(event.subject());
317 });
318 break;
Hyunsun Moon0d457362017-06-27 17:19:41 +0900319 case OPENSTACK_NODE_INCOMPLETE:
Hyunsun Moon44aac662017-02-18 02:07:01 +0900320 log.warn("{} is changed to INCOMPLETE state", osNode);
321 break;
Hyunsun Moon0d457362017-06-27 17:19:41 +0900322 case OPENSTACK_NODE_CREATED:
323 case OPENSTACK_NODE_UPDATED:
324 case OPENSTACK_NODE_REMOVED:
Jian Li9d676fb2018-03-02 15:25:36 +0900325 // not reacts to the events other than complete and incomplete states
326 break;
Hyunsun Moon44aac662017-02-18 02:07:01 +0900327 default:
Jian Li9d676fb2018-03-02 15:25:36 +0900328 log.warn("Unsupported openstack node event type");
Hyunsun Moon44aac662017-02-18 02:07:01 +0900329 break;
330 }
331 }
332
333 private void processCompleteNode(OpenstackNode osNode) {
Hyunsun Moon0d457362017-06-27 17:19:41 +0900334 deviceService.getPorts(osNode.intgBridge()).stream()
Hyunsun Moon44aac662017-02-18 02:07:01 +0900335 .filter(port -> port.annotations().value(PORT_NAME)
336 .startsWith(PORT_NAME_PREFIX_VM) &&
337 port.isEnabled())
338 .forEach(port -> {
339 log.debug("Instance port {} is detected from {}",
340 port.annotations().value(PORT_NAME),
341 osNode.hostname());
342 processPortAdded(port);
343 });
344
Daniel Parkc4d06402018-05-28 15:57:37 +0900345 PORT_NAME_PREFIX_MAP.values().forEach(portNamePrefix -> {
346 deviceService.getPorts(osNode.intgBridge()).stream()
347 .filter(port -> port.annotations().value(PORT_NAME)
348 .startsWith(portNamePrefix) &&
349 port.isEnabled())
350 .forEach(port -> {
351 log.debug("Instance port {} is detected from {}",
352 port.annotations().value(portNamePrefix),
353 osNode.hostname());
354 processPortAdded(port);
355 });
356 });
357
Hyunsun Moon44aac662017-02-18 02:07:01 +0900358 Tools.stream(hostService.getHosts())
359 .filter(host -> deviceService.getPort(
360 host.location().deviceId(),
361 host.location().port()) == null)
362 .forEach(host -> {
363 log.info("Remove stale host {}", host.id());
364 hostProvider.hostVanished(host.id());
365 });
366 }
367 }
368}