blob: e2e3fbe3553f10813d440fc197713ca6a740beb9 [file] [log] [blame]
Thomas Vachuska781d18b2014-10-27 10:31:25 -07001/*
Brian O'Connora09fe5b2017-08-03 21:12:30 -07002 * Copyright 2015-present Open Networking Foundation
Thomas Vachuska781d18b2014-10-27 10:31:25 -07003 *
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07004 * 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
Thomas Vachuska781d18b2014-10-27 10:31:25 -07007 *
Thomas Vachuska4f1a60c2014-10-28 13:39:07 -07008 * 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.
Thomas Vachuska781d18b2014-10-27 10:31:25 -070015 */
Brian O'Connorabafb502014-12-02 22:26:20 -080016package org.onosproject.provider.lldp.impl;
alshabib7911a052014-10-16 17:49:37 -070017
Ray Milkeyd84f89b2018-08-17 14:54:17 -070018import com.google.common.collect.ImmutableMap;
19import com.google.common.collect.ImmutableSet;
20import com.google.common.collect.Maps;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080021import org.onlab.packet.Ethernet;
Thomas Vachuska6519e6f2015-03-11 02:29:31 -070022import org.onosproject.cfg.ComponentConfigService;
Ayaka Koshibe12c8c082015-12-08 12:48:46 -080023import org.onosproject.cluster.ClusterMetadataService;
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -070024import org.onosproject.cluster.ClusterService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080025import org.onosproject.core.ApplicationId;
26import org.onosproject.core.CoreService;
Brian O'Connorabafb502014-12-02 22:26:20 -080027import org.onosproject.mastership.MastershipEvent;
28import org.onosproject.mastership.MastershipListener;
29import org.onosproject.mastership.MastershipService;
30import org.onosproject.net.ConnectPoint;
31import org.onosproject.net.Device;
32import org.onosproject.net.DeviceId;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070033import org.onosproject.net.LinkKey;
Brian O'Connorabafb502014-12-02 22:26:20 -080034import org.onosproject.net.Port;
Naoki Shiota399a0b32015-11-15 20:36:13 -060035import org.onosproject.net.config.ConfigFactory;
36import org.onosproject.net.config.NetworkConfigEvent;
37import org.onosproject.net.config.NetworkConfigListener;
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -070038import org.onosproject.net.config.NetworkConfigRegistry;
Brian O'Connorabafb502014-12-02 22:26:20 -080039import org.onosproject.net.device.DeviceEvent;
HIGUCHI Yuta1979f552015-12-28 21:24:26 -080040import org.onosproject.net.device.DeviceEvent.Type;
Brian O'Connorabafb502014-12-02 22:26:20 -080041import org.onosproject.net.device.DeviceListener;
42import org.onosproject.net.device.DeviceService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080043import org.onosproject.net.flow.DefaultTrafficSelector;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080044import org.onosproject.net.flow.TrafficSelector;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070045import org.onosproject.net.link.DefaultLinkDescription;
Brian O'Connorabafb502014-12-02 22:26:20 -080046import org.onosproject.net.link.LinkProviderRegistry;
47import org.onosproject.net.link.LinkProviderService;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070048import org.onosproject.net.link.LinkService;
Ray Milkeyd9bbde82016-06-09 11:35:00 -070049import org.onosproject.net.link.ProbedLinkProvider;
Brian O'Connorabafb502014-12-02 22:26:20 -080050import org.onosproject.net.packet.PacketContext;
Jonathan Hart3cfce8e2015-01-14 16:43:27 -080051import org.onosproject.net.packet.PacketPriority;
Brian O'Connorabafb502014-12-02 22:26:20 -080052import org.onosproject.net.packet.PacketProcessor;
53import org.onosproject.net.packet.PacketService;
54import org.onosproject.net.provider.AbstractProvider;
55import org.onosproject.net.provider.ProviderId;
Ray Milkey957390e2016-02-09 10:02:46 -080056import org.onosproject.provider.lldpcommon.LinkDiscovery;
Ray Milkeyd9bbde82016-06-09 11:35:00 -070057import org.onosproject.provider.lldpcommon.LinkDiscoveryContext;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080058import org.osgi.service.component.ComponentContext;
Ray Milkeyd84f89b2018-08-17 14:54:17 -070059import org.osgi.service.component.annotations.Activate;
60import org.osgi.service.component.annotations.Component;
61import org.osgi.service.component.annotations.Deactivate;
62import org.osgi.service.component.annotations.Modified;
63import org.osgi.service.component.annotations.Reference;
64import org.osgi.service.component.annotations.ReferenceCardinality;
alshabib7911a052014-10-16 17:49:37 -070065import org.slf4j.Logger;
66
Ray Milkeyd84f89b2018-08-17 14:54:17 -070067import java.util.Dictionary;
68import java.util.EnumSet;
69import java.util.LinkedList;
70import java.util.List;
71import java.util.Map;
72import java.util.Optional;
73import java.util.Properties;
74import java.util.Set;
75import java.util.concurrent.ConcurrentHashMap;
76import java.util.concurrent.ExecutorService;
77import java.util.concurrent.ScheduledExecutorService;
Marc De Leenheer0bfc2a12016-02-02 22:46:27 -080078
79import static com.google.common.base.Strings.isNullOrEmpty;
80import static java.util.concurrent.Executors.newSingleThreadScheduledExecutor;
81import static java.util.concurrent.TimeUnit.SECONDS;
82import static org.onlab.packet.Ethernet.TYPE_BSN;
83import static org.onlab.packet.Ethernet.TYPE_LLDP;
84import static org.onlab.util.Tools.get;
85import static org.onlab.util.Tools.groupedThreads;
Marc De Leenheer0bfc2a12016-02-02 22:46:27 -080086import static org.onosproject.net.Link.Type.DIRECT;
87import static org.onosproject.net.config.basics.SubjectFactories.APP_SUBJECT_FACTORY;
88import static org.onosproject.net.config.basics.SubjectFactories.CONNECT_POINT_SUBJECT_FACTORY;
89import static org.onosproject.net.config.basics.SubjectFactories.DEVICE_SUBJECT_FACTORY;
Thomas Vachuska4167c3f2018-10-16 07:16:31 -070090import static org.onosproject.provider.lldp.impl.OsgiPropertyConstants.*;
Marc De Leenheer0bfc2a12016-02-02 22:46:27 -080091import static org.slf4j.LoggerFactory.getLogger;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080092
alshabib7911a052014-10-16 17:49:37 -070093/**
Thomas Vachuska05453c92015-09-09 14:40:49 -070094 * Provider which uses LLDP and BDDP packets to detect network infrastructure links.
alshabib7911a052014-10-16 17:49:37 -070095 */
Thomas Vachuska4167c3f2018-10-16 07:16:31 -070096@Component(immediate = true,
97 property = {
98 PROP_ENABLED + ":Boolean=" + ENABLED_DEFAULT,
99 PROP_USE_BDDP + ":Boolean=" + USE_BDDP_DEFAULT,
100 PROP_PROBE_RATE + ":Integer=" + PROBE_RATE_DEFAULT,
101 PROP_STALE_LINK_AGE + ":Integer=" + STALE_LINK_AGE_DEFAULT,
102 })
Ayaka Koshibe48229222016-05-16 18:04:26 -0700103public class LldpLinkProvider extends AbstractProvider implements ProbedLinkProvider {
alshabib7911a052014-10-16 17:49:37 -0700104
Thomas Vachuskafc52fec2015-05-18 19:13:56 -0700105 private static final String PROVIDER_NAME = "org.onosproject.provider.lldp";
106
Thomas Vachuska05453c92015-09-09 14:40:49 -0700107 private static final String FORMAT =
108 "Settings: enabled={}, useBDDP={}, probeRate={}, " +
Naoki Shiota399a0b32015-11-15 20:36:13 -0600109 "staleLinkAge={}";
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800110
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700111 // When a Device/Port has this annotation, do not send out LLDP/BDDP
112 public static final String NO_LLDP = "no-lldp";
113
alshabib7911a052014-10-16 17:49:37 -0700114 private final Logger log = getLogger(getClass());
115
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700116 @Reference(cardinality = ReferenceCardinality.MANDATORY)
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800117 protected CoreService coreService;
118
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700119 @Reference(cardinality = ReferenceCardinality.MANDATORY)
alshabib7911a052014-10-16 17:49:37 -0700120 protected LinkProviderRegistry providerRegistry;
121
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700122 @Reference(cardinality = ReferenceCardinality.MANDATORY)
alshabib7911a052014-10-16 17:49:37 -0700123 protected DeviceService deviceService;
124
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700125 @Reference(cardinality = ReferenceCardinality.MANDATORY)
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700126 protected LinkService linkService;
127
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700128 @Reference(cardinality = ReferenceCardinality.MANDATORY)
Marc De Leenheer8b3e80b2015-03-06 14:27:03 -0800129 protected PacketService packetService;
alshabib7911a052014-10-16 17:49:37 -0700130
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700131 @Reference(cardinality = ReferenceCardinality.MANDATORY)
alshabib875d6262014-10-17 16:19:40 -0700132 protected MastershipService masterService;
133
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700134 @Reference(cardinality = ReferenceCardinality.MANDATORY)
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700135 protected ComponentConfigService cfgService;
136
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700137 @Reference(cardinality = ReferenceCardinality.MANDATORY)
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700138 protected ClusterService clusterService;
139
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700140 @Reference(cardinality = ReferenceCardinality.MANDATORY)
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700141 protected NetworkConfigRegistry cfgRegistry;
142
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700143 @Reference(cardinality = ReferenceCardinality.MANDATORY)
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800144 protected ClusterMetadataService clusterMetadataService;
145
alshabib7911a052014-10-16 17:49:37 -0700146 private LinkProviderService providerService;
147
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800148 private ScheduledExecutorService executor;
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700149 protected ExecutorService eventExecutor;
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800150
Ray Milkeye80e18f2016-06-02 16:44:14 -0700151 private boolean shuttingDown = false;
152
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700153 // TODO: Add sanity checking for the configurable params based on the delays
154 private static final long DEVICE_SYNC_DELAY = 5;
155 private static final long LINK_PRUNER_DELAY = 3;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700156
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700157 //@Property(name = PROP_ENABLED, boolValue = true,
158 // label = "If false, link discovery is disabled")
Thomas Vachuska4167c3f2018-10-16 07:16:31 -0700159 private boolean enabled = ENABLED_DEFAULT;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700160
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700161 //@Property(name = PROP_USE_BDDP, boolValue = true,
162 // label = "Use BDDP for link discovery")
Thomas Vachuska4167c3f2018-10-16 07:16:31 -0700163 private boolean useBddp = USE_BDDP_DEFAULT;
alshabib7911a052014-10-16 17:49:37 -0700164
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700165 //@Property(name = PROP_PROBE_RATE, intValue = DEFAULT_PROBE_RATE,
166 // label = "LLDP and BDDP probe rate specified in millis")
Thomas Vachuska4167c3f2018-10-16 07:16:31 -0700167 private int probeRate = PROBE_RATE_DEFAULT;
Saurav Dasc313c402015-02-27 10:09:47 -0800168
Ray Milkeyd84f89b2018-08-17 14:54:17 -0700169 //@Property(name = PROP_STALE_LINK_AGE, intValue = DEFAULT_STALE_LINK_AGE,
170 // label = "Number of millis beyond which links will be considered stale")
Thomas Vachuska4167c3f2018-10-16 07:16:31 -0700171 private int staleLinkAge = STALE_LINK_AGE_DEFAULT;
alshabib7911a052014-10-16 17:49:37 -0700172
Ray Milkey957390e2016-02-09 10:02:46 -0800173 private final LinkDiscoveryContext context = new InternalDiscoveryContext();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800174 private final InternalRoleListener roleListener = new InternalRoleListener();
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700175 private final InternalDeviceListener deviceListener = new InternalDeviceListener();
176 private final InternalPacketProcessor packetProcessor = new InternalPacketProcessor();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800177
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700178 // Device link discovery helpers.
alshabib7911a052014-10-16 17:49:37 -0700179 protected final Map<DeviceId, LinkDiscovery> discoverers = new ConcurrentHashMap<>();
180
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700181 // Most recent time a tracked link was seen; links are tracked if their
182 // destination connection point is mastered by this controller instance.
183 private final Map<LinkKey, Long> linkTimes = Maps.newConcurrentMap();
184
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800185 private ApplicationId appId;
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800186
Naoki Shiota399a0b32015-11-15 20:36:13 -0600187 static final SuppressionRules DEFAULT_RULES
Yuta HIGUCHI09697d02017-03-03 16:53:39 -0800188 = new SuppressionRules(EnumSet.of(Device.Type.ROADM,
189 Device.Type.FIBER_SWITCH,
190 Device.Type.OPTICAL_AMPLIFIER,
191 Device.Type.OTN),
Naoki Shiota399a0b32015-11-15 20:36:13 -0600192 ImmutableMap.of(NO_LLDP, SuppressionRules.ANY_VALUE));
193
194 private SuppressionRules rules = LldpLinkProvider.DEFAULT_RULES;
195
196 public static final String CONFIG_KEY = "suppression";
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800197 public static final String FEATURE_NAME = "linkDiscovery";
Naoki Shiota399a0b32015-11-15 20:36:13 -0600198
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800199 private final Set<ConfigFactory<?, ?>> factories = ImmutableSet.of(
Naoki Shiota399a0b32015-11-15 20:36:13 -0600200 new ConfigFactory<ApplicationId, SuppressionConfig>(APP_SUBJECT_FACTORY,
201 SuppressionConfig.class,
202 CONFIG_KEY) {
203 @Override
204 public SuppressionConfig createConfig() {
205 return new SuppressionConfig();
206 }
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800207 },
208 new ConfigFactory<DeviceId, LinkDiscoveryFromDevice>(DEVICE_SUBJECT_FACTORY,
209 LinkDiscoveryFromDevice.class, FEATURE_NAME) {
210 @Override
211 public LinkDiscoveryFromDevice createConfig() {
212 return new LinkDiscoveryFromDevice();
213 }
214 },
215 new ConfigFactory<ConnectPoint, LinkDiscoveryFromPort>(CONNECT_POINT_SUBJECT_FACTORY,
216 LinkDiscoveryFromPort.class, FEATURE_NAME) {
217 @Override
218 public LinkDiscoveryFromPort createConfig() {
219 return new LinkDiscoveryFromPort();
220 }
Naoki Shiota399a0b32015-11-15 20:36:13 -0600221 }
222 );
223
224 private final InternalConfigListener cfgListener = new InternalConfigListener();
225
alshabib7911a052014-10-16 17:49:37 -0700226 /**
227 * Creates an OpenFlow link provider.
228 */
Jonathan Hartb35540a2015-11-17 09:30:56 -0800229 public LldpLinkProvider() {
Thomas Vachuskafc52fec2015-05-18 19:13:56 -0700230 super(new ProviderId("lldp", PROVIDER_NAME));
alshabib7911a052014-10-16 17:49:37 -0700231 }
232
Sho SHIMIZU9efeb812016-08-18 09:29:20 -0700233 private String buildSrcMac() {
Ayaka Koshibe48229222016-05-16 18:04:26 -0700234 String defMac = ProbedLinkProvider.defaultMac();
Ray Milkey55f513f2017-12-01 15:58:21 -0800235 if (clusterMetadataService == null) {
236 log.debug("No cluster metadata service is available. Using default value {}", defMac);
237 return defMac;
238 }
239
240 String srcMac = ProbedLinkProvider.fingerprintMac(clusterMetadataService.getClusterMetadata());
Ayaka Koshibe48229222016-05-16 18:04:26 -0700241 if (srcMac.equals(defMac)) {
242 log.warn("Couldn't generate fingerprint. Using default value {}", defMac);
243 return defMac;
244 }
245 log.trace("Generated MAC address {}", srcMac);
246 return srcMac;
247 }
248
alshabib7911a052014-10-16 17:49:37 -0700249 @Activate
Saurav Dasc313c402015-02-27 10:09:47 -0800250 public void activate(ComponentContext context) {
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700251 eventExecutor = newSingleThreadScheduledExecutor(groupedThreads("onos/linkevents", "events-%d", log));
Ray Milkeye80e18f2016-06-02 16:44:14 -0700252 shuttingDown = false;
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700253 cfgService.registerProperties(getClass());
Thomas Vachuskafc52fec2015-05-18 19:13:56 -0700254 appId = coreService.registerApplication(PROVIDER_NAME);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600255
256 cfgRegistry.addListener(cfgListener);
257 factories.forEach(cfgRegistry::registerConfigFactory);
258
Madan Jampanic6371882016-06-03 21:30:17 -0700259 SuppressionConfig cfg = cfgRegistry.getConfig(appId, SuppressionConfig.class);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600260 if (cfg == null) {
261 // If no configuration is found, register default.
Madan Jampanic6371882016-06-03 21:30:17 -0700262 cfg = this.setDefaultSuppressionConfig();
Naoki Shiota399a0b32015-11-15 20:36:13 -0600263 }
264 cfgListener.reconfigureSuppressionRules(cfg);
265
Saurav Dasc313c402015-02-27 10:09:47 -0800266 modified(context);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700267 log.info("Started");
268 }
269
Thomas Vachuskaaad8b1d2015-12-11 10:36:53 -0800270 private SuppressionConfig setDefaultSuppressionConfig() {
271 SuppressionConfig cfg = cfgRegistry.addConfig(appId, SuppressionConfig.class);
272 cfg.deviceTypes(DEFAULT_RULES.getSuppressedDeviceType())
273 .annotation(DEFAULT_RULES.getSuppressedAnnotation())
274 .apply();
275 return cfg;
276 }
277
Thomas Vachuska05453c92015-09-09 14:40:49 -0700278 @Deactivate
279 public void deactivate() {
Ray Milkeye80e18f2016-06-02 16:44:14 -0700280 shuttingDown = true;
Naoki Shiota399a0b32015-11-15 20:36:13 -0600281 cfgRegistry.removeListener(cfgListener);
282 factories.forEach(cfgRegistry::unregisterConfigFactory);
283
Thomas Vachuska05453c92015-09-09 14:40:49 -0700284 cfgService.unregisterProperties(getClass(), false);
285 disable();
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700286 eventExecutor.shutdownNow();
287 eventExecutor = null;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700288 log.info("Stopped");
289 }
290
291 @Modified
292 public void modified(ComponentContext context) {
293 Dictionary<?, ?> properties = context != null ? context.getProperties() : new Properties();
294
295 boolean newEnabled, newUseBddp;
296 int newProbeRate, newStaleLinkAge;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700297 try {
298 String s = get(properties, PROP_ENABLED);
299 newEnabled = isNullOrEmpty(s) || Boolean.parseBoolean(s.trim());
300
301 s = get(properties, PROP_USE_BDDP);
302 newUseBddp = isNullOrEmpty(s) || Boolean.parseBoolean(s.trim());
303
304 s = get(properties, PROP_PROBE_RATE);
305 newProbeRate = isNullOrEmpty(s) ? probeRate : Integer.parseInt(s.trim());
306
307 s = get(properties, PROP_STALE_LINK_AGE);
308 newStaleLinkAge = isNullOrEmpty(s) ? staleLinkAge : Integer.parseInt(s.trim());
309
Thomas Vachuska05453c92015-09-09 14:40:49 -0700310 } catch (NumberFormatException e) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700311 log.warn("Component configuration had invalid values", e);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700312 newEnabled = enabled;
Jonathan Hartb35540a2015-11-17 09:30:56 -0800313 newUseBddp = useBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700314 newProbeRate = probeRate;
315 newStaleLinkAge = staleLinkAge;
Saurav Dasc313c402015-02-27 10:09:47 -0800316 }
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800317
Thomas Vachuska05453c92015-09-09 14:40:49 -0700318 boolean wasEnabled = enabled;
319
320 enabled = newEnabled;
Jonathan Hartb35540a2015-11-17 09:30:56 -0800321 useBddp = newUseBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700322 probeRate = newProbeRate;
323 staleLinkAge = newStaleLinkAge;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700324
325 if (!wasEnabled && enabled) {
326 enable();
327 } else if (wasEnabled && !enabled) {
328 disable();
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700329 } else {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700330 if (enabled) {
331 // update all discovery helper state
332 loadDevices();
333 }
Thomas Vachuska05453c92015-09-09 14:40:49 -0700334 }
335
Naoki Shiota399a0b32015-11-15 20:36:13 -0600336 log.info(FORMAT, enabled, useBddp, probeRate, staleLinkAge);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700337 }
338
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700339 /**
340 * Enables link discovery processing.
341 */
Thomas Vachuska05453c92015-09-09 14:40:49 -0700342 private void enable() {
alshabib7911a052014-10-16 17:49:37 -0700343 providerService = providerRegistry.register(this);
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800344 masterService.addListener(roleListener);
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700345 deviceService.addListener(deviceListener);
346 packetService.addProcessor(packetProcessor, PacketProcessor.advisor(0));
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800347
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700348 loadDevices();
Thomas Vachuska05453c92015-09-09 14:40:49 -0700349
HIGUCHI Yutad9e01052016-04-14 09:31:42 -0700350 executor = newSingleThreadScheduledExecutor(groupedThreads("onos/link", "discovery-%d", log));
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700351 executor.scheduleAtFixedRate(new SyncDeviceInfoTask(),
352 DEVICE_SYNC_DELAY, DEVICE_SYNC_DELAY, SECONDS);
353 executor.scheduleAtFixedRate(new LinkPrunerTask(),
354 LINK_PRUNER_DELAY, LINK_PRUNER_DELAY, SECONDS);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700355
Thomas Vachuska05453c92015-09-09 14:40:49 -0700356 requestIntercepts();
357 }
358
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700359 /**
360 * Disables link discovery processing.
361 */
Thomas Vachuska05453c92015-09-09 14:40:49 -0700362 private void disable() {
363 withdrawIntercepts();
364
365 providerRegistry.unregister(this);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700366 masterService.removeListener(roleListener);
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700367 deviceService.removeListener(deviceListener);
368 packetService.removeProcessor(packetProcessor);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700369
370 if (executor != null) {
371 executor.shutdownNow();
372 }
373 discoverers.values().forEach(LinkDiscovery::stop);
374 discoverers.clear();
Jon Hall125c3f22017-08-09 13:46:28 -0700375 linkTimes.clear();
Thomas Vachuska05453c92015-09-09 14:40:49 -0700376
377 providerService = null;
378 }
379
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700380 /**
381 * Loads available devices and registers their ports to be probed.
382 */
383 private void loadDevices() {
Ray Milkey0f87d482016-07-06 11:49:19 -0700384 if (!enabled || deviceService == null) {
Naoki Shiota399a0b32015-11-15 20:36:13 -0600385 return;
386 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700387 deviceService.getAvailableDevices()
388 .forEach(d -> updateDevice(d)
389 .ifPresent(ld -> updatePorts(ld, d.id())));
390 }
391
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800392 private boolean isBlacklisted(DeviceId did) {
393 LinkDiscoveryFromDevice cfg = cfgRegistry.getConfig(did, LinkDiscoveryFromDevice.class);
394 if (cfg == null) {
395 return false;
396 }
397 return !cfg.enabled();
398 }
399
400 private boolean isBlacklisted(ConnectPoint cp) {
401 // if parent device is blacklisted, so is the port
402 if (isBlacklisted(cp.deviceId())) {
403 return true;
404 }
405 LinkDiscoveryFromPort cfg = cfgRegistry.getConfig(cp, LinkDiscoveryFromPort.class);
406 if (cfg == null) {
407 return false;
408 }
409 return !cfg.enabled();
410 }
411
412 private boolean isBlacklisted(Port port) {
413 return isBlacklisted(new ConnectPoint(port.element().id(), port.number()));
414 }
415
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700416 /**
417 * Updates discovery helper for specified device.
418 *
419 * Adds and starts a discovery helper for specified device if enabled,
420 * calls {@link #removeDevice(DeviceId)} otherwise.
421 *
422 * @param device device to add
423 * @return discovery helper if discovery is enabled for the device
424 */
425 private Optional<LinkDiscovery> updateDevice(Device device) {
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800426 if (device == null) {
427 return Optional.empty();
428 }
Jon Hall125c3f22017-08-09 13:46:28 -0700429 if (!masterService.isLocalMaster(device.id())) {
430 // Reset the last seen time for all links to this device
431 // then stop discovery for this device
432 List<LinkKey> updateLinks = new LinkedList<>();
433 linkTimes.forEach((link, time) -> {
434 if (link.dst().deviceId().equals(device.id())) {
435 updateLinks.add(link);
436 }
437 });
438 updateLinks.forEach(link -> linkTimes.remove(link));
439 removeDevice(device.id());
440 return Optional.empty();
441 }
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800442 if (rules.isSuppressed(device) || isBlacklisted(device.id())) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700443 log.trace("LinkDiscovery from {} disabled by configuration", device.id());
444 removeDevice(device.id());
445 return Optional.empty();
446 }
Ayaka Koshibe3ddb7b22015-12-10 17:32:59 -0800447
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700448 LinkDiscovery ld = discoverers.computeIfAbsent(device.id(),
449 did -> new LinkDiscovery(device, context));
450 if (ld.isStopped()) {
451 ld.start();
452 }
453 return Optional.of(ld);
454 }
455
456 /**
457 * Removes after stopping discovery helper for specified device.
458 * @param deviceId device to remove
459 */
460 private void removeDevice(final DeviceId deviceId) {
461 discoverers.computeIfPresent(deviceId, (did, ld) -> {
462 ld.stop();
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700463 return null;
464 });
465
466 }
467
468 /**
469 * Updates ports of the specified device to the specified discovery helper.
470 */
471 private void updatePorts(LinkDiscovery discoverer, DeviceId deviceId) {
472 deviceService.getPorts(deviceId).forEach(p -> updatePort(discoverer, p));
473 }
474
475 /**
476 * Updates discovery helper state of the specified port.
477 *
478 * Adds a port to the discovery helper if up and discovery is enabled,
479 * or calls {@link #removePort(Port)} otherwise.
480 */
481 private void updatePort(LinkDiscovery discoverer, Port port) {
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800482 if (port == null) {
483 return;
484 }
HIGUCHI Yutab853b3f2015-11-17 18:43:20 -0800485 if (port.number().isLogical()) {
486 // silently ignore logical ports
487 return;
488 }
Naoki Shiota399a0b32015-11-15 20:36:13 -0600489
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800490 if (rules.isSuppressed(port) || isBlacklisted(port)) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700491 log.trace("LinkDiscovery from {} disabled by configuration", port);
492 removePort(port);
493 return;
494 }
495
496 // check if enabled and turn off discovery?
497 if (!port.isEnabled()) {
498 removePort(port);
499 return;
500 }
501
HIGUCHI Yutab853b3f2015-11-17 18:43:20 -0800502 discoverer.addPort(port);
alshabib7911a052014-10-16 17:49:37 -0700503 }
504
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700505 /**
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700506 * Removes a port from the specified discovery helper.
507 * @param port the port
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700508 */
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700509 private void removePort(Port port) {
510 if (port.element() instanceof Device) {
511 Device d = (Device) port.element();
512 LinkDiscovery ld = discoverers.get(d.id());
513 if (ld != null) {
514 ld.removePort(port.number());
Jonathan Hart45066bc2015-07-28 11:18:34 -0700515 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700516 } else {
517 log.warn("Attempted to remove non-Device port", port);
Jonathan Hart45066bc2015-07-28 11:18:34 -0700518 }
519 }
520
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700521 /**
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700522 * Requests packet intercepts.
Charles M.C. Chane148de82015-05-06 12:38:21 +0800523 */
Thomas Vachuska27bee092015-06-23 19:03:10 -0700524 private void requestIntercepts() {
525 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
Thomas Vachuska347cc872015-09-23 10:25:29 -0700526 selector.matchEthType(TYPE_LLDP);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700527 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
Jonathan Hart3cfce8e2015-01-14 16:43:27 -0800528
Thomas Vachuska347cc872015-09-23 10:25:29 -0700529 selector.matchEthType(TYPE_BSN);
Jonathan Hartb35540a2015-11-17 09:30:56 -0800530 if (useBddp) {
Thomas Vachuska27bee092015-06-23 19:03:10 -0700531 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
532 } else {
533 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800534 }
535 }
536
Thomas Vachuska27bee092015-06-23 19:03:10 -0700537 /**
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700538 * Withdraws packet intercepts.
Thomas Vachuska27bee092015-06-23 19:03:10 -0700539 */
540 private void withdrawIntercepts() {
541 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
Thomas Vachuska347cc872015-09-23 10:25:29 -0700542 selector.matchEthType(TYPE_LLDP);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700543 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Thomas Vachuska347cc872015-09-23 10:25:29 -0700544 selector.matchEthType(TYPE_BSN);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700545 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
546 }
547
Naoki Shiota399a0b32015-11-15 20:36:13 -0600548 protected SuppressionRules rules() {
549 return rules;
550 }
551
552 protected void updateRules(SuppressionRules newRules) {
553 if (!rules.equals(newRules)) {
554 rules = newRules;
555 loadDevices();
556 }
557 }
558
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700559 /**
560 * Processes device mastership role changes.
561 */
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800562 private class InternalRoleListener implements MastershipListener {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800563 @Override
564 public void event(MastershipEvent event) {
Jordan Halterman9d982a52017-12-11 15:27:37 -0800565 if (event.type() == MastershipEvent.Type.MASTER_CHANGED) {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800566 // only need new master events
Jon Hall7a8bfc62016-05-26 17:59:04 -0700567 eventExecutor.execute(() -> {
568 DeviceId deviceId = event.subject();
569 Device device = deviceService.getDevice(deviceId);
570 if (device == null) {
571 log.debug("Device {} doesn't exist, or isn't there yet", deviceId);
572 return;
573 }
Jordan Halterman9d982a52017-12-11 15:27:37 -0800574 updateDevice(device).ifPresent(ld -> updatePorts(ld, device.id()));
Jon Hall7a8bfc62016-05-26 17:59:04 -0700575 });
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800576 }
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800577 }
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800578 }
alshabib7911a052014-10-16 17:49:37 -0700579
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700580 private class DeviceEventProcessor implements Runnable {
581
582 DeviceEvent event;
583
584 DeviceEventProcessor(DeviceEvent event) {
585 this.event = event;
586 }
587
alshabib7911a052014-10-16 17:49:37 -0700588 @Override
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700589 public void run() {
alshabib7911a052014-10-16 17:49:37 -0700590 Device device = event.subject();
alshabibacd91832014-10-17 14:38:41 -0700591 Port port = event.port();
alshabibdfc7afb2014-10-21 20:13:27 -0700592 if (device == null) {
593 log.error("Device is null.");
594 return;
595 }
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700596 log.trace("{} {} {}", event.type(), event.subject(), event);
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700597 final DeviceId deviceId = device.id();
alshabib7911a052014-10-16 17:49:37 -0700598 switch (event.type()) {
599 case DEVICE_ADDED:
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700600 case DEVICE_UPDATED:
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700601 updateDevice(device).ifPresent(ld -> updatePorts(ld, deviceId));
alshabib7911a052014-10-16 17:49:37 -0700602 break;
603 case PORT_ADDED:
604 case PORT_UPDATED:
Yuta HIGUCHIf6725882014-10-29 15:25:51 -0700605 if (port.isEnabled()) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700606 updateDevice(device).ifPresent(ld -> updatePort(ld, port));
alshabib7911a052014-10-16 17:49:37 -0700607 } else {
Yuta HIGUCHIf6725882014-10-29 15:25:51 -0700608 log.debug("Port down {}", port);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700609 removePort(port);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600610 providerService.linksVanished(new ConnectPoint(port.element().id(),
611 port.number()));
alshabib7911a052014-10-16 17:49:37 -0700612 }
613 break;
614 case PORT_REMOVED:
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700615 log.debug("Port removed {}", port);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700616 removePort(port);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600617 providerService.linksVanished(new ConnectPoint(port.element().id(),
618 port.number()));
alshabib7911a052014-10-16 17:49:37 -0700619 break;
620 case DEVICE_REMOVED:
621 case DEVICE_SUSPENDED:
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700622 log.debug("Device removed {}", deviceId);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700623 removeDevice(deviceId);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600624 providerService.linksVanished(deviceId);
alshabib7911a052014-10-16 17:49:37 -0700625 break;
626 case DEVICE_AVAILABILITY_CHANGED:
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700627 if (deviceService.isAvailable(deviceId)) {
628 log.debug("Device up {}", deviceId);
alshabibe3af2652015-12-01 23:05:34 -0800629 updateDevice(device).ifPresent(ld -> updatePorts(ld, deviceId));
alshabib7911a052014-10-16 17:49:37 -0700630 } else {
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700631 log.debug("Device down {}", deviceId);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700632 removeDevice(deviceId);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600633 providerService.linksVanished(deviceId);
alshabib7911a052014-10-16 17:49:37 -0700634 }
635 break;
Jonathan Hart9de692c2015-04-23 11:45:47 -0700636 case PORT_STATS_UPDATED:
637 break;
alshabib7911a052014-10-16 17:49:37 -0700638 default:
639 log.debug("Unknown event {}", event);
640 }
641 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700642 }
alshabib7911a052014-10-16 17:49:37 -0700643
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700644 /**
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700645 * Processes device events.
646 */
647 private class InternalDeviceListener implements DeviceListener {
648 @Override
649 public void event(DeviceEvent event) {
650 if (event.type() == Type.PORT_STATS_UPDATED) {
651 return;
652 }
653
654 Runnable deviceEventProcessor = new DeviceEventProcessor(event);
655
656 eventExecutor.execute(deviceEventProcessor);
657 }
658 }
659
660 /**
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700661 * Processes incoming packets.
662 */
663 private class InternalPacketProcessor implements PacketProcessor {
alshabib7911a052014-10-16 17:49:37 -0700664 @Override
665 public void process(PacketContext context) {
Thomas Vachuska347cc872015-09-23 10:25:29 -0700666 if (context == null || context.isHandled()) {
alshabib4a179dc2014-10-17 17:17:01 -0700667 return;
668 }
Thomas Vachuska347cc872015-09-23 10:25:29 -0700669
670 Ethernet eth = context.inPacket().parsed();
671 if (eth == null || (eth.getEtherType() != TYPE_LLDP && eth.getEtherType() != TYPE_BSN)) {
672 return;
673 }
674
Thomas Vachuska96f3ea72015-09-08 13:50:12 -0700675 LinkDiscovery ld = discoverers.get(context.inPacket().receivedFrom().deviceId());
alshabib7911a052014-10-16 17:49:37 -0700676 if (ld == null) {
677 return;
678 }
679
Jonathan Hartb35540a2015-11-17 09:30:56 -0800680 if (ld.handleLldp(context)) {
alshabib7911a052014-10-16 17:49:37 -0700681 context.block();
682 }
683 }
684 }
685
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700686 /**
687 * Auxiliary task to keep device ports up to date.
688 */
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800689 private final class SyncDeviceInfoTask implements Runnable {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800690 @Override
691 public void run() {
692 if (Thread.currentThread().isInterrupted()) {
693 log.info("Interrupted, quitting");
694 return;
695 }
696 // check what deviceService sees, to see if we are missing anything
697 try {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700698 loadDevices();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800699 } catch (Exception e) {
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700700 // Catch all exceptions to avoid task being suppressed
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800701 log.error("Exception thrown during synchronization process", e);
702 }
703 }
704 }
705
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700706 /**
707 * Auxiliary task for pruning stale links.
708 */
709 private class LinkPrunerTask implements Runnable {
710 @Override
711 public void run() {
712 if (Thread.currentThread().isInterrupted()) {
713 log.info("Interrupted, quitting");
714 return;
715 }
716
717 try {
718 // TODO: There is still a slight possibility of mastership
719 // change occurring right with link going stale. This will
720 // result in the stale link not being pruned.
721 Maps.filterEntries(linkTimes, e -> {
722 if (!masterService.isLocalMaster(e.getKey().dst().deviceId())) {
723 return true;
724 }
725 if (isStale(e.getValue())) {
726 providerService.linkVanished(new DefaultLinkDescription(e.getKey().src(),
727 e.getKey().dst(),
728 DIRECT));
729 return true;
730 }
731 return false;
732 }).clear();
733
734 } catch (Exception e) {
735 // Catch all exceptions to avoid task being suppressed
Ray Milkeye80e18f2016-06-02 16:44:14 -0700736 if (!shuttingDown) {
737 // Error condition
738 log.error("Exception thrown during link pruning process", e);
739 } else {
740 // Provider is shutting down, the error can be ignored
741 log.trace("Shutting down, ignoring error", e);
742 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700743 }
744 }
745
746 private boolean isStale(long lastSeen) {
747 return lastSeen < System.currentTimeMillis() - staleLinkAge;
748 }
749 }
750
751 /**
752 * Provides processing context for the device link discovery helpers.
753 */
Ray Milkey957390e2016-02-09 10:02:46 -0800754 private class InternalDiscoveryContext implements LinkDiscoveryContext {
Thomas Vachuska05453c92015-09-09 14:40:49 -0700755 @Override
756 public MastershipService mastershipService() {
757 return masterService;
758 }
759
760 @Override
761 public LinkProviderService providerService() {
762 return providerService;
763 }
764
765 @Override
766 public PacketService packetService() {
767 return packetService;
768 }
769
770 @Override
771 public long probeRate() {
772 return probeRate;
773 }
774
775 @Override
Jonathan Hartb35540a2015-11-17 09:30:56 -0800776 public boolean useBddp() {
777 return useBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700778 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700779
780 @Override
781 public void touchLink(LinkKey key) {
782 linkTimes.put(key, System.currentTimeMillis());
783 }
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800784
785 @Override
Ayaka Koshibe48229222016-05-16 18:04:26 -0700786 public DeviceService deviceService() {
787 return deviceService;
Ayaka Koshibe3ddb7b22015-12-10 17:32:59 -0800788 }
789
790 @Override
Ayaka Koshibe48229222016-05-16 18:04:26 -0700791 public String fingerprint() {
792 return buildSrcMac();
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800793 }
Thomas Vachuska05453c92015-09-09 14:40:49 -0700794 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700795
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800796 static final EnumSet<NetworkConfigEvent.Type> CONFIG_CHANGED
797 = EnumSet.of(NetworkConfigEvent.Type.CONFIG_ADDED,
798 NetworkConfigEvent.Type.CONFIG_UPDATED,
799 NetworkConfigEvent.Type.CONFIG_REMOVED);
800
Naoki Shiota399a0b32015-11-15 20:36:13 -0600801 private class InternalConfigListener implements NetworkConfigListener {
802
803 private synchronized void reconfigureSuppressionRules(SuppressionConfig cfg) {
804 if (cfg == null) {
Ray Milkey0a8ee912016-06-13 09:58:12 -0700805 log.debug("Suppression Config is null.");
Naoki Shiota399a0b32015-11-15 20:36:13 -0600806 return;
807 }
808
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800809 SuppressionRules newRules = new SuppressionRules(cfg.deviceTypes(),
Naoki Shiota399a0b32015-11-15 20:36:13 -0600810 cfg.annotation());
811
812 updateRules(newRules);
813 }
814
Jon Hall125c3f22017-08-09 13:46:28 -0700815 private boolean isRelevantDeviceEvent(NetworkConfigEvent event) {
816 return event.configClass() == LinkDiscoveryFromDevice.class &&
817 CONFIG_CHANGED.contains(event.type());
818 }
819
820 private boolean isRelevantPortEvent(NetworkConfigEvent event) {
821 return event.configClass() == LinkDiscoveryFromPort.class &&
822 CONFIG_CHANGED.contains(event.type());
823 }
824
825 private boolean isRelevantSuppressionEvent(NetworkConfigEvent event) {
826 return (event.configClass().equals(SuppressionConfig.class) &&
827 (event.type() == NetworkConfigEvent.Type.CONFIG_ADDED ||
828 event.type() == NetworkConfigEvent.Type.CONFIG_UPDATED));
829 }
830
Naoki Shiota399a0b32015-11-15 20:36:13 -0600831 @Override
832 public void event(NetworkConfigEvent event) {
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700833 eventExecutor.execute(() -> {
Jon Hall125c3f22017-08-09 13:46:28 -0700834 if (isRelevantDeviceEvent(event)) {
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800835 if (event.subject() instanceof DeviceId) {
836 final DeviceId did = (DeviceId) event.subject();
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800837 Device device = deviceService.getDevice(did);
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800838 updateDevice(device).ifPresent(ld -> updatePorts(ld, did));
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800839 }
Jon Hall125c3f22017-08-09 13:46:28 -0700840 } else if (isRelevantPortEvent(event)) {
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800841 if (event.subject() instanceof ConnectPoint) {
842 ConnectPoint cp = (ConnectPoint) event.subject();
843 if (cp.elementId() instanceof DeviceId) {
844 final DeviceId did = (DeviceId) cp.elementId();
845 Device device = deviceService.getDevice(did);
846 Port port = deviceService.getPort(did, cp.port());
847 updateDevice(device).ifPresent(ld -> updatePort(ld, port));
848 }
849 }
Jon Hall125c3f22017-08-09 13:46:28 -0700850 } else if (isRelevantSuppressionEvent(event)) {
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800851 SuppressionConfig cfg = cfgRegistry.getConfig(appId, SuppressionConfig.class);
852 reconfigureSuppressionRules(cfg);
853 log.trace("Network config reconfigured");
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800854 }
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800855 });
Naoki Shiota399a0b32015-11-15 20:36:13 -0600856 }
857 }
alshabib7911a052014-10-16 17:49:37 -0700858}