blob: 8fbe81400696dc591340bf6884f12b39ed50c48b [file] [log] [blame]
Steven Burrows57e24e92016-08-04 18:38:24 +01001/*
2 * Copyright 2016-present Open Networking Laboratory
3 *
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,
56 nodes: this.regionNodes.bind(this)
57 });
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 Burrows247ab152017-03-29 13:55:54 +010070 isLoadComplete: function() {
71 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 Burrows68d6f952017-03-10 13:53:35 +000083 layerOrder: this.regionData.layerOrder
Steven Burrowsaf96a212016-12-28 12:57:02 +000084 });
85
Steven Burrows32ce1d92017-04-13 13:18:44 -070086 this.sortMultiLinks();
87
Steven Burrows68d6f952017-03-10 13:53:35 +000088 this.model.set({ subregions: t2sr.createSubRegionCollection(this.regionData.subregions, this) });
89 this.model.set({ devices: t2ds.createDeviceCollection(this.regionData.devices, this) });
90 this.model.set({ hosts: t2hs.createHostCollection(this.regionData.hosts, this) });
91 this.model.set({ peerRegions: t2pr.createCollection(this.peers, this) });
92 this.model.set({ links: t2ls.createLinkCollection(this.regionData.links, this) });
Steven Burrowsaf96a212016-12-28 12:57:02 +000093
94 // Hide Breadcrumbs if there are no subregions configured in the root region
95 if (this.isRootRegion() && !this.model.get('subregions').models.length) {
96 t2bcs.hide();
97 }
Steven Burrows3cc0c372017-02-10 15:15:24 +000098
Steven Burrowsb11a8b82017-03-10 16:00:31 +000099 this.layout.createForceLayout();
Steven Burrows6501de92017-04-12 15:10:34 -0700100 this.displayNoDevs();
Steven Burrowsb43c1a92017-03-07 17:13:28 +0000101 },
102 clear: function () {
Steven Burrowsb43c1a92017-03-07 17:13:28 +0000103 this.regionData = null;
Steven Burrows6de27f42017-03-30 16:21:27 +0100104 this.createEmptyModel();
Steven Burrowsaf96a212016-12-28 12:57:02 +0000105 },
Steven Burrows32ce1d92017-04-13 13:18:44 -0700106 removePort: function (key) {
107 var regex = new RegExp('^[^/]*');
108 return regex.exec(key)[0];
109 },
110 sortMultiLinks: function () {
111 var _this = this,
112 deviceConnections = {};
113
114 _.each(this.regionData.links, function (link) {
Steven Burrows32ce1d92017-04-13 13:18:44 -0700115
Steven Burrows9ce0e402017-04-20 16:39:17 -0400116 var epA = _this.removePort(link.epA),
117 epB = _this.removePort(link.epB),
118 key = epA + '~' + epB,
119 collection = deviceConnections[key] || [],
120 dup = _.find(collection, link);
Steven Burrows32ce1d92017-04-13 13:18:44 -0700121
Steven Burrows0bc66652017-04-20 13:04:10 -0400122 // TODO: Investigate why region contains dup links?!?!
123 // FIXME: This shouldn't be needed - The backend is sending dups
124 // and this is preventing the client thinking its a multilink
Steven Burrows9ce0e402017-04-20 16:39:17 -0400125 if (!dup) {
126 collection.push(link);
Steven Burrows0bc66652017-04-20 13:04:10 -0400127 }
Steven Burrows9ce0e402017-04-20 16:39:17 -0400128
129 deviceConnections[key] = collection;
Steven Burrows32ce1d92017-04-13 13:18:44 -0700130 });
131
Steven Burrows9ce0e402017-04-20 16:39:17 -0400132 _.forIn(deviceConnections, function (collection) {
133 if (collection.length > 1) {
134 _.each(collection, function (link, index) {
Steven Burrows32ce1d92017-04-13 13:18:44 -0700135 link.multiline = {
Steven Burrows9ce0e402017-04-20 16:39:17 -0400136 deviceLinks: collection.length,
Steven Burrows32ce1d92017-04-13 13:18:44 -0700137 index: index
Steven Burrows9ce0e402017-04-20 16:39:17 -0400138 }
139 });
Steven Burrows32ce1d92017-04-13 13:18:44 -0700140 }
Steven Burrows9ce0e402017-04-20 16:39:17 -0400141 })
142
Steven Burrows32ce1d92017-04-13 13:18:44 -0700143 },
Steven Burrowsaf96a212016-12-28 12:57:02 +0000144 isRootRegion: function () {
145 return this.model.get('id') === ROOT;
146 },
147 findNodeById: function (link, id) {
148 if (link.get('type') !== 'UiEdgeLink') {
Steven Burrows32ce1d92017-04-13 13:18:44 -0700149 id = this.removePort(id);
Steven Burrowsaf96a212016-12-28 12:57:02 +0000150 }
151 return this.model.get('devices').get(id) ||
152 this.model.get('hosts').get(id) ||
153 this.model.get('subregions').get(id);
154 },
155 regionNodes: function () {
Steven Burrowsaf96a212016-12-28 12:57:02 +0000156 if (this.model) {
157 return [].concat(
158 this.model.get('devices').models,
159 this.model.get('hosts').models,
Steven Burrows68d6f952017-03-10 13:53:35 +0000160 this.model.get('subregions').models,
161 this.model.get('peerRegions').models
Steven Burrowsaf96a212016-12-28 12:57:02 +0000162 );
163 }
Steven Burrowsaf96a212016-12-28 12:57:02 +0000164 return [];
165 },
166 regionLinks: function () {
Steven Burrowsb11a8b82017-03-10 16:00:31 +0000167 return this.model.get('links').models;
Steven Burrowsaf96a212016-12-28 12:57:02 +0000168 },
Steven Burrowsb15a3942017-01-17 17:25:04 +0000169 getLink: function (linkId) {
170 return this.model.get('links').get(linkId);
171 },
Steven Burrows42eb9e22017-02-06 14:20:24 +0000172 getDevice: function (deviceId) {
173 return this.model.get('devices').get(deviceId);
174 },
Steven Burrowsaf96a212016-12-28 12:57:02 +0000175 filterRegionNodes: function (predicate) {
176 var nodes = this.regionNodes();
177 return _.filter(nodes, predicate);
178 },
179 deselectAllNodes: function () {
180 var selected = this.filterRegionNodes(function (node) {
181 return node.get('selected', true);
182 });
183
184 if (selected.length) {
185
186 selected.forEach(function (node) {
187 node.deselect();
188 });
189
190 t2dps().el.hide();
191 return true;
192 }
193
194 return false;
195 },
196 deselectLink: function () {
197 var selected = _.filter(this.regionLinks(), function (link) {
198 return link.get('selected', true);
199 });
200
201 if (selected.length) {
202
203 selected.forEach(function (link) {
204 link.deselect();
205 });
206
207 t2dps().el.hide();
208 return true;
209 }
210
211 return false;
Steven Burrowsb15a3942017-01-17 17:25:04 +0000212 },
Steven Burrowsaea509d2017-04-12 14:17:47 -0700213 toggleHosts: function () {
214 var state = this.lookupPrefState('hosts');
215 this.updatePrefState('hosts', !state);
Steven Burrowsb15a3942017-01-17 17:25:04 +0000216
Steven Burrowsaea509d2017-04-12 14:17:47 -0700217 _.each(this.model.get('hosts').models, function (host) {
218 host.setVisibility();
219 });
220
221 _.each(this.model.get('links').models, function (link) {
222 link.setVisibility();
223 });
Steven Burrows02e67f42017-04-13 13:59:43 -0700224
225 return !state;
226 },
227 toggleOfflineDevices: function () {
228 var state = this.lookupPrefState('offline_devices');
229 this.updatePrefState('offline_devices', !state);
230 _.each(this.regionNodes(), function (node) {
231 node.setOfflineVisibility();
232 });
233
234 return !state;
Steven Burrowsaea509d2017-04-12 14:17:47 -0700235 },
Steven Burrowsb15a3942017-01-17 17:25:04 +0000236 update: function (event) {
Steven Burrows42eb9e22017-02-06 14:20:24 +0000237
Steven Burrows6de27f42017-03-30 16:21:27 +0100238 if (!this.isLoadComplete()){
239 this.layout.createForceLayout();
240 }
241
Steven Burrowsb15a3942017-01-17 17:25:04 +0000242 if (this[event.type]) {
243 this[event.type](event);
Steven Burrowsb15a3942017-01-17 17:25:04 +0000244 } else {
245 $log.error("Unhanded topology update", event);
246 }
Steven Burrows42eb9e22017-02-06 14:20:24 +0000247
248 this.layout.update()
Steven Burrows6501de92017-04-12 15:10:34 -0700249 this.displayNoDevs();
250 },
251 displayNoDevs: function () {
252 if (this.regionNodes().length > 0) {
253 t2ndcs.hide();
254 } else {
255 t2ndcs.show();
256 }
Steven Burrowsb15a3942017-01-17 17:25:04 +0000257 },
258
259 // Topology update event handlers
260 LINK_ADDED_OR_UPDATED: function (event) {
Steven Burrows6de27f42017-03-30 16:21:27 +0100261
262 var regionLinks = this.model.get('links'),
263 device;
264
265 if (!regionLinks) {
266 this.model.set({ links: t2ls.createLinkCollection([], this) })
267 }
268
Steven Burrowsb15a3942017-01-17 17:25:04 +0000269 if (event.memo === 'added') {
270 var link = this.model.get('links').add(event.data);
271 link.createLink();
Steven Burrows6de27f42017-03-30 16:21:27 +0100272 $log.debug('Added Link', link);
Steven Burrowsb15a3942017-01-17 17:25:04 +0000273 }
274 },
275 LINK_REMOVED: function (event) {
276 var link = this.getLink(event.subject);
277 link.remove();
Steven Burrows42eb9e22017-02-06 14:20:24 +0000278 this.model.get('links').remove(link);
279 },
280 DEVICE_ADDED_OR_UPDATED: function (event) {
281
Steven Burrows6de27f42017-03-30 16:21:27 +0100282 var regionDevices = this.model.get('devices'),
283 device;
284
285 if (!regionDevices) {
286 this.model.set({ devices: t2ds.createDeviceCollection([], this) })
287 }
Steven Burrows42eb9e22017-02-06 14:20:24 +0000288
289 if (event.memo === 'added') {
290 device = this.model.get('devices').add(event.data);
Simon Hunteb3cf542017-02-10 13:18:41 -0800291 $log.debug('Added device', device);
Steven Burrows42eb9e22017-02-06 14:20:24 +0000292 } else if (event.memo === 'updated') {
293 device = this.getDevice(event.subject);
294 device.set(event.data);
295 }
296 },
297 DEVICE_REMOVED: function (event) {
298 device.remove();
Steven Burrows6de27f42017-03-30 16:21:27 +0100299 },
300 HOST_ADDED_OR_UPDATED: function (event) {
301 var regionHosts = this.model.get('hosts'),
302 host;
303
304 if (!regionHosts) {
305 this.model.set({ hosts: t2hs.createHostCollection([], this) })
306 }
307
308 if (event.memo === 'added') {
309 host = this.model.get('hosts').add(event.data);
310 $log.debug('Added host', host);
311 }
312 },
313 REGION_ADDED_OR_UPDATED: function (event) {
314 var regionSubRegions = this.model.get('subregions'),
315 region;
316
317 if (!regionSubRegions) {
318 this.model.set({ subregions: t2sr.createSubRegionCollection([], this) })
319 }
320
321 if (event.memo === 'added') {
322 region = this.model.get('subregions').add(event.data);
323 $log.debug('Added region', region);
324 }
Steven Burrowsaf96a212016-12-28 12:57:02 +0000325 }
326 });
327
328 function getInstance() {
329 return instance || new Region();
330 }
331
332 return getInstance();
Steven Burrowsb15a3942017-01-17 17:25:04 +0000333
Steven Burrows57e24e92016-08-04 18:38:24 +0100334 }]);
335
336})();