blob: 7027fc2b2d46a4c2dce47a8c7ecf4f6edfe87acd [file] [log] [blame]
Thomas Vachuska329af532015-03-10 02:08:33 -07001/*
Brian O'Connora09fe5b2017-08-03 21:12:30 -07002 * Copyright 2015-present Open Networking Foundation
Thomas Vachuska329af532015-03-10 02:08:33 -07003 *
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.ui.impl;
17
18import com.fasterxml.jackson.databind.JsonNode;
Thomas Vachuska329af532015-03-10 02:08:33 -070019import com.fasterxml.jackson.databind.node.ArrayNode;
20import com.fasterxml.jackson.databind.node.ObjectNode;
Simon Hunt879ce452017-08-10 23:32:00 -070021import com.google.common.collect.ImmutableSet;
Thomas Vachuska329af532015-03-10 02:08:33 -070022import org.onlab.osgi.ServiceDirectory;
23import org.onlab.packet.IpAddress;
Simon Hunta58d8942017-08-11 12:51:14 -070024import org.onlab.packet.VlanId;
Simon Hunt95d56fd2015-11-12 11:06:44 -080025import org.onlab.util.DefaultHashMap;
Thomas Vachuska329af532015-03-10 02:08:33 -070026import org.onosproject.cluster.ClusterEvent;
Thomas Vachuska329af532015-03-10 02:08:33 -070027import org.onosproject.cluster.ControllerNode;
28import org.onosproject.cluster.NodeId;
29import org.onosproject.core.CoreService;
cheng fan35dc0f22015-06-10 06:02:47 +080030import org.onosproject.incubator.net.tunnel.OpticalTunnelEndPoint;
31import org.onosproject.incubator.net.tunnel.Tunnel;
Thomas Vachuska329af532015-03-10 02:08:33 -070032import org.onosproject.net.Annotated;
33import org.onosproject.net.AnnotationKeys;
34import org.onosproject.net.Annotations;
35import org.onosproject.net.ConnectPoint;
36import org.onosproject.net.DefaultEdgeLink;
37import org.onosproject.net.Device;
38import org.onosproject.net.DeviceId;
39import org.onosproject.net.EdgeLink;
Simon Huntf4fd2a22016-08-10 15:41:09 -070040import org.onosproject.net.ElementId;
Thomas Vachuska329af532015-03-10 02:08:33 -070041import org.onosproject.net.Host;
42import org.onosproject.net.HostId;
43import org.onosproject.net.HostLocation;
44import org.onosproject.net.Link;
Thomas Vachuska329af532015-03-10 02:08:33 -070045import org.onosproject.net.device.DeviceEvent;
Thomas Vachuska329af532015-03-10 02:08:33 -070046import org.onosproject.net.host.HostEvent;
Thomas Vachuska329af532015-03-10 02:08:33 -070047import org.onosproject.net.link.LinkEvent;
Thomas Vachuska329af532015-03-10 02:08:33 -070048import org.onosproject.net.provider.ProviderId;
Thomas Vachuska329af532015-03-10 02:08:33 -070049import org.onosproject.net.topology.Topology;
Simon Huntd2747a02015-04-30 22:41:16 -070050import org.onosproject.ui.JsonUtils;
Thomas Vachuska329af532015-03-10 02:08:33 -070051import org.onosproject.ui.UiConnection;
Simon Hunta0ddb022015-05-01 09:53:01 -070052import org.onosproject.ui.UiMessageHandler;
Simon Hunted804d52016-03-30 09:51:40 -070053import org.onosproject.ui.impl.topo.util.ServicesBundle;
Simon Hunt879ce452017-08-10 23:32:00 -070054import org.onosproject.ui.lion.LionBundle;
Simon Hunt0af1ec32015-07-24 12:17:55 -070055import org.onosproject.ui.topo.PropertyPanel;
Thomas Vachuska329af532015-03-10 02:08:33 -070056import org.slf4j.Logger;
57import org.slf4j.LoggerFactory;
58
Thomas Vachuska329af532015-03-10 02:08:33 -070059import java.util.Collection;
60import java.util.Collections;
Thomas Vachuska329af532015-03-10 02:08:33 -070061import java.util.Iterator;
Thomas Vachuska329af532015-03-10 02:08:33 -070062import java.util.Map;
chengfanc553c952016-07-22 15:48:23 +080063import java.util.Optional;
Simon Huntf4fd2a22016-08-10 15:41:09 -070064import java.util.Set;
Thomas Vachuska329af532015-03-10 02:08:33 -070065import java.util.concurrent.ConcurrentHashMap;
66
Thomas Vachuska329af532015-03-10 02:08:33 -070067import static com.google.common.base.Strings.isNullOrEmpty;
Thomas Vachuska329af532015-03-10 02:08:33 -070068import static org.onosproject.net.PortNumber.portNumber;
Simon Hunt3a0598f2015-08-04 19:59:04 -070069import static org.onosproject.ui.topo.TopoConstants.CoreButtons;
Simon Hunt879ce452017-08-10 23:32:00 -070070import static org.onosproject.ui.topo.TopoConstants.Properties.DEVICES;
71import static org.onosproject.ui.topo.TopoConstants.Properties.FLOWS;
Simon Hunta58d8942017-08-11 12:51:14 -070072import static org.onosproject.ui.topo.TopoConstants.Properties.GRID_X;
73import static org.onosproject.ui.topo.TopoConstants.Properties.GRID_Y;
Simon Hunt879ce452017-08-10 23:32:00 -070074import static org.onosproject.ui.topo.TopoConstants.Properties.HOSTS;
Simon Hunta58d8942017-08-11 12:51:14 -070075import static org.onosproject.ui.topo.TopoConstants.Properties.HW_VERSION;
Simon Hunt879ce452017-08-10 23:32:00 -070076import static org.onosproject.ui.topo.TopoConstants.Properties.INTENTS;
Simon Hunta58d8942017-08-11 12:51:14 -070077import static org.onosproject.ui.topo.TopoConstants.Properties.IP;
78import static org.onosproject.ui.topo.TopoConstants.Properties.LATITUDE;
Simon Hunt879ce452017-08-10 23:32:00 -070079import static org.onosproject.ui.topo.TopoConstants.Properties.LINKS;
Simon Hunta58d8942017-08-11 12:51:14 -070080import static org.onosproject.ui.topo.TopoConstants.Properties.LONGITUDE;
81import static org.onosproject.ui.topo.TopoConstants.Properties.MAC;
82import static org.onosproject.ui.topo.TopoConstants.Properties.PORTS;
83import static org.onosproject.ui.topo.TopoConstants.Properties.PROTOCOL;
84import static org.onosproject.ui.topo.TopoConstants.Properties.SERIAL_NUMBER;
85import static org.onosproject.ui.topo.TopoConstants.Properties.SW_VERSION;
Simon Hunt879ce452017-08-10 23:32:00 -070086import static org.onosproject.ui.topo.TopoConstants.Properties.TOPOLOGY_SSCS;
87import static org.onosproject.ui.topo.TopoConstants.Properties.TUNNELS;
Simon Hunta58d8942017-08-11 12:51:14 -070088import static org.onosproject.ui.topo.TopoConstants.Properties.URI;
89import static org.onosproject.ui.topo.TopoConstants.Properties.VENDOR;
Simon Hunt879ce452017-08-10 23:32:00 -070090import static org.onosproject.ui.topo.TopoConstants.Properties.VERSION;
Simon Hunta58d8942017-08-11 12:51:14 -070091import static org.onosproject.ui.topo.TopoConstants.Properties.VLAN;
92import static org.onosproject.ui.topo.TopoConstants.Properties.VLAN_NONE;
Simon Huntd3ceffa2015-08-25 12:44:35 -070093import static org.onosproject.ui.topo.TopoUtils.compactLinkString;
Thomas Vachuska329af532015-03-10 02:08:33 -070094
95/**
96 * Facility for creating messages bound for the topology viewer.
97 */
Simon Hunta0ddb022015-05-01 09:53:01 -070098public abstract class TopologyViewMessageHandlerBase extends UiMessageHandler {
Thomas Vachuska329af532015-03-10 02:08:33 -070099
Simon Huntf4fd2a22016-08-10 15:41:09 -0700100 private static final String NO_GEO_VALUE = "0.0";
Simon Hunt10618f62017-06-15 19:30:52 -0700101 private static final String DASH = "-";
Simon Hunta58d8942017-08-11 12:51:14 -0700102 private static final String SLASH = " / ";
Simon Hunt10618f62017-06-15 19:30:52 -0700103
104 // nav paths are the view names for hot-link navigation from topo view...
105 private static final String DEVICE_NAV_PATH = "device";
106 private static final String HOST_NAV_PATH = "host";
Simon Huntf4fd2a22016-08-10 15:41:09 -0700107
Simon Hunta58d8942017-08-11 12:51:14 -0700108 // link panel label keys
109 private static final String LPL_FRIENDLY = "lp_label_friendly";
110 private static final String LPL_A_TYPE = "lp_label_a_type";
111 private static final String LPL_A_ID = "lp_label_a_id";
112 private static final String LPL_A_FRIENDLY = "lp_label_a_friendly";
113 private static final String LPL_A_PORT = "lp_label_a_port";
114 private static final String LPL_B_TYPE = "lp_label_b_type";
115 private static final String LPL_B_ID = "lp_label_b_id";
116 private static final String LPL_B_FRIENDLY = "lp_label_b_friendly";
117 private static final String LPL_B_PORT = "lp_label_b_port";
118 private static final String LPL_A2B = "lp_label_a2b";
119 private static final String LPL_B2A = "lp_label_b2a";
120 private static final String LPV_NO_LINK = "lp_value_no_link";
121
122 // other Lion keys
123 private static final String HOST = "host";
124 private static final String DEVICE = "device";
125 private static final String EXPECTED = "expected";
126 private static final String NOT_EXPECTED = "not_expected";
127
Simon Hunt95d56fd2015-11-12 11:06:44 -0800128 // default to an "add" event...
129 private static final DefaultHashMap<ClusterEvent.Type, String> CLUSTER_EVENT =
130 new DefaultHashMap<>("addInstance");
131
132 // default to an "update" event...
133 private static final DefaultHashMap<DeviceEvent.Type, String> DEVICE_EVENT =
134 new DefaultHashMap<>("updateDevice");
135 private static final DefaultHashMap<LinkEvent.Type, String> LINK_EVENT =
136 new DefaultHashMap<>("updateLink");
137 private static final DefaultHashMap<HostEvent.Type, String> HOST_EVENT =
138 new DefaultHashMap<>("updateHost");
139
140 // but call out specific events that we care to differentiate...
141 static {
142 CLUSTER_EVENT.put(ClusterEvent.Type.INSTANCE_REMOVED, "removeInstance");
143
144 DEVICE_EVENT.put(DeviceEvent.Type.DEVICE_ADDED, "addDevice");
145 DEVICE_EVENT.put(DeviceEvent.Type.DEVICE_REMOVED, "removeDevice");
146
147 LINK_EVENT.put(LinkEvent.Type.LINK_ADDED, "addLink");
148 LINK_EVENT.put(LinkEvent.Type.LINK_REMOVED, "removeLink");
149
150 HOST_EVENT.put(HostEvent.Type.HOST_ADDED, "addHost");
151 HOST_EVENT.put(HostEvent.Type.HOST_REMOVED, "removeHost");
152 HOST_EVENT.put(HostEvent.Type.HOST_MOVED, "moveHost");
153 }
154
Simon Hunta58d8942017-08-11 12:51:14 -0700155 private static final DefaultHashMap<Device.Type, String> DEVICE_GLYPHS =
156 new DefaultHashMap<>("m_unknown");
157
158 static {
159 DEVICE_GLYPHS.put(Device.Type.SWITCH, "m_switch");
160 DEVICE_GLYPHS.put(Device.Type.ROUTER, "m_router");
161 DEVICE_GLYPHS.put(Device.Type.ROADM, "m_roadm");
162 DEVICE_GLYPHS.put(Device.Type.OTN, "m_otn");
163 DEVICE_GLYPHS.put(Device.Type.ROADM_OTN, "m_roadm_otn");
164 DEVICE_GLYPHS.put(Device.Type.BALANCER, "m_balancer");
165 DEVICE_GLYPHS.put(Device.Type.IPS, "m_ips");
166 DEVICE_GLYPHS.put(Device.Type.IDS, "m_ids");
167 DEVICE_GLYPHS.put(Device.Type.CONTROLLER, "m_controller");
168 DEVICE_GLYPHS.put(Device.Type.VIRTUAL, "m_virtual");
169 DEVICE_GLYPHS.put(Device.Type.FIBER_SWITCH, "m_fiberSwitch");
170 DEVICE_GLYPHS.put(Device.Type.MICROWAVE, "m_microwave");
171 DEVICE_GLYPHS.put(Device.Type.OLT, "m_olt");
172 DEVICE_GLYPHS.put(Device.Type.ONU, "m_onu");
173 DEVICE_GLYPHS.put(Device.Type.OPTICAL_AMPLIFIER, "unknown"); // TODO glyph needed
174 DEVICE_GLYPHS.put(Device.Type.OTHER, "m_other");
175 }
176
177 private static final String DEFAULT_HOST_GLYPH = "m_endstation";
178 private static final String LINK_GLYPH = "m_ports";
179
180
Ray Milkey9c9cde42018-01-12 14:22:06 -0800181 static final Logger log =
Simon Huntd2747a02015-04-30 22:41:16 -0700182 LoggerFactory.getLogger(TopologyViewMessageHandlerBase.class);
Thomas Vachuska329af532015-03-10 02:08:33 -0700183
Simon Huntd2747a02015-04-30 22:41:16 -0700184 private static final ProviderId PID =
185 new ProviderId("core", "org.onosproject.core", true);
Thomas Vachuska329af532015-03-10 02:08:33 -0700186
Thomas Vachuska329af532015-03-10 02:08:33 -0700187 // TODO: extract into an external & durable state; good enough for now and demo
188 private static Map<String, ObjectNode> metaUi = new ConcurrentHashMap<>();
189
190 /**
Thomas Vachuska329af532015-03-10 02:08:33 -0700191 * Returns read-only view of the meta-ui information.
192 *
193 * @return map of id to meta-ui mementos
194 */
195 static Map<String, ObjectNode> getMetaUi() {
196 return Collections.unmodifiableMap(metaUi);
197 }
198
Simon Hunt879ce452017-08-10 23:32:00 -0700199 private static final String LION_TOPO = "core.view.Topo";
200
201 private static final Set<String> REQ_LION_BUNDLES = ImmutableSet.of(
202 LION_TOPO
203 );
Simon Hunt1911fe42017-05-02 18:25:58 -0700204
205 protected ServicesBundle services;
206
207 private String version;
208
209
Thomas Vachuska329af532015-03-10 02:08:33 -0700210 @Override
211 public void init(UiConnection connection, ServiceDirectory directory) {
212 super.init(connection, directory);
Simon Hunt1911fe42017-05-02 18:25:58 -0700213 services = new ServicesBundle(directory);
214 setVersionString(directory);
215 }
Thomas Vachuska329af532015-03-10 02:08:33 -0700216
Simon Hunt1911fe42017-05-02 18:25:58 -0700217 // Creates a palatable version string to display on the summary panel
218 private void setVersionString(ServiceDirectory directory) {
Thomas Vachuska329af532015-03-10 02:08:33 -0700219 String ver = directory.get(CoreService.class).version().toString();
220 version = ver.replace(".SNAPSHOT", "*").replaceFirst("~.*$", "");
221 }
222
Simon Hunt879ce452017-08-10 23:32:00 -0700223 @Override
224 public Set<String> requiredLionBundles() {
225 return REQ_LION_BUNDLES;
226 }
227
Simon Hunt10618f62017-06-15 19:30:52 -0700228 // Returns the first of the given set of IP addresses as a string.
Thomas Vachuska329af532015-03-10 02:08:33 -0700229 private String ip(Set<IpAddress> ipAddresses) {
230 Iterator<IpAddress> it = ipAddresses.iterator();
231 return it.hasNext() ? it.next().toString() : "unknown";
232 }
233
234 // Produces JSON structure from annotations.
235 private JsonNode props(Annotations annotations) {
Simon Huntda580882015-05-12 20:58:18 -0700236 ObjectNode props = objectNode();
Thomas Vachuska329af532015-03-10 02:08:33 -0700237 if (annotations != null) {
238 for (String key : annotations.keys()) {
239 props.put(key, annotations.value(key));
240 }
241 }
242 return props;
243 }
244
245 // Produces an informational log message event bound to the client.
Simon Hunt36740d02017-06-07 11:25:51 -0700246 protected ObjectNode info(String message) {
247 return message("info", message);
Thomas Vachuska329af532015-03-10 02:08:33 -0700248 }
249
250 // Produces a warning log message event bound to the client.
Simon Hunt36740d02017-06-07 11:25:51 -0700251 protected ObjectNode warning(String message) {
252 return message("warning", message);
Thomas Vachuska329af532015-03-10 02:08:33 -0700253 }
254
255 // Produces an error log message event bound to the client.
Simon Hunt36740d02017-06-07 11:25:51 -0700256 protected ObjectNode error(String message) {
257 return message("error", message);
Thomas Vachuska329af532015-03-10 02:08:33 -0700258 }
259
260 // Produces a log message event bound to the client.
Simon Hunt36740d02017-06-07 11:25:51 -0700261 private ObjectNode message(String severity, String message) {
Simon Huntda580882015-05-12 20:58:18 -0700262 ObjectNode payload = objectNode()
Simon Huntd2747a02015-04-30 22:41:16 -0700263 .put("severity", severity)
264 .put("message", message);
Thomas Vachuska329af532015-03-10 02:08:33 -0700265
Simon Hunt36740d02017-06-07 11:25:51 -0700266 return JsonUtils.envelope("message", payload);
Thomas Vachuska329af532015-03-10 02:08:33 -0700267 }
268
Thomas Vachuska329af532015-03-10 02:08:33 -0700269 // Produces a cluster instance message to the client.
Simon Hunt95d56fd2015-11-12 11:06:44 -0800270 protected ObjectNode instanceMessage(ClusterEvent event, String msgType) {
Thomas Vachuska329af532015-03-10 02:08:33 -0700271 ControllerNode node = event.subject();
Simon Hunt1911fe42017-05-02 18:25:58 -0700272 int switchCount = services.mastership().getDevicesOf(node.id()).size();
Simon Huntda580882015-05-12 20:58:18 -0700273 ObjectNode payload = objectNode()
Thomas Vachuska329af532015-03-10 02:08:33 -0700274 .put("id", node.id().toString())
275 .put("ip", node.ip().toString())
Simon Hunt1911fe42017-05-02 18:25:58 -0700276 .put("online", services.cluster().getState(node.id()).isActive())
277 .put("ready", services.cluster().getState(node.id()).isReady())
278 .put("uiAttached", node.equals(services.cluster().getLocalNode()))
Thomas Vachuska329af532015-03-10 02:08:33 -0700279 .put("switches", switchCount);
280
Simon Huntda580882015-05-12 20:58:18 -0700281 ArrayNode labels = arrayNode();
Thomas Vachuska329af532015-03-10 02:08:33 -0700282 labels.add(node.id().toString());
283 labels.add(node.ip().toString());
284
285 // Add labels, props and stuff the payload into envelope.
286 payload.set("labels", labels);
287 addMetaUi(node.id().toString(), payload);
288
Simon Hunt95d56fd2015-11-12 11:06:44 -0800289 String type = msgType != null ? msgType : CLUSTER_EVENT.get(event.type());
Simon Hunt36740d02017-06-07 11:25:51 -0700290 return JsonUtils.envelope(type, payload);
Thomas Vachuska329af532015-03-10 02:08:33 -0700291 }
292
293 // Produces a device event message to the client.
294 protected ObjectNode deviceMessage(DeviceEvent event) {
295 Device device = event.subject();
Simon Hunt1e20dae2016-10-28 11:26:26 -0700296 String uiType = device.annotations().value(AnnotationKeys.UI_TYPE);
297 String devType = uiType != null ? uiType :
Laszlo Pappe1579fa2018-03-08 09:18:48 +0000298 services.driver().getDriver(device.id()).getProperty(AnnotationKeys.UI_TYPE);
299 if (devType == null) {
300 devType = device.type().toString().toLowerCase();
301 }
Simon Hunt10618f62017-06-15 19:30:52 -0700302 String name = device.annotations().value(AnnotationKeys.NAME);
303 name = isNullOrEmpty(name) ? device.id().toString() : name;
Simon Hunt1e20dae2016-10-28 11:26:26 -0700304
Simon Huntda580882015-05-12 20:58:18 -0700305 ObjectNode payload = objectNode()
Thomas Vachuska329af532015-03-10 02:08:33 -0700306 .put("id", device.id().toString())
Simon Hunt1e20dae2016-10-28 11:26:26 -0700307 .put("type", devType)
Simon Hunt1911fe42017-05-02 18:25:58 -0700308 .put("online", services.device().isAvailable(device.id()))
Thomas Vachuska329af532015-03-10 02:08:33 -0700309 .put("master", master(device.id()));
310
Simon Hunt10618f62017-06-15 19:30:52 -0700311 payload.set("labels", labels("", name, device.id().toString()));
Thomas Vachuska329af532015-03-10 02:08:33 -0700312 payload.set("props", props(device.annotations()));
313 addGeoLocation(device, payload);
Thomas Vachuskac67a9912018-03-06 14:37:45 -0800314 addGridLocation(device, payload);
Thomas Vachuska329af532015-03-10 02:08:33 -0700315 addMetaUi(device.id().toString(), payload);
316
Simon Hunt95d56fd2015-11-12 11:06:44 -0800317 String type = DEVICE_EVENT.get(event.type());
Simon Hunt36740d02017-06-07 11:25:51 -0700318 return JsonUtils.envelope(type, payload);
Thomas Vachuska329af532015-03-10 02:08:33 -0700319 }
320
321 // Produces a link event message to the client.
322 protected ObjectNode linkMessage(LinkEvent event) {
323 Link link = event.subject();
Simon Huntda580882015-05-12 20:58:18 -0700324 ObjectNode payload = objectNode()
Thomas Vachuska329af532015-03-10 02:08:33 -0700325 .put("id", compactLinkString(link))
326 .put("type", link.type().toString().toLowerCase())
Ray Milkeyb7f0f642016-01-22 16:08:14 -0800327 .put("expected", link.isExpected())
Thomas Vachuska329af532015-03-10 02:08:33 -0700328 .put("online", link.state() == Link.State.ACTIVE)
329 .put("linkWidth", 1.2)
330 .put("src", link.src().deviceId().toString())
331 .put("srcPort", link.src().port().toString())
332 .put("dst", link.dst().deviceId().toString())
333 .put("dstPort", link.dst().port().toString());
Simon Hunt95d56fd2015-11-12 11:06:44 -0800334 String type = LINK_EVENT.get(event.type());
Simon Hunt36740d02017-06-07 11:25:51 -0700335 return JsonUtils.envelope(type, payload);
Thomas Vachuska329af532015-03-10 02:08:33 -0700336 }
337
338 // Produces a host event message to the client.
339 protected ObjectNode hostMessage(HostEvent event) {
340 Host host = event.subject();
Charles Chan33f28a92015-11-13 13:12:38 -0800341 Host prevHost = event.prevSubject();
Simon Hunt1e20dae2016-10-28 11:26:26 -0700342 String hostType = host.annotations().value(AnnotationKeys.UI_TYPE);
Simon Hunt10618f62017-06-15 19:30:52 -0700343 String ip = ip(host.ipAddresses());
Simon Hunt95d56fd2015-11-12 11:06:44 -0800344
Simon Huntda580882015-05-12 20:58:18 -0700345 ObjectNode payload = objectNode()
Thomas Vachuska329af532015-03-10 02:08:33 -0700346 .put("id", host.id().toString())
Simon Hunt12c79ed2017-09-12 11:58:44 -0700347 .put("type", isNullOrEmpty(hostType) ? "endstation" : hostType);
Simon Hunt10618f62017-06-15 19:30:52 -0700348
Simon Hunt12c79ed2017-09-12 11:58:44 -0700349 // set most recent connect point (and previous if we know it)
Simon Huntda580882015-05-12 20:58:18 -0700350 payload.set("cp", hostConnect(host.location()));
Charles Chan33f28a92015-11-13 13:12:38 -0800351 if (prevHost != null && prevHost.location() != null) {
352 payload.set("prevCp", hostConnect(prevHost.location()));
Simon Hunt95d56fd2015-11-12 11:06:44 -0800353 }
Simon Hunt12c79ed2017-09-12 11:58:44 -0700354
355 // set ALL connect points
356 addAllCps(host.locations(), payload);
357
Thomas Vachuska528ea952017-10-09 18:25:35 +0200358 payload.set("labels", labels(nameForHost(host), ip, host.mac().toString(), ""));
Thomas Vachuska329af532015-03-10 02:08:33 -0700359 payload.set("props", props(host.annotations()));
360 addGeoLocation(host, payload);
Thomas Vachuskac67a9912018-03-06 14:37:45 -0800361 addGridLocation(host, payload);
Thomas Vachuska329af532015-03-10 02:08:33 -0700362 addMetaUi(host.id().toString(), payload);
363
Simon Hunt95d56fd2015-11-12 11:06:44 -0800364 String type = HOST_EVENT.get(event.type());
Simon Hunt36740d02017-06-07 11:25:51 -0700365 return JsonUtils.envelope(type, payload);
Thomas Vachuska329af532015-03-10 02:08:33 -0700366 }
367
Simon Hunt12c79ed2017-09-12 11:58:44 -0700368 private void addAllCps(Set<HostLocation> locations, ObjectNode payload) {
369 ArrayNode cps = arrayNode();
370 locations.forEach(loc -> cps.add(hostConnect(loc)));
371 payload.set("allCps", cps);
372 }
373
Thomas Vachuska329af532015-03-10 02:08:33 -0700374 // Encodes the specified host location into a JSON object.
Simon Huntda580882015-05-12 20:58:18 -0700375 private ObjectNode hostConnect(HostLocation location) {
376 return objectNode()
Thomas Vachuska329af532015-03-10 02:08:33 -0700377 .put("device", location.deviceId().toString())
378 .put("port", location.port().toLong());
379 }
380
381 // Encodes the specified list of labels a JSON array.
Simon Huntda580882015-05-12 20:58:18 -0700382 private ArrayNode labels(String... labels) {
383 ArrayNode json = arrayNode();
Thomas Vachuska329af532015-03-10 02:08:33 -0700384 for (String label : labels) {
385 json.add(label);
386 }
387 return json;
388 }
389
390 // Returns the name of the master node for the specified device id.
391 private String master(DeviceId deviceId) {
Simon Hunt1911fe42017-05-02 18:25:58 -0700392 NodeId master = services.mastership().getMasterFor(deviceId);
Thomas Vachuska329af532015-03-10 02:08:33 -0700393 return master != null ? master.toString() : "";
394 }
395
396 // Generates an edge link from the specified host location.
397 private EdgeLink edgeLink(Host host, boolean ingress) {
398 return new DefaultEdgeLink(PID, new ConnectPoint(host.id(), portNumber(0)),
399 host.location(), ingress);
400 }
401
402 // Adds meta UI information for the specified object.
403 private void addMetaUi(String id, ObjectNode payload) {
404 ObjectNode meta = metaUi.get(id);
405 if (meta != null) {
406 payload.set("metaUi", meta);
407 }
408 }
409
410 // Adds a geo location JSON to the specified payload object.
411 private void addGeoLocation(Annotated annotated, ObjectNode payload) {
412 Annotations annotations = annotated.annotations();
413 if (annotations == null) {
414 return;
415 }
416
Simon Huntfd7106c2016-02-09 15:05:26 -0800417 String slat = annotations.value(AnnotationKeys.LATITUDE);
Simon Huntf27a9292017-05-04 17:36:26 -0700418 String slng = annotations.value(AnnotationKeys.LONGITUDE);
Simon Huntf4fd2a22016-08-10 15:41:09 -0700419 boolean validLat = slat != null && !slat.equals(NO_GEO_VALUE);
Simon Huntf27a9292017-05-04 17:36:26 -0700420 boolean validLng = slng != null && !slng.equals(NO_GEO_VALUE);
Simon Huntf4fd2a22016-08-10 15:41:09 -0700421 if (validLat && validLng) {
422 try {
Simon Huntfd7106c2016-02-09 15:05:26 -0800423 double lat = Double.parseDouble(slat);
Simon Huntf27a9292017-05-04 17:36:26 -0700424 double lng = Double.parseDouble(slng);
Simon Huntda580882015-05-12 20:58:18 -0700425 ObjectNode loc = objectNode()
Simon Huntf27a9292017-05-04 17:36:26 -0700426 .put("locType", "geo")
427 .put("latOrY", lat)
428 .put("longOrX", lng);
Thomas Vachuska329af532015-03-10 02:08:33 -0700429 payload.set("location", loc);
Simon Huntf4fd2a22016-08-10 15:41:09 -0700430 } catch (NumberFormatException e) {
Simon Huntf27a9292017-05-04 17:36:26 -0700431 log.warn("Invalid geo data: latitude={}, longitude={}", slat, slng);
Thomas Vachuska329af532015-03-10 02:08:33 -0700432 }
Thomas Vachuska329af532015-03-10 02:08:33 -0700433 }
434 }
435
Thomas Vachuskac67a9912018-03-06 14:37:45 -0800436 // Adds a grid location JSON to the specified payload object.
437 private void addGridLocation(Annotated annotated, ObjectNode payload) {
438 Annotations annotations = annotated.annotations();
439 if (annotations == null) {
440 return;
441 }
442
443 String xs = annotations.value(AnnotationKeys.GRID_X);
444 String ys = annotations.value(AnnotationKeys.GRID_Y);
445 if (xs != null && ys != null) {
446 try {
447 double x = Double.parseDouble(xs);
448 double y = Double.parseDouble(ys);
449 ObjectNode loc = objectNode()
450 .put("locType", "grid")
451 .put("latOrY", y)
452 .put("longOrX", x);
453 payload.set("location", loc);
454 } catch (NumberFormatException e) {
455 log.warn("Invalid grid data: x={}, y={}", xs, ys);
456 }
457 }
458 }
459
Thomas Vachuska329af532015-03-10 02:08:33 -0700460 // Updates meta UI information for the specified object.
Simon Huntd2747a02015-04-30 22:41:16 -0700461 protected void updateMetaUi(ObjectNode payload) {
462 metaUi.put(JsonUtils.string(payload, "id"),
463 JsonUtils.node(payload, "memento"));
Thomas Vachuska329af532015-03-10 02:08:33 -0700464 }
465
Simon Hunta17fa672015-08-19 18:42:22 -0700466
Simon Huntb745ca62015-07-28 15:37:11 -0700467 // -----------------------------------------------------------------------
468 // Create models of the data to return, that overlays can adjust / augment
469
Simon Hunta58d8942017-08-11 12:51:14 -0700470 private String lookupGlyph(Device device) {
471 return DEVICE_GLYPHS.get(device.type());
472 }
473
474
Simon Huntb745ca62015-07-28 15:37:11 -0700475 // Returns property panel model for summary response.
Simon Hunt8a0429a2017-01-06 16:52:47 -0800476 protected PropertyPanel summmaryMessage() {
Simon Hunta58d8942017-08-11 12:51:14 -0700477 // chose NOT to add debug messages, since this is called every few seconds
Simon Hunt1911fe42017-05-02 18:25:58 -0700478 Topology topology = services.topology().currentTopology();
Simon Hunt879ce452017-08-10 23:32:00 -0700479 LionBundle lion = getLionBundle(LION_TOPO);
480 String panelTitle = lion.getSafe("title_panel_summary");
Simon Hunt0af1ec32015-07-24 12:17:55 -0700481
Simon Hunta58d8942017-08-11 12:51:14 -0700482 return new PropertyPanel(panelTitle, "bird")
Simon Hunt879ce452017-08-10 23:32:00 -0700483 .addProp(VERSION, lion.getSafe(VERSION), version)
Simon Hunt1911fe42017-05-02 18:25:58 -0700484 .addSeparator()
Simon Hunt879ce452017-08-10 23:32:00 -0700485 .addProp(DEVICES, lion.getSafe(DEVICES), services.device().getDeviceCount())
486 .addProp(LINKS, lion.getSafe(LINKS), topology.linkCount())
487 .addProp(HOSTS, lion.getSafe(HOSTS), services.host().getHostCount())
488 .addProp(TOPOLOGY_SSCS, lion.getSafe(TOPOLOGY_SSCS), topology.clusterCount())
Simon Hunt1911fe42017-05-02 18:25:58 -0700489 .addSeparator()
Simon Hunt879ce452017-08-10 23:32:00 -0700490 .addProp(INTENTS, lion.getSafe(INTENTS), services.intent().getIntentCount())
491 .addProp(TUNNELS, lion.getSafe(TUNNELS), services.tunnel().tunnelCount())
492 .addProp(FLOWS, lion.getSafe(FLOWS), services.flow().getFlowRuleCount());
Thomas Vachuska329af532015-03-10 02:08:33 -0700493 }
494
Simon Hunta58d8942017-08-11 12:51:14 -0700495
496 private String friendlyDevice(DeviceId deviceId) {
Simon Hunt1911fe42017-05-02 18:25:58 -0700497 Device device = services.device().getDevice(deviceId);
Thomas Vachuska329af532015-03-10 02:08:33 -0700498 Annotations annot = device.annotations();
499 String name = annot.value(AnnotationKeys.NAME);
Simon Hunta58d8942017-08-11 12:51:14 -0700500 return isNullOrEmpty(name) ? deviceId.toString() : name;
501 }
502
503 // Generates a property panel model for device details response
504 protected PropertyPanel deviceDetails(DeviceId deviceId) {
505 log.debug("generate prop panel data for device {}", deviceId);
506 Device device = services.device().getDevice(deviceId);
507 Annotations annot = device.annotations();
508 String proto = annot.value(AnnotationKeys.PROTOCOL);
509 String title = friendlyDevice(deviceId);
510 LionBundle lion = getLionBundle(LION_TOPO);
511
512 PropertyPanel pp = new PropertyPanel(title, lookupGlyph(device))
513 .navPath(DEVICE_NAV_PATH)
514 .id(deviceId.toString());
515 addDeviceBasicProps(pp, deviceId, device, proto, lion);
516 addLocationProps(pp, annot, lion);
517 addDeviceCountStats(pp, deviceId, lion);
518 addDeviceCoreButtons(pp);
519 return pp;
520 }
521
522 private void addDeviceBasicProps(PropertyPanel pp, DeviceId deviceId,
523 Device device, String proto, LionBundle lion) {
524 pp.addProp(URI, lion.getSafe(URI), deviceId.toString())
525 .addProp(VENDOR, lion.getSafe(VENDOR), device.manufacturer())
526 .addProp(HW_VERSION, lion.getSafe(HW_VERSION), device.hwVersion())
527 .addProp(SW_VERSION, lion.getSafe(SW_VERSION), device.swVersion())
528 .addProp(SERIAL_NUMBER, lion.getSafe(SERIAL_NUMBER), device.serialNumber())
529 .addProp(PROTOCOL, lion.getSafe(PROTOCOL), proto)
530 .addSeparator();
531 }
532
533 // only add location properties if we have them
534 private void addLocationProps(PropertyPanel pp, Annotations annot,
535 LionBundle lion) {
536 String slat = annot.value(AnnotationKeys.LATITUDE);
537 String slng = annot.value(AnnotationKeys.LONGITUDE);
538 String sgrY = annot.value(AnnotationKeys.GRID_Y);
539 String sgrX = annot.value(AnnotationKeys.GRID_X);
540
541 boolean validLat = slat != null && !slat.equals(NO_GEO_VALUE);
542 boolean validLng = slng != null && !slng.equals(NO_GEO_VALUE);
543 if (validLat && validLng) {
544 pp.addProp(LATITUDE, lion.getSafe(LATITUDE), slat)
545 .addProp(LONGITUDE, lion.getSafe(LONGITUDE), slng)
546 .addSeparator();
547
548 } else if (sgrY != null && sgrX != null) {
549 pp.addProp(GRID_Y, lion.getSafe(GRID_Y), sgrY)
550 .addProp(GRID_X, lion.getSafe(GRID_X), sgrX)
551 .addSeparator();
552 }
553 // else, no location
554 }
555
556 private void addDeviceCountStats(PropertyPanel pp, DeviceId deviceId, LionBundle lion) {
Simon Hunt1911fe42017-05-02 18:25:58 -0700557 int portCount = services.device().getPorts(deviceId).size();
Thomas Vachuska329af532015-03-10 02:08:33 -0700558 int flowCount = getFlowCount(deviceId);
cheng fan35dc0f22015-06-10 06:02:47 +0800559 int tunnelCount = getTunnelCount(deviceId);
Simon Huntb745ca62015-07-28 15:37:11 -0700560
Simon Hunta58d8942017-08-11 12:51:14 -0700561 pp.addProp(PORTS, lion.getSafe(PORTS), portCount)
562 .addProp(FLOWS, lion.getSafe(FLOWS), flowCount)
563 .addProp(TUNNELS, lion.getSafe(TUNNELS), tunnelCount);
564 }
Simon Huntb745ca62015-07-28 15:37:11 -0700565
Simon Hunta58d8942017-08-11 12:51:14 -0700566 private void addDeviceCoreButtons(PropertyPanel pp) {
567 pp.addButton(CoreButtons.SHOW_DEVICE_VIEW)
Simon Hunt1911fe42017-05-02 18:25:58 -0700568 .addButton(CoreButtons.SHOW_FLOW_VIEW)
569 .addButton(CoreButtons.SHOW_PORT_VIEW)
570 .addButton(CoreButtons.SHOW_GROUP_VIEW)
571 .addButton(CoreButtons.SHOW_METER_VIEW);
Thomas Vachuska329af532015-03-10 02:08:33 -0700572 }
573
574 protected int getFlowCount(DeviceId deviceId) {
Thomas Vachuskaa8e74772018-02-26 11:33:35 -0800575 return services.flow().getFlowRuleCount(deviceId);
Thomas Vachuska329af532015-03-10 02:08:33 -0700576 }
577
cheng fan35dc0f22015-06-10 06:02:47 +0800578 protected int getTunnelCount(DeviceId deviceId) {
579 int count = 0;
Simon Hunt1911fe42017-05-02 18:25:58 -0700580 Collection<Tunnel> tunnels = services.tunnel().queryAllTunnels();
cheng fan35dc0f22015-06-10 06:02:47 +0800581 for (Tunnel tunnel : tunnels) {
chengfanc553c952016-07-22 15:48:23 +0800582 //Only OpticalTunnelEndPoint has a device
583 if (!(tunnel.src() instanceof OpticalTunnelEndPoint) ||
584 !(tunnel.dst() instanceof OpticalTunnelEndPoint)) {
585 continue;
586 }
587
588 Optional<ElementId> srcElementId = ((OpticalTunnelEndPoint) tunnel.src()).elementId();
589 Optional<ElementId> dstElementId = ((OpticalTunnelEndPoint) tunnel.dst()).elementId();
590 if (!srcElementId.isPresent() || !dstElementId.isPresent()) {
591 continue;
592 }
593 DeviceId srcDeviceId = (DeviceId) srcElementId.get();
594 DeviceId dstDeviceId = (DeviceId) dstElementId.get();
595 if (srcDeviceId.equals(deviceId) || dstDeviceId.equals(deviceId)) {
cheng fan35dc0f22015-06-10 06:02:47 +0800596 count++;
597 }
598 }
599 return count;
600 }
601
Simon Hunt10618f62017-06-15 19:30:52 -0700602 private boolean useDefaultName(String annotName) {
603 return isNullOrEmpty(annotName) || DASH.equals(annotName);
604 }
605
606 private String nameForHost(Host host) {
607 String name = host.annotations().value(AnnotationKeys.NAME);
608 return useDefaultName(name) ? ip(host.ipAddresses()) : name;
609 }
610
Simon Hunta58d8942017-08-11 12:51:14 -0700611 private String glyphForHost(Annotations annot) {
612 String uiType = annot.value(AnnotationKeys.UI_TYPE);
613 return isNullOrEmpty(uiType) ? DEFAULT_HOST_GLYPH : uiType;
Thomas Vachuska329af532015-03-10 02:08:33 -0700614 }
615
Simon Hunta58d8942017-08-11 12:51:14 -0700616 // Generates a property panel model for a host details response
617 protected PropertyPanel hostDetails(HostId hostId) {
618 log.debug("generate prop panel data for host {}", hostId);
619 Host host = services.host().getHost(hostId);
620 Annotations annot = host.annotations();
621 String glyphId = glyphForHost(annot);
622 LionBundle lion = getLionBundle(LION_TOPO);
623
624 PropertyPanel pp = new PropertyPanel(nameForHost(host), glyphId)
625 .navPath(HOST_NAV_PATH)
626 .id(hostId.toString());
627 addHostBasicProps(pp, host, lion);
628 addLocationProps(pp, annot, lion);
629 return pp;
630 }
631
632 private void addHostBasicProps(PropertyPanel pp, Host host, LionBundle lion) {
633 pp.addProp(LPL_FRIENDLY, lion.getSafe(LPL_FRIENDLY), nameForHost(host))
634 .addProp(MAC, lion.getSafe(MAC), host.mac())
635 .addProp(IP, lion.getSafe(IP), host.ipAddresses(), "[\\[\\]]")
636 .addProp(VLAN, lion.getSafe(VLAN), displayVlan(host.vlan(), lion))
637 .addSeparator();
638 }
639
640 private String displayVlan(VlanId vlan, LionBundle lion) {
641 return VlanId.NONE.equals(vlan) ? lion.getSafe(VLAN_NONE) : vlan.toString();
642 }
643
644 // Generates a property panel model for a link details response (edge-link)
645 protected PropertyPanel edgeLinkDetails(HostId hid, ConnectPoint cp) {
646 log.debug("generate prop panel data for edgelink {} {}", hid, cp);
647 LionBundle lion = getLionBundle(LION_TOPO);
648 String title = lion.getSafe("title_edge_link");
649
650 PropertyPanel pp = new PropertyPanel(title, LINK_GLYPH);
651 addLinkHostProps(pp, hid, lion);
652 addLinkCpBProps(pp, cp, lion);
653 return pp;
654 }
655
656 // Generates a property panel model for a link details response (infra-link)
657 protected PropertyPanel infraLinkDetails(ConnectPoint cpA, ConnectPoint cpB) {
658 log.debug("generate prop panel data for infralink {} {}", cpA, cpB);
659 LionBundle lion = getLionBundle(LION_TOPO);
660 String title = lion.getSafe("title_infra_link");
661
662 PropertyPanel pp = new PropertyPanel(title, LINK_GLYPH);
663 addLinkCpAProps(pp, cpA, lion);
664 addLinkCpBProps(pp, cpB, lion);
665 addLinkBackingProps(pp, cpA, cpB, lion);
666 return pp;
667 }
668
669 private void addLinkHostProps(PropertyPanel pp, HostId hostId, LionBundle lion) {
670 Host host = services.host().getHost(hostId);
671
672 pp.addProp(LPL_A_TYPE, lion.getSafe(LPL_A_TYPE), lion.getSafe(HOST))
673 .addProp(LPL_A_ID, lion.getSafe(LPL_A_ID), hostId.toString())
674 .addProp(LPL_A_FRIENDLY, lion.getSafe(LPL_A_FRIENDLY), nameForHost(host))
675 .addSeparator();
676 }
677
678 private void addLinkCpAProps(PropertyPanel pp, ConnectPoint cp, LionBundle lion) {
679 DeviceId did = cp.deviceId();
680
681 pp.addProp(LPL_A_TYPE, lion.getSafe(LPL_A_TYPE), lion.getSafe(DEVICE))
682 .addProp(LPL_A_ID, lion.getSafe(LPL_A_ID), did.toString())
683 .addProp(LPL_A_FRIENDLY, lion.getSafe(LPL_A_FRIENDLY), friendlyDevice(did))
684 .addProp(LPL_A_PORT, lion.getSafe(LPL_A_PORT), cp.port().toLong())
685 .addSeparator();
686 }
687
688 private void addLinkCpBProps(PropertyPanel pp, ConnectPoint cp, LionBundle lion) {
689 DeviceId did = cp.deviceId();
690
691 pp.addProp(LPL_B_TYPE, lion.getSafe(LPL_B_TYPE), lion.getSafe(DEVICE))
692 .addProp(LPL_B_ID, lion.getSafe(LPL_B_ID), did.toString())
693 .addProp(LPL_B_FRIENDLY, lion.getSafe(LPL_B_FRIENDLY), friendlyDevice(did))
694 .addProp(LPL_B_PORT, lion.getSafe(LPL_B_PORT), cp.port().toLong())
695 .addSeparator();
696 }
697
698 private void addLinkBackingProps(PropertyPanel pp, ConnectPoint cpA,
699 ConnectPoint cpB, LionBundle lion) {
700 Link a2b = services.link().getLink(cpA, cpB);
701 Link b2a = services.link().getLink(cpB, cpA);
702
703 pp.addProp(LPL_A2B, lion.getSafe(LPL_A2B), linkPropString(a2b, lion))
704 .addProp(LPL_B2A, lion.getSafe(LPL_B2A), linkPropString(b2a, lion));
705 }
706
707 private String linkPropString(Link link, LionBundle lion) {
708 if (link == null) {
709 return lion.getSafe(LPV_NO_LINK);
710 }
711 return lion.getSafe(link.type()) + SLASH +
712 lion.getSafe(link.state()) + SLASH +
713 lion.getSafe(link.isExpected() ? EXPECTED : NOT_EXPECTED);
714 }
Thomas Vachuska329af532015-03-10 02:08:33 -0700715}