blob: 95429c066b4136a9a87448656f2128b8bf296c08 [file] [log] [blame]
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -08001/*
Brian O'Connor5ab426f2016-04-09 01:19:45 -07002 * Copyright 2015-present Open Networking Laboratory
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -08003 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16package org.onosproject.pathpainter;
17
18import com.fasterxml.jackson.databind.node.ObjectNode;
19import com.google.common.collect.ImmutableList;
20import com.google.common.collect.ImmutableSet;
Andrea Campanella8583e6b2015-12-01 21:24:45 -080021import com.google.common.collect.Sets;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080022import org.onlab.osgi.ServiceDirectory;
23import org.onosproject.net.DeviceId;
Thomas Vachuska08bef152015-12-02 17:08:59 -080024import org.onosproject.net.DisjointPath;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080025import org.onosproject.net.ElementId;
26import org.onosproject.net.HostId;
27import org.onosproject.net.Link;
28import org.onosproject.net.Path;
Andrea Campanellac87fba72015-12-04 11:30:59 -080029import org.onosproject.net.device.DeviceService;
30import org.onosproject.net.topology.GeoDistanceLinkWeight;
31import org.onosproject.net.topology.LinkWeight;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080032import org.onosproject.net.topology.PathService;
Andrea Campanella48c674c2015-12-04 14:48:04 -080033import org.onosproject.net.topology.TopologyEvent;
34import org.onosproject.net.topology.TopologyListener;
35import org.onosproject.net.topology.TopologyService;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080036import org.onosproject.ui.RequestHandler;
37import org.onosproject.ui.UiConnection;
38import org.onosproject.ui.UiMessageHandler;
Andrea Campanella490e8392015-12-03 12:18:11 -080039import org.onosproject.ui.topo.DeviceHighlight;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080040import org.onosproject.ui.topo.Highlights;
Andrea Campanella490e8392015-12-03 12:18:11 -080041import org.onosproject.ui.topo.HostHighlight;
42import org.onosproject.ui.topo.NodeBadge;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080043import org.slf4j.Logger;
44import org.slf4j.LoggerFactory;
45
46import java.util.Collection;
47import java.util.List;
48import java.util.Set;
49
Simon Hunt8a0429a2017-01-06 16:52:47 -080050import static org.onosproject.ui.topo.TopoJson.highlightsMessage;
51
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080052/**
Andrea Campanella86d08632015-12-07 16:27:59 -080053 * ONOS UI PathPainter Topology-Overlay message handler.
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080054 */
55public class PathPainterTopovMessageHandler extends UiMessageHandler {
56
Andrea Campanella48c674c2015-12-04 14:48:04 -080057 private static final String PAINTER_CLEAR = "ppTopovClear";
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080058 private static final String PAINTER_SET_SRC = "ppTopovSetSrc";
59 private static final String PAINTER_SET_DST = "ppTopovSetDst";
60 private static final String PAINTER_SWAP_SRC_DST = "ppTopovSwapSrcDst";
61 private static final String PAINTER_SET_MODE = "ppTopovSetMode";
62
63 private static final String PAINTER_NEXT_PATH = "ppTopovNextPath";
64 private static final String PAINTER_PREV_PATH = "ppTopovPrevPath";
65
66 private static final String ID = "id";
67 private static final String MODE = "mode";
Andrea Campanella490e8392015-12-03 12:18:11 -080068 private static final String TYPE = "type";
69 private static final String SWITCH = "switch";
70 private static final String ENDSTATION = "endstation";
Simon Hunt8a0429a2017-01-06 16:52:47 -080071 private static final String DST = "Dst";
72 private static final String SRC = "Src";
Andrea Campanella86d08632015-12-07 16:27:59 -080073 // Delay for showHighlights event processing on GUI client side to
74 // account for addLink animation.
Simon Hunt8a0429a2017-01-06 16:52:47 -080075 private static final int DELAY_MS = 1100;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080076
Andrea Campanella48c674c2015-12-04 14:48:04 -080077 private final TopologyListener topologyListener = new InternalTopologyListener();
78
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080079 private Set<Link> allPathLinks;
Andrea Campanella48c674c2015-12-04 14:48:04 -080080 private boolean listenersRemoved;
Andrea Campanella86d08632015-12-07 16:27:59 -080081 private LinkWeight linkData;
82 private int highlightDelay;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080083
84 private enum Mode {
Andrea Campanellac87fba72015-12-04 11:30:59 -080085 SHORTEST, DISJOINT, GEODATA, SRLG, INVALID
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080086 }
87
88 private final Logger log = LoggerFactory.getLogger(getClass());
89
90 private PathService pathService;
91
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080092 private ElementId src, dst;
Andrea Campanella490e8392015-12-03 12:18:11 -080093 private String srcType, dstType;
Andrea Campanella8583e6b2015-12-01 21:24:45 -080094 private Mode currentMode = Mode.SHORTEST;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -080095 private List<Path> paths;
96 private int pathIndex;
97
Andrea Campanella48c674c2015-12-04 14:48:04 -080098 protected TopologyService topologyService;
99
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800100
Simon Hunt8a0429a2017-01-06 16:52:47 -0800101 // ===============-=-=-=-=-=-======================-=-=-=-=-=-=-===========
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800102
103
104 @Override
105 public void init(UiConnection connection, ServiceDirectory directory) {
106 super.init(connection, directory);
107 pathService = directory.get(PathService.class);
Andrea Campanella48c674c2015-12-04 14:48:04 -0800108 topologyService = directory.get(TopologyService.class);
Andrea Campanellac87fba72015-12-04 11:30:59 -0800109 linkData = new GeoDistanceLinkWeight(directory.get(DeviceService.class));
Andrea Campanella48c674c2015-12-04 14:48:04 -0800110 addListeners();
111 }
112
113
114 @Override
115 public void destroy() {
116 removeListeners();
117 super.destroy();
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800118 }
119
120 @Override
121 protected Collection<RequestHandler> createRequestHandlers() {
122 return ImmutableSet.of(
Andrea Campanella48c674c2015-12-04 14:48:04 -0800123 new ClearHandler(),
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800124 new SetSrcHandler(),
125 new SetDstHandler(),
Andrea Campanella0c17a0a2015-12-01 09:53:51 -0800126 new SwapSrcDstHandler(),
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800127 new NextPathHandler(),
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800128 new PrevPathHandler(),
129 new SetModeHandler()
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800130 );
131 }
132
133 // === -------------------------
134 // === Handler classes
135
Andrea Campanella48c674c2015-12-04 14:48:04 -0800136 private final class ClearHandler extends RequestHandler {
137
138 public ClearHandler() {
139 super(PAINTER_CLEAR);
140 }
141
142 @Override
Simon Hunt8a0429a2017-01-06 16:52:47 -0800143 public void process(ObjectNode payload) {
Andrea Campanella48c674c2015-12-04 14:48:04 -0800144 src = null;
145 dst = null;
Simon Hunt8a0429a2017-01-06 16:52:47 -0800146 sendMessage(highlightsMessage(new Highlights()));
Andrea Campanella48c674c2015-12-04 14:48:04 -0800147 }
148 }
149
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800150 private final class SetSrcHandler extends RequestHandler {
Andrea Campanella490e8392015-12-03 12:18:11 -0800151
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800152 public SetSrcHandler() {
153 super(PAINTER_SET_SRC);
154 }
155
156 @Override
Simon Hunt8a0429a2017-01-06 16:52:47 -0800157 public void process(ObjectNode payload) {
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800158 String id = string(payload, ID);
159 src = elementId(id);
Andrea Campanella490e8392015-12-03 12:18:11 -0800160 srcType = string(payload, TYPE);
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800161 if (src.equals(dst)) {
162 dst = null;
163 }
Simon Hunt8a0429a2017-01-06 16:52:47 -0800164
165 sendMessage(highlightsMessage(
166 addBadge(new Highlights(), srcType, src.toString(), SRC))
167 );
Andrea Campanella490e8392015-12-03 12:18:11 -0800168 findAndSendPaths(currentMode);
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800169 }
170 }
171
172 private final class SetDstHandler extends RequestHandler {
173 public SetDstHandler() {
174 super(PAINTER_SET_DST);
175 }
176
177 @Override
Simon Hunt8a0429a2017-01-06 16:52:47 -0800178 public void process(ObjectNode payload) {
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800179 String id = string(payload, ID);
180 dst = elementId(id);
Andrea Campanella490e8392015-12-03 12:18:11 -0800181 dstType = string(payload, TYPE);
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800182 if (src.equals(dst)) {
183 src = null;
184 }
Andrea Campanella490e8392015-12-03 12:18:11 -0800185
Simon Hunt8a0429a2017-01-06 16:52:47 -0800186 sendMessage(highlightsMessage(
187 addBadge(new Highlights(), dstType, dst.toString(), DST))
188 );
Andrea Campanella490e8392015-12-03 12:18:11 -0800189 findAndSendPaths(currentMode);
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800190 }
191 }
192
Andrea Campanella0c17a0a2015-12-01 09:53:51 -0800193 private final class SwapSrcDstHandler extends RequestHandler {
194 public SwapSrcDstHandler() {
195 super(PAINTER_SWAP_SRC_DST);
196 }
197
198 @Override
Simon Hunt8a0429a2017-01-06 16:52:47 -0800199 public void process(ObjectNode payload) {
Andrea Campanella0c17a0a2015-12-01 09:53:51 -0800200 ElementId temp = src;
201 src = dst;
202 dst = temp;
Andrea Campanella490e8392015-12-03 12:18:11 -0800203 String s = srcType;
204 srcType = dstType;
205 dstType = s;
206 findAndSendPaths(currentMode);
Andrea Campanella0c17a0a2015-12-01 09:53:51 -0800207 }
208 }
209
Andrea Campanellac87fba72015-12-04 11:30:59 -0800210
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800211 private final class NextPathHandler extends RequestHandler {
212 public NextPathHandler() {
213 super(PAINTER_NEXT_PATH);
214 }
215
216 @Override
Simon Hunt8a0429a2017-01-06 16:52:47 -0800217 public void process(ObjectNode payload) {
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800218 pathIndex = (pathIndex >= paths.size() - 1 ? 0 : pathIndex + 1);
219 hilightAndSendPaths();
220 }
221 }
222
223 private final class PrevPathHandler extends RequestHandler {
224 public PrevPathHandler() {
225 super(PAINTER_PREV_PATH);
226 }
227
228 @Override
Simon Hunt8a0429a2017-01-06 16:52:47 -0800229 public void process(ObjectNode payload) {
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800230 pathIndex = (pathIndex <= 0 ? paths.size() - 1 : pathIndex - 1);
231 hilightAndSendPaths();
232 }
233 }
234
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800235 private final class SetModeHandler extends RequestHandler {
236 public SetModeHandler() {
237 super(PAINTER_SET_MODE);
238 }
239
240 @Override
Simon Hunt8a0429a2017-01-06 16:52:47 -0800241 public void process(ObjectNode payload) {
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800242 String mode = string(payload, MODE);
Andrea Campanellac87fba72015-12-04 11:30:59 -0800243 switch (mode) {
244 case "shortest":
245 currentMode = Mode.SHORTEST;
246 break;
247 case "disjoint":
248 currentMode = Mode.DISJOINT;
249 break;
250 case "geodata":
251 currentMode = Mode.GEODATA;
252 break;
253 case "srlg":
254 currentMode = Mode.SRLG;
255 break;
256 default:
257 currentMode = Mode.INVALID;
258 break;
259 }
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800260 //TODO: add support for SRLG
Andrea Campanella490e8392015-12-03 12:18:11 -0800261 findAndSendPaths(currentMode);
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800262 }
263 }
264
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800265 // === ------------
266
267 private ElementId elementId(String id) {
268 try {
269 return DeviceId.deviceId(id);
270 } catch (IllegalArgumentException e) {
271 return HostId.hostId(id);
272 }
273 }
274
Andrea Campanella490e8392015-12-03 12:18:11 -0800275 private void findAndSendPaths(Mode mode) {
Yuta HIGUCHIa0972cb2016-07-20 20:43:12 -0700276 log.debug("src={}; dst={}; mode={}", src, dst, currentMode);
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800277 if (src != null && dst != null) {
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800278 pathIndex = 0;
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800279 ImmutableSet.Builder<Link> builder = ImmutableSet.builder();
Andrea Campanella490e8392015-12-03 12:18:11 -0800280 if (mode.equals(Mode.SHORTEST)) {
281 paths = ImmutableList.copyOf(pathService.getPaths(src, dst));
282 allPathLinks = buildPaths(builder).build();
283 } else if (mode.equals(Mode.DISJOINT)) {
284 paths = ImmutableList.copyOf(pathService.getDisjointPaths(src, dst));
285 allPathLinks = buildDisjointPaths(builder).build();
Andrea Campanellac87fba72015-12-04 11:30:59 -0800286 } else if (mode.equals(Mode.GEODATA)) {
287 paths = ImmutableList.copyOf(pathService.getPaths(src, dst, linkData));
288 allPathLinks = buildPaths(builder).build();
Andrea Campanella490e8392015-12-03 12:18:11 -0800289 } else {
Simon Hunt8a0429a2017-01-06 16:52:47 -0800290 log.warn("Unsupported MODE");
Andrea Campanella490e8392015-12-03 12:18:11 -0800291 }
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800292 } else {
293 paths = ImmutableList.of();
294 allPathLinks = ImmutableSet.of();
295 }
296 hilightAndSendPaths();
Andrea Campanella490e8392015-12-03 12:18:11 -0800297
298 }
299
300 private ImmutableSet.Builder<Link> buildPaths(ImmutableSet.Builder<Link> pathBuilder) {
301 paths.forEach(path -> path.links().forEach(pathBuilder::add));
302 return pathBuilder;
303 }
304
305 private ImmutableSet.Builder<Link> buildDisjointPaths(ImmutableSet.Builder<Link> pathBuilder) {
306 paths.forEach(path -> {
307 DisjointPath dp = (DisjointPath) path;
308 pathBuilder.addAll(dp.primary().links());
309 pathBuilder.addAll(dp.backup().links());
310 });
311 return pathBuilder;
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800312 }
313
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800314 private void hilightAndSendPaths() {
315 PathLinkMap linkMap = new PathLinkMap();
316 allPathLinks.forEach(linkMap::add);
317
Thomas Vachuska08bef152015-12-02 17:08:59 -0800318 Set<Link> selectedPathLinks;
319
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800320 // Prepare two working sets; one containing selected path links and
321 // the other containing all paths links.
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800322 if (currentMode.equals(Mode.DISJOINT)) {
Andrea Campanella490e8392015-12-03 12:18:11 -0800323 DisjointPath dp = (DisjointPath) paths.get(pathIndex);
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800324 selectedPathLinks = paths.isEmpty() ?
Andrea Campanella490e8392015-12-03 12:18:11 -0800325 ImmutableSet.of() : Sets.newHashSet(dp.primary().links());
Thomas Vachuska08bef152015-12-02 17:08:59 -0800326 selectedPathLinks.addAll(dp.backup().links());
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800327 } else {
328 selectedPathLinks = paths.isEmpty() ?
Thomas Vachuska08bef152015-12-02 17:08:59 -0800329 ImmutableSet.of() : ImmutableSet.copyOf(paths.get(pathIndex).links());
Andrea Campanella8583e6b2015-12-01 21:24:45 -0800330 }
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800331 Highlights highlights = new Highlights();
Andrea Campanella86d08632015-12-07 16:27:59 -0800332 if (highlightDelay > 0) {
333 highlights.delay(highlightDelay);
334 }
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800335 for (PathLink plink : linkMap.biLinks()) {
336 plink.computeHilight(selectedPathLinks, allPathLinks);
337 highlights.add(plink.highlight(null));
338 }
Andrea Campanella490e8392015-12-03 12:18:11 -0800339 if (src != null) {
340 highlights = addBadge(highlights, srcType, src.toString(), SRC);
341 }
342 if (dst != null) {
343 highlights = addBadge(highlights, dstType, dst.toString(), DST);
344 }
Simon Hunt8a0429a2017-01-06 16:52:47 -0800345 sendMessage(highlightsMessage(highlights));
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800346 }
347
Andrea Campanella490e8392015-12-03 12:18:11 -0800348 private Highlights addBadge(Highlights highlights, String type, String elemId, String src) {
349 if (SWITCH.equals(type)) {
350 highlights = addDeviceBadge(highlights, elemId, src);
351 } else if (ENDSTATION.equals(type)) {
352 highlights = addHostBadge(highlights, elemId, src);
353 }
354 return highlights;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800355 }
356
Andrea Campanella490e8392015-12-03 12:18:11 -0800357 private Highlights addDeviceBadge(Highlights h, String elemId, String type) {
358 DeviceHighlight dh = new DeviceHighlight(elemId);
359 dh.setBadge(createBadge(type));
360 h.add(dh);
361 return h;
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800362 }
Andrea Campanella490e8392015-12-03 12:18:11 -0800363
364 private Highlights addHostBadge(Highlights h, String elemId, String type) {
365 HostHighlight hh = new HostHighlight(elemId);
366 hh.setBadge(createBadge(type));
367 h.add(hh);
368 return h;
369 }
370
371 private NodeBadge createBadge(String type) {
372 return NodeBadge.text(type);
373 }
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800374
Andrea Campanella48c674c2015-12-04 14:48:04 -0800375 private synchronized void addListeners() {
376 listenersRemoved = false;
377 topologyService.addListener(topologyListener);
378 }
Andrea Campanella86d08632015-12-07 16:27:59 -0800379
Andrea Campanella48c674c2015-12-04 14:48:04 -0800380 private synchronized void removeListeners() {
381 if (!listenersRemoved) {
382 listenersRemoved = true;
383 topologyService.removeListener(topologyListener);
384 }
385 }
386
387 // Link event listener.
388 private class InternalTopologyListener implements TopologyListener {
389 @Override
390 public void event(TopologyEvent event) {
Andrea Campanella86d08632015-12-07 16:27:59 -0800391 highlightDelay = DELAY_MS;
Andrea Campanella48c674c2015-12-04 14:48:04 -0800392 findAndSendPaths(currentMode);
Andrea Campanella86d08632015-12-07 16:27:59 -0800393 highlightDelay = 0;
Andrea Campanella48c674c2015-12-04 14:48:04 -0800394 }
395 }
396
Thomas Vachuskab4d3ff72015-12-01 09:53:51 -0800397}