blob: 0e4f7ab178cb8f356b6a40dfa80020d1a7a0df93 [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;
55import org.slf4j.Logger;
56import org.slf4j.LoggerFactory;
57
58import java.util.Set;
59import java.util.concurrent.ExecutorService;
60import java.util.concurrent.Executors;
61import java.util.stream.Collectors;
62
63import static org.onlab.util.Tools.groupedThreads;
64import static org.onosproject.net.AnnotationKeys.PORT_NAME;
65import static org.onosproject.openstacknetworking.api.Constants.*;
66import static org.onosproject.openstacknetworking.impl.HostBasedInstancePort.ANNOTATION_CREATE_TIME;
67import static org.onosproject.openstacknetworking.impl.HostBasedInstancePort.ANNOTATION_NETWORK_ID;
68import static org.onosproject.openstacknetworking.impl.HostBasedInstancePort.ANNOTATION_PORT_ID;
69
70@Service
71@Component(immediate = true)
72public final class OpenstackSwitchingHostProvider extends AbstractProvider implements HostProvider {
73
74 private final Logger log = LoggerFactory.getLogger(getClass());
75
76 private static final String PORT_NAME_PREFIX_VM = "tap";
77 private static final String ERR_ADD_HOST = "Failed to add host: ";
daniel park5fe30792017-08-22 15:42:54 +090078 private static final String ANNOTATION_SEGMENT_ID = "segId";
Hyunsun Moon44aac662017-02-18 02:07:01 +090079
80 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Ray Milkey9c9cde42018-01-12 14:22:06 -080081 CoreService coreService;
Hyunsun Moon44aac662017-02-18 02:07:01 +090082
83 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Ray Milkey9c9cde42018-01-12 14:22:06 -080084 DeviceService deviceService;
Hyunsun Moon44aac662017-02-18 02:07:01 +090085
86 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Ray Milkey9c9cde42018-01-12 14:22:06 -080087 HostProviderRegistry hostProviderRegistry;
Hyunsun Moon44aac662017-02-18 02:07:01 +090088
89 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Ray Milkey9c9cde42018-01-12 14:22:06 -080090 HostService hostService;
Hyunsun Moon44aac662017-02-18 02:07:01 +090091
92 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Ray Milkey9c9cde42018-01-12 14:22:06 -080093 MastershipService mastershipService;
Hyunsun Moon44aac662017-02-18 02:07:01 +090094
95 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Ray Milkey9c9cde42018-01-12 14:22:06 -080096 OpenstackNetworkService osNetworkService;
Hyunsun Moon44aac662017-02-18 02:07:01 +090097
98 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Ray Milkey9c9cde42018-01-12 14:22:06 -080099 OpenstackNodeService osNodeService;
Hyunsun Moon44aac662017-02-18 02:07:01 +0900100
101 private final ExecutorService deviceEventExecutor =
102 Executors.newSingleThreadExecutor(groupedThreads("openstacknetworking", "device-event"));
103 private final ExecutorService configEventExecutor =
104 Executors.newSingleThreadExecutor(groupedThreads("openstacknetworking", "config-event"));
105 private final InternalDeviceListener internalDeviceListener = new InternalDeviceListener();
106 private final InternalOpenstackNodeListener internalNodeListener = new InternalOpenstackNodeListener();
107
108 private HostProviderService hostProvider;
109
110 /**
111 * Creates OpenStack switching host provider.
112 */
113 public OpenstackSwitchingHostProvider() {
Frank Wangad237e72017-06-03 11:39:44 +0800114 super(new ProviderId("sona", OPENSTACK_NETWORKING_APP_ID));
Hyunsun Moon44aac662017-02-18 02:07:01 +0900115 }
116
117 @Activate
Ray Milkey9c9cde42018-01-12 14:22:06 -0800118 void activate() {
Hyunsun Moon44aac662017-02-18 02:07:01 +0900119 coreService.registerApplication(OPENSTACK_NETWORKING_APP_ID);
120 deviceService.addListener(internalDeviceListener);
121 osNodeService.addListener(internalNodeListener);
122 hostProvider = hostProviderRegistry.register(this);
123
124 log.info("Started");
125 }
126
127 @Deactivate
Ray Milkey9c9cde42018-01-12 14:22:06 -0800128 void deactivate() {
Hyunsun Moon44aac662017-02-18 02:07:01 +0900129 hostProviderRegistry.unregister(this);
130 osNodeService.removeListener(internalNodeListener);
131 deviceService.removeListener(internalDeviceListener);
132
133 deviceEventExecutor.shutdown();
134 configEventExecutor.shutdown();
135
136 log.info("Stopped");
137 }
138
139 @Override
140 public void triggerProbe(Host host) {
141 // no probe is required
142 }
143
144 private void processPortAdded(Port port) {
145 // TODO check the node state is COMPLETE
146 org.openstack4j.model.network.Port osPort = osNetworkService.port(port);
147 if (osPort == null) {
148 log.warn(ERR_ADD_HOST + "OpenStack port for {} not found", port);
149 return;
150 }
151
152 Network osNet = osNetworkService.network(osPort.getNetworkId());
153 if (osNet == null) {
154 log.warn(ERR_ADD_HOST + "OpenStack network {} not found",
155 osPort.getNetworkId());
156 return;
157 }
158
159 if (osPort.getFixedIps().isEmpty()) {
160 log.warn(ERR_ADD_HOST + "no fixed IP for port {}", osPort.getId());
161 return;
162 }
163
164 MacAddress macAddr = MacAddress.valueOf(osPort.getMacAddress());
165 Set<IpAddress> fixedIps = osPort.getFixedIps().stream()
166 .map(ip -> IpAddress.valueOf(ip.getIpAddress()))
167 .collect(Collectors.toSet());
168 ConnectPoint connectPoint = new ConnectPoint(port.element().id(), port.number());
169
170 DefaultAnnotations.Builder annotations = DefaultAnnotations.builder()
171 .set(ANNOTATION_NETWORK_ID, osPort.getNetworkId())
172 .set(ANNOTATION_PORT_ID, osPort.getId())
daniel park5fe30792017-08-22 15:42:54 +0900173 .set(ANNOTATION_CREATE_TIME, String.valueOf(System.currentTimeMillis()))
174 .set(ANNOTATION_SEGMENT_ID, osNet.getProviderSegID());
Hyunsun Moon44aac662017-02-18 02:07:01 +0900175
176 HostDescription hostDesc = new DefaultHostDescription(
177 macAddr,
178 VlanId.NONE,
179 new HostLocation(connectPoint, System.currentTimeMillis()),
180 fixedIps,
181 annotations.build());
182
183 HostId hostId = HostId.hostId(macAddr);
184 hostProvider.hostDetected(hostId, hostDesc, false);
185 }
186
187 private void processPortRemoved(Port port) {
188 ConnectPoint connectPoint = new ConnectPoint(port.element().id(), port.number());
189 hostService.getConnectedHosts(connectPoint).forEach(host -> {
190 hostProvider.hostVanished(host.id());
191 });
192 }
193
194 private class InternalDeviceListener implements DeviceListener {
195
196 @Override
197 public boolean isRelevant(DeviceEvent event) {
198 Device device = event.subject();
199 if (!mastershipService.isLocalMaster(device.id())) {
200 // do not allow to proceed without mastership
201 return false;
202 }
203 Port port = event.port();
204 if (port == null) {
205 return false;
206 }
207 String portName = port.annotations().value(PORT_NAME);
208 if (Strings.isNullOrEmpty(portName) ||
209 !portName.startsWith(PORT_NAME_PREFIX_VM)) {
210 // handles Nova created port event only
211 return false;
212 }
213 return true;
214 }
215
216 @Override
217 public void event(DeviceEvent event) {
218 switch (event.type()) {
219 case PORT_UPDATED:
220 if (!event.port().isEnabled()) {
221 deviceEventExecutor.execute(() -> {
222 log.debug("Instance port {} is removed from {}",
223 event.port().annotations().value(PORT_NAME),
224 event.subject().id());
225 processPortRemoved(event.port());
226 });
227 } else if (event.port().isEnabled()) {
228 deviceEventExecutor.execute(() -> {
229 log.debug("Instance Port {} is detected from {}",
230 event.port().annotations().value(PORT_NAME),
231 event.subject().id());
232 processPortAdded(event.port());
233 });
234 }
235 break;
236 case PORT_ADDED:
237 deviceEventExecutor.execute(() -> {
238 log.debug("Instance port {} is detected from {}",
239 event.port().annotations().value(PORT_NAME),
240 event.subject().id());
241 processPortAdded(event.port());
242 });
243 break;
Hyunsun Moonb7a9cd22017-02-24 11:12:53 +0900244 case PORT_REMOVED:
245 deviceEventExecutor.execute(() -> {
246 log.debug("Instance port {} is removed from {}",
247 event.port().annotations().value(PORT_NAME),
248 event.subject().id());
249 processPortRemoved(event.port());
250 });
Hyunsun Moon44aac662017-02-18 02:07:01 +0900251 default:
252 break;
253 }
254 }
255 }
256
257 private class InternalOpenstackNodeListener implements OpenstackNodeListener {
258
259 @Override
260 public void event(OpenstackNodeEvent event) {
261 OpenstackNode osNode = event.subject();
262 // TODO check leadership of the node and make only the leader process
263
264 switch (event.type()) {
Hyunsun Moon0d457362017-06-27 17:19:41 +0900265 case OPENSTACK_NODE_COMPLETE:
Hyunsun Moon44aac662017-02-18 02:07:01 +0900266 deviceEventExecutor.execute(() -> {
267 log.info("COMPLETE node {} is detected", osNode.hostname());
268 processCompleteNode(event.subject());
269 });
270 break;
Hyunsun Moon0d457362017-06-27 17:19:41 +0900271 case OPENSTACK_NODE_INCOMPLETE:
Hyunsun Moon44aac662017-02-18 02:07:01 +0900272 log.warn("{} is changed to INCOMPLETE state", osNode);
273 break;
Hyunsun Moon0d457362017-06-27 17:19:41 +0900274 case OPENSTACK_NODE_CREATED:
275 case OPENSTACK_NODE_UPDATED:
276 case OPENSTACK_NODE_REMOVED:
Hyunsun Moon44aac662017-02-18 02:07:01 +0900277 default:
278 break;
279 }
280 }
281
282 private void processCompleteNode(OpenstackNode osNode) {
Hyunsun Moon0d457362017-06-27 17:19:41 +0900283 deviceService.getPorts(osNode.intgBridge()).stream()
Hyunsun Moon44aac662017-02-18 02:07:01 +0900284 .filter(port -> port.annotations().value(PORT_NAME)
285 .startsWith(PORT_NAME_PREFIX_VM) &&
286 port.isEnabled())
287 .forEach(port -> {
288 log.debug("Instance port {} is detected from {}",
289 port.annotations().value(PORT_NAME),
290 osNode.hostname());
291 processPortAdded(port);
292 });
293
294 Tools.stream(hostService.getHosts())
295 .filter(host -> deviceService.getPort(
296 host.location().deviceId(),
297 host.location().port()) == null)
298 .forEach(host -> {
299 log.info("Remove stale host {}", host.id());
300 hostProvider.hostVanished(host.id());
301 });
302 }
303 }
304}