blob: d5225d23a0247a21b353b6b6dc2ba0886396c661 [file] [log] [blame]
Steven Burrows57e24e92016-08-04 18:38:24 +01001/*
Brian O'Connora09fe5b2017-08-03 21:12:30 -07002 * Copyright 2016-present Open Networking Foundation
Steven Burrows57e24e92016-08-04 18:38:24 +01003 *
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 */
16
17/*
18 ONOS GUI -- Topology Region Module.
19 Module that holds the current region in memory
20 */
21
22(function () {
23 'use strict';
24
Steven Burrows6deb4ce2016-08-26 16:06:23 +010025 // Injected Services
Steven Burrowsdfa52b02016-09-02 13:50:43 +010026 var Model;
Steven Burrows57e24e92016-08-04 18:38:24 +010027
Steven Burrowsdfa52b02016-09-02 13:50:43 +010028 // Internal
Steven Burrowsb15a3942017-01-17 17:25:04 +000029 var instance;
Steven Burrows57e24e92016-08-04 18:38:24 +010030
Steven Burrows8ea5dea2016-12-27 13:28:41 +000031 // 'static' vars
32 var ROOT = '(root)';
33
Steven Burrows57e24e92016-08-04 18:38:24 +010034 angular.module('ovTopo2')
Steven Burrowsaf96a212016-12-28 12:57:02 +000035 .factory('Topo2RegionService', [
36 '$log', 'Topo2Model', 'Topo2SubRegionService', 'Topo2DeviceService',
Steven Burrows1aa4f582016-12-13 15:05:41 -050037 'Topo2HostService', 'Topo2LinkService', 'Topo2ZoomService', 'Topo2DetailsPanelService',
Steven Burrows86b74fc2017-02-22 00:15:16 +000038 'Topo2BreadcrumbService', 'Topo2ViewController', 'Topo2SpriteLayerService', 'Topo2MapService',
Steven Burrows6501de92017-04-12 15:10:34 -070039 'Topo2MapConfigService', 'Topo2PeerRegionService', 'Topo2NoDevicesConnectedService',
Steven Burrowsb11a8b82017-03-10 16:00:31 +000040 function ($log, _Model_, t2sr, t2ds, t2hs, t2ls, t2zs, t2dps, t2bcs, ViewController,
Steven Burrows6501de92017-04-12 15:10:34 -070041 t2sls, t2ms, t2mcs, t2pr, t2ndcs) {
Steven Burrows57e24e92016-08-04 18:38:24 +010042
Steven Burrowsdfa52b02016-09-02 13:50:43 +010043 Model = _Model_;
Steven Burrows57e24e92016-08-04 18:38:24 +010044
Steven Burrowsaf96a212016-12-28 12:57:02 +000045 var Region = ViewController.extend({
46 initialize: function () {
47 instance = this;
48 this.model = null;
Steven Burrows68d6f952017-03-10 13:53:35 +000049
Steven Burrowsb43c1a92017-03-07 17:13:28 +000050 this.bgRendered = false;
Steven Burrows68d6f952017-03-10 13:53:35 +000051 this.regionData = null;
52 this.peers = null;
Steven Burrowsb11a8b82017-03-10 16:00:31 +000053
54 var RegionModel = Model.extend({
55 findNodeById: this.findNodeById,
Steven Burrows1c2a9682017-07-14 16:52:46 +010056 nodes: this.regionNodes.bind(this),
Steven Burrowsb11a8b82017-03-10 16:00:31 +000057 });
58
59 this.model = new RegionModel();
Steven Burrows6de27f42017-03-30 16:21:27 +010060 this.createEmptyModel();
61
62 },
63 createEmptyModel: function () {
64 this.model.set({ subregions: t2sr.createSubRegionCollection([], this) });
65 this.model.set({ devices: t2ds.createDeviceCollection([], this) });
66 this.model.set({ hosts: t2hs.createHostCollection([], this) });
67 this.model.set({ peerRegions: t2pr.createCollection([], this) });
68 this.model.set({ links: t2ls.createLinkCollection([], this) });
Steven Burrowsaf96a212016-12-28 12:57:02 +000069 },
Steven Burrows1c2a9682017-07-14 16:52:46 +010070 isLoadComplete: function () {
Steven Burrows247ab152017-03-29 13:55:54 +010071 return this.bgRendered && this.regionData && this.peers;
72 },
Steven Burrows68d6f952017-03-10 13:53:35 +000073 loaded: function (key, value) {
74 this[key] = value;
Steven Burrows247ab152017-03-29 13:55:54 +010075 if (this.isLoadComplete()) {
Steven Burrowsb43c1a92017-03-07 17:13:28 +000076 this.startRegion();
77 }
78 },
79 startRegion: function () {
Steven Burrows57e24e92016-08-04 18:38:24 +010080
Steven Burrowsaf96a212016-12-28 12:57:02 +000081 this.model.set({
Steven Burrowsb11a8b82017-03-10 16:00:31 +000082 id: this.regionData.id,
Steven Burrows1c2a9682017-07-14 16:52:46 +010083 layerOrder: this.regionData.layerOrder,
Steven Burrowsaf96a212016-12-28 12:57:02 +000084 });
85
Steven Burrows32ce1d92017-04-13 13:18:44 -070086 this.sortMultiLinks();
Steven Burrows4a1b5532017-04-28 11:08:07 -040087 this.assignPeerLocations();
Steven Burrows32ce1d92017-04-13 13:18:44 -070088
Steven Burrows2b29ca42017-05-12 10:58:05 -040089 // TODO: RegionLinks are dublicated in JSON Payload
90
Steven Burrows68d6f952017-03-10 13:53:35 +000091 this.model.set({ subregions: t2sr.createSubRegionCollection(this.regionData.subregions, this) });
92 this.model.set({ devices: t2ds.createDeviceCollection(this.regionData.devices, this) });
93 this.model.set({ hosts: t2hs.createHostCollection(this.regionData.hosts, this) });
94 this.model.set({ peerRegions: t2pr.createCollection(this.peers, this) });
95 this.model.set({ links: t2ls.createLinkCollection(this.regionData.links, this) });
Steven Burrowsaf96a212016-12-28 12:57:02 +000096
97 // Hide Breadcrumbs if there are no subregions configured in the root region
98 if (this.isRootRegion() && !this.model.get('subregions').models.length) {
99 t2bcs.hide();
100 }
Steven Burrows3cc0c372017-02-10 15:15:24 +0000101
Steven Burrowsb11a8b82017-03-10 16:00:31 +0000102 this.layout.createForceLayout();
Steven Burrows6501de92017-04-12 15:10:34 -0700103 this.displayNoDevs();
Steven Burrowsb43c1a92017-03-07 17:13:28 +0000104 },
105 clear: function () {
Steven Burrowsb43c1a92017-03-07 17:13:28 +0000106 this.regionData = null;
Steven Burrows6de27f42017-03-30 16:21:27 +0100107 this.createEmptyModel();
Steven Burrowsaf96a212016-12-28 12:57:02 +0000108 },
Steven Burrows32ce1d92017-04-13 13:18:44 -0700109 removePort: function (key) {
110 var regex = new RegExp('^[^/]*');
111 return regex.exec(key)[0];
112 },
Steven Burrows4a1b5532017-04-28 11:08:07 -0400113 assignPeerLocations: function () {
114 var _this = this;
115 _.each(this.regionData.peerLocations, function (location, id) {
116 _.each(_this.peers, function (peer) {
117 if (peer.id === id) {
118 peer.location = location;
119 }
Steven Burrows1c2a9682017-07-14 16:52:46 +0100120 });
Steven Burrows4a1b5532017-04-28 11:08:07 -0400121 });
122 },
Steven Burrows32ce1d92017-04-13 13:18:44 -0700123 sortMultiLinks: function () {
124 var _this = this,
125 deviceConnections = {};
126
127 _.each(this.regionData.links, function (link) {
Steven Burrows32ce1d92017-04-13 13:18:44 -0700128
Steven Burrows9ce0e402017-04-20 16:39:17 -0400129 var epA = _this.removePort(link.epA),
130 epB = _this.removePort(link.epB),
131 key = epA + '~' + epB,
132 collection = deviceConnections[key] || [],
133 dup = _.find(collection, link);
Steven Burrows32ce1d92017-04-13 13:18:44 -0700134
Steven Burrows0bc66652017-04-20 13:04:10 -0400135 // TODO: Investigate why region contains dup links?!?!
136 // FIXME: This shouldn't be needed - The backend is sending dups
137 // and this is preventing the client thinking its a multilink
Steven Burrows9ce0e402017-04-20 16:39:17 -0400138 if (!dup) {
139 collection.push(link);
Steven Burrows0bc66652017-04-20 13:04:10 -0400140 }
Steven Burrows9ce0e402017-04-20 16:39:17 -0400141
142 deviceConnections[key] = collection;
Steven Burrows32ce1d92017-04-13 13:18:44 -0700143 });
144
Steven Burrows9ce0e402017-04-20 16:39:17 -0400145 _.forIn(deviceConnections, function (collection) {
146 if (collection.length > 1) {
147 _.each(collection, function (link, index) {
Steven Burrows32ce1d92017-04-13 13:18:44 -0700148 link.multiline = {
Steven Burrows9ce0e402017-04-20 16:39:17 -0400149 deviceLinks: collection.length,
Steven Burrows1c2a9682017-07-14 16:52:46 +0100150 index: index,
151 };
Steven Burrows9ce0e402017-04-20 16:39:17 -0400152 });
Steven Burrows32ce1d92017-04-13 13:18:44 -0700153 }
Steven Burrows1c2a9682017-07-14 16:52:46 +0100154 });
Steven Burrows32ce1d92017-04-13 13:18:44 -0700155 },
Steven Burrowsaf96a212016-12-28 12:57:02 +0000156 isRootRegion: function () {
157 return this.model.get('id') === ROOT;
158 },
159 findNodeById: function (link, id) {
160 if (link.get('type') !== 'UiEdgeLink') {
Steven Burrows32ce1d92017-04-13 13:18:44 -0700161 id = this.removePort(id);
Steven Burrowsaf96a212016-12-28 12:57:02 +0000162 }
163 return this.model.get('devices').get(id) ||
164 this.model.get('hosts').get(id) ||
165 this.model.get('subregions').get(id);
166 },
167 regionNodes: function () {
Steven Burrowsaf96a212016-12-28 12:57:02 +0000168 if (this.model) {
169 return [].concat(
170 this.model.get('devices').models,
171 this.model.get('hosts').models,
Steven Burrows68d6f952017-03-10 13:53:35 +0000172 this.model.get('subregions').models,
173 this.model.get('peerRegions').models
Steven Burrowsaf96a212016-12-28 12:57:02 +0000174 );
175 }
Steven Burrowsaf96a212016-12-28 12:57:02 +0000176 return [];
177 },
178 regionLinks: function () {
Steven Burrowsb11a8b82017-03-10 16:00:31 +0000179 return this.model.get('links').models;
Steven Burrowsaf96a212016-12-28 12:57:02 +0000180 },
Steven Burrowsb15a3942017-01-17 17:25:04 +0000181 getLink: function (linkId) {
182 return this.model.get('links').get(linkId);
183 },
Steven Burrows42eb9e22017-02-06 14:20:24 +0000184 getDevice: function (deviceId) {
185 return this.model.get('devices').get(deviceId);
186 },
Steven Burrowsaf96a212016-12-28 12:57:02 +0000187 filterRegionNodes: function (predicate) {
188 var nodes = this.regionNodes();
189 return _.filter(nodes, predicate);
190 },
191 deselectAllNodes: function () {
192 var selected = this.filterRegionNodes(function (node) {
193 return node.get('selected', true);
194 });
195
196 if (selected.length) {
197
198 selected.forEach(function (node) {
199 node.deselect();
200 });
201
202 t2dps().el.hide();
203 return true;
204 }
205
206 return false;
207 },
208 deselectLink: function () {
209 var selected = _.filter(this.regionLinks(), function (link) {
210 return link.get('selected', true);
211 });
212
213 if (selected.length) {
214
215 selected.forEach(function (link) {
216 link.deselect();
217 });
218
219 t2dps().el.hide();
220 return true;
221 }
222
223 return false;
Steven Burrowsb15a3942017-01-17 17:25:04 +0000224 },
Steven Burrowsaea509d2017-04-12 14:17:47 -0700225 toggleHosts: function () {
226 var state = this.lookupPrefState('hosts');
227 this.updatePrefState('hosts', !state);
Steven Burrowsb15a3942017-01-17 17:25:04 +0000228
Steven Burrowsaea509d2017-04-12 14:17:47 -0700229 _.each(this.model.get('hosts').models, function (host) {
230 host.setVisibility();
231 });
232
233 _.each(this.model.get('links').models, function (link) {
234 link.setVisibility();
235 });
Steven Burrows02e67f42017-04-13 13:59:43 -0700236
237 return !state;
238 },
239 toggleOfflineDevices: function () {
240 var state = this.lookupPrefState('offline_devices');
241 this.updatePrefState('offline_devices', !state);
242 _.each(this.regionNodes(), function (node) {
243 node.setOfflineVisibility();
244 });
245
246 return !state;
Steven Burrowsaea509d2017-04-12 14:17:47 -0700247 },
Steven Burrowsb15a3942017-01-17 17:25:04 +0000248 update: function (event) {
Steven Burrows42eb9e22017-02-06 14:20:24 +0000249
Steven Burrows1c2a9682017-07-14 16:52:46 +0100250 if (!this.isLoadComplete()) {
Steven Burrows6de27f42017-03-30 16:21:27 +0100251 this.layout.createForceLayout();
252 }
253
Steven Burrowsb15a3942017-01-17 17:25:04 +0000254 if (this[event.type]) {
255 this[event.type](event);
Steven Burrowsb15a3942017-01-17 17:25:04 +0000256 } else {
Steven Burrows1c2a9682017-07-14 16:52:46 +0100257 $log.error('Unhanded topology update', event);
Steven Burrowsb15a3942017-01-17 17:25:04 +0000258 }
Steven Burrows42eb9e22017-02-06 14:20:24 +0000259
Steven Burrows1c2a9682017-07-14 16:52:46 +0100260 this.layout.update();
Steven Burrows6501de92017-04-12 15:10:34 -0700261 this.displayNoDevs();
262 },
263 displayNoDevs: function () {
264 if (this.regionNodes().length > 0) {
265 t2ndcs.hide();
266 } else {
267 t2ndcs.show();
268 }
Steven Burrowsb15a3942017-01-17 17:25:04 +0000269 },
270
271 // Topology update event handlers
272 LINK_ADDED_OR_UPDATED: function (event) {
Steven Burrows6de27f42017-03-30 16:21:27 +0100273
Steven Burrows1c2a9682017-07-14 16:52:46 +0100274 var regionLinks = this.model.get('links');
Steven Burrows6de27f42017-03-30 16:21:27 +0100275
276 if (!regionLinks) {
Steven Burrows1c2a9682017-07-14 16:52:46 +0100277 this.model.set({ links: t2ls.createLinkCollection([], this) });
Steven Burrows6de27f42017-03-30 16:21:27 +0100278 }
279
Steven Burrowsb15a3942017-01-17 17:25:04 +0000280 if (event.memo === 'added') {
281 var link = this.model.get('links').add(event.data);
282 link.createLink();
Steven Burrows6de27f42017-03-30 16:21:27 +0100283 $log.debug('Added Link', link);
Steven Burrowsb15a3942017-01-17 17:25:04 +0000284 }
285 },
286 LINK_REMOVED: function (event) {
287 var link = this.getLink(event.subject);
288 link.remove();
Steven Burrows42eb9e22017-02-06 14:20:24 +0000289 this.model.get('links').remove(link);
290 },
291 DEVICE_ADDED_OR_UPDATED: function (event) {
292
Steven Burrows6de27f42017-03-30 16:21:27 +0100293 var regionDevices = this.model.get('devices'),
294 device;
295
296 if (!regionDevices) {
Steven Burrows1c2a9682017-07-14 16:52:46 +0100297 this.model.set({ devices: t2ds.createDeviceCollection([], this) });
Steven Burrows6de27f42017-03-30 16:21:27 +0100298 }
Steven Burrows42eb9e22017-02-06 14:20:24 +0000299
300 if (event.memo === 'added') {
301 device = this.model.get('devices').add(event.data);
Simon Hunteb3cf542017-02-10 13:18:41 -0800302 $log.debug('Added device', device);
Steven Burrows42eb9e22017-02-06 14:20:24 +0000303 } else if (event.memo === 'updated') {
304 device = this.getDevice(event.subject);
305 device.set(event.data);
306 }
307 },
308 DEVICE_REMOVED: function (event) {
309 device.remove();
Steven Burrows6de27f42017-03-30 16:21:27 +0100310 },
311 HOST_ADDED_OR_UPDATED: function (event) {
312 var regionHosts = this.model.get('hosts'),
313 host;
314
315 if (!regionHosts) {
Steven Burrows1c2a9682017-07-14 16:52:46 +0100316 this.model.set({ hosts: t2hs.createHostCollection([], this) });
Steven Burrows6de27f42017-03-30 16:21:27 +0100317 }
318
319 if (event.memo === 'added') {
320 host = this.model.get('hosts').add(event.data);
321 $log.debug('Added host', host);
322 }
323 },
324 REGION_ADDED_OR_UPDATED: function (event) {
325 var regionSubRegions = this.model.get('subregions'),
326 region;
327
328 if (!regionSubRegions) {
Steven Burrows1c2a9682017-07-14 16:52:46 +0100329 this.model.set({ subregions: t2sr.createSubRegionCollection([], this) });
Steven Burrows6de27f42017-03-30 16:21:27 +0100330 }
331
332 if (event.memo === 'added') {
333 region = this.model.get('subregions').add(event.data);
334 $log.debug('Added region', region);
335 }
Steven Burrows1c2a9682017-07-14 16:52:46 +0100336 },
Steven Burrowsaf96a212016-12-28 12:57:02 +0000337 });
338
339 function getInstance() {
340 return instance || new Region();
341 }
342
343 return getInstance();
Steven Burrowsb15a3942017-01-17 17:25:04 +0000344
Steven Burrows57e24e92016-08-04 18:38:24 +0100345 }]);
346
347})();