blob: c3c49cfe40c14d0e8c906ce936ef377c006ae3ae [file] [log] [blame]
Tyler Scottcdfcde82015-09-14 16:13:29 -06001//Run when the document loads AND we have the config loaded.
Tyler Scott66a965d2016-02-29 15:41:33 -07002(function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -06003 "use strict";
4 var config;
Tyler Scott384f37a2016-05-05 16:09:05 -06005 var conversions;
Tyler Scottcdfcde82015-09-14 16:13:29 -06006 Promise.all([
Tyler Scott66a965d2016-02-29 15:41:33 -07007 new Promise(function(resolve, reject) {
8 $.ajax('../config.json').done(function(data) {
9 config = data;
10 resolve();
11 }).fail(function() {
12 console.error("Failed to get config.");
13 ga('send', 'event', 'error', 'config');
14 reject();
15 });
Tyler Scott384f37a2016-05-05 16:09:05 -060016 }),
Tyler Scott66a965d2016-02-29 15:41:33 -070017 new Promise(function(resolve, reject) {
18 var timeout = setTimeout(function() {
19 console.error("Document never loaded? Something bad has happened!");
20 reject();
21 }, 10000);
22 $(function() {
23 clearTimeout(timeout);
24 resolve();
25 });
Tyler Scott384f37a2016-05-05 16:09:05 -060026 }),
27 new Promise(function(resolve, reject) {
28 $.getJSON('../conversions.json').done(function(data) {
29 conversions = data;
30 resolve();
31 }).fail(function(){
32 console.error("Failed to get conversions.");
33 ga('send', 'event', 'error', 'config');
34 //reject(); We will continue anyways. We don't need this functionality.
35 conversions = {};
36 resolve();
37 });
38 })
Tyler Scott66a965d2016-02-29 15:41:33 -070039 ]).then(function() {
Tyler Scott2118a412015-10-29 15:17:05 -060040
Tyler Scott66a965d2016-02-29 15:41:33 -070041 var getParameterByName = function(name) {
42 name = name.replace(/[\[]/, "\\[").replace(/[\]]/, "\\]");
43 var regex = new RegExp("[\\?&]" + name + "=([^&#]*)");
44 var results = regex.exec(location.search);
45 return results === null ? "" : decodeURIComponent(results[1].replace(/\+/g, " "));
46 }
Tyler Scott2118a412015-10-29 15:17:05 -060047
48 //Overwrite config if present. Any failure will just cause this to be skipped.
Tyler Scott66a965d2016-02-29 15:41:33 -070049 try {
Tyler Scott2118a412015-10-29 15:17:05 -060050 var configParam = JSON.parse(getParameterByName('config'));
51 config = jQuery.extend(true, config, configParam);
Tyler Scott66a965d2016-02-29 15:41:33 -070052 } catch (e) {
Tyler Scott2118a412015-10-29 15:17:05 -060053 console.warn("Failure in config overwrite, skipping.", e);
54 }
55
Tyler Scott384f37a2016-05-05 16:09:05 -060056 new Atmos(config, conversions);
Tyler Scott66a965d2016-02-29 15:41:33 -070057 }, function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -060058 console.error("Failed to initialize!");
59 ga('send', 'event', 'error', 'init');
60 });
61})();
62
Tyler Scott66a965d2016-02-29 15:41:33 -070063var Atmos = (function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -060064 "use strict";
65
66 var closeButton = '<button type="button" class="close" data-dismiss="alert" aria-label="Close"><span aria-hidden="true">&times;</span></button>';
67
Tyler Scott66a965d2016-02-29 15:41:33 -070068 var guid = function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -060069 var d = new Date().getTime();
70 var uuid = 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) {
Tyler Scott66a965d2016-02-29 15:41:33 -070071 var r = (d + Math.random() * 16) % 16 | 0;
72 d = Math.floor(d / 16);
73 return (c == 'x' ? r : (r & 0x3 | 0x8)).toString(16);
Tyler Scottcdfcde82015-09-14 16:13:29 -060074 });
75 return uuid;
Tyler Scott66a965d2016-02-29 15:41:33 -070076 }
Tyler Scottcdfcde82015-09-14 16:13:29 -060077
78 /**
79 * Atmos
80 * @version 2.0
Tyler Scott66a965d2016-02-29 15:41:33 -070081
Tyler Scottcdfcde82015-09-14 16:13:29 -060082 * Configures an Atmos object. This manages the atmos interface.
Tyler Scott66a965d2016-02-29 15:41:33 -070083
Tyler Scottcdfcde82015-09-14 16:13:29 -060084 * @constructor
85 * @param {string} catalog - NDN path
86 * @param {Object} config - Object of configuration options for a Face.
87 */
Tyler Scott384f37a2016-05-05 16:09:05 -060088 var Atmos = function(config, conversions) {
Tyler Scottcdfcde82015-09-14 16:13:29 -060089
90 //Internal variables.
91 this.results = [];
92 this.resultCount = Infinity;
Tyler Scott66a965d2016-02-29 15:41:33 -070093 this.name = null ;
Tyler Scottcdfcde82015-09-14 16:13:29 -060094 this.page = 0;
95 this.resultsPerPage = 25;
96 this.retrievedSegments = 0;
97
98 //Config/init
99 this.config = config;
Tyler Scott384f37a2016-05-05 16:09:05 -0600100 this.conversions = conversions;
Tyler Scottcdfcde82015-09-14 16:13:29 -0600101
102 this.catalog = config['global']['catalogPrefix'];
Chengyu Fan7b978f82015-12-09 17:03:23 -0700103 this.catalogPrefix = new Name(this.catalog);
Tyler Scottcdfcde82015-09-14 16:13:29 -0600104
105 this.face = new Face(config['global']['faceConfig']);
106
107 //Easy access dom variables
108 this.categories = $('#side-menu');
109 this.resultTable = $('#resultTable');
110 this.filters = $('#filters');
111 this.searchInput = $('#search');
112 this.searchBar = $('#searchBar');
113 this.searchButton = $('#searchButton');
114 this.resultMenu = $('.resultMenu');
115 this.alerts = $('#alerts');
116 this.requestForm = $('#requestForm');
117
118 var scope = this;
119
Tyler Scott66a965d2016-02-29 15:41:33 -0700120 $('.requestSelectedButton').click(function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600121 ga('send', 'event', 'button', 'click', 'request');
122 scope.request(scope.resultTable.find('.resultSelector:checked:not([disabled])').parent().parent());
123 });
124
125 this.filterSetup();
126
127 //Init autocomplete
Tyler Scott66a965d2016-02-29 15:41:33 -0700128 this.searchInput.autoComplete(function(field, callback) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600129 ga('send', 'event', 'search', 'autocomplete');
Tyler Scott66a965d2016-02-29 15:41:33 -0700130 scope.autoComplete(field, function(data) {
Tyler Scott94458992015-09-24 14:16:28 -0700131 var list = data.next;
132 var last = data.lastComponent === true;
Tyler Scott66a965d2016-02-29 15:41:33 -0700133 callback(list.map(function(element) {
134 return field + element + (last ? "/" : "");
135 //Don't add trailing slash for last component.
Tyler Scottcdfcde82015-09-14 16:13:29 -0600136 }));
137 });
138 });
139
140 //Handle search
Tyler Scott66a965d2016-02-29 15:41:33 -0700141 this.searchBar.submit(function(e) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600142 ga('send', 'event', 'search', 'submit');
143 e.preventDefault();
Tyler Scott66a965d2016-02-29 15:41:33 -0700144 if (scope.searchInput.val().length === 0) {
145 if (!scope.searchBar.hasClass('has-error')) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600146 scope.searchBar.addClass('has-error').append('<span class="help-block">Search path is required!</span>');
147 }
148 return;
149 } else {
Tyler Scott66a965d2016-02-29 15:41:33 -0700150 scope.searchBar.removeClass('has-error').find('.help-block').fadeOut(function() {
151 $(this).remove()
152 });
Tyler Scottcdfcde82015-09-14 16:13:29 -0600153 }
154 scope.pathSearch();
155 });
156
Tyler Scott66a965d2016-02-29 15:41:33 -0700157 this.searchButton.click(function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600158 console.log("Search Button Pressed");
159 ga('send', 'event', 'button', 'click', 'search');
160 scope.search();
161 });
162
163 //Result navigation handlers
Tyler Scott66a965d2016-02-29 15:41:33 -0700164 this.resultMenu.find('.next').click(function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600165 ga('send', 'event', 'button', 'click', 'next');
Tyler Scott66a965d2016-02-29 15:41:33 -0700166 if (!$(this).hasClass('disabled')) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600167 scope.getResults(scope.page + 1);
168 }
169 });
Tyler Scott66a965d2016-02-29 15:41:33 -0700170 this.resultMenu.find('.previous').click(function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600171 ga('send', 'event', 'button', 'click', 'previous');
Tyler Scott66a965d2016-02-29 15:41:33 -0700172 if (!$(this).hasClass('disabled')) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600173 scope.getResults(scope.page - 1);
174 }
175 });
Tyler Scott66a965d2016-02-29 15:41:33 -0700176 this.resultMenu.find('.clearResults').click(function() {
Tyler Scotte8dac702015-10-13 14:33:25 -0600177 ga('send', 'event', 'button', 'click', 'resultClear');
178 scope.clearResults();
Tyler Scott66a965d2016-02-29 15:41:33 -0700179 $('#results').fadeOut(function() {
Tyler Scotte8dac702015-10-13 14:33:25 -0600180 $(this).addClass('hidden');
181 });
182 });
Tyler Scottcdfcde82015-09-14 16:13:29 -0600183
184 //Change the number of results per page handler
Tyler Scott66a965d2016-02-29 15:41:33 -0700185 var rpps = $('.resultsPerPageSelector').click(function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600186
187 var t = $(this);
188
Tyler Scott66a965d2016-02-29 15:41:33 -0700189 if (t.hasClass('active')) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600190 return;
191 }
192
193 rpps.find('.active').removeClass('active');
194 t.addClass('active');
195 scope.resultsPerPage = Number(t.text());
Tyler Scott66a965d2016-02-29 15:41:33 -0700196 scope.getResults(0);
197 //Force return to page 1;
Tyler Scottcdfcde82015-09-14 16:13:29 -0600198
199 });
200
201 //Init tree search
Tyler Scott66a965d2016-02-29 15:41:33 -0700202 $('#treeSearch div').treeExplorer(function(path, callback) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600203 console.log("Tree Explorer request", path);
204 ga('send', 'event', 'tree', 'request');
Tyler Scott66a965d2016-02-29 15:41:33 -0700205 scope.autoComplete(path, function(data) {
Tyler Scottb59e6de2015-09-18 14:46:30 -0600206 var list = data.next;
207 var last = (data.lastComponent === true);
Tyler Scottbb42ed22015-10-21 17:02:56 -0600208
209 if (last) {
210 console.log("Redirecting last element request to a search.");
211 scope.clearResults();
Tyler Scott66a965d2016-02-29 15:41:33 -0700212 scope.query(scope.catalog, {
213 '??': path
214 },
215 function(interest, data) {
Tyler Scottbb42ed22015-10-21 17:02:56 -0600216 console.log("Search response", interest, data);
Chengyu Fan7b978f82015-12-09 17:03:23 -0700217 scope.name = interest.getName();
Tyler Scottbb42ed22015-10-21 17:02:56 -0600218 scope.getResults(0);
Tyler Scott66a965d2016-02-29 15:41:33 -0700219 }, function(interest) {
Tyler Scottbb42ed22015-10-21 17:02:56 -0600220 console.warn("Failed to retrieve final component.", interest, path);
221 scope.createAlert("Failed to request final component. " + path + " See console for details.");
222 });
Tyler Scott66a965d2016-02-29 15:41:33 -0700223 return;
224 //Don't call the callback
Tyler Scottbb42ed22015-10-21 17:02:56 -0600225 }
226
Tyler Scottcdfcde82015-09-14 16:13:29 -0600227 console.log("Autocomplete response", list);
Tyler Scott66a965d2016-02-29 15:41:33 -0700228 callback(list.map(function(element) {
229 return (path == "/" ? "/" : "") + element + "/";
Tyler Scottcdfcde82015-09-14 16:13:29 -0600230 }));
Chengyu Fan7b978f82015-12-09 17:03:23 -0700231 });
Tyler Scottcdfcde82015-09-14 16:13:29 -0600232 });
233
Tyler Scott66a965d2016-02-29 15:41:33 -0700234 $('#treeSearch').on('click', '.treeSearch', function() {
Tyler Scottb59e6de2015-09-18 14:46:30 -0600235 var t = $(this);
236
237 scope.clearResults();
238
239 var path = t.parent().parent().attr('id');
240
Tyler Scottbb42ed22015-10-21 17:02:56 -0600241 console.log("Tree search:", path);
Tyler Scottb59e6de2015-09-18 14:46:30 -0600242
Tyler Scott66a965d2016-02-29 15:41:33 -0700243 scope.query(scope.catalog, {
244 '??': path
245 },
246 function(interest, data) {
247 //Success
Tyler Scottb59e6de2015-09-18 14:46:30 -0600248 console.log("Tree search response", interest, data);
249
Chengyu Fan7b978f82015-12-09 17:03:23 -0700250 scope.name = interest.getName();
Tyler Scottb59e6de2015-09-18 14:46:30 -0600251
252 scope.getResults(0);
253 },
Tyler Scott66a965d2016-02-29 15:41:33 -0700254 function(interest) {
255 //Failure
Tyler Scottb59e6de2015-09-18 14:46:30 -0600256 console.warn("Request failed! Timeout", interest);
Tyler Scott66a965d2016-02-29 15:41:33 -0700257 scope.createAlert("Request timed out.\"" + interest.getName().toUri() + "\" See console for details.");
Tyler Scottb59e6de2015-09-18 14:46:30 -0600258 });
259
260 });
261
Tyler Scottcdfcde82015-09-14 16:13:29 -0600262 this.setupRequestForm();
263
Tyler Scott48f92cd2015-10-16 18:31:20 -0600264 this.resultTable.popover({
Tyler Scott66a965d2016-02-29 15:41:33 -0700265 selector: ".metaDataLink",
266 content: function() {
Tyler Scottf95a97c2016-06-21 12:09:22 -0600267 $('.metaDataLink').not(this).popover('destroy');
Tyler Scott48f92cd2015-10-16 18:31:20 -0600268 return scope.getMetaData(this);
269 },
270 title: "Metadata",
271 html: true,
272 trigger: 'click',
273 placement: 'bottom'
274 });
275
Tyler Scott66a965d2016-02-29 15:41:33 -0700276 this.resultTable.on('click', '.metaDataLink', function(e) {
Tyler Scott48f92cd2015-10-16 18:31:20 -0600277 //This prevents the page from scrolling when you click on a name.
278 e.preventDefault();
279 });
280
Tyler Scott66a965d2016-02-29 15:41:33 -0700281 this.resultTable.on('click', '.subsetButton', function() {
Tyler Scottf355e452015-11-06 21:35:14 -0700282 var metaData = $(this).siblings('pre').text();
283 var exp = /netcdf ([\w-]+)/;
284 var match = exp.exec(metaData);
Tyler Scott66a965d2016-02-29 15:41:33 -0700285 var filename = match[0].replace(/netcdf /, '') + '.nc';
286 scope.request(null , filename);
Tyler Scottf355e452015-11-06 21:35:14 -0700287 });
288
Tyler Scottf95a97c2016-06-21 12:09:22 -0600289 //Allow the title to change the tab
290 $('#brand-title').click(function(){
291 //Correct active class on tabs.
292 $('#explore-tab').removeClass('active');
293 $('#search-tab').addClass('active');
294 });
295
Tyler Scott66a965d2016-02-29 15:41:33 -0700296 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600297
Tyler Scott66a965d2016-02-29 15:41:33 -0700298 Atmos.prototype.clearResults = function() {
299 this.results = [];
300 //Drop any old results.
Tyler Scottcdfcde82015-09-14 16:13:29 -0600301 this.retrievedSegments = 0;
302 this.resultCount = Infinity;
303 this.page = 0;
304 this.resultTable.empty();
Tyler Scott66a965d2016-02-29 15:41:33 -0700305 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600306
Tyler Scott66a965d2016-02-29 15:41:33 -0700307 Atmos.prototype.pathSearch = function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600308 var value = this.searchInput.val();
309
310 this.clearResults();
311
312 var scope = this;
313
Tyler Scott66a965d2016-02-29 15:41:33 -0700314 this.query(this.catalog, {
315 "??": value
316 },
317 function(interest, data) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600318 console.log("Query response:", interest, data);
319
Tyler Scott66a965d2016-02-29 15:41:33 -0700320 scope.name = interest.getName();
Tyler Scottcdfcde82015-09-14 16:13:29 -0600321
322 scope.getResults(0);
323
324 },
Tyler Scott66a965d2016-02-29 15:41:33 -0700325 function(interest) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600326 console.warn("Request failed! Timeout", interest);
327 scope.createAlert("Request timed out. \"" + interest.getName().toUri() + "\" See console for details.");
328 });
329
Tyler Scott66a965d2016-02-29 15:41:33 -0700330 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600331
Tyler Scott66a965d2016-02-29 15:41:33 -0700332 Atmos.prototype.search = function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600333
334 var filters = this.getFilters();
335
336 console.log("Search started!", this.searchInput.val(), filters);
337
338 console.log("Initiating query");
339
340 this.clearResults();
341
342 var scope = this;
343
344 this.query(this.catalog, filters,
Tyler Scott66a965d2016-02-29 15:41:33 -0700345 function(interest, data) {
346 //Response function
Tyler Scottcdfcde82015-09-14 16:13:29 -0600347 console.log("Query Response:", interest, data);
348
Chengyu Fan7b978f82015-12-09 17:03:23 -0700349 scope.name = interest.getName();
Tyler Scottcdfcde82015-09-14 16:13:29 -0600350
351 scope.getResults(0);
352
Tyler Scott66a965d2016-02-29 15:41:33 -0700353 }, function(interest) {
354 //Timeout function
Tyler Scottd980a292015-10-13 15:16:34 -0600355 console.warn("Request failed after 3 attempts!", interest);
356 scope.createAlert("Request failed after 3 attempts. \"" + interest.getName().toUri() + "\" See console for details.");
Tyler Scottcdfcde82015-09-14 16:13:29 -0600357 });
358
Tyler Scott66a965d2016-02-29 15:41:33 -0700359 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600360
Tyler Scott66a965d2016-02-29 15:41:33 -0700361 Atmos.prototype.autoComplete = function(field, callback) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600362
363 var scope = this;
364
Tyler Scott66a965d2016-02-29 15:41:33 -0700365 var result = {};
Tyler Scottcdfcde82015-09-14 16:13:29 -0600366
Tyler Scott66a965d2016-02-29 15:41:33 -0700367 const getAll = function(interest, data) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600368
Tyler Scott66a965d2016-02-29 15:41:33 -0700369 if (data.getContent().length !== 0) {
370 var resp = JSON.parse(data.getContent().toString().replace(/[\n\0]/g, ""));
371 if (result.next) {
372 result.next = result.next.concat(resp.next);
373 } else {
374 result = resp;
Tyler Scottd61bf832015-11-30 16:36:17 -0700375 }
Tyler Scott66a965d2016-02-29 15:41:33 -0700376 } else {
377 callback(result);
378 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600379
Tyler Scott66a965d2016-02-29 15:41:33 -0700380 var name = data.getName();
381 var segment = name.components[name.getComponentCount() - 1];
382 if (segment.toSegment() !== data.getMetaInfo().getFinalBlockId().toSegment()){
383 name = name.getPrefix(-1); //Remove segment
384 name.appendSegment(segment.toSegment() + 1);
385 scope.expressInterest(name, getAll, function() {
386 console.warn("Autocomplete timed out, results may be incomplete.");
387 callback(result);
388 //Return if we get a timeout.
389 });
390 } else {
391 callback(result);
392 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600393
Tyler Scott66a965d2016-02-29 15:41:33 -0700394 }
395
396 this.query(this.catalog, {
397 "?": field
398 }, getAll);
399
400 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600401
402 Atmos.prototype.showResults = function(resultIndex) {
403
404 var results = this.results.slice(this.resultsPerPage * resultIndex, this.resultsPerPage * (resultIndex + 1));
405
406 var resultDOM = $(
Tyler Scott66a965d2016-02-29 15:41:33 -0700407 results.reduce(function(prev, current) {
408 prev.push('<tr><td><input class="resultSelector" type="checkbox"></td><td class="popover-container"><a href="#" class="metaDataLink">');
409 prev.push(current);
410 prev.push('</a></td></tr>');
411 return prev;
412 }, ['<tr><th><input id="resultSelectAll" type="checkbox" title="Select All"> Select</th><th>Name</th></tr>']).join('')
Tyler Scottcdfcde82015-09-14 16:13:29 -0600413 );
414
Tyler Scott66a965d2016-02-29 15:41:33 -0700415 resultDOM.find('#resultSelectAll').click(function() {
416 if ($(this).is(':checked')) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600417 resultDOM.find('.resultSelector:not([disabled])').prop('checked', true);
418 } else {
419 resultDOM.find('.resultSelector:not([disabled])').prop('checked', false);
420 }
421 });
422
423 this.resultTable.hide().empty().append(resultDOM).slideDown('slow');
424
425 this.resultMenu.find('.pageNumber').text(resultIndex + 1);
426 this.resultMenu.find('.pageLength').text(this.resultsPerPage * resultIndex + results.length);
427
428 if (this.resultsPerPage * (resultIndex + 1) >= this.resultCount) {
429 this.resultMenu.find('.next').addClass('disabled');
Tyler Scott66a965d2016-02-29 15:41:33 -0700430 } else if (resultIndex === 0) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600431 this.resultMenu.find('.next').removeClass('disabled');
432 }
433
Tyler Scott66a965d2016-02-29 15:41:33 -0700434 if (resultIndex === 0) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600435 this.resultMenu.find('.previous').addClass('disabled');
436 } else if (resultIndex === 1) {
437 this.resultMenu.find('.previous').removeClass('disabled');
438 }
439
Tyler Scott66a965d2016-02-29 15:41:33 -0700440 $.scrollTo("#results", 500, {
441 interrupt: true
442 });
Tyler Scottf355e452015-11-06 21:35:14 -0700443
Tyler Scott66a965d2016-02-29 15:41:33 -0700444 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600445
Tyler Scott66a965d2016-02-29 15:41:33 -0700446 Atmos.prototype.getResults = function(index) {
Chengyu Fan7b978f82015-12-09 17:03:23 -0700447 var scope = this;
Tyler Scottcdfcde82015-09-14 16:13:29 -0600448
Tyler Scott66a965d2016-02-29 15:41:33 -0700449 if ($('#results').hasClass('hidden')) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600450 $('#results').removeClass('hidden').slideDown();
451 }
452
Tyler Scott66a965d2016-02-29 15:41:33 -0700453 if ((scope.results.length === scope.resultCount) || (scope.resultsPerPage * (index + 1) < scope.results.length)) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600454 //console.log("We already have index", index);
Chengyu Fan7b978f82015-12-09 17:03:23 -0700455 scope.page = index;
456 scope.showResults(index);
Tyler Scottcdfcde82015-09-14 16:13:29 -0600457 return;
458 }
459
Tyler Scott66a965d2016-02-29 15:41:33 -0700460 if (scope.name === null ) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600461 console.error("This shouldn't be reached! We are getting results before a search has occured!");
462 throw new Error("Illegal State");
463 }
464
Chengyu Fan7b978f82015-12-09 17:03:23 -0700465 var interestName = new Name(scope.name);
Tyler Scottcdfcde82015-09-14 16:13:29 -0600466
Chengyu Fan7b978f82015-12-09 17:03:23 -0700467 // Interest name should be /<catalog-prefix>/query/<query-param>/<version>/<#seq>
468 if (scope.name.size() === (scope.catalogPrefix.size() + 3)) {
469 interestName = interestName.appendSegment(scope.retrievedSegments++);
Tyler Scott66a965d2016-02-29 15:41:33 -0700470 //console.log("Requesting data index: (", scope.retrievedSegments - 1, ") at ", interestName.toUri());
Chengyu Fan7b978f82015-12-09 17:03:23 -0700471 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600472
Tyler Scott66a965d2016-02-29 15:41:33 -0700473 this.expressInterest(interestName,
474 function(interest, data) {
475 //Response
Tyler Scottcdfcde82015-09-14 16:13:29 -0600476
Tyler Scott66a965d2016-02-29 15:41:33 -0700477 if (data.getContent().length === 0) {
478 scope.resultMenu.find('.totalResults').text(0);
479 scope.resultMenu.find('.pageNumber').text(0);
480 scope.resultMenu.find('.pageLength').text(0);
481 console.log("Empty response.");
482 scope.resultTable.html("<tr><td>Empty response. This usually means no results.</td></tr>");
483 return;
Tyler Scottcdfcde82015-09-14 16:13:29 -0600484 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600485
Tyler Scott66a965d2016-02-29 15:41:33 -0700486 var content = JSON.parse(data.getContent().toString().replace(/[\n\0]/g, ""));
487
488 if (!content.results) {
489 scope.resultMenu.find('.totalResults').text(0);
490 scope.resultMenu.find('.pageNumber').text(0);
491 scope.resultMenu.find('.pageLength').text(0);
492 console.log("No results were found!");
493 scope.resultTable.html("<tr><td>No Results</td></tr>");
494 return;
495 }
496
497 scope.results = scope.results.concat(content.results);
498
499 scope.resultCount = content.resultCount;
500
501 scope.resultMenu.find('.totalResults').text(scope.resultCount);
502
503 scope.page = index;
504
505 // reset scope.name
506 scope.name = new Name(data.getName().getPrefix(scope.catalogPrefix.size() + 3));
507
508 scope.getResults(index);
509 //Keep calling this until we have enough data.
510
511 }, function() {}//Ignore failure
512 );
513
514 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600515
516 Atmos.prototype.query = function(prefix, parameters, callback, timeout) {
517
518 var queryPrefix = new Name(prefix);
519 queryPrefix.append("query");
520
521 var jsonString = JSON.stringify(parameters);
522 queryPrefix.append(jsonString);
523
Tyler Scott66a965d2016-02-29 15:41:33 -0700524 this.expressInterest(queryPrefix, callback, timeout);
Tyler Scottcdfcde82015-09-14 16:13:29 -0600525
Tyler Scott66a965d2016-02-29 15:41:33 -0700526 }
Tyler Scottd61bf832015-11-30 16:36:17 -0700527
Tyler Scott66a965d2016-02-29 15:41:33 -0700528
529 Atmos.prototype.expressInterest = function(name, success, failure) {
530
531 var interest = new Interest(name);
532 interest.setInterestLifetimeMilliseconds(1500);
533 interest.setMustBeFresh(true);
534
535 const face = this.face;
536
537 async.retry(4, function(done) {
538 face.expressInterest(interest, function(interest, data) {
Tyler Scottd61bf832015-11-30 16:36:17 -0700539 done();
Tyler Scott66a965d2016-02-29 15:41:33 -0700540 success(interest, data);
541 }, function(interest) {
542 done("Interest timed out 4 times.", interest);
543 });
544 }, function(err, interest) {
545 if (err) {
546 console.log(err, interest);
547 failure(interest);
Tyler Scott8724e422015-10-13 17:59:07 -0600548 }
Tyler Scottd61bf832015-11-30 16:36:17 -0700549 });
Tyler Scottcdfcde82015-09-14 16:13:29 -0600550
Tyler Scott66a965d2016-02-29 15:41:33 -0700551 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600552
553 /**
554 * This function returns a map of all the categories active filters.
555 * @return {Object<string, string>}
556 */
Tyler Scott66a965d2016-02-29 15:41:33 -0700557 Atmos.prototype.getFilters = function() {
558 var filters = this.filters.children().toArray().reduce(function(prev, current) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600559 var data = $(current).text().split(/:/);
560 prev[data[0]] = data[1];
561 return prev;
Tyler Scott66a965d2016-02-29 15:41:33 -0700562 }, {});
563 //Collect a map<category, filter>.
Tyler Scottcdfcde82015-09-14 16:13:29 -0600564 //TODO Make the return value map<category, Array<filter>>
565 return filters;
Tyler Scott66a965d2016-02-29 15:41:33 -0700566 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600567
568 /**
569 * Creates a closable alert for the user.
Tyler Scott66a965d2016-02-29 15:41:33 -0700570
Tyler Scottcdfcde82015-09-14 16:13:29 -0600571 * @param {string} message
572 * @param {string} type - Override the alert type.
573 */
574 Atmos.prototype.createAlert = function(message, type) {
575
576 var alert = $('<div class="alert"><div>');
Tyler Scott66a965d2016-02-29 15:41:33 -0700577 alert.addClass(type ? type : 'alert-info');
Tyler Scottcdfcde82015-09-14 16:13:29 -0600578 alert.text(message);
579 alert.append(closeButton);
580
581 this.alerts.append(alert);
Tyler Scott66a965d2016-02-29 15:41:33 -0700582 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600583
584 /**
585 * Requests all of the names represented by the buttons in the elements list.
Tyler Scott66a965d2016-02-29 15:41:33 -0700586
Tyler Scottcdfcde82015-09-14 16:13:29 -0600587 * @param elements {Array<jQuery>} A list of the table row elements
Tyler Scottf355e452015-11-06 21:35:14 -0700588 * @param subsetFileName {String} If present then do a subsetting request instead.
Tyler Scottcdfcde82015-09-14 16:13:29 -0600589 */
Tyler Scott66a965d2016-02-29 15:41:33 -0700590 Atmos.prototype.request = function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600591
592 //Pseudo globals.
593 var keyChain;
594 var certificateName;
595 var keyAdded = false;
596
Tyler Scott66a965d2016-02-29 15:41:33 -0700597 return function(elements, subsetFilename) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600598
599 var names = [];
Tyler Scott66a965d2016-02-29 15:41:33 -0700600 $(elements).find('.metaDataLink').each(function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600601 var name = $(this).text();
602 names.push(name);
Tyler Scottf355e452015-11-06 21:35:14 -0700603 });
604
605 var subset = false;
606
Tyler Scott66a965d2016-02-29 15:41:33 -0700607 if (!subsetFilename) {
Tyler Scottf355e452015-11-06 21:35:14 -0700608 $('#subsetting').hide();
609 } else {
610 $('#subsetting').show();
611 subset = true;
612 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600613
614 var scope = this;
Tyler Scott384f37a2016-05-05 16:09:05 -0600615
616 //FIXME The following is temporary, it allows people to direct download from
617 //a single host with a small set of names. It is to demo the functionality but
618 //could use improvement. (Multiple servers, non static list, etc)
619 var directDls = $('#directDownloadList').empty();
620 names.forEach(function(name){
621 if (scope.conversions[name]){ //If the name exists in the conversions.
622 var ele = $('<a href="http://atmos-mwsc.ucar.edu/ucar/' + conversions[name] + '" class="list-group-item>' + name + '</a>');
623 directDls.append(ele);
624 }
625 });
626
Tyler Scott66a965d2016-02-29 15:41:33 -0700627 this.requestForm.on('submit', function(e) {
628 //This will be registered for the next submit from the form.
Tyler Scottcdfcde82015-09-14 16:13:29 -0600629 e.preventDefault();
630
Tyler Scottf355e452015-11-06 21:35:14 -0700631 $('#request .alert').remove();
632
633 var variables = [];
Tyler Scott66a965d2016-02-29 15:41:33 -0700634 if (subset) {
635 $('#subsetVariables .row').each(function() {
Tyler Scottf355e452015-11-06 21:35:14 -0700636 var t = $(this);
637 var values = {};
Tyler Scott66a965d2016-02-29 15:41:33 -0700638 t.find('.values input').each(function() {
Tyler Scottf355e452015-11-06 21:35:14 -0700639 var t = $(this);
640 values[t.attr('name')] = t.val();
641 });
Tyler Scott66a965d2016-02-29 15:41:33 -0700642 variables.push({
643 variable: t.find('.variable').val(),
644 values: values
645 });
Tyler Scottf355e452015-11-06 21:35:14 -0700646 });
647 }
648
Tyler Scottcdfcde82015-09-14 16:13:29 -0600649 //Form checking
650 var dest = scope.requestForm.find('#requestDest .active');
Tyler Scott66a965d2016-02-29 15:41:33 -0700651 if (dest.length !== 1) {
Tyler Scottf355e452015-11-06 21:35:14 -0700652 var alert = $('<div class="alert alert-warning">A destination is required!' + closeButton + '<div>');
653 $('#request > .panel-body').append(alert);
Tyler Scottcdfcde82015-09-14 16:13:29 -0600654 return;
655 }
656
Tyler Scott66a965d2016-02-29 15:41:33 -0700657 $('#request').modal('hide');
658 //Initial params are ok. We can close the form.
Tyler Scottb59e6de2015-09-18 14:46:30 -0600659
660 scope.cleanRequestForm();
Tyler Scottcdfcde82015-09-14 16:13:29 -0600661
Tyler Scott66a965d2016-02-29 15:41:33 -0700662 $(this).off(e);
663 //Don't fire this again, the request must be regenerated
Tyler Scottcdfcde82015-09-14 16:13:29 -0600664
665 //Key setup
Tyler Scott66a965d2016-02-29 15:41:33 -0700666 if (!keyAdded) {
667 if (!scope.config.retrieval.demoKey || !scope.config.retrieval.demoKey.pub || !scope.config.retrieval.demoKey.priv) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600668 scope.createAlert("This host was not configured to handle retrieval! See console for details.", 'alert-danger');
669 console.error("Missing/invalid key! This must be configured in the config on the server.", scope.config.demoKey);
670 return;
671 }
672
673 //FIXME base64 may or may not exist in other browsers. Need a new polyfill.
Tyler Scott66a965d2016-02-29 15:41:33 -0700674 var pub = new Buffer(base64.toByteArray(scope.config.retrieval.demoKey.pub));
675 //MUST be a Buffer (Buffer != Uint8Array)
Tyler Scottcdfcde82015-09-14 16:13:29 -0600676 var priv = new Buffer(base64.toByteArray(scope.config.retrieval.demoKey.priv));
677
678 var identityStorage = new MemoryIdentityStorage();
679 var privateKeyStorage = new MemoryPrivateKeyStorage();
Tyler Scott66a965d2016-02-29 15:41:33 -0700680 keyChain = new KeyChain(new IdentityManager(identityStorage,privateKeyStorage),
681 new SelfVerifyPolicyManager(identityStorage));
Tyler Scottcdfcde82015-09-14 16:13:29 -0600682
683 var keyName = new Name("/retrieve/DSK-123");
684 certificateName = keyName.getSubName(0, keyName.size() - 1)
Tyler Scott66a965d2016-02-29 15:41:33 -0700685 .append("KEY").append(keyName.get(-1))
686 .append("ID-CERT").append("0");
Tyler Scottcdfcde82015-09-14 16:13:29 -0600687
Tyler Scott66a965d2016-02-29 15:41:33 -0700688 identityStorage.addKey(keyName, KeyType.RSA, new Blob(pub,false));
Tyler Scottcdfcde82015-09-14 16:13:29 -0600689 privateKeyStorage.setKeyPairForKeyName(keyName, KeyType.RSA, pub, priv);
690
691 scope.face.setCommandSigningInfo(keyChain, certificateName);
692
693 keyAdded = true;
694
695 }
696
697 //Retrieval
698 var retrievePrefix = new Name("/catalog/ui/" + guid());
699
Tyler Scottcdfcde82015-09-14 16:13:29 -0600700 scope.face.registerPrefix(retrievePrefix,
Tyler Scott66a965d2016-02-29 15:41:33 -0700701 function(prefix, interest, face, interestFilterId, filter) {
702 //On Interest
703 //This function will exist until the page exits but will likely only be used once.
Tyler Scottcdfcde82015-09-14 16:13:29 -0600704
Tyler Scott66a965d2016-02-29 15:41:33 -0700705 var data = new Data(interest.getName());
706 var content;
707 if (subset) {
708 content = JSON.stringify({
709 name: subsetFilename,
710 subset: variables
Tyler Scottd61bf832015-11-30 16:36:17 -0700711 });
Tyler Scott66a965d2016-02-29 15:41:33 -0700712 } else {
713 content = JSON.stringify(names);
Tyler Scottcdfcde82015-09-14 16:13:29 -0600714 }
Tyler Scott66a965d2016-02-29 15:41:33 -0700715 //Blob breaks the data! Don't use it
716 data.setContent(content);
717 //TODO Packetize this.
718 keyChain.sign(data, certificateName);
719
720 try {
721 face.putData(data);
722 console.log("Responded for", interest.getName().toUri(), data);
723 scope.createAlert("Data retrieval has initiated.", "alert-success");
724 } catch (e) {
725 console.error("Failed to respond to", interest.getName().toUri(), data);
726 scope.createAlert("Data retrieval failed.");
727 }
728
729 }, function(prefix) {
730 //On fail
731 scope.createAlert("Failed to register the retrieval URI! See console for details.", "alert-danger");
732 console.error("Failed to register URI:", prefix.toUri(), prefix);
733 }, function(prefix, registeredPrefixId) {
734 //On success
735 var name = new Name(dest.text());
736 name.append(prefix);
737
738 scope.expressInterest(name,
739 function(interest, data) {
740 //Success
741 console.log("Request for", name.toUri(), "succeeded.", interest, data);
742 },
743 function() {
744 console.warn("Failed to request from retrieve agent.");
745 }
746 );
747 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600748 );
749
750 });
Tyler Scott66a965d2016-02-29 15:41:33 -0700751 $('#request').modal();
752 //This forces the form to be the only option.
Tyler Scottcdfcde82015-09-14 16:13:29 -0600753
Tyler Scott66a965d2016-02-29 15:41:33 -0700754 }
755
Tyler Scottcdfcde82015-09-14 16:13:29 -0600756 }();
757
758 Atmos.prototype.filterSetup = function() {
759 //Filter setup
760
761 var prefix = new Name(this.catalog).append("filters-initialization");
762
763 var scope = this;
764
Tyler Scott66a965d2016-02-29 15:41:33 -0700765 this.getAll(prefix, function(data) {
766 //Success
767 var raw = JSON.parse(data.replace(/[\n\0]/g, ''));
768 //Remove null byte and parse
Tyler Scottcdfcde82015-09-14 16:13:29 -0600769
770 console.log("Filter categories:", raw);
771
Tyler Scott66a965d2016-02-29 15:41:33 -0700772 $.each(raw, function(index, object) {
773 //Unpack list of objects
774 $.each(object, function(category, searchOptions) {
775 //Unpack category from object (We don't know what it is called)
Tyler Scottcdfcde82015-09-14 16:13:29 -0600776 //Create the category
777 var e = $('<li><a href="#">' + category.replace(/_/g, " ") + '</a><ul class="subnav nav nav-pills nav-stacked"></ul></li>');
778
779 var sub = e.find('ul.subnav');
Tyler Scott66a965d2016-02-29 15:41:33 -0700780 $.each(searchOptions, function(index, name) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600781 //Create the filter list inside the category
782 var item = $('<li><a href="#">' + name + '</a></li>');
783 sub.append(item);
Tyler Scott66a965d2016-02-29 15:41:33 -0700784 item.click(function() {
785 //Click on the side menu filters
786 if (item.hasClass('active')) {
787 //Does the filter already exist?
Tyler Scottcdfcde82015-09-14 16:13:29 -0600788 item.removeClass('active');
789 scope.filters.find(':contains(' + category + ':' + name + ')').remove();
Tyler Scott66a965d2016-02-29 15:41:33 -0700790 } else {
791 //Add a filter
Tyler Scottcdfcde82015-09-14 16:13:29 -0600792 item.addClass('active');
793 var filter = $('<span class="label label-default"></span>');
794 filter.text(category + ':' + name);
795
796 scope.filters.append(filter);
797
Tyler Scott66a965d2016-02-29 15:41:33 -0700798 filter.click(function() {
799 //Click on a filter
Tyler Scottcdfcde82015-09-14 16:13:29 -0600800 filter.remove();
801 item.removeClass('active');
802 });
803 }
804
805 });
806 });
807
808 //Toggle the menus. (Only respond when the immediate tab is clicked.)
Tyler Scott66a965d2016-02-29 15:41:33 -0700809 e.find('> a').click(function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600810 scope.categories.find('.subnav').slideUp();
811 var t = $(this).siblings('.subnav');
Tyler Scott66a965d2016-02-29 15:41:33 -0700812 if (!t.is(':visible')) {
813 //If the sub menu is not visible
814 t.slideDown(function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600815 t.triggerHandler('focus');
Tyler Scott66a965d2016-02-29 15:41:33 -0700816 });
817 //Make it visible and look at it.
Tyler Scottcdfcde82015-09-14 16:13:29 -0600818 }
819 });
820
821 scope.categories.append(e);
822
823 });
824 });
825
Tyler Scott66a965d2016-02-29 15:41:33 -0700826 }, function(interest) {
827 //Timeout
Tyler Scottcdfcde82015-09-14 16:13:29 -0600828 scope.createAlert("Failed to initialize the filters!", "alert-danger");
829 console.error("Failed to initialize filters!", interest);
830 ga('send', 'event', 'error', 'filters');
831 });
832
Tyler Scott66a965d2016-02-29 15:41:33 -0700833 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600834
835 /**
836 * This function retrieves all segments in order until it knows it has reached the last one.
837 * It then returns the final joined result.
Tyler Scott66a965d2016-02-29 15:41:33 -0700838
Tyler Scottf355e452015-11-06 21:35:14 -0700839 * @param prefix {String|Name} The ndn name we are retrieving.
840 * @param callback {function(String)} if successful, will call the callback with a string of data.
841 * @param failure {function(Interest)} if unsuccessful, will call failure with the last failed interest.
842 * @param stop {boolean} stop if no finalBlock.
Tyler Scottcdfcde82015-09-14 16:13:29 -0600843 */
Tyler Scott66a965d2016-02-29 15:41:33 -0700844 Atmos.prototype.getAll = function(prefix, callback, failure, stop) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600845
846 var scope = this;
847 var d = [];
848
Tyler Scottd61bf832015-11-30 16:36:17 -0700849 var name = new Name(prefix);
850 var segment = 0;
Tyler Scott8724e422015-10-13 17:59:07 -0600851
Tyler Scott66a965d2016-02-29 15:41:33 -0700852 var request = function() {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600853
Tyler Scottd61bf832015-11-30 16:36:17 -0700854 var n2 = new Name(name);
855 n2.appendSegment(segment);
Tyler Scottcdfcde82015-09-14 16:13:29 -0600856
Tyler Scottf95a97c2016-06-21 12:09:22 -0600857 scope.expressInterest(n2, handleData, function(err, interest){failure(interest)});
Tyler Scott66a965d2016-02-29 15:41:33 -0700858 //Forward to handleData and ignore error
859 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600860
Tyler Scott66a965d2016-02-29 15:41:33 -0700861 var handleData = function(interest, data) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600862
863 d.push(data.getContent().toString());
864
Tyler Scottf355e452015-11-06 21:35:14 -0700865 var hasFinalBlock = data.getMetaInfo().getFinalBlockId().value.length === 0;
866 var finalBlockStop = hasFinalBlock && stop;
867
868 if (finalBlockStop ||
Tyler Scott66a965d2016-02-29 15:41:33 -0700869 (!hasFinalBlock && interest.getName().get(-1).toSegment() == data.getMetaInfo().getFinalBlockId().toSegment())) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600870 callback(d.join(""));
871 } else {
Tyler Scottd61bf832015-11-30 16:36:17 -0700872 segment++;
873 request();
Tyler Scottcdfcde82015-09-14 16:13:29 -0600874 }
875
Tyler Scott66a965d2016-02-29 15:41:33 -0700876 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600877
Tyler Scottd61bf832015-11-30 16:36:17 -0700878 request();
Tyler Scottcdfcde82015-09-14 16:13:29 -0600879
Tyler Scott66a965d2016-02-29 15:41:33 -0700880 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600881
Tyler Scott66a965d2016-02-29 15:41:33 -0700882 Atmos.prototype.cleanRequestForm = function() {
Tyler Scottb59e6de2015-09-18 14:46:30 -0600883 $('#requestDest').prev().removeClass('btn-success').addClass('btn-default');
884 $('#requestDropText').text('Destination');
885 $('#requestDest .active').removeClass('active');
Tyler Scottf355e452015-11-06 21:35:14 -0700886 $('#subsetMenu').attr('class', 'collapse');
887 $('#subsetVariables').empty();
888 $('#request .alert').alert('close').remove();
Tyler Scott66a965d2016-02-29 15:41:33 -0700889 }
Tyler Scottb59e6de2015-09-18 14:46:30 -0600890
Tyler Scott66a965d2016-02-29 15:41:33 -0700891 Atmos.prototype.setupRequestForm = function() {
Tyler Scottb59e6de2015-09-18 14:46:30 -0600892
893 var scope = this;
894
Tyler Scott66a965d2016-02-29 15:41:33 -0700895 this.requestForm.find('#requestCancel').click(function() {
896 $('#request').unbind('submit')//Removes all event handlers.
897 .modal('hide');
898 //Hides the form.
Tyler Scottb59e6de2015-09-18 14:46:30 -0600899 scope.cleanRequestForm();
Tyler Scottcdfcde82015-09-14 16:13:29 -0600900 });
901
Tyler Scott66a965d2016-02-29 15:41:33 -0700902 var dests = $(this.config['retrieval']['destinations'].reduce(function(prev, current) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600903 prev.push('<li><a href="#">');
904 prev.push(current);
905 prev.push("</a></li>");
906 return prev;
907 }, []).join(""));
908
909 this.requestForm.find('#requestDest').append(dests)
Tyler Scott66a965d2016-02-29 15:41:33 -0700910 .on('click', 'a', function(e) {
Tyler Scottcdfcde82015-09-14 16:13:29 -0600911 $('#requestDest .active').removeClass('active');
Tyler Scottb59e6de2015-09-18 14:46:30 -0600912 var t = $(this);
913 t.parent().addClass('active');
914 $('#requestDropText').text(t.text());
915 $('#requestDest').prev().removeClass('btn-default').addClass('btn-success');
Tyler Scottcdfcde82015-09-14 16:13:29 -0600916 });
917
Tyler Scott66a965d2016-02-29 15:41:33 -0700918 var addVariable = function(selector) {
919 var ele = $(selector).clone().attr('id', '');
920 ele.find('.close').click(function() {
Tyler Scottf355e452015-11-06 21:35:14 -0700921 ele.remove();
922 });
923 $('#subsetVariables').append(ele);
Tyler Scott66a965d2016-02-29 15:41:33 -0700924 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600925
Tyler Scott66a965d2016-02-29 15:41:33 -0700926 $('#subsetAddVariableBtn').click(function() {
Tyler Scottf355e452015-11-06 21:35:14 -0700927 addVariable('#customTemplate');
928 });
Tyler Scott66a965d2016-02-29 15:41:33 -0700929 $('#subsetAddTimeVariable').click(function() {
Tyler Scottf355e452015-11-06 21:35:14 -0700930 addVariable('#timeTemplate');
931 });
Tyler Scott66a965d2016-02-29 15:41:33 -0700932 $('#subsetAddLocVariable').click(function() {
Tyler Scottf355e452015-11-06 21:35:14 -0700933 addVariable('#locationTemplate');
934 });
Tyler Scottcdfcde82015-09-14 16:13:29 -0600935
Tyler Scott66a965d2016-02-29 15:41:33 -0700936 }
Tyler Scottcdfcde82015-09-14 16:13:29 -0600937
Tyler Scott66a965d2016-02-29 15:41:33 -0700938 Atmos.prototype.getMetaData = (function() {
Tyler Scott48f92cd2015-10-16 18:31:20 -0600939
940 var cache = {};
941
942 return function(element) {
943 var name = $(element).text();
944
Tyler Scottf355e452015-11-06 21:35:14 -0700945 ga('send', 'event', 'request', 'metaData');
946
947 var subsetButton = '<button class="btn btn-default subsetButton" type="button">Subset</button>';
948
Tyler Scott48f92cd2015-10-16 18:31:20 -0600949 if (cache[name]) {
Tyler Scottf355e452015-11-06 21:35:14 -0700950 return [subsetButton, '<pre class="metaData">', cache[name], '</pre>'].join('');
Tyler Scott48f92cd2015-10-16 18:31:20 -0600951 }
952
953 var prefix = new Name(name).append("metadata");
Tyler Scott66a965d2016-02-29 15:41:33 -0700954 var id = guid();
955 //We need an id because the return MUST be a string.
Tyler Scott48f92cd2015-10-16 18:31:20 -0600956 var ret = '<div id="' + id + '"><span class="fa fa-spinner fa-spin"></span></div>';
957
Tyler Scott66a965d2016-02-29 15:41:33 -0700958 this.getAll(prefix, function(data) {
Tyler Scott48f92cd2015-10-16 18:31:20 -0600959 var el = $('<pre class="metaData"></pre>');
960 el.text(data);
Tyler Scottf355e452015-11-06 21:35:14 -0700961 var container = $('<div></div>');
962 container.append($(subsetButton));
963 container.append(el);
964 $('#' + id).empty().append(container);
Tyler Scott48f92cd2015-10-16 18:31:20 -0600965 cache[name] = data;
Tyler Scott66a965d2016-02-29 15:41:33 -0700966 }, function(interest) {
Tyler Scott48f92cd2015-10-16 18:31:20 -0600967 $('#' + id).text("The metadata is unavailable for this name.");
968 console.log("Data is unavailable for " + name);
969 });
970
971 return ret;
972
Tyler Scott66a965d2016-02-29 15:41:33 -0700973 }
Tyler Scott48f92cd2015-10-16 18:31:20 -0600974 })();
975
Tyler Scottcdfcde82015-09-14 16:13:29 -0600976 return Atmos;
977
978})();