blob: 6179c146bd7833e780cd61f8409836b64e496a8c [file] [log] [blame]
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07001/*
2 * Copyright 2014 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 */
Brian O'Connorabafb502014-12-02 22:26:20 -080016package org.onosproject.net.host.impl;
tomdb0d03f2014-08-27 16:34:15 -070017
18import org.apache.felix.scr.annotations.Activate;
19import org.apache.felix.scr.annotations.Component;
20import org.apache.felix.scr.annotations.Deactivate;
tom5f38b3a2014-08-27 23:50:54 -070021import org.apache.felix.scr.annotations.Reference;
22import org.apache.felix.scr.annotations.ReferenceCardinality;
tomdb0d03f2014-08-27 16:34:15 -070023import org.apache.felix.scr.annotations.Service;
Simon Huntff663742015-05-14 13:33:05 -070024import org.onlab.packet.IpAddress;
25import org.onlab.packet.MacAddress;
26import org.onlab.packet.VlanId;
Thomas Vachuska42e8cce2015-07-29 19:25:18 -070027import org.onosproject.net.provider.AbstractListenerProviderRegistry;
Changhoon Yoon541ef712015-05-23 17:18:34 +090028import org.onosproject.core.Permission;
Sahil Lele3a0cdd52015-07-21 14:16:31 -070029import org.onosproject.incubator.net.config.NetworkConfigEvent;
30import org.onosproject.incubator.net.config.NetworkConfigListener;
31import org.onosproject.incubator.net.config.NetworkConfigService;
32import org.onosproject.incubator.net.config.basics.BasicHostConfig;
Brian O'Connorabafb502014-12-02 22:26:20 -080033import org.onosproject.net.ConnectPoint;
34import org.onosproject.net.DeviceId;
35import org.onosproject.net.Host;
36import org.onosproject.net.HostId;
37import org.onosproject.net.device.DeviceService;
38import org.onosproject.net.host.HostAdminService;
39import org.onosproject.net.host.HostDescription;
40import org.onosproject.net.host.HostEvent;
41import org.onosproject.net.host.HostListener;
42import org.onosproject.net.host.HostProvider;
43import org.onosproject.net.host.HostProviderRegistry;
44import org.onosproject.net.host.HostProviderService;
45import org.onosproject.net.host.HostService;
46import org.onosproject.net.host.HostStore;
47import org.onosproject.net.host.HostStoreDelegate;
48import org.onosproject.net.host.PortAddresses;
49import org.onosproject.net.packet.PacketService;
Brian O'Connorabafb502014-12-02 22:26:20 -080050import org.onosproject.net.provider.AbstractProviderService;
tomdb0d03f2014-08-27 16:34:15 -070051import org.slf4j.Logger;
tom5f38b3a2014-08-27 23:50:54 -070052
Simon Huntff663742015-05-14 13:33:05 -070053import java.util.Set;
54
55import static com.google.common.base.Preconditions.checkNotNull;
Sahil Lele3a0cdd52015-07-21 14:16:31 -070056import static com.google.common.base.Preconditions.checkState;
Changhoon Yoon541ef712015-05-23 17:18:34 +090057import static org.onosproject.security.AppGuard.checkPermission;
Thomas Vachuska42e8cce2015-07-29 19:25:18 -070058import static org.slf4j.LoggerFactory.getLogger;
Simon Huntff663742015-05-14 13:33:05 -070059
tomdb0d03f2014-08-27 16:34:15 -070060/**
61 * Provides basic implementation of the host SB & NB APIs.
62 */
63@Component(immediate = true)
64@Service
tom202175a2014-09-19 19:00:11 -070065public class HostManager
Thomas Vachuska42e8cce2015-07-29 19:25:18 -070066 extends AbstractListenerProviderRegistry<HostEvent, HostListener, HostProvider, HostProviderService>
tom89b63c52014-09-16 09:19:51 -070067 implements HostService, HostAdminService, HostProviderRegistry {
tomdb0d03f2014-08-27 16:34:15 -070068
tom5f38b3a2014-08-27 23:50:54 -070069 private final Logger log = getLogger(getClass());
tomdb0d03f2014-08-27 16:34:15 -070070
Thomas Vachuska42e8cce2015-07-29 19:25:18 -070071 public static final String HOST_ID_NULL = "Host ID cannot be null";
tom5f38b3a2014-08-27 23:50:54 -070072
Sahil Lele3a0cdd52015-07-21 14:16:31 -070073 private final NetworkConfigListener networkConfigListener = new InternalNetworkConfigListener();
74
tomc78acee2014-09-24 15:16:55 -070075 private HostStoreDelegate delegate = new InternalStoreDelegate();
76
tom5bcc9462014-09-19 10:11:31 -070077 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
78 protected HostStore store;
tom7869ad92014-09-09 14:32:08 -070079
tom5f38b3a2014-08-27 23:50:54 -070080 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jonathan Hart70da5122014-10-01 16:37:42 -070081 protected DeviceService deviceService;
82
83 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
84 protected PacketService packetService;
85
Sahil Lele3a0cdd52015-07-21 14:16:31 -070086 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
87 protected NetworkConfigService networkConfigService;
88
Jonathan Hart70da5122014-10-01 16:37:42 -070089 private HostMonitor monitor;
tomdb0d03f2014-08-27 16:34:15 -070090
91 @Activate
92 public void activate() {
tomc78acee2014-09-24 15:16:55 -070093 store.setDelegate(delegate);
tom96dfcab2014-08-28 09:26:03 -070094 eventDispatcher.addSink(HostEvent.class, listenerRegistry);
Sahil Lele3a0cdd52015-07-21 14:16:31 -070095 networkConfigService.addListener(networkConfigListener);
Thomas Vachuska42e8cce2015-07-29 19:25:18 -070096 monitor = new HostMonitor(deviceService, packetService, this);
Jonathan Hart8f6f1ea2014-10-03 16:05:19 -070097 monitor.start();
Thomas Vachuska42e8cce2015-07-29 19:25:18 -070098 log.info("Started");
tomdb0d03f2014-08-27 16:34:15 -070099 }
100
101 @Deactivate
102 public void deactivate() {
tomc78acee2014-09-24 15:16:55 -0700103 store.unsetDelegate(delegate);
tom5f38b3a2014-08-27 23:50:54 -0700104 eventDispatcher.removeSink(HostEvent.class);
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700105 networkConfigService.removeListener(networkConfigListener);
tomdb0d03f2014-08-27 16:34:15 -0700106 log.info("Stopped");
107 }
108
109 @Override
tom5f38b3a2014-08-27 23:50:54 -0700110 protected HostProviderService createProviderService(HostProvider provider) {
Jonathan Hart70da5122014-10-01 16:37:42 -0700111 monitor.registerHostProvider(provider);
tom5f38b3a2014-08-27 23:50:54 -0700112 return new InternalHostProviderService(provider);
tomdb0d03f2014-08-27 16:34:15 -0700113 }
114
tom7869ad92014-09-09 14:32:08 -0700115 @Override
116 public int getHostCount() {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900117 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700118 return store.getHostCount();
119 }
120
121 @Override
122 public Iterable<Host> getHosts() {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900123 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700124 return store.getHosts();
125 }
126
127 @Override
128 public Host getHost(HostId hostId) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900129 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700130 checkNotNull(hostId, HOST_ID_NULL);
131 return store.getHost(hostId);
132 }
133
134 @Override
Ayaka Koshibea9c199f2014-09-16 16:21:40 -0700135 public Set<Host> getHostsByVlan(VlanId vlanId) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900136 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700137 return store.getHosts(vlanId);
138 }
139
140 @Override
Ayaka Koshibea9c199f2014-09-16 16:21:40 -0700141 public Set<Host> getHostsByMac(MacAddress mac) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900142 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700143 checkNotNull(mac, "MAC address cannot be null");
144 return store.getHosts(mac);
145 }
146
147 @Override
Pavlin Radoslavov33f228a2014-10-27 19:33:16 -0700148 public Set<Host> getHostsByIp(IpAddress ip) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900149 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700150 checkNotNull(ip, "IP address cannot be null");
151 return store.getHosts(ip);
152 }
153
154 @Override
155 public Set<Host> getConnectedHosts(ConnectPoint connectPoint) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900156 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700157 checkNotNull(connectPoint, "Connection point cannot be null");
158 return store.getConnectedHosts(connectPoint);
159 }
160
161 @Override
162 public Set<Host> getConnectedHosts(DeviceId deviceId) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900163 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700164 checkNotNull(deviceId, "Device ID cannot be null");
165 return store.getConnectedHosts(deviceId);
166 }
167
168 @Override
Jonathan Hartac60c082014-09-23 08:55:17 -0700169 public void startMonitoringIp(IpAddress ip) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900170 checkPermission(Permission.HOST_EVENT);
Jonathan Hart70da5122014-10-01 16:37:42 -0700171 monitor.addMonitoringFor(ip);
Jonathan Hartfca736c2014-09-19 17:26:59 -0700172 }
173
174 @Override
Jonathan Hartac60c082014-09-23 08:55:17 -0700175 public void stopMonitoringIp(IpAddress ip) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900176 checkPermission(Permission.HOST_EVENT);
Jonathan Hart70da5122014-10-01 16:37:42 -0700177 monitor.stopMonitoring(ip);
Jonathan Hartac60c082014-09-23 08:55:17 -0700178 }
179
180 @Override
181 public void requestMac(IpAddress ip) {
Thomas Vachuska42e8cce2015-07-29 19:25:18 -0700182 // FIXME!!!! Auto-generated method stub
tom7869ad92014-09-09 14:32:08 -0700183 }
184
tom89b63c52014-09-16 09:19:51 -0700185 @Override
186 public void removeHost(HostId hostId) {
187 checkNotNull(hostId, HOST_ID_NULL);
188 HostEvent event = store.removeHost(hostId);
189 if (event != null) {
tom89b63c52014-09-16 09:19:51 -0700190 post(event);
191 }
192 }
193
Jonathan Hartac60c082014-09-23 08:55:17 -0700194 @Override
Jonathan Hart09585c62014-09-23 16:58:04 -0700195 public void bindAddressesToPort(PortAddresses addresses) {
196 store.updateAddressBindings(addresses);
Jonathan Hartac60c082014-09-23 08:55:17 -0700197 }
198
199 @Override
Jonathan Hart09585c62014-09-23 16:58:04 -0700200 public void unbindAddressesFromPort(PortAddresses portAddresses) {
201 store.removeAddressBindings(portAddresses);
202 }
203
204 @Override
205 public void clearAddresses(ConnectPoint connectPoint) {
206 store.clearAddressBindings(connectPoint);
Jonathan Hartac60c082014-09-23 08:55:17 -0700207 }
208
209 @Override
210 public Set<PortAddresses> getAddressBindings() {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900211 checkPermission(Permission.HOST_READ);
Jonathan Hart43c182c2014-09-23 11:13:42 -0700212 return store.getAddressBindings();
Jonathan Hartac60c082014-09-23 08:55:17 -0700213 }
214
215 @Override
Jonathan Harta887ba82014-11-03 15:20:52 -0800216 public Set<PortAddresses> getAddressBindingsForPort(ConnectPoint connectPoint) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900217 checkPermission(Permission.HOST_READ);
Jonathan Hart43c182c2014-09-23 11:13:42 -0700218 return store.getAddressBindingsForPort(connectPoint);
Jonathan Hartac60c082014-09-23 08:55:17 -0700219 }
220
tomdb0d03f2014-08-27 16:34:15 -0700221 // Personalized host provider service issued to the supplied provider.
tom7869ad92014-09-09 14:32:08 -0700222 private class InternalHostProviderService
223 extends AbstractProviderService<HostProvider>
tomdb0d03f2014-08-27 16:34:15 -0700224 implements HostProviderService {
tomcfde0622014-09-09 11:02:42 -0700225 InternalHostProviderService(HostProvider provider) {
tomdb0d03f2014-08-27 16:34:15 -0700226 super(provider);
227 }
228
229 @Override
tom7869ad92014-09-09 14:32:08 -0700230 public void hostDetected(HostId hostId, HostDescription hostDescription) {
231 checkNotNull(hostId, HOST_ID_NULL);
232 checkValidity();
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700233 hostDescription = validateHost(hostDescription, hostId);
tom7869ad92014-09-09 14:32:08 -0700234 HostEvent event = store.createOrUpdateHost(provider().id(), hostId,
235 hostDescription);
236 if (event != null) {
tom7869ad92014-09-09 14:32:08 -0700237 post(event);
238 }
tomdb0d03f2014-08-27 16:34:15 -0700239 }
240
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700241 // returns a HostDescription made from the union of the BasicHostConfig
242 // annotations if it exists
243 private HostDescription validateHost(HostDescription hostDescription, HostId hostId) {
244 BasicHostConfig cfg = networkConfigService.getConfig(hostId, BasicHostConfig.class);
245 checkState(cfg == null || cfg.isAllowed(), "Host {} is not allowed", hostId);
Ayaka Koshibe5373e762015-08-06 12:31:44 -0700246
247 return BasicHostOperator.combine(cfg, hostDescription);
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700248 }
249
tomdb0d03f2014-08-27 16:34:15 -0700250 @Override
tom7869ad92014-09-09 14:32:08 -0700251 public void hostVanished(HostId hostId) {
252 checkNotNull(hostId, HOST_ID_NULL);
253 checkValidity();
254 HostEvent event = store.removeHost(hostId);
255 if (event != null) {
tom7869ad92014-09-09 14:32:08 -0700256 post(event);
257 }
tomdb0d03f2014-08-27 16:34:15 -0700258 }
259 }
tom7869ad92014-09-09 14:32:08 -0700260
tomc78acee2014-09-24 15:16:55 -0700261 // Store delegate to re-post events emitted from the store.
262 private class InternalStoreDelegate implements HostStoreDelegate {
263 @Override
264 public void notify(HostEvent event) {
265 post(event);
266 }
267 }
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700268
269 // listens for NetworkConfigEvents of type BasicHostConfig and removes
270 // links that the config does not allow
271 private class InternalNetworkConfigListener implements NetworkConfigListener {
272 @Override
273 public void event(NetworkConfigEvent event) {
274 if ((event.type() == NetworkConfigEvent.Type.CONFIG_ADDED ||
275 event.type() == NetworkConfigEvent.Type.CONFIG_UPDATED) &&
276 event.configClass().equals(BasicHostConfig.class)) {
277 log.info("Detected Host network config event {}", event.type());
278 kickOutBadHost(((HostId) event.subject()));
279 }
280 }
281 }
282
283 // checks if the specified host is allowed by the BasicHostConfig
284 // and if not, removes it
285 private void kickOutBadHost(HostId hostId) {
286 BasicHostConfig cfg = networkConfigService.getConfig(hostId, BasicHostConfig.class);
287 if (cfg != null && !cfg.isAllowed()) {
288 Host badHost = getHost(hostId);
289 if (badHost != null) {
290 removeHost(hostId);
291 } else {
292 log.info("Failed removal: Host {} does not exist", hostId);
293 }
294 }
295 }
tomdb0d03f2014-08-27 16:34:15 -0700296}