blob: 99d401acf3227307e0270d084d5c84820ee61d47 [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;
Jonathan Hart4cb39882015-08-12 23:50:55 -040027import org.onosproject.incubator.net.intf.InterfaceService;
Thomas Vachuska42e8cce2015-07-29 19:25:18 -070028import org.onosproject.net.provider.AbstractListenerProviderRegistry;
Changhoon Yoon541ef712015-05-23 17:18:34 +090029import org.onosproject.core.Permission;
Ray Milkeya4122362015-08-18 15:19:08 -070030import org.onosproject.net.config.NetworkConfigEvent;
31import org.onosproject.net.config.NetworkConfigListener;
32import org.onosproject.net.config.NetworkConfigService;
Thomas Vachuska4998caa2015-08-26 13:28:38 -070033import org.onosproject.net.config.basics.BasicHostConfig;
Brian O'Connorabafb502014-12-02 22:26:20 -080034import org.onosproject.net.ConnectPoint;
35import org.onosproject.net.DeviceId;
36import org.onosproject.net.Host;
37import org.onosproject.net.HostId;
38import org.onosproject.net.device.DeviceService;
39import org.onosproject.net.host.HostAdminService;
40import org.onosproject.net.host.HostDescription;
41import org.onosproject.net.host.HostEvent;
42import org.onosproject.net.host.HostListener;
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.host.HostStore;
48import org.onosproject.net.host.HostStoreDelegate;
49import org.onosproject.net.host.PortAddresses;
50import org.onosproject.net.packet.PacketService;
Brian O'Connorabafb502014-12-02 22:26:20 -080051import org.onosproject.net.provider.AbstractProviderService;
tomdb0d03f2014-08-27 16:34:15 -070052import org.slf4j.Logger;
tom5f38b3a2014-08-27 23:50:54 -070053
Simon Huntff663742015-05-14 13:33:05 -070054import java.util.Set;
55
56import static com.google.common.base.Preconditions.checkNotNull;
Sahil Lele3a0cdd52015-07-21 14:16:31 -070057import static com.google.common.base.Preconditions.checkState;
Changhoon Yoon541ef712015-05-23 17:18:34 +090058import static org.onosproject.security.AppGuard.checkPermission;
Thomas Vachuska42e8cce2015-07-29 19:25:18 -070059import static org.slf4j.LoggerFactory.getLogger;
Simon Huntff663742015-05-14 13:33:05 -070060
tomdb0d03f2014-08-27 16:34:15 -070061/**
62 * Provides basic implementation of the host SB & NB APIs.
63 */
64@Component(immediate = true)
65@Service
tom202175a2014-09-19 19:00:11 -070066public class HostManager
Thomas Vachuska42e8cce2015-07-29 19:25:18 -070067 extends AbstractListenerProviderRegistry<HostEvent, HostListener, HostProvider, HostProviderService>
tom89b63c52014-09-16 09:19:51 -070068 implements HostService, HostAdminService, HostProviderRegistry {
tomdb0d03f2014-08-27 16:34:15 -070069
tom5f38b3a2014-08-27 23:50:54 -070070 private final Logger log = getLogger(getClass());
tomdb0d03f2014-08-27 16:34:15 -070071
Thomas Vachuska42e8cce2015-07-29 19:25:18 -070072 public static final String HOST_ID_NULL = "Host ID cannot be null";
tom5f38b3a2014-08-27 23:50:54 -070073
Sahil Lele3a0cdd52015-07-21 14:16:31 -070074 private final NetworkConfigListener networkConfigListener = new InternalNetworkConfigListener();
75
tomc78acee2014-09-24 15:16:55 -070076 private HostStoreDelegate delegate = new InternalStoreDelegate();
77
tom5bcc9462014-09-19 10:11:31 -070078 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
79 protected HostStore store;
tom7869ad92014-09-09 14:32:08 -070080
tom5f38b3a2014-08-27 23:50:54 -070081 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Jonathan Hart70da5122014-10-01 16:37:42 -070082 protected DeviceService deviceService;
83
84 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
85 protected PacketService packetService;
86
Sahil Lele3a0cdd52015-07-21 14:16:31 -070087 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
88 protected NetworkConfigService networkConfigService;
89
Jonathan Hart4cb39882015-08-12 23:50:55 -040090 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
91 protected InterfaceService interfaceService;
92
Jonathan Hart70da5122014-10-01 16:37:42 -070093 private HostMonitor monitor;
tomdb0d03f2014-08-27 16:34:15 -070094
95 @Activate
96 public void activate() {
tomc78acee2014-09-24 15:16:55 -070097 store.setDelegate(delegate);
tom96dfcab2014-08-28 09:26:03 -070098 eventDispatcher.addSink(HostEvent.class, listenerRegistry);
Sahil Lele3a0cdd52015-07-21 14:16:31 -070099 networkConfigService.addListener(networkConfigListener);
Jonathan Hart4cb39882015-08-12 23:50:55 -0400100 monitor = new HostMonitor(packetService, this, interfaceService);
Jonathan Hart8f6f1ea2014-10-03 16:05:19 -0700101 monitor.start();
Thomas Vachuska42e8cce2015-07-29 19:25:18 -0700102 log.info("Started");
tomdb0d03f2014-08-27 16:34:15 -0700103 }
104
105 @Deactivate
106 public void deactivate() {
tomc78acee2014-09-24 15:16:55 -0700107 store.unsetDelegate(delegate);
tom5f38b3a2014-08-27 23:50:54 -0700108 eventDispatcher.removeSink(HostEvent.class);
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700109 networkConfigService.removeListener(networkConfigListener);
tomdb0d03f2014-08-27 16:34:15 -0700110 log.info("Stopped");
111 }
112
113 @Override
tom5f38b3a2014-08-27 23:50:54 -0700114 protected HostProviderService createProviderService(HostProvider provider) {
Jonathan Hart70da5122014-10-01 16:37:42 -0700115 monitor.registerHostProvider(provider);
tom5f38b3a2014-08-27 23:50:54 -0700116 return new InternalHostProviderService(provider);
tomdb0d03f2014-08-27 16:34:15 -0700117 }
118
tom7869ad92014-09-09 14:32:08 -0700119 @Override
120 public int getHostCount() {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900121 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700122 return store.getHostCount();
123 }
124
125 @Override
126 public Iterable<Host> getHosts() {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900127 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700128 return store.getHosts();
129 }
130
131 @Override
132 public Host getHost(HostId hostId) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900133 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700134 checkNotNull(hostId, HOST_ID_NULL);
135 return store.getHost(hostId);
136 }
137
138 @Override
Ayaka Koshibea9c199f2014-09-16 16:21:40 -0700139 public Set<Host> getHostsByVlan(VlanId vlanId) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900140 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700141 return store.getHosts(vlanId);
142 }
143
144 @Override
Ayaka Koshibea9c199f2014-09-16 16:21:40 -0700145 public Set<Host> getHostsByMac(MacAddress mac) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900146 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700147 checkNotNull(mac, "MAC address cannot be null");
148 return store.getHosts(mac);
149 }
150
151 @Override
Pavlin Radoslavov33f228a2014-10-27 19:33:16 -0700152 public Set<Host> getHostsByIp(IpAddress ip) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900153 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700154 checkNotNull(ip, "IP address cannot be null");
155 return store.getHosts(ip);
156 }
157
158 @Override
159 public Set<Host> getConnectedHosts(ConnectPoint connectPoint) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900160 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700161 checkNotNull(connectPoint, "Connection point cannot be null");
162 return store.getConnectedHosts(connectPoint);
163 }
164
165 @Override
166 public Set<Host> getConnectedHosts(DeviceId deviceId) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900167 checkPermission(Permission.HOST_READ);
tom7869ad92014-09-09 14:32:08 -0700168 checkNotNull(deviceId, "Device ID cannot be null");
169 return store.getConnectedHosts(deviceId);
170 }
171
172 @Override
Jonathan Hartac60c082014-09-23 08:55:17 -0700173 public void startMonitoringIp(IpAddress ip) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900174 checkPermission(Permission.HOST_EVENT);
Jonathan Hart70da5122014-10-01 16:37:42 -0700175 monitor.addMonitoringFor(ip);
Jonathan Hartfca736c2014-09-19 17:26:59 -0700176 }
177
178 @Override
Jonathan Hartac60c082014-09-23 08:55:17 -0700179 public void stopMonitoringIp(IpAddress ip) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900180 checkPermission(Permission.HOST_EVENT);
Jonathan Hart70da5122014-10-01 16:37:42 -0700181 monitor.stopMonitoring(ip);
Jonathan Hartac60c082014-09-23 08:55:17 -0700182 }
183
184 @Override
185 public void requestMac(IpAddress ip) {
Thomas Vachuska42e8cce2015-07-29 19:25:18 -0700186 // FIXME!!!! Auto-generated method stub
tom7869ad92014-09-09 14:32:08 -0700187 }
188
tom89b63c52014-09-16 09:19:51 -0700189 @Override
190 public void removeHost(HostId hostId) {
191 checkNotNull(hostId, HOST_ID_NULL);
192 HostEvent event = store.removeHost(hostId);
193 if (event != null) {
tom89b63c52014-09-16 09:19:51 -0700194 post(event);
195 }
196 }
197
Jonathan Hartac60c082014-09-23 08:55:17 -0700198 @Override
Jonathan Hart09585c62014-09-23 16:58:04 -0700199 public void bindAddressesToPort(PortAddresses addresses) {
200 store.updateAddressBindings(addresses);
Jonathan Hartac60c082014-09-23 08:55:17 -0700201 }
202
203 @Override
Jonathan Hart09585c62014-09-23 16:58:04 -0700204 public void unbindAddressesFromPort(PortAddresses portAddresses) {
205 store.removeAddressBindings(portAddresses);
206 }
207
208 @Override
209 public void clearAddresses(ConnectPoint connectPoint) {
210 store.clearAddressBindings(connectPoint);
Jonathan Hartac60c082014-09-23 08:55:17 -0700211 }
212
213 @Override
214 public Set<PortAddresses> getAddressBindings() {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900215 checkPermission(Permission.HOST_READ);
Jonathan Hart43c182c2014-09-23 11:13:42 -0700216 return store.getAddressBindings();
Jonathan Hartac60c082014-09-23 08:55:17 -0700217 }
218
219 @Override
Jonathan Harta887ba82014-11-03 15:20:52 -0800220 public Set<PortAddresses> getAddressBindingsForPort(ConnectPoint connectPoint) {
Changhoon Yoon541ef712015-05-23 17:18:34 +0900221 checkPermission(Permission.HOST_READ);
Jonathan Hart43c182c2014-09-23 11:13:42 -0700222 return store.getAddressBindingsForPort(connectPoint);
Jonathan Hartac60c082014-09-23 08:55:17 -0700223 }
224
tomdb0d03f2014-08-27 16:34:15 -0700225 // Personalized host provider service issued to the supplied provider.
tom7869ad92014-09-09 14:32:08 -0700226 private class InternalHostProviderService
227 extends AbstractProviderService<HostProvider>
tomdb0d03f2014-08-27 16:34:15 -0700228 implements HostProviderService {
tomcfde0622014-09-09 11:02:42 -0700229 InternalHostProviderService(HostProvider provider) {
tomdb0d03f2014-08-27 16:34:15 -0700230 super(provider);
231 }
232
233 @Override
tom7869ad92014-09-09 14:32:08 -0700234 public void hostDetected(HostId hostId, HostDescription hostDescription) {
235 checkNotNull(hostId, HOST_ID_NULL);
236 checkValidity();
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700237 hostDescription = validateHost(hostDescription, hostId);
tom7869ad92014-09-09 14:32:08 -0700238 HostEvent event = store.createOrUpdateHost(provider().id(), hostId,
239 hostDescription);
240 if (event != null) {
tom7869ad92014-09-09 14:32:08 -0700241 post(event);
242 }
tomdb0d03f2014-08-27 16:34:15 -0700243 }
244
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700245 // returns a HostDescription made from the union of the BasicHostConfig
246 // annotations if it exists
247 private HostDescription validateHost(HostDescription hostDescription, HostId hostId) {
248 BasicHostConfig cfg = networkConfigService.getConfig(hostId, BasicHostConfig.class);
249 checkState(cfg == null || cfg.isAllowed(), "Host {} is not allowed", hostId);
Ayaka Koshibe5373e762015-08-06 12:31:44 -0700250
251 return BasicHostOperator.combine(cfg, hostDescription);
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700252 }
253
tomdb0d03f2014-08-27 16:34:15 -0700254 @Override
tom7869ad92014-09-09 14:32:08 -0700255 public void hostVanished(HostId hostId) {
256 checkNotNull(hostId, HOST_ID_NULL);
257 checkValidity();
258 HostEvent event = store.removeHost(hostId);
259 if (event != null) {
tom7869ad92014-09-09 14:32:08 -0700260 post(event);
261 }
tomdb0d03f2014-08-27 16:34:15 -0700262 }
263 }
tom7869ad92014-09-09 14:32:08 -0700264
tomc78acee2014-09-24 15:16:55 -0700265 // Store delegate to re-post events emitted from the store.
266 private class InternalStoreDelegate implements HostStoreDelegate {
267 @Override
268 public void notify(HostEvent event) {
269 post(event);
270 }
271 }
Sahil Lele3a0cdd52015-07-21 14:16:31 -0700272
273 // listens for NetworkConfigEvents of type BasicHostConfig and removes
274 // links that the config does not allow
275 private class InternalNetworkConfigListener implements NetworkConfigListener {
276 @Override
277 public void event(NetworkConfigEvent event) {
278 if ((event.type() == NetworkConfigEvent.Type.CONFIG_ADDED ||
279 event.type() == NetworkConfigEvent.Type.CONFIG_UPDATED) &&
280 event.configClass().equals(BasicHostConfig.class)) {
281 log.info("Detected Host network config event {}", event.type());
282 kickOutBadHost(((HostId) event.subject()));
283 }
284 }
285 }
286
287 // checks if the specified host is allowed by the BasicHostConfig
288 // and if not, removes it
289 private void kickOutBadHost(HostId hostId) {
290 BasicHostConfig cfg = networkConfigService.getConfig(hostId, BasicHostConfig.class);
291 if (cfg != null && !cfg.isAllowed()) {
292 Host badHost = getHost(hostId);
293 if (badHost != null) {
294 removeHost(hostId);
295 } else {
296 log.info("Failed removal: Host {} does not exist", hostId);
297 }
298 }
299 }
tomdb0d03f2014-08-27 16:34:15 -0700300}