2015-02-04 10:46:36 +01:00
|
|
|
/*! Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
|
2014-07-15 13:39:22 +02:00
|
|
|
|
2014-02-18 20:00:00 +01:00
|
|
|
/**
|
|
|
|
* Icinga.Loader
|
|
|
|
*
|
|
|
|
* This is where we take care of XHR requests, responses and failures.
|
|
|
|
*/
|
2014-03-04 14:08:29 +01:00
|
|
|
(function(Icinga, $) {
|
2014-02-18 20:00:00 +01:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
Icinga.Loader = function (icinga) {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* YES, we need Icinga
|
|
|
|
*/
|
|
|
|
this.icinga = icinga;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Our base url
|
|
|
|
*/
|
|
|
|
this.baseUrl = icinga.config.baseUrl;
|
|
|
|
|
|
|
|
this.failureNotice = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pending requests
|
|
|
|
*/
|
|
|
|
this.requests = {};
|
|
|
|
|
2014-06-20 16:18:10 +02:00
|
|
|
this.iconCache = {};
|
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
this.autorefreshEnabled = true;
|
|
|
|
};
|
2014-02-18 20:00:00 +01:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
Icinga.Loader.prototype = {
|
|
|
|
|
|
|
|
initialize: function () {
|
|
|
|
this.icinga.timer.register(this.autorefresh, this, 500);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Load the given URL to the given target
|
|
|
|
*
|
2015-08-20 14:38:03 +02:00
|
|
|
* @param {string} url URL to be loaded
|
|
|
|
* @param {object} target Target jQuery element
|
|
|
|
* @param {object} data Optional parameters, usually for POST requests
|
|
|
|
* @param {string} method HTTP method, default is 'GET'
|
|
|
|
* @param {string} action How to handle the response ('replace' or 'append'), default is 'replace'
|
|
|
|
* @param {boolean} autorefresh Whether the cause is a autorefresh or not
|
|
|
|
* @param {object} progressTimer A timer to be stopped when the request is done
|
2014-03-04 14:08:29 +01:00
|
|
|
*/
|
2015-08-20 14:38:03 +02:00
|
|
|
loadUrl: function (url, $target, data, method, action, autorefresh, progressTimer) {
|
2014-03-04 14:08:29 +01:00
|
|
|
var id = null;
|
|
|
|
|
|
|
|
// Default method is GET
|
|
|
|
if ('undefined' === typeof method) {
|
|
|
|
method = 'GET';
|
2014-02-18 20:00:00 +01:00
|
|
|
}
|
2014-04-01 10:43:47 +02:00
|
|
|
if ('undefined' === typeof action) {
|
|
|
|
action = 'replace';
|
|
|
|
}
|
2014-03-17 17:34:37 +01:00
|
|
|
if ('undefined' === typeof autorefresh) {
|
|
|
|
autorefresh = false;
|
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
|
|
|
|
this.icinga.logger.debug('Loading ', url, ' to ', $target);
|
|
|
|
|
|
|
|
// We should do better and ignore requests without target and/or id
|
|
|
|
if (typeof $target !== 'undefined' && $target.attr('id')) {
|
|
|
|
id = $target.attr('id');
|
2014-02-18 20:00:00 +01:00
|
|
|
}
|
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
// If we have a pending request for the same target...
|
2014-06-25 20:15:44 +02:00
|
|
|
if (typeof this.requests[id] !== 'undefined') {
|
2014-03-17 17:10:03 +01:00
|
|
|
if (autorefresh) {
|
|
|
|
return false;
|
|
|
|
}
|
2015-08-18 09:45:01 +02:00
|
|
|
// ... ignore the new request if it is already pending with the same URL. Only abort GETs, as those
|
|
|
|
// are the only methods that are guaranteed to return the same value
|
|
|
|
if (this.requests[id].url === url && method === 'GET') {
|
2014-03-04 14:08:29 +01:00
|
|
|
this.icinga.logger.debug('Request to ', url, ' is already running for ', $target);
|
|
|
|
return this.requests[id];
|
|
|
|
}
|
2015-08-18 09:45:01 +02:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
// ...or abort the former request otherwise
|
|
|
|
this.icinga.logger.debug(
|
|
|
|
'Aborting pending request loading ',
|
|
|
|
url,
|
|
|
|
' to ',
|
|
|
|
$target
|
|
|
|
);
|
2014-02-18 20:00:00 +01:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
this.requests[id].abort();
|
|
|
|
}
|
2014-02-18 20:00:00 +01:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
// Not sure whether we need this Accept-header
|
|
|
|
var headers = { 'X-Icinga-Accept': 'text/html' };
|
2014-02-18 20:00:00 +01:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
// Ask for a new window id in case we don't already have one
|
|
|
|
if (this.icinga.ui.hasWindowId()) {
|
|
|
|
headers['X-Icinga-WindowId'] = this.icinga.ui.getWindowId();
|
|
|
|
} else {
|
|
|
|
headers['X-Icinga-WindowId'] = 'undefined';
|
|
|
|
}
|
2014-02-18 20:00:00 +01:00
|
|
|
|
2015-07-15 15:25:40 +02:00
|
|
|
// This is jQuery's default content type
|
|
|
|
var contentType = 'application/x-www-form-urlencoded; charset=UTF-8';
|
|
|
|
|
2015-07-22 13:29:44 +02:00
|
|
|
var isFormData = typeof window.FormData !== 'undefined' && data instanceof window.FormData;
|
2015-07-15 15:25:40 +02:00
|
|
|
if (isFormData) {
|
|
|
|
// Setting false is mandatory as the form's data
|
|
|
|
// won't be recognized by the server otherwise
|
|
|
|
contentType = false;
|
|
|
|
}
|
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
var self = this;
|
|
|
|
var req = $.ajax({
|
|
|
|
type : method,
|
|
|
|
url : url,
|
|
|
|
data : data,
|
|
|
|
headers: headers,
|
2015-07-15 15:25:40 +02:00
|
|
|
context: self,
|
|
|
|
contentType: contentType,
|
|
|
|
processData: ! isFormData
|
2014-03-04 14:08:29 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
req.$target = $target;
|
|
|
|
req.url = url;
|
|
|
|
req.done(this.onResponse);
|
|
|
|
req.fail(this.onFailure);
|
|
|
|
req.complete(this.onComplete);
|
2014-03-17 17:10:03 +01:00
|
|
|
req.autorefresh = autorefresh;
|
2014-04-01 10:43:47 +02:00
|
|
|
req.action = action;
|
2015-02-16 15:56:50 +01:00
|
|
|
req.addToHistory = true;
|
2015-08-20 14:38:03 +02:00
|
|
|
req.progressTimer = progressTimer;
|
2015-02-02 16:34:29 +01:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
if (id) {
|
|
|
|
this.requests[id] = req;
|
|
|
|
}
|
2014-05-20 16:58:58 +02:00
|
|
|
if (! autorefresh) {
|
|
|
|
req.$target.addClass('impact');
|
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
this.icinga.ui.refreshDebug();
|
|
|
|
return req;
|
|
|
|
},
|
|
|
|
|
2015-07-20 10:52:28 +02:00
|
|
|
/**
|
|
|
|
* Mimic XHR form submission by using an iframe
|
|
|
|
*
|
|
|
|
* @param {object} $form The form being submitted
|
|
|
|
* @param {string} action The form's action URL
|
|
|
|
* @param {object} $target The target container
|
|
|
|
*/
|
|
|
|
submitFormToIframe: function ($form, action, $target) {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
$form.prop('action', self.icinga.utils.addUrlParams(action, {
|
2015-07-21 15:43:47 +02:00
|
|
|
'_frameUpload': true
|
2015-07-20 10:52:28 +02:00
|
|
|
}));
|
|
|
|
$form.prop('target', 'fileupload-frame-target');
|
|
|
|
$('#fileupload-frame-target').on('load', function (event) {
|
|
|
|
var $frame = $(event.target);
|
2015-07-21 15:43:47 +02:00
|
|
|
var $contents = $frame.contents();
|
|
|
|
|
|
|
|
var $redirectMeta = $contents.find('meta[name="redirectUrl"]');
|
|
|
|
if ($redirectMeta.length) {
|
2015-07-21 16:38:52 +02:00
|
|
|
self.redirectToUrl($redirectMeta.attr('content'), $target);
|
2015-07-21 15:43:47 +02:00
|
|
|
} else {
|
|
|
|
// Fetch the frame's new content and paste it into the target
|
|
|
|
self.renderContentToContainer(
|
|
|
|
$contents.find('body').html(),
|
|
|
|
$target,
|
|
|
|
'replace'
|
|
|
|
);
|
|
|
|
}
|
2015-07-20 10:52:28 +02:00
|
|
|
|
2015-07-21 15:43:47 +02:00
|
|
|
$frame.prop('src', 'about:blank'); // Clear the frame's dom
|
2015-07-20 10:52:28 +02:00
|
|
|
$frame.off('load'); // Unbind the event as it's set on demand
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
/**
|
|
|
|
* Create an URL relative to the Icinga base Url, still unused
|
|
|
|
*
|
|
|
|
* @param {string} url Relative url
|
|
|
|
*/
|
|
|
|
url: function (url) {
|
|
|
|
if (typeof url === 'undefined') {
|
|
|
|
return this.baseUrl;
|
|
|
|
}
|
|
|
|
return this.baseUrl + url;
|
|
|
|
},
|
2014-02-18 20:00:00 +01:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
stopPendingRequestsFor: function ($el) {
|
|
|
|
var id;
|
2014-06-24 07:01:37 +02:00
|
|
|
if (typeof $el === 'undefined' || ! (id = $el.attr('id'))) {
|
2014-03-04 14:08:29 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-06-24 07:01:37 +02:00
|
|
|
if (typeof this.requests[id] !== 'undefined') {
|
2014-03-04 14:08:29 +01:00
|
|
|
this.requests[id].abort();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2014-03-17 17:10:03 +01:00
|
|
|
filterAutorefreshingContainers: function () {
|
|
|
|
return $(this).data('icingaRefresh') > 0;
|
|
|
|
},
|
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
autorefresh: function () {
|
|
|
|
var self = this;
|
|
|
|
if (self.autorefreshEnabled !== true) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-03-17 17:10:03 +01:00
|
|
|
$('.container').filter(this.filterAutorefreshingContainers).each(function (idx, el) {
|
2014-03-04 14:08:29 +01:00
|
|
|
var $el = $(el);
|
|
|
|
var id = $el.attr('id');
|
2014-06-25 20:15:44 +02:00
|
|
|
if (typeof self.requests[id] !== 'undefined') {
|
2014-03-17 17:10:03 +01:00
|
|
|
self.icinga.logger.debug('No refresh, request pending for ', id);
|
2014-03-04 14:08:29 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var interval = $el.data('icingaRefresh');
|
|
|
|
var lastUpdate = $el.data('lastUpdate');
|
|
|
|
|
|
|
|
if (typeof interval === 'undefined' || ! interval) {
|
|
|
|
self.icinga.logger.info('No interval, setting default', id);
|
|
|
|
interval = 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof lastUpdate === 'undefined' || ! lastUpdate) {
|
|
|
|
self.icinga.logger.info('No lastUpdate, setting one', id);
|
|
|
|
$el.data('lastUpdate',(new Date()).getTime());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
interval = interval * 1000;
|
|
|
|
|
2014-03-06 13:08:11 +01:00
|
|
|
// TODO:
|
2014-03-04 14:08:29 +01:00
|
|
|
if ((lastUpdate + interval) > (new Date()).getTime()) {
|
|
|
|
// self.icinga.logger.info(
|
|
|
|
// 'Skipping refresh',
|
|
|
|
// id,
|
|
|
|
// lastUpdate,
|
|
|
|
// interval,
|
|
|
|
// (new Date()).getTime()
|
|
|
|
// );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-04-01 10:43:47 +02:00
|
|
|
if (self.loadUrl($el.data('icingaUrl'), $el, undefined, undefined, undefined, true) === false) {
|
2014-03-17 17:10:03 +01:00
|
|
|
self.icinga.logger.debug(
|
|
|
|
'NOT autorefreshing ' + id + ', even if ' + interval + ' ms passed. Request pending?'
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
self.icinga.logger.debug(
|
|
|
|
'Autorefreshing ' + id + ' ' + interval + ' ms passed'
|
|
|
|
);
|
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
el = null;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2014-03-08 16:20:30 +01:00
|
|
|
/**
|
|
|
|
* Disable the autorefresh mechanism
|
|
|
|
*/
|
2014-03-04 14:08:29 +01:00
|
|
|
disableAutorefresh: function () {
|
|
|
|
this.autorefreshEnabled = false;
|
|
|
|
},
|
|
|
|
|
2014-03-08 16:20:30 +01:00
|
|
|
/**
|
|
|
|
* Enable the autorefresh mechanism
|
|
|
|
*/
|
2014-03-04 14:08:29 +01:00
|
|
|
enableAutorefresh: function () {
|
|
|
|
this.autorefreshEnabled = true;
|
|
|
|
},
|
|
|
|
|
2014-06-05 20:46:11 +02:00
|
|
|
processNotificationHeader: function(req) {
|
|
|
|
var header = req.getResponseHeader('X-Icinga-Notification');
|
2014-06-20 13:22:53 +02:00
|
|
|
var self = this;
|
2014-06-05 21:37:12 +02:00
|
|
|
if (! header) return false;
|
2014-06-20 13:22:53 +02:00
|
|
|
var list = header.split('&');
|
|
|
|
$.each(list, function(idx, el) {
|
|
|
|
var parts = decodeURIComponent(el).split(' ');
|
|
|
|
self.createNotice(parts.shift(), parts.join(' '));
|
|
|
|
});
|
2014-06-05 20:46:11 +02:00
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
2014-06-22 20:06:45 +02:00
|
|
|
addUrlFlag: function(url, flag)
|
|
|
|
{
|
|
|
|
if (url.match(/\?/)) {
|
|
|
|
return url + '&' + flag;
|
|
|
|
} else {
|
|
|
|
return url + '?' + flag;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2015-03-11 21:26:44 +01:00
|
|
|
/**
|
|
|
|
* Process the X-Icinga-Redirect HTTP Response Header
|
|
|
|
*
|
|
|
|
* If the response includes the X-Icinga-Redirect header, redirects to the URL associated with the header.
|
|
|
|
*
|
|
|
|
* @param {object} req Current request
|
|
|
|
*
|
|
|
|
* @returns {boolean} Whether we're about to redirect
|
|
|
|
*/
|
2014-06-05 21:37:12 +02:00
|
|
|
processRedirectHeader: function(req) {
|
2015-03-11 21:24:56 +01:00
|
|
|
var icinga = this.icinga,
|
|
|
|
redirect = req.getResponseHeader('X-Icinga-Redirect');
|
|
|
|
|
|
|
|
if (! redirect) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-08-19 10:00:29 +02:00
|
|
|
redirect = decodeURIComponent(redirect);
|
|
|
|
if (redirect.match(/__SELF__/)) {
|
2015-03-11 21:24:56 +01:00
|
|
|
if (req.autorefresh) {
|
|
|
|
// Redirect to the current window's URL in case it's an auto-refresh request. If authenticated
|
|
|
|
// externally this ensures seamless re-login if the session's expired
|
|
|
|
redirect = redirect.replace(
|
|
|
|
/__SELF__/,
|
|
|
|
encodeURIComponent(
|
|
|
|
document.location.pathname + document.location.search + document.location.hash
|
|
|
|
)
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
// Redirect to the URL which required authentication. When clicking a link this ensures that we
|
|
|
|
// redirect to the link's URL instead of the current window's URL (see above)
|
|
|
|
redirect = redirect.replace(/__SELF__/, req.url);
|
|
|
|
}
|
2014-08-19 10:00:29 +02:00
|
|
|
}
|
2015-03-11 21:24:56 +01:00
|
|
|
|
2015-09-25 13:58:52 +02:00
|
|
|
this.redirectToUrl(redirect, req.$target, req.url, req.getResponseHeader('X-Icinga-Rerender-Layout'));
|
2015-07-21 16:37:21 +02:00
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Redirect to the given url
|
|
|
|
*
|
|
|
|
* @param {string} url
|
|
|
|
* @param {object} $target
|
2015-09-25 13:58:52 +02:00
|
|
|
* @param {string] origin
|
2015-07-21 16:37:21 +02:00
|
|
|
* @param {boolean} rerenderLayout
|
|
|
|
*/
|
2015-09-25 13:58:52 +02:00
|
|
|
redirectToUrl: function (url, $target, origin, rerenderLayout) {
|
2015-07-21 16:37:21 +02:00
|
|
|
var icinga = this.icinga;
|
|
|
|
|
|
|
|
if (typeof rerenderLayout === 'undefined') {
|
|
|
|
rerenderLayout = false;
|
|
|
|
}
|
|
|
|
|
2014-06-30 16:19:05 +02:00
|
|
|
icinga.logger.debug(
|
2015-07-21 16:37:21 +02:00
|
|
|
'Got redirect for ', $target, ', URL was ' + url
|
2014-06-05 21:37:12 +02:00
|
|
|
);
|
2014-06-22 20:06:45 +02:00
|
|
|
|
2015-07-21 16:37:21 +02:00
|
|
|
if (rerenderLayout) {
|
|
|
|
var parts = url.split(/#!/);
|
|
|
|
url = parts.shift();
|
|
|
|
var redirectionUrl = this.addUrlFlag(url, 'renderLayout');
|
2014-08-19 10:00:29 +02:00
|
|
|
var r = this.loadUrl(redirectionUrl, $('#layout'));
|
2015-10-02 11:45:42 +02:00
|
|
|
r.historyUrl = url;
|
2014-08-19 10:00:29 +02:00
|
|
|
if (parts.length) {
|
|
|
|
r.loadNext = parts;
|
2015-03-13 04:04:58 +01:00
|
|
|
} else if (!! document.location.hash) {
|
|
|
|
// Retain detail URL if the layout is rerendered
|
|
|
|
parts = document.location.hash.split('#!').splice(1);
|
|
|
|
if (parts.length) {
|
2015-09-25 13:58:52 +02:00
|
|
|
r.loadNext = $.grep(parts, function (url) {
|
|
|
|
if (url !== origin) {
|
|
|
|
icinga.logger.debug('Retaining detail url ' + url);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
icinga.logger.debug('Discarding detail url ' + url + ' as it\'s the origin of the redirect');
|
|
|
|
return false;
|
|
|
|
});
|
2015-03-13 04:04:58 +01:00
|
|
|
}
|
2014-08-19 10:00:29 +02:00
|
|
|
}
|
2014-06-22 20:06:45 +02:00
|
|
|
} else {
|
2015-07-21 16:37:21 +02:00
|
|
|
if (url.match(/#!/)) {
|
|
|
|
var parts = url.split(/#!/);
|
2014-11-01 01:23:55 +01:00
|
|
|
icinga.ui.layout2col();
|
|
|
|
this.loadUrl(parts.shift(), $('#col1'));
|
|
|
|
this.loadUrl(parts.shift(), $('#col2'));
|
|
|
|
} else {
|
2015-07-21 16:37:21 +02:00
|
|
|
if ($target.attr('id') === 'col2') { // TODO: multicol
|
|
|
|
if ($('#col1').data('icingaUrl').split('?')[0] === url.split('?')[0]) {
|
2014-11-01 01:23:55 +01:00
|
|
|
icinga.ui.layout1col();
|
2015-07-21 16:37:21 +02:00
|
|
|
$target = $('#col1');
|
2014-11-01 01:23:55 +01:00
|
|
|
delete(this.requests['col2']);
|
|
|
|
}
|
2014-06-30 16:19:05 +02:00
|
|
|
}
|
|
|
|
|
2015-07-21 16:37:21 +02:00
|
|
|
this.loadUrl(url, $target);
|
2014-11-01 01:23:55 +01:00
|
|
|
}
|
2014-06-22 20:06:45 +02:00
|
|
|
}
|
2014-06-05 21:37:12 +02:00
|
|
|
},
|
|
|
|
|
2014-06-20 16:18:10 +02:00
|
|
|
cacheLoadedIcons: function($container) {
|
|
|
|
// TODO: this is just a prototype, disabled for now
|
|
|
|
return;
|
|
|
|
|
|
|
|
var self = this;
|
|
|
|
$('img.icon', $container).each(function(idx, img) {
|
|
|
|
var src = $(img).attr('src');
|
|
|
|
if (typeof self.iconCache[src] !== 'undefined') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var cache = new Image();
|
|
|
|
cache.src = src
|
|
|
|
self.iconCache[src] = cache;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
/**
|
|
|
|
* Handle successful XHR response
|
|
|
|
*/
|
|
|
|
onResponse: function (data, textStatus, req) {
|
|
|
|
var self = this;
|
|
|
|
if (this.failureNotice !== null) {
|
2015-01-30 15:25:03 +01:00
|
|
|
if (! this.failureNotice.hasClass('fading-out')) {
|
|
|
|
this.failureNotice.remove();
|
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
this.failureNotice = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
var url = req.url;
|
|
|
|
this.icinga.logger.debug(
|
|
|
|
'Got response for ', req.$target, ', URL was ' + url
|
2014-02-18 20:00:00 +01:00
|
|
|
);
|
2014-06-05 20:46:11 +02:00
|
|
|
this.processNotificationHeader(req);
|
|
|
|
|
2014-06-22 20:01:10 +02:00
|
|
|
var cssreload = req.getResponseHeader('X-Icinga-Reload-Css');
|
|
|
|
if (cssreload) {
|
|
|
|
this.icinga.ui.reloadCss();
|
|
|
|
}
|
|
|
|
|
2015-08-18 15:54:57 +02:00
|
|
|
if (req.getResponseHeader('X-Icinga-Redirect')) {
|
|
|
|
return;
|
|
|
|
}
|
2014-02-18 20:00:00 +01:00
|
|
|
|
2014-03-26 19:01:20 +01:00
|
|
|
// div helps getting an XML tree
|
2014-06-06 14:41:57 +02:00
|
|
|
var $resp = $('<div>' + req.responseText + '</div>');
|
2014-03-04 14:08:29 +01:00
|
|
|
var active = false;
|
2014-03-25 13:13:42 +01:00
|
|
|
var rendered = false;
|
2014-03-28 20:57:24 +01:00
|
|
|
var classes;
|
2014-03-04 14:08:29 +01:00
|
|
|
|
2015-02-02 15:03:00 +01:00
|
|
|
if (req.autorefresh) {
|
2014-03-04 14:08:29 +01:00
|
|
|
// TODO: next container url
|
|
|
|
active = $('[href].active', req.$target).attr('href');
|
|
|
|
}
|
|
|
|
|
|
|
|
var target = req.getResponseHeader('X-Icinga-Container');
|
|
|
|
var newBody = false;
|
2014-06-22 19:55:50 +02:00
|
|
|
var oldNotifications = false;
|
2014-03-04 14:08:29 +01:00
|
|
|
if (target) {
|
2014-03-08 12:07:49 +01:00
|
|
|
if (target === 'ignore') {
|
|
|
|
return;
|
|
|
|
}
|
2014-03-08 23:58:19 +01:00
|
|
|
// If we change the target, oncomplete will fail to clean up
|
|
|
|
// This fixes the problem, not using req.$target would be better
|
|
|
|
delete this.requests[req.$target.attr('id')];
|
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
req.$target = $('#' + target);
|
2014-06-22 19:55:50 +02:00
|
|
|
if (target === 'layout') {
|
|
|
|
oldNotifications = $('#notifications li').detach();
|
|
|
|
}
|
2014-06-21 02:27:27 +02:00
|
|
|
// We assume target === 'layout' right now. This might not be correct
|
|
|
|
this.icinga.ui.layout1col();
|
2014-03-04 14:08:29 +01:00
|
|
|
newBody = true;
|
|
|
|
}
|
|
|
|
|
2014-03-17 17:10:03 +01:00
|
|
|
var moduleName = req.getResponseHeader('X-Icinga-Module');
|
2014-03-28 20:57:24 +01:00
|
|
|
classes = $.grep(req.$target.classes(), function (el) {
|
|
|
|
if (el === 'icinga-module' || el.match(/^module\-/)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
});
|
2014-03-17 17:10:03 +01:00
|
|
|
if (moduleName) {
|
|
|
|
req.$target.data('icingaModule', moduleName);
|
2014-03-28 20:57:24 +01:00
|
|
|
classes.push('icinga-module');
|
|
|
|
classes.push('module-' + moduleName);
|
2014-03-17 17:10:03 +01:00
|
|
|
} else {
|
|
|
|
req.$target.removeData('icingaModule');
|
2014-07-08 20:10:44 +02:00
|
|
|
if (req.$target.attr('data-icinga-module')) {
|
|
|
|
req.$target.removeAttr('data-icinga-module');
|
|
|
|
}
|
2014-03-17 17:10:03 +01:00
|
|
|
}
|
2014-03-28 20:57:24 +01:00
|
|
|
req.$target.attr('class', classes.join(' '));
|
2014-03-17 17:10:03 +01:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
var title = req.getResponseHeader('X-Icinga-Title');
|
2014-05-09 12:13:49 +02:00
|
|
|
if (title && ! req.autorefresh && req.$target.closest('.dashboard').length === 0) {
|
2014-05-28 23:39:34 +02:00
|
|
|
this.icinga.ui.setTitle(decodeURIComponent(title));
|
2014-03-04 14:08:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var refresh = req.getResponseHeader('X-Icinga-Refresh');
|
|
|
|
if (refresh) {
|
|
|
|
req.$target.data('icingaRefresh', refresh);
|
|
|
|
} else {
|
2014-03-26 19:18:00 +01:00
|
|
|
req.$target.removeData('icingaRefresh');
|
2014-07-08 20:10:44 +02:00
|
|
|
if (req.$target.attr('data-icinga-refresh')) {
|
|
|
|
req.$target.removeAttr('data-icinga-refresh');
|
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set a window identifier if the server asks us to do so
|
|
|
|
var windowId = req.getResponseHeader('X-Icinga-WindowId');
|
|
|
|
if (windowId) {
|
|
|
|
this.icinga.ui.setWindowId(windowId);
|
|
|
|
}
|
2014-03-08 00:19:02 +01:00
|
|
|
|
2014-03-25 13:13:42 +01:00
|
|
|
// Handle search requests, still hardcoded.
|
|
|
|
if (req.url.match(/^\/search/) &&
|
2014-03-26 19:01:20 +01:00
|
|
|
req.$target.data('icingaUrl').match(/^\/search/) &&
|
|
|
|
$('.dashboard', $resp).length > 0 &&
|
|
|
|
$('.dashboard .container', req.$target).length > 0)
|
2014-03-04 14:08:29 +01:00
|
|
|
{
|
|
|
|
// TODO: We need dashboard pane and container identifiers (not ids)
|
|
|
|
var targets = [];
|
2014-03-25 13:13:42 +01:00
|
|
|
$('.dashboard .container', req.$target).each(function (idx, el) {
|
2014-03-04 14:08:29 +01:00
|
|
|
targets.push($(el));
|
|
|
|
});
|
|
|
|
|
|
|
|
var i = 0;
|
2014-03-25 13:13:42 +01:00
|
|
|
// Searching for '.dashboard .container' in $resp doesn't dork?!
|
2014-03-26 19:01:20 +01:00
|
|
|
$('.dashboard .container', $resp).each(function (idx, el) {
|
2014-03-04 14:08:29 +01:00
|
|
|
var $el = $(el);
|
2014-03-26 19:01:20 +01:00
|
|
|
if ($el.hasClass('dashboard')) {
|
|
|
|
return;
|
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
var url = $el.data('icingaUrl');
|
|
|
|
targets[i].data('icingaUrl', url);
|
|
|
|
var title = $('h1', $el).first();
|
|
|
|
$('h1', targets[i]).first().replaceWith(title);
|
|
|
|
|
|
|
|
self.loadUrl(url, targets[i]);
|
|
|
|
i++;
|
|
|
|
});
|
2014-03-25 13:13:42 +01:00
|
|
|
rendered = true;
|
2014-03-04 14:08:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
req.$target.data('icingaUrl', req.url);
|
|
|
|
|
2014-04-16 19:43:56 +02:00
|
|
|
this.icinga.ui.initializeTriStates($resp);
|
2014-03-26 14:56:35 +01:00
|
|
|
|
2015-08-18 15:54:57 +02:00
|
|
|
if (rendered) {
|
|
|
|
return;
|
2014-03-04 14:08:29 +01:00
|
|
|
}
|
2014-03-25 13:13:42 +01:00
|
|
|
|
2015-08-20 14:38:03 +02:00
|
|
|
if (typeof req.progressTimer !== 'undefined') {
|
|
|
|
this.icinga.timer.unregister(req.progressTimer);
|
|
|
|
}
|
|
|
|
|
2014-03-26 19:01:20 +01:00
|
|
|
// .html() removes outer div we added above
|
2014-06-20 13:52:05 +02:00
|
|
|
this.renderContentToContainer($resp.html(), req.$target, req.action, req.autorefresh);
|
2014-06-22 19:55:50 +02:00
|
|
|
if (oldNotifications) {
|
|
|
|
oldNotifications.appendTo($('#notifications'));
|
|
|
|
}
|
2014-03-20 12:24:15 +01:00
|
|
|
if (url.match(/#/)) {
|
2015-08-06 11:51:57 +02:00
|
|
|
this.icinga.ui.focusElement(url.split(/#/)[1], req.$target);
|
2014-03-20 12:24:15 +01:00
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
if (newBody) {
|
|
|
|
this.icinga.ui.fixDebugVisibility().triggerWindowResize();
|
|
|
|
}
|
2014-06-20 16:18:10 +02:00
|
|
|
self.cacheLoadedIcons(req.$target);
|
2014-03-04 14:08:29 +01:00
|
|
|
},
|
|
|
|
|
2014-03-08 16:20:30 +01:00
|
|
|
/**
|
|
|
|
* Regardless of whether a request succeeded of failed, clean up
|
|
|
|
*/
|
2014-03-04 14:08:29 +01:00
|
|
|
onComplete: function (req, textStatus) {
|
2015-02-02 15:18:05 +01:00
|
|
|
// Remove 'impact' class if there was such
|
|
|
|
if (req.$target.hasClass('impact')) {
|
|
|
|
req.$target.removeClass('impact');
|
|
|
|
}
|
|
|
|
|
2015-02-02 15:03:00 +01:00
|
|
|
if (! req.autorefresh) {
|
|
|
|
// TODO: Hook for response/url?
|
|
|
|
var url = req.url;
|
2015-08-25 16:55:10 +02:00
|
|
|
|
|
|
|
if (req.$target[0].id === 'col1') {
|
|
|
|
self.icinga.behaviors.navigation.trySetActiveByUrl(url);
|
|
|
|
}
|
|
|
|
|
2015-02-02 15:03:00 +01:00
|
|
|
var $forms = $('[action="' + this.icinga.utils.parseUrl(url).path + '"]');
|
|
|
|
var $matches = $.merge($('[href="' + url + '"]'), $forms);
|
|
|
|
$matches.each(function (idx, el) {
|
|
|
|
var $el = $(el);
|
|
|
|
if ($el.closest('#menu').length) {
|
|
|
|
if ($el.is('form')) {
|
|
|
|
$('input', $el).addClass('active');
|
|
|
|
}
|
2015-08-25 16:55:10 +02:00
|
|
|
// Interrupt .each, only one menu item shall be active
|
2015-02-02 15:03:00 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2015-08-26 15:23:58 +02:00
|
|
|
// Update history when necessary
|
2015-08-28 16:04:57 +02:00
|
|
|
if (! req.autorefresh && req.addToHistory) {
|
2015-07-06 16:56:44 +02:00
|
|
|
if (req.$target.hasClass('container')) {
|
2014-06-21 01:57:59 +02:00
|
|
|
// We only want to care about top-level containers
|
|
|
|
if (req.$target.parent().closest('.container').length === 0) {
|
|
|
|
this.icinga.history.pushCurrentState();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Request wasn't for a container, so it's usually the body
|
|
|
|
// or the full layout. Push request URL to history:
|
2015-10-02 11:45:42 +02:00
|
|
|
var url = typeof req.historyUrl !== 'undefined' ? req.historyUrl : req.url;
|
|
|
|
this.icinga.history.pushUrl(url);
|
2014-06-20 16:16:04 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-26 11:35:45 +01:00
|
|
|
req.$target.data('lastUpdate', (new Date()).getTime());
|
2014-03-04 14:08:29 +01:00
|
|
|
delete this.requests[req.$target.attr('id')];
|
2014-03-08 00:19:02 +01:00
|
|
|
this.icinga.ui.fadeNotificationsAway();
|
2014-08-19 10:00:29 +02:00
|
|
|
|
2015-10-02 11:45:55 +02:00
|
|
|
this.processRedirectHeader(req);
|
|
|
|
|
2015-09-25 13:58:52 +02:00
|
|
|
if (typeof req.loadNext !== 'undefined' && req.loadNext.length) {
|
2014-08-19 10:00:29 +02:00
|
|
|
if ($('#col2').length) {
|
2015-10-01 10:30:18 +02:00
|
|
|
var r = this.loadUrl(req.loadNext[0], $('#col2'));
|
|
|
|
r.addToHistory = req.addToHistory;
|
2014-08-19 10:00:29 +02:00
|
|
|
this.icinga.ui.layout2col();
|
|
|
|
} else {
|
|
|
|
this.icinga.logger.error('Failed to load URL for #col2', req.loadNext);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-11 17:41:24 +02:00
|
|
|
req.$target.trigger('rendered');
|
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
this.icinga.ui.refreshDebug();
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle failed XHR response
|
|
|
|
*/
|
|
|
|
onFailure: function (req, textStatus, errorThrown) {
|
|
|
|
var url = req.url;
|
|
|
|
|
2015-02-02 16:34:29 +01:00
|
|
|
/*
|
|
|
|
* Test if a manual actions comes in and autorefresh is active: Stop refreshing
|
|
|
|
*/
|
2015-08-26 15:23:58 +02:00
|
|
|
if (req.addToHistory && ! req.autorefresh) {
|
2015-02-02 16:34:29 +01:00
|
|
|
req.$target.data('icingaRefresh', 0);
|
|
|
|
req.$target.data('icingaUrl', url);
|
2015-10-02 11:45:55 +02:00
|
|
|
icinga.history.pushCurrentState();
|
2015-02-02 16:34:29 +01:00
|
|
|
}
|
|
|
|
|
2015-08-20 14:38:03 +02:00
|
|
|
if (typeof req.progressTimer !== 'undefined') {
|
|
|
|
this.icinga.timer.unregister(req.progressTimer);
|
|
|
|
}
|
|
|
|
|
2015-01-30 14:50:25 +01:00
|
|
|
if (req.status > 0) {
|
2014-05-09 14:08:38 +02:00
|
|
|
this.icinga.logger.error(
|
|
|
|
req.status,
|
|
|
|
errorThrown + ':',
|
2015-04-16 12:13:31 +02:00
|
|
|
$(req.responseText).text().replace(/\s+/g, ' ').slice(0, 100)
|
2014-05-09 14:08:38 +02:00
|
|
|
);
|
2014-03-04 14:08:29 +01:00
|
|
|
this.renderContentToContainer(
|
2014-03-25 13:13:42 +01:00
|
|
|
req.responseText,
|
2014-04-01 10:43:47 +02:00
|
|
|
req.$target,
|
2014-06-20 13:52:05 +02:00
|
|
|
req.action,
|
|
|
|
req.autorefresh
|
2014-02-18 20:00:00 +01:00
|
|
|
);
|
2014-03-04 14:08:29 +01:00
|
|
|
} else {
|
|
|
|
if (errorThrown === 'abort') {
|
2014-05-05 16:17:21 +02:00
|
|
|
this.icinga.logger.debug(
|
2014-03-04 14:08:29 +01:00
|
|
|
'Request to ' + url + ' has been aborted for ',
|
|
|
|
req.$target
|
|
|
|
);
|
2015-02-16 15:56:50 +01:00
|
|
|
|
|
|
|
// Aborted requests should not be added to browser history
|
|
|
|
req.addToHistory = false;
|
2014-03-04 14:08:29 +01:00
|
|
|
} else {
|
|
|
|
if (this.failureNotice === null) {
|
|
|
|
this.failureNotice = this.createNotice(
|
|
|
|
'error',
|
2014-05-29 11:26:02 +02:00
|
|
|
'The connection to the Icinga web server was lost at ' +
|
2014-03-04 14:08:29 +01:00
|
|
|
this.icinga.utils.timeShort() +
|
2014-03-08 00:19:02 +01:00
|
|
|
'.',
|
|
|
|
true
|
2014-03-04 14:08:29 +01:00
|
|
|
);
|
2014-03-06 13:08:11 +01:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
this.icinga.ui.fixControls();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.icinga.logger.error(
|
|
|
|
'Failed to contact web server loading ',
|
|
|
|
url,
|
|
|
|
' for ',
|
|
|
|
req.$target
|
|
|
|
);
|
|
|
|
}
|
2014-02-18 20:00:00 +01:00
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
},
|
|
|
|
|
2014-03-08 16:20:30 +01:00
|
|
|
/**
|
|
|
|
* Create a notification. Can be improved.
|
|
|
|
*/
|
2014-03-08 12:17:56 +01:00
|
|
|
createNotice: function (severity, message, persist) {
|
2014-03-08 00:19:02 +01:00
|
|
|
var c = severity;
|
|
|
|
if (persist) {
|
|
|
|
c += ' persist';
|
|
|
|
}
|
2014-03-08 12:07:49 +01:00
|
|
|
var $notice = $(
|
2014-03-08 00:19:02 +01:00
|
|
|
'<li class="' + c + '">' + message + '</li>'
|
2014-03-04 14:08:29 +01:00
|
|
|
).appendTo($('#notifications'));
|
2015-01-30 15:25:03 +01:00
|
|
|
|
2014-03-08 12:07:49 +01:00
|
|
|
this.icinga.ui.fixControls();
|
2015-01-30 15:25:03 +01:00
|
|
|
|
|
|
|
if (!persist) {
|
|
|
|
this.icinga.ui.fadeNotificationsAway();
|
|
|
|
}
|
|
|
|
|
2014-03-08 12:07:49 +01:00
|
|
|
return $notice;
|
2014-03-04 14:08:29 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Smoothly render given HTML to given container
|
|
|
|
*/
|
2014-06-20 13:52:05 +02:00
|
|
|
renderContentToContainer: function (content, $container, action, autorefresh) {
|
2014-03-04 14:08:29 +01:00
|
|
|
// Container update happens here
|
|
|
|
var scrollPos = false;
|
2014-06-24 07:01:37 +02:00
|
|
|
var self = this;
|
2014-03-04 14:08:29 +01:00
|
|
|
var containerId = $container.attr('id');
|
|
|
|
if (typeof containerId !== 'undefined') {
|
2014-09-02 20:11:37 +02:00
|
|
|
if (autorefresh) {
|
|
|
|
scrollPos = $container.scrollTop();
|
|
|
|
} else {
|
|
|
|
scrollPos = 0;
|
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
}
|
2015-02-17 09:24:29 +01:00
|
|
|
if (autorefresh && $.contains($container[0], document.activeElement)) {
|
|
|
|
var origFocus = self.icinga.utils.getDomPath(document.activeElement);
|
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
|
2014-11-01 01:12:59 +01:00
|
|
|
$container.trigger('beforerender');
|
|
|
|
|
2014-09-19 15:48:44 +02:00
|
|
|
var discard = false;
|
|
|
|
$.each(self.icinga.behaviors, function(name, behavior) {
|
|
|
|
if (behavior.renderHook) {
|
|
|
|
var changed = behavior.renderHook(content, $container, action, autorefresh);
|
|
|
|
if (!changed) {
|
|
|
|
discard = true;
|
|
|
|
} else {
|
|
|
|
content = changed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
if (discard) {
|
2014-06-20 13:52:05 +02:00
|
|
|
return;
|
|
|
|
}
|
2014-05-28 17:59:42 +02:00
|
|
|
|
|
|
|
// TODO: We do not want to wrap this twice...
|
|
|
|
var $content = $('<div>' + content + '</div>');
|
|
|
|
|
2014-06-20 13:52:05 +02:00
|
|
|
// Disable all click events while rendering
|
2014-11-01 01:11:12 +01:00
|
|
|
// (Disabling disabled, was ways too slow)
|
|
|
|
// $('*').click(function (event) {
|
|
|
|
// event.stopImmediatePropagation();
|
|
|
|
// event.stopPropagation();
|
|
|
|
// event.preventDefault();
|
|
|
|
// });
|
2014-06-20 13:52:05 +02:00
|
|
|
|
2014-06-24 07:01:37 +02:00
|
|
|
$('.container', $container).each(function() {
|
|
|
|
self.stopPendingRequestsFor($(this));
|
|
|
|
});
|
2014-06-20 13:52:05 +02:00
|
|
|
|
2014-03-04 14:08:29 +01:00
|
|
|
if (false &&
|
|
|
|
$('.dashboard', $content).length > 0 &&
|
|
|
|
$('.dashboard', $container).length === 0
|
|
|
|
) {
|
|
|
|
// $('.dashboard', $content)
|
|
|
|
// $container.html(content);
|
|
|
|
|
|
|
|
} else {
|
2015-04-15 13:23:34 +02:00
|
|
|
if ($container.closest('.dashboard').length) {
|
|
|
|
var title = $('h1', $container).first().detach();
|
2014-03-04 14:08:29 +01:00
|
|
|
$container.html(title).append(content);
|
2014-04-01 10:43:47 +02:00
|
|
|
} else if (action === 'replace') {
|
2014-03-04 14:08:29 +01:00
|
|
|
$container.html(content);
|
2014-04-01 10:43:47 +02:00
|
|
|
} else {
|
|
|
|
$container.append(content);
|
2014-03-04 14:08:29 +01:00
|
|
|
}
|
|
|
|
}
|
2014-06-24 06:40:08 +02:00
|
|
|
this.icinga.ui.assignUniqueContainerIds();
|
2014-03-20 12:24:15 +01:00
|
|
|
|
2015-02-17 09:24:29 +01:00
|
|
|
if (origFocus && origFocus.length > 0 && origFocus[0] !== '') {
|
2015-02-04 17:57:31 +01:00
|
|
|
setTimeout(function() {
|
|
|
|
$(self.icinga.utils.getElementByDomPath(origFocus)).focus();
|
|
|
|
}, 0);
|
2014-03-04 14:08:29 +01:00
|
|
|
}
|
|
|
|
|
2014-10-14 15:50:15 +02:00
|
|
|
if (scrollPos !== false) {
|
|
|
|
$container.scrollTop(scrollPos);
|
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
var icinga = this.icinga;
|
2014-09-12 09:09:21 +02:00
|
|
|
//icinga.events.applyHandlers($container);
|
2014-03-04 14:08:29 +01:00
|
|
|
icinga.ui.initializeControls($container);
|
|
|
|
icinga.ui.fixControls();
|
|
|
|
|
2014-11-01 01:11:12 +01:00
|
|
|
// Re-enable all click events (disabled as of performance reasons)
|
|
|
|
// $('*').off('click');
|
2014-03-04 14:08:29 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* On shutdown we kill all pending requests
|
|
|
|
*/
|
|
|
|
destroy: function() {
|
|
|
|
$.each(this.requests, function(id, request) {
|
|
|
|
request.abort();
|
|
|
|
});
|
|
|
|
this.icinga = null;
|
|
|
|
this.requests = {};
|
2014-02-18 20:00:00 +01:00
|
|
|
}
|
2014-03-04 14:08:29 +01:00
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}(Icinga, jQuery));
|