blob: eddf4b485fe73b39a4c132e2ff1dce65a4f6b143 [file] [log] [blame]
Thomas Vachuska781d18b2014-10-27 10:31:25 -07001/*
Brian O'Connor5ab426f2016-04-09 01:19:45 -07002 * Copyright 2015-present Open Networking Laboratory
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 Milkeyd9bbde82016-06-09 11:35:00 -070018import java.util.Dictionary;
19import java.util.EnumSet;
20import java.util.Map;
21import java.util.Optional;
22import java.util.Properties;
23import java.util.Set;
24import java.util.concurrent.ConcurrentHashMap;
25import java.util.concurrent.ExecutorService;
26import java.util.concurrent.ScheduledExecutorService;
27
alshabib7911a052014-10-16 17:49:37 -070028import org.apache.felix.scr.annotations.Activate;
29import org.apache.felix.scr.annotations.Component;
30import org.apache.felix.scr.annotations.Deactivate;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080031import org.apache.felix.scr.annotations.Modified;
32import org.apache.felix.scr.annotations.Property;
alshabib7911a052014-10-16 17:49:37 -070033import org.apache.felix.scr.annotations.Reference;
34import org.apache.felix.scr.annotations.ReferenceCardinality;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080035import org.onlab.packet.Ethernet;
Thomas Vachuska6519e6f2015-03-11 02:29:31 -070036import org.onosproject.cfg.ComponentConfigService;
Ayaka Koshibe12c8c082015-12-08 12:48:46 -080037import org.onosproject.cluster.ClusterMetadataService;
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -070038import org.onosproject.cluster.ClusterService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080039import org.onosproject.core.ApplicationId;
40import org.onosproject.core.CoreService;
Brian O'Connorabafb502014-12-02 22:26:20 -080041import org.onosproject.mastership.MastershipEvent;
42import org.onosproject.mastership.MastershipListener;
43import org.onosproject.mastership.MastershipService;
44import org.onosproject.net.ConnectPoint;
45import org.onosproject.net.Device;
46import org.onosproject.net.DeviceId;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070047import org.onosproject.net.LinkKey;
Brian O'Connorabafb502014-12-02 22:26:20 -080048import org.onosproject.net.Port;
Naoki Shiota399a0b32015-11-15 20:36:13 -060049import org.onosproject.net.config.ConfigFactory;
50import org.onosproject.net.config.NetworkConfigEvent;
51import org.onosproject.net.config.NetworkConfigListener;
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -070052import org.onosproject.net.config.NetworkConfigRegistry;
Brian O'Connorabafb502014-12-02 22:26:20 -080053import org.onosproject.net.device.DeviceEvent;
HIGUCHI Yuta1979f552015-12-28 21:24:26 -080054import org.onosproject.net.device.DeviceEvent.Type;
Brian O'Connorabafb502014-12-02 22:26:20 -080055import org.onosproject.net.device.DeviceListener;
56import org.onosproject.net.device.DeviceService;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080057import org.onosproject.net.flow.DefaultTrafficSelector;
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -080058import org.onosproject.net.flow.TrafficSelector;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070059import org.onosproject.net.link.DefaultLinkDescription;
Brian O'Connorabafb502014-12-02 22:26:20 -080060import org.onosproject.net.link.LinkProviderRegistry;
61import org.onosproject.net.link.LinkProviderService;
Thomas Vachuskae4ebac92015-09-10 11:39:05 -070062import org.onosproject.net.link.LinkService;
Ray Milkeyd9bbde82016-06-09 11:35:00 -070063import org.onosproject.net.link.ProbedLinkProvider;
Brian O'Connorabafb502014-12-02 22:26:20 -080064import org.onosproject.net.packet.PacketContext;
Jonathan Hart3cfce8e2015-01-14 16:43:27 -080065import org.onosproject.net.packet.PacketPriority;
Brian O'Connorabafb502014-12-02 22:26:20 -080066import org.onosproject.net.packet.PacketProcessor;
67import org.onosproject.net.packet.PacketService;
68import org.onosproject.net.provider.AbstractProvider;
69import org.onosproject.net.provider.ProviderId;
Ray Milkey957390e2016-02-09 10:02:46 -080070import org.onosproject.provider.lldpcommon.LinkDiscovery;
Ray Milkeyd9bbde82016-06-09 11:35:00 -070071import org.onosproject.provider.lldpcommon.LinkDiscoveryContext;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080072import org.osgi.service.component.ComponentContext;
alshabib7911a052014-10-16 17:49:37 -070073import org.slf4j.Logger;
74
Ray Milkeyd9bbde82016-06-09 11:35:00 -070075import com.google.common.collect.ImmutableMap;
76import com.google.common.collect.ImmutableSet;
77import com.google.common.collect.Maps;
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;
90import static org.slf4j.LoggerFactory.getLogger;
Yuta HIGUCHI41289382014-12-19 17:47:12 -080091
alshabib7911a052014-10-16 17:49:37 -070092/**
Thomas Vachuska05453c92015-09-09 14:40:49 -070093 * Provider which uses LLDP and BDDP packets to detect network infrastructure links.
alshabib7911a052014-10-16 17:49:37 -070094 */
95@Component(immediate = true)
Ayaka Koshibe48229222016-05-16 18:04:26 -070096public class LldpLinkProvider extends AbstractProvider implements ProbedLinkProvider {
alshabib7911a052014-10-16 17:49:37 -070097
Thomas Vachuskafc52fec2015-05-18 19:13:56 -070098 private static final String PROVIDER_NAME = "org.onosproject.provider.lldp";
99
Thomas Vachuska05453c92015-09-09 14:40:49 -0700100 private static final String FORMAT =
101 "Settings: enabled={}, useBDDP={}, probeRate={}, " +
Naoki Shiota399a0b32015-11-15 20:36:13 -0600102 "staleLinkAge={}";
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800103
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700104 // When a Device/Port has this annotation, do not send out LLDP/BDDP
105 public static final String NO_LLDP = "no-lldp";
106
Thomas Vachuskaaad8b1d2015-12-11 10:36:53 -0800107 private static final int MAX_RETRIES = 5;
108 private static final int RETRY_DELAY = 1_000; // millis
109
alshabib7911a052014-10-16 17:49:37 -0700110 private final Logger log = getLogger(getClass());
111
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800112 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
113 protected CoreService coreService;
114
115 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
alshabib7911a052014-10-16 17:49:37 -0700116 protected LinkProviderRegistry providerRegistry;
117
118 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
119 protected DeviceService deviceService;
120
121 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700122 protected LinkService linkService;
123
124 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
Marc De Leenheer8b3e80b2015-03-06 14:27:03 -0800125 protected PacketService packetService;
alshabib7911a052014-10-16 17:49:37 -0700126
alshabib875d6262014-10-17 16:19:40 -0700127 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
128 protected MastershipService masterService;
129
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700130 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
131 protected ComponentConfigService cfgService;
132
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700133 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
134 protected ClusterService clusterService;
135
136 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
137 protected NetworkConfigRegistry cfgRegistry;
138
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800139 @Reference(cardinality = ReferenceCardinality.MANDATORY_UNARY)
140 protected ClusterMetadataService clusterMetadataService;
141
alshabib7911a052014-10-16 17:49:37 -0700142 private LinkProviderService providerService;
143
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800144 private ScheduledExecutorService executor;
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700145 protected ExecutorService eventExecutor;
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800146
Ray Milkeye80e18f2016-06-02 16:44:14 -0700147 private boolean shuttingDown = false;
148
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700149 // TODO: Add sanity checking for the configurable params based on the delays
150 private static final long DEVICE_SYNC_DELAY = 5;
151 private static final long LINK_PRUNER_DELAY = 3;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700152
153 private static final String PROP_ENABLED = "enabled";
154 @Property(name = PROP_ENABLED, boolValue = true,
155 label = "If false, link discovery is disabled")
156 private boolean enabled = false;
157
158 private static final String PROP_USE_BDDP = "useBDDP";
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700159 @Property(name = PROP_USE_BDDP, boolValue = true,
160 label = "Use BDDP for link discovery")
Jonathan Hartb35540a2015-11-17 09:30:56 -0800161 private boolean useBddp = true;
alshabib7911a052014-10-16 17:49:37 -0700162
Thomas Vachuska05453c92015-09-09 14:40:49 -0700163 private static final String PROP_PROBE_RATE = "probeRate";
Thomas Vachuska9e51fd02016-01-04 16:51:28 -0800164 private static final int DEFAULT_PROBE_RATE = 3000;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700165 @Property(name = PROP_PROBE_RATE, intValue = DEFAULT_PROBE_RATE,
166 label = "LLDP and BDDP probe rate specified in millis")
167 private int probeRate = DEFAULT_PROBE_RATE;
Saurav Dasc313c402015-02-27 10:09:47 -0800168
Thomas Vachuska05453c92015-09-09 14:40:49 -0700169 private static final String PROP_STALE_LINK_AGE = "staleLinkAge";
Thomas Vachuska9e51fd02016-01-04 16:51:28 -0800170 private static final int DEFAULT_STALE_LINK_AGE = 10000;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700171 @Property(name = PROP_STALE_LINK_AGE, intValue = DEFAULT_STALE_LINK_AGE,
172 label = "Number of millis beyond which links will be considered stale")
173 private int staleLinkAge = DEFAULT_STALE_LINK_AGE;
alshabib7911a052014-10-16 17:49:37 -0700174
Ray Milkey957390e2016-02-09 10:02:46 -0800175 private final LinkDiscoveryContext context = new InternalDiscoveryContext();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800176 private final InternalRoleListener roleListener = new InternalRoleListener();
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700177 private final InternalDeviceListener deviceListener = new InternalDeviceListener();
178 private final InternalPacketProcessor packetProcessor = new InternalPacketProcessor();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800179
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700180 // Device link discovery helpers.
alshabib7911a052014-10-16 17:49:37 -0700181 protected final Map<DeviceId, LinkDiscovery> discoverers = new ConcurrentHashMap<>();
182
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700183 // Most recent time a tracked link was seen; links are tracked if their
184 // destination connection point is mastered by this controller instance.
185 private final Map<LinkKey, Long> linkTimes = Maps.newConcurrentMap();
186
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800187 private ApplicationId appId;
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800188
Naoki Shiota399a0b32015-11-15 20:36:13 -0600189 static final SuppressionRules DEFAULT_RULES
Rimon Ashkenazy8ebfff02016-02-01 11:56:36 +0200190 = new SuppressionRules(EnumSet.of(Device.Type.ROADM, Device.Type.FIBER_SWITCH, Device.Type.OTN),
Naoki Shiota399a0b32015-11-15 20:36:13 -0600191 ImmutableMap.of(NO_LLDP, SuppressionRules.ANY_VALUE));
192
193 private SuppressionRules rules = LldpLinkProvider.DEFAULT_RULES;
194
195 public static final String CONFIG_KEY = "suppression";
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800196 public static final String FEATURE_NAME = "linkDiscovery";
Naoki Shiota399a0b32015-11-15 20:36:13 -0600197
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800198 private final Set<ConfigFactory<?, ?>> factories = ImmutableSet.of(
Naoki Shiota399a0b32015-11-15 20:36:13 -0600199 new ConfigFactory<ApplicationId, SuppressionConfig>(APP_SUBJECT_FACTORY,
200 SuppressionConfig.class,
201 CONFIG_KEY) {
202 @Override
203 public SuppressionConfig createConfig() {
204 return new SuppressionConfig();
205 }
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800206 },
207 new ConfigFactory<DeviceId, LinkDiscoveryFromDevice>(DEVICE_SUBJECT_FACTORY,
208 LinkDiscoveryFromDevice.class, FEATURE_NAME) {
209 @Override
210 public LinkDiscoveryFromDevice createConfig() {
211 return new LinkDiscoveryFromDevice();
212 }
213 },
214 new ConfigFactory<ConnectPoint, LinkDiscoveryFromPort>(CONNECT_POINT_SUBJECT_FACTORY,
215 LinkDiscoveryFromPort.class, FEATURE_NAME) {
216 @Override
217 public LinkDiscoveryFromPort createConfig() {
218 return new LinkDiscoveryFromPort();
219 }
Naoki Shiota399a0b32015-11-15 20:36:13 -0600220 }
221 );
222
223 private final InternalConfigListener cfgListener = new InternalConfigListener();
224
alshabib7911a052014-10-16 17:49:37 -0700225 /**
226 * Creates an OpenFlow link provider.
227 */
Jonathan Hartb35540a2015-11-17 09:30:56 -0800228 public LldpLinkProvider() {
Thomas Vachuskafc52fec2015-05-18 19:13:56 -0700229 super(new ProviderId("lldp", PROVIDER_NAME));
alshabib7911a052014-10-16 17:49:37 -0700230 }
231
Sho SHIMIZU9efeb812016-08-18 09:29:20 -0700232 private String buildSrcMac() {
Ayaka Koshibe48229222016-05-16 18:04:26 -0700233 String srcMac = ProbedLinkProvider.fingerprintMac(clusterMetadataService.getClusterMetadata());
234 String defMac = ProbedLinkProvider.defaultMac();
235 if (srcMac.equals(defMac)) {
236 log.warn("Couldn't generate fingerprint. Using default value {}", defMac);
237 return defMac;
238 }
239 log.trace("Generated MAC address {}", srcMac);
240 return srcMac;
241 }
242
alshabib7911a052014-10-16 17:49:37 -0700243 @Activate
Saurav Dasc313c402015-02-27 10:09:47 -0800244 public void activate(ComponentContext context) {
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700245 eventExecutor = newSingleThreadScheduledExecutor(groupedThreads("onos/linkevents", "events-%d", log));
Ray Milkeye80e18f2016-06-02 16:44:14 -0700246 shuttingDown = false;
Thomas Vachuska6519e6f2015-03-11 02:29:31 -0700247 cfgService.registerProperties(getClass());
Thomas Vachuskafc52fec2015-05-18 19:13:56 -0700248 appId = coreService.registerApplication(PROVIDER_NAME);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600249
250 cfgRegistry.addListener(cfgListener);
251 factories.forEach(cfgRegistry::registerConfigFactory);
252
Madan Jampanic6371882016-06-03 21:30:17 -0700253 SuppressionConfig cfg = cfgRegistry.getConfig(appId, SuppressionConfig.class);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600254 if (cfg == null) {
255 // If no configuration is found, register default.
Madan Jampanic6371882016-06-03 21:30:17 -0700256 cfg = this.setDefaultSuppressionConfig();
Naoki Shiota399a0b32015-11-15 20:36:13 -0600257 }
258 cfgListener.reconfigureSuppressionRules(cfg);
259
Saurav Dasc313c402015-02-27 10:09:47 -0800260 modified(context);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700261 log.info("Started");
262 }
263
Thomas Vachuskaaad8b1d2015-12-11 10:36:53 -0800264 private SuppressionConfig setDefaultSuppressionConfig() {
265 SuppressionConfig cfg = cfgRegistry.addConfig(appId, SuppressionConfig.class);
266 cfg.deviceTypes(DEFAULT_RULES.getSuppressedDeviceType())
267 .annotation(DEFAULT_RULES.getSuppressedAnnotation())
268 .apply();
269 return cfg;
270 }
271
Thomas Vachuska05453c92015-09-09 14:40:49 -0700272 @Deactivate
273 public void deactivate() {
Ray Milkeye80e18f2016-06-02 16:44:14 -0700274 shuttingDown = true;
Naoki Shiota399a0b32015-11-15 20:36:13 -0600275 cfgRegistry.removeListener(cfgListener);
276 factories.forEach(cfgRegistry::unregisterConfigFactory);
277
Thomas Vachuska05453c92015-09-09 14:40:49 -0700278 cfgService.unregisterProperties(getClass(), false);
279 disable();
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700280 eventExecutor.shutdownNow();
281 eventExecutor = null;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700282 log.info("Stopped");
283 }
284
285 @Modified
286 public void modified(ComponentContext context) {
287 Dictionary<?, ?> properties = context != null ? context.getProperties() : new Properties();
288
289 boolean newEnabled, newUseBddp;
290 int newProbeRate, newStaleLinkAge;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700291 try {
292 String s = get(properties, PROP_ENABLED);
293 newEnabled = isNullOrEmpty(s) || Boolean.parseBoolean(s.trim());
294
295 s = get(properties, PROP_USE_BDDP);
296 newUseBddp = isNullOrEmpty(s) || Boolean.parseBoolean(s.trim());
297
298 s = get(properties, PROP_PROBE_RATE);
299 newProbeRate = isNullOrEmpty(s) ? probeRate : Integer.parseInt(s.trim());
300
301 s = get(properties, PROP_STALE_LINK_AGE);
302 newStaleLinkAge = isNullOrEmpty(s) ? staleLinkAge : Integer.parseInt(s.trim());
303
Thomas Vachuska05453c92015-09-09 14:40:49 -0700304 } catch (NumberFormatException e) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700305 log.warn("Component configuration had invalid values", e);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700306 newEnabled = enabled;
Jonathan Hartb35540a2015-11-17 09:30:56 -0800307 newUseBddp = useBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700308 newProbeRate = probeRate;
309 newStaleLinkAge = staleLinkAge;
Saurav Dasc313c402015-02-27 10:09:47 -0800310 }
Yuta HIGUCHI41289382014-12-19 17:47:12 -0800311
Thomas Vachuska05453c92015-09-09 14:40:49 -0700312 boolean wasEnabled = enabled;
313
314 enabled = newEnabled;
Jonathan Hartb35540a2015-11-17 09:30:56 -0800315 useBddp = newUseBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700316 probeRate = newProbeRate;
317 staleLinkAge = newStaleLinkAge;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700318
319 if (!wasEnabled && enabled) {
320 enable();
321 } else if (wasEnabled && !enabled) {
322 disable();
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700323 } else {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700324 if (enabled) {
325 // update all discovery helper state
326 loadDevices();
327 }
Thomas Vachuska05453c92015-09-09 14:40:49 -0700328 }
329
Naoki Shiota399a0b32015-11-15 20:36:13 -0600330 log.info(FORMAT, enabled, useBddp, probeRate, staleLinkAge);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700331 }
332
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700333 /**
334 * Enables link discovery processing.
335 */
Thomas Vachuska05453c92015-09-09 14:40:49 -0700336 private void enable() {
alshabib7911a052014-10-16 17:49:37 -0700337 providerService = providerRegistry.register(this);
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800338 masterService.addListener(roleListener);
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700339 deviceService.addListener(deviceListener);
340 packetService.addProcessor(packetProcessor, PacketProcessor.advisor(0));
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800341
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700342 loadDevices();
Thomas Vachuska05453c92015-09-09 14:40:49 -0700343
HIGUCHI Yutad9e01052016-04-14 09:31:42 -0700344 executor = newSingleThreadScheduledExecutor(groupedThreads("onos/link", "discovery-%d", log));
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700345 executor.scheduleAtFixedRate(new SyncDeviceInfoTask(),
346 DEVICE_SYNC_DELAY, DEVICE_SYNC_DELAY, SECONDS);
347 executor.scheduleAtFixedRate(new LinkPrunerTask(),
348 LINK_PRUNER_DELAY, LINK_PRUNER_DELAY, SECONDS);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700349
Thomas Vachuska05453c92015-09-09 14:40:49 -0700350 requestIntercepts();
351 }
352
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700353 /**
354 * Disables link discovery processing.
355 */
Thomas Vachuska05453c92015-09-09 14:40:49 -0700356 private void disable() {
357 withdrawIntercepts();
358
359 providerRegistry.unregister(this);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700360 masterService.removeListener(roleListener);
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700361 deviceService.removeListener(deviceListener);
362 packetService.removeProcessor(packetProcessor);
Thomas Vachuska05453c92015-09-09 14:40:49 -0700363
364 if (executor != null) {
365 executor.shutdownNow();
366 }
367 discoverers.values().forEach(LinkDiscovery::stop);
368 discoverers.clear();
369
370 providerService = null;
371 }
372
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700373 /**
374 * Loads available devices and registers their ports to be probed.
375 */
376 private void loadDevices() {
Ray Milkey0f87d482016-07-06 11:49:19 -0700377 if (!enabled || deviceService == null) {
Naoki Shiota399a0b32015-11-15 20:36:13 -0600378 return;
379 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700380 deviceService.getAvailableDevices()
381 .forEach(d -> updateDevice(d)
382 .ifPresent(ld -> updatePorts(ld, d.id())));
383 }
384
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800385 private boolean isBlacklisted(DeviceId did) {
386 LinkDiscoveryFromDevice cfg = cfgRegistry.getConfig(did, LinkDiscoveryFromDevice.class);
387 if (cfg == null) {
388 return false;
389 }
390 return !cfg.enabled();
391 }
392
393 private boolean isBlacklisted(ConnectPoint cp) {
394 // if parent device is blacklisted, so is the port
395 if (isBlacklisted(cp.deviceId())) {
396 return true;
397 }
398 LinkDiscoveryFromPort cfg = cfgRegistry.getConfig(cp, LinkDiscoveryFromPort.class);
399 if (cfg == null) {
400 return false;
401 }
402 return !cfg.enabled();
403 }
404
405 private boolean isBlacklisted(Port port) {
406 return isBlacklisted(new ConnectPoint(port.element().id(), port.number()));
407 }
408
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700409 /**
410 * Updates discovery helper for specified device.
411 *
412 * Adds and starts a discovery helper for specified device if enabled,
413 * calls {@link #removeDevice(DeviceId)} otherwise.
414 *
415 * @param device device to add
416 * @return discovery helper if discovery is enabled for the device
417 */
418 private Optional<LinkDiscovery> updateDevice(Device device) {
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800419 if (device == null) {
420 return Optional.empty();
421 }
422 if (rules.isSuppressed(device) || isBlacklisted(device.id())) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700423 log.trace("LinkDiscovery from {} disabled by configuration", device.id());
424 removeDevice(device.id());
425 return Optional.empty();
426 }
Ayaka Koshibe3ddb7b22015-12-10 17:32:59 -0800427
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700428 LinkDiscovery ld = discoverers.computeIfAbsent(device.id(),
429 did -> new LinkDiscovery(device, context));
430 if (ld.isStopped()) {
431 ld.start();
432 }
433 return Optional.of(ld);
434 }
435
436 /**
437 * Removes after stopping discovery helper for specified device.
438 * @param deviceId device to remove
439 */
440 private void removeDevice(final DeviceId deviceId) {
441 discoverers.computeIfPresent(deviceId, (did, ld) -> {
442 ld.stop();
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700443 return null;
444 });
445
446 }
447
448 /**
449 * Updates ports of the specified device to the specified discovery helper.
450 */
451 private void updatePorts(LinkDiscovery discoverer, DeviceId deviceId) {
452 deviceService.getPorts(deviceId).forEach(p -> updatePort(discoverer, p));
453 }
454
455 /**
456 * Updates discovery helper state of the specified port.
457 *
458 * Adds a port to the discovery helper if up and discovery is enabled,
459 * or calls {@link #removePort(Port)} otherwise.
460 */
461 private void updatePort(LinkDiscovery discoverer, Port port) {
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800462 if (port == null) {
463 return;
464 }
HIGUCHI Yutab853b3f2015-11-17 18:43:20 -0800465 if (port.number().isLogical()) {
466 // silently ignore logical ports
467 return;
468 }
Naoki Shiota399a0b32015-11-15 20:36:13 -0600469
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800470 if (rules.isSuppressed(port) || isBlacklisted(port)) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700471 log.trace("LinkDiscovery from {} disabled by configuration", port);
472 removePort(port);
473 return;
474 }
475
476 // check if enabled and turn off discovery?
477 if (!port.isEnabled()) {
478 removePort(port);
479 return;
480 }
481
HIGUCHI Yutab853b3f2015-11-17 18:43:20 -0800482 discoverer.addPort(port);
alshabib7911a052014-10-16 17:49:37 -0700483 }
484
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700485 /**
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700486 * Removes a port from the specified discovery helper.
487 * @param port the port
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700488 */
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700489 private void removePort(Port port) {
490 if (port.element() instanceof Device) {
491 Device d = (Device) port.element();
492 LinkDiscovery ld = discoverers.get(d.id());
493 if (ld != null) {
494 ld.removePort(port.number());
Jonathan Hart45066bc2015-07-28 11:18:34 -0700495 }
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700496 } else {
497 log.warn("Attempted to remove non-Device port", port);
Jonathan Hart45066bc2015-07-28 11:18:34 -0700498 }
499 }
500
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700501 /**
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700502 * Requests packet intercepts.
Charles M.C. Chane148de82015-05-06 12:38:21 +0800503 */
Thomas Vachuska27bee092015-06-23 19:03:10 -0700504 private void requestIntercepts() {
505 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
Thomas Vachuska347cc872015-09-23 10:25:29 -0700506 selector.matchEthType(TYPE_LLDP);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700507 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
Jonathan Hart3cfce8e2015-01-14 16:43:27 -0800508
Thomas Vachuska347cc872015-09-23 10:25:29 -0700509 selector.matchEthType(TYPE_BSN);
Jonathan Hartb35540a2015-11-17 09:30:56 -0800510 if (useBddp) {
Thomas Vachuska27bee092015-06-23 19:03:10 -0700511 packetService.requestPackets(selector.build(), PacketPriority.CONTROL, appId);
512 } else {
513 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Pavlin Radoslavovd36a74b2015-01-09 11:59:07 -0800514 }
515 }
516
Thomas Vachuska27bee092015-06-23 19:03:10 -0700517 /**
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700518 * Withdraws packet intercepts.
Thomas Vachuska27bee092015-06-23 19:03:10 -0700519 */
520 private void withdrawIntercepts() {
521 TrafficSelector.Builder selector = DefaultTrafficSelector.builder();
Thomas Vachuska347cc872015-09-23 10:25:29 -0700522 selector.matchEthType(TYPE_LLDP);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700523 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
Thomas Vachuska347cc872015-09-23 10:25:29 -0700524 selector.matchEthType(TYPE_BSN);
Thomas Vachuska27bee092015-06-23 19:03:10 -0700525 packetService.cancelPackets(selector.build(), PacketPriority.CONTROL, appId);
526 }
527
Naoki Shiota399a0b32015-11-15 20:36:13 -0600528 protected SuppressionRules rules() {
529 return rules;
530 }
531
532 protected void updateRules(SuppressionRules newRules) {
533 if (!rules.equals(newRules)) {
534 rules = newRules;
535 loadDevices();
536 }
537 }
538
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700539 /**
540 * Processes device mastership role changes.
541 */
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800542 private class InternalRoleListener implements MastershipListener {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800543 @Override
544 public void event(MastershipEvent event) {
Jon Hall7a8bfc62016-05-26 17:59:04 -0700545 if (MastershipEvent.Type.MASTER_CHANGED.equals(event.type())) {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800546 // only need new master events
Jon Hall7a8bfc62016-05-26 17:59:04 -0700547 eventExecutor.execute(() -> {
548 DeviceId deviceId = event.subject();
549 Device device = deviceService.getDevice(deviceId);
550 if (device == null) {
551 log.debug("Device {} doesn't exist, or isn't there yet", deviceId);
552 return;
553 }
554 if (clusterService.getLocalNode().id().equals(event.roleInfo().master())) {
555 updateDevice(device).ifPresent(ld -> updatePorts(ld, device.id()));
556 }
557 });
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800558 }
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800559 }
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800560 }
alshabib7911a052014-10-16 17:49:37 -0700561
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700562 private class DeviceEventProcessor implements Runnable {
563
564 DeviceEvent event;
565
566 DeviceEventProcessor(DeviceEvent event) {
567 this.event = event;
568 }
569
alshabib7911a052014-10-16 17:49:37 -0700570 @Override
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700571 public void run() {
alshabib7911a052014-10-16 17:49:37 -0700572 Device device = event.subject();
alshabibacd91832014-10-17 14:38:41 -0700573 Port port = event.port();
alshabibdfc7afb2014-10-21 20:13:27 -0700574 if (device == null) {
575 log.error("Device is null.");
576 return;
577 }
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700578 log.trace("{} {} {}", event.type(), event.subject(), event);
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700579 final DeviceId deviceId = device.id();
alshabib7911a052014-10-16 17:49:37 -0700580 switch (event.type()) {
581 case DEVICE_ADDED:
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700582 case DEVICE_UPDATED:
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700583 updateDevice(device).ifPresent(ld -> updatePorts(ld, deviceId));
alshabib7911a052014-10-16 17:49:37 -0700584 break;
585 case PORT_ADDED:
586 case PORT_UPDATED:
Yuta HIGUCHIf6725882014-10-29 15:25:51 -0700587 if (port.isEnabled()) {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700588 updateDevice(device).ifPresent(ld -> updatePort(ld, port));
alshabib7911a052014-10-16 17:49:37 -0700589 } else {
Yuta HIGUCHIf6725882014-10-29 15:25:51 -0700590 log.debug("Port down {}", port);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700591 removePort(port);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600592 providerService.linksVanished(new ConnectPoint(port.element().id(),
593 port.number()));
alshabib7911a052014-10-16 17:49:37 -0700594 }
595 break;
596 case PORT_REMOVED:
Yuta HIGUCHIeb24e9d2014-10-26 19:34:20 -0700597 log.debug("Port removed {}", port);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700598 removePort(port);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600599 providerService.linksVanished(new ConnectPoint(port.element().id(),
600 port.number()));
alshabib7911a052014-10-16 17:49:37 -0700601 break;
602 case DEVICE_REMOVED:
603 case DEVICE_SUSPENDED:
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700604 log.debug("Device removed {}", deviceId);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700605 removeDevice(deviceId);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600606 providerService.linksVanished(deviceId);
alshabib7911a052014-10-16 17:49:37 -0700607 break;
608 case DEVICE_AVAILABILITY_CHANGED:
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700609 if (deviceService.isAvailable(deviceId)) {
610 log.debug("Device up {}", deviceId);
alshabibe3af2652015-12-01 23:05:34 -0800611 updateDevice(device).ifPresent(ld -> updatePorts(ld, deviceId));
alshabib7911a052014-10-16 17:49:37 -0700612 } else {
Yuta HIGUCHId19f6702014-10-31 15:23:25 -0700613 log.debug("Device down {}", deviceId);
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700614 removeDevice(deviceId);
Naoki Shiota399a0b32015-11-15 20:36:13 -0600615 providerService.linksVanished(deviceId);
alshabib7911a052014-10-16 17:49:37 -0700616 }
617 break;
Jonathan Hart9de692c2015-04-23 11:45:47 -0700618 case PORT_STATS_UPDATED:
619 break;
alshabib7911a052014-10-16 17:49:37 -0700620 default:
621 log.debug("Unknown event {}", event);
622 }
623 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700624 }
alshabib7911a052014-10-16 17:49:37 -0700625
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700626 /**
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700627 * Processes device events.
628 */
629 private class InternalDeviceListener implements DeviceListener {
630 @Override
631 public void event(DeviceEvent event) {
632 if (event.type() == Type.PORT_STATS_UPDATED) {
633 return;
634 }
635
636 Runnable deviceEventProcessor = new DeviceEventProcessor(event);
637
638 eventExecutor.execute(deviceEventProcessor);
639 }
640 }
641
642 /**
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700643 * Processes incoming packets.
644 */
645 private class InternalPacketProcessor implements PacketProcessor {
alshabib7911a052014-10-16 17:49:37 -0700646 @Override
647 public void process(PacketContext context) {
Thomas Vachuska347cc872015-09-23 10:25:29 -0700648 if (context == null || context.isHandled()) {
alshabib4a179dc2014-10-17 17:17:01 -0700649 return;
650 }
Thomas Vachuska347cc872015-09-23 10:25:29 -0700651
652 Ethernet eth = context.inPacket().parsed();
653 if (eth == null || (eth.getEtherType() != TYPE_LLDP && eth.getEtherType() != TYPE_BSN)) {
654 return;
655 }
656
Thomas Vachuska96f3ea72015-09-08 13:50:12 -0700657 LinkDiscovery ld = discoverers.get(context.inPacket().receivedFrom().deviceId());
alshabib7911a052014-10-16 17:49:37 -0700658 if (ld == null) {
659 return;
660 }
661
Jonathan Hartb35540a2015-11-17 09:30:56 -0800662 if (ld.handleLldp(context)) {
alshabib7911a052014-10-16 17:49:37 -0700663 context.block();
664 }
665 }
666 }
667
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700668 /**
669 * Auxiliary task to keep device ports up to date.
670 */
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800671 private final class SyncDeviceInfoTask implements Runnable {
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800672 @Override
673 public void run() {
674 if (Thread.currentThread().isInterrupted()) {
675 log.info("Interrupted, quitting");
676 return;
677 }
678 // check what deviceService sees, to see if we are missing anything
679 try {
HIGUCHI Yuta9a9edf82015-10-21 11:23:20 -0700680 loadDevices();
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800681 } catch (Exception e) {
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700682 // Catch all exceptions to avoid task being suppressed
Ayaka Koshibeccfa94c2014-11-20 11:15:52 -0800683 log.error("Exception thrown during synchronization process", e);
684 }
685 }
686 }
687
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700688 /**
689 * Auxiliary task for pruning stale links.
690 */
691 private class LinkPrunerTask implements Runnable {
692 @Override
693 public void run() {
694 if (Thread.currentThread().isInterrupted()) {
695 log.info("Interrupted, quitting");
696 return;
697 }
698
699 try {
700 // TODO: There is still a slight possibility of mastership
701 // change occurring right with link going stale. This will
702 // result in the stale link not being pruned.
703 Maps.filterEntries(linkTimes, e -> {
704 if (!masterService.isLocalMaster(e.getKey().dst().deviceId())) {
705 return true;
706 }
707 if (isStale(e.getValue())) {
708 providerService.linkVanished(new DefaultLinkDescription(e.getKey().src(),
709 e.getKey().dst(),
710 DIRECT));
711 return true;
712 }
713 return false;
714 }).clear();
715
716 } catch (Exception e) {
717 // Catch all exceptions to avoid task being suppressed
Ray Milkeye80e18f2016-06-02 16:44:14 -0700718 if (!shuttingDown) {
719 // Error condition
720 log.error("Exception thrown during link pruning process", e);
721 } else {
722 // Provider is shutting down, the error can be ignored
723 log.trace("Shutting down, ignoring error", e);
724 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700725 }
726 }
727
728 private boolean isStale(long lastSeen) {
729 return lastSeen < System.currentTimeMillis() - staleLinkAge;
730 }
731 }
732
733 /**
734 * Provides processing context for the device link discovery helpers.
735 */
Ray Milkey957390e2016-02-09 10:02:46 -0800736 private class InternalDiscoveryContext implements LinkDiscoveryContext {
Thomas Vachuska05453c92015-09-09 14:40:49 -0700737 @Override
738 public MastershipService mastershipService() {
739 return masterService;
740 }
741
742 @Override
743 public LinkProviderService providerService() {
744 return providerService;
745 }
746
747 @Override
748 public PacketService packetService() {
749 return packetService;
750 }
751
752 @Override
753 public long probeRate() {
754 return probeRate;
755 }
756
757 @Override
Jonathan Hartb35540a2015-11-17 09:30:56 -0800758 public boolean useBddp() {
759 return useBddp;
Thomas Vachuska05453c92015-09-09 14:40:49 -0700760 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700761
762 @Override
763 public void touchLink(LinkKey key) {
764 linkTimes.put(key, System.currentTimeMillis());
765 }
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800766
767 @Override
Ayaka Koshibe48229222016-05-16 18:04:26 -0700768 public DeviceService deviceService() {
769 return deviceService;
Ayaka Koshibe3ddb7b22015-12-10 17:32:59 -0800770 }
771
772 @Override
Ayaka Koshibe48229222016-05-16 18:04:26 -0700773 public String fingerprint() {
774 return buildSrcMac();
Ayaka Koshibe12c8c082015-12-08 12:48:46 -0800775 }
Thomas Vachuska05453c92015-09-09 14:40:49 -0700776 }
Thomas Vachuskae4ebac92015-09-10 11:39:05 -0700777
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800778 static final EnumSet<NetworkConfigEvent.Type> CONFIG_CHANGED
779 = EnumSet.of(NetworkConfigEvent.Type.CONFIG_ADDED,
780 NetworkConfigEvent.Type.CONFIG_UPDATED,
781 NetworkConfigEvent.Type.CONFIG_REMOVED);
782
Naoki Shiota399a0b32015-11-15 20:36:13 -0600783 private class InternalConfigListener implements NetworkConfigListener {
784
785 private synchronized void reconfigureSuppressionRules(SuppressionConfig cfg) {
786 if (cfg == null) {
Ray Milkey0a8ee912016-06-13 09:58:12 -0700787 log.debug("Suppression Config is null.");
Naoki Shiota399a0b32015-11-15 20:36:13 -0600788 return;
789 }
790
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800791 SuppressionRules newRules = new SuppressionRules(cfg.deviceTypes(),
Naoki Shiota399a0b32015-11-15 20:36:13 -0600792 cfg.annotation());
793
794 updateRules(newRules);
795 }
796
797 @Override
798 public void event(NetworkConfigEvent event) {
Ray Milkeyd9bbde82016-06-09 11:35:00 -0700799 eventExecutor.execute(() -> {
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800800 if (event.configClass() == LinkDiscoveryFromDevice.class &&
801 CONFIG_CHANGED.contains(event.type())) {
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800802
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800803 if (event.subject() instanceof DeviceId) {
804 final DeviceId did = (DeviceId) event.subject();
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800805 Device device = deviceService.getDevice(did);
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800806 updateDevice(device).ifPresent(ld -> updatePorts(ld, did));
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800807 }
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800808
809 } else if (event.configClass() == LinkDiscoveryFromPort.class &&
810 CONFIG_CHANGED.contains(event.type())) {
811
812 if (event.subject() instanceof ConnectPoint) {
813 ConnectPoint cp = (ConnectPoint) event.subject();
814 if (cp.elementId() instanceof DeviceId) {
815 final DeviceId did = (DeviceId) cp.elementId();
816 Device device = deviceService.getDevice(did);
817 Port port = deviceService.getPort(did, cp.port());
818 updateDevice(device).ifPresent(ld -> updatePort(ld, port));
819 }
820 }
821
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800822 } else if (event.configClass().equals(SuppressionConfig.class) &&
823 (event.type() == NetworkConfigEvent.Type.CONFIG_ADDED ||
824 event.type() == NetworkConfigEvent.Type.CONFIG_UPDATED)) {
825 SuppressionConfig cfg = cfgRegistry.getConfig(appId, SuppressionConfig.class);
826 reconfigureSuppressionRules(cfg);
827 log.trace("Network config reconfigured");
HIGUCHI Yutad9fe3a32015-11-24 18:52:25 -0800828 }
Thomas Vachuskab5f6f522016-03-01 13:52:10 -0800829 });
Naoki Shiota399a0b32015-11-15 20:36:13 -0600830 }
831 }
alshabib7911a052014-10-16 17:49:37 -0700832}