/*! Icinga Web 2 | (c) 2014 Icinga Development Team | GPLv2+ */ /** * Icinga.Loader * * This is where we take care of XHR requests, responses and failures. */ (function(Icinga, $) { '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 = {}; this.iconCache = {}; /** * Whether auto-refresh is enabled */ this.autorefreshEnabled = true; /** * Whether auto-refresh is suspended due to visibility of page */ this.autorefreshSuspended = false; }; Icinga.Loader.prototype = { initialize: function () { this.icinga.timer.register(this.autorefresh, this, 500); }, /** * Load the given URL to the given target * * @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 */ loadUrl: function (url, $target, data, method, action, autorefresh, progressTimer) { var id = null; // Default method is GET if ('undefined' === typeof method) { method = 'GET'; } if ('undefined' === typeof action) { action = 'replace'; } if ('undefined' === typeof autorefresh) { autorefresh = false; } 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'); } // If we have a pending request for the same target... if (typeof this.requests[id] !== 'undefined') { if (autorefresh) { return false; } // ... 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') { this.icinga.logger.debug('Request to ', url, ' is already running for ', $target); return this.requests[id]; } // ...or abort the former request otherwise this.icinga.logger.debug( 'Aborting pending request loading ', url, ' to ', $target ); this.requests[id].abort(); } // Not sure whether we need this Accept-header var headers = { 'X-Icinga-Accept': 'text/html' }; // Ask for a new window id in case we don't already have one if (this.icinga.ui.hasWindowId()) { var windowId = this.icinga.ui.getWindowId(); var containerId = this.icinga.ui.getUniqueContainerId($target); if (containerId) { windowId = windowId + '_' + containerId; } headers['X-Icinga-WindowId'] = windowId; } else { headers['X-Icinga-WindowId'] = 'undefined'; } // This is jQuery's default content type var contentType = 'application/x-www-form-urlencoded; charset=UTF-8'; var isFormData = typeof window.FormData !== 'undefined' && data instanceof window.FormData; if (isFormData) { // Setting false is mandatory as the form's data // won't be recognized by the server otherwise contentType = false; } var _this = this; var req = $.ajax({ type : method, url : url, data : data, headers: headers, context: _this, contentType: contentType, processData: ! isFormData }); req.$target = $target; req.url = url; req.done(this.onResponse); req.fail(this.onFailure); req.complete(this.onComplete); req.autorefresh = autorefresh; req.method = method; req.action = action; req.addToHistory = true; req.progressTimer = progressTimer; if (url.match(/#/)) { req.forceFocus = url.split(/#/)[1]; } if (id) { this.requests[id] = req; } if (! autorefresh) { req.$target.addClass('impact'); } this.icinga.ui.refreshDebug(); return req; }, /** * 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 _this = this; $form.prop('action', _this.icinga.utils.addUrlParams(action, { '_frameUpload': true })); $form.prop('target', 'fileupload-frame-target'); $('#fileupload-frame-target').on('load', function (event) { var $frame = $(event.target); var $contents = $frame.contents(); var $redirectMeta = $contents.find('meta[name="redirectUrl"]'); if ($redirectMeta.length) { _this.redirectToUrl($redirectMeta.attr('content'), $target); } else { // Fetch the frame's new content and paste it into the target _this.renderContentToContainer( $contents.find('body').html(), $target, 'replace' ); } $frame.prop('src', 'about:blank'); // Clear the frame's dom $frame.off('load'); // Unbind the event as it's set on demand }); }, /** * 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; }, stopPendingRequestsFor: function ($el) { var id; if (typeof $el === 'undefined' || ! (id = $el.attr('id'))) { return; } if (typeof this.requests[id] !== 'undefined') { this.requests[id].abort(); } }, filterAutorefreshingContainers: function () { return $(this).data('icingaRefresh') > 0; }, autorefresh: function () { var _this = this; $('.container').filter(this.filterAutorefreshingContainers).each(function (idx, el) { var $el = $(el); var id = $el.attr('id'); // Always request application-state if (id !== 'application-state' && (! _this.autorefreshEnabled || _this.autorefreshSuspended)) { // Continue return true; } if (typeof _this.requests[id] !== 'undefined') { _this.icinga.logger.debug('No refresh, request pending for ', id); return; } var interval = $el.data('icingaRefresh'); var lastUpdate = $el.data('lastUpdate'); if (typeof interval === 'undefined' || ! interval) { _this.icinga.logger.info('No interval, setting default', id); interval = 10; } if (typeof lastUpdate === 'undefined' || ! lastUpdate) { _this.icinga.logger.info('No lastUpdate, setting one', id); $el.data('lastUpdate',(new Date()).getTime()); return; } interval = interval * 1000; // TODO: if ((lastUpdate + interval) > (new Date()).getTime()) { // self.icinga.logger.info( // 'Skipping refresh', // id, // lastUpdate, // interval, // (new Date()).getTime() // ); return; } if (_this.loadUrl($el.data('icingaUrl'), $el, undefined, undefined, undefined, true) === false) { _this.icinga.logger.debug( 'NOT autorefreshing ' + id + ', even if ' + interval + ' ms passed. Request pending?' ); } else { _this.icinga.logger.debug( 'Autorefreshing ' + id + ' ' + interval + ' ms passed' ); } el = null; }); }, /** * Disable the autorefresh mechanism */ disableAutorefresh: function () { this.autorefreshEnabled = false; }, /** * Enable the autorefresh mechanism */ enableAutorefresh: function () { this.autorefreshEnabled = true; }, processNotificationHeader: function(req) { var header = req.getResponseHeader('X-Icinga-Notification'); var _this = this; if (! header) return false; var list = header.split('&'); $.each(list, function(idx, el) { var parts = decodeURIComponent(el).split(' '); _this.createNotice(parts.shift(), parts.join(' ')); }); return true; }, addUrlFlag: function(url, flag) { if (url.match(/\?/)) { return url + '&' + flag; } else { return url + '?' + flag; } }, /** * 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 */ processRedirectHeader: function(req) { var icinga = this.icinga, redirect = req.getResponseHeader('X-Icinga-Redirect'); if (! redirect) { return false; } redirect = decodeURIComponent(redirect); if (redirect.match(/__SELF__/)) { 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); } } var useHttp = req.getResponseHeader('X-Icinga-Redirect-Http'); if (useHttp === 'yes') { window.location.replace(redirect); return true; } this.redirectToUrl(redirect, req.$target, req); return true; }, /** * Redirect to the given url * * @param {string} url * @param {object} $target * @param {XMLHttpRequest} referrer */ redirectToUrl: function (url, $target, referrer) { var icinga = this.icinga, rerenderLayout, autoRefreshInterval, forceFocus, origin; if (typeof referrer !== 'undefined') { rerenderLayout = referrer.getResponseHeader('X-Icinga-Rerender-Layout'); autoRefreshInterval = referrer.autoRefreshInterval; forceFocus = referrer.forceFocus; origin = referrer.url; } icinga.logger.debug( 'Got redirect for ', $target, ', URL was ' + url ); if (rerenderLayout) { var parts = url.split(/#!/); url = parts.shift(); var redirectionUrl = this.addUrlFlag(url, 'renderLayout'); var r = this.loadUrl(redirectionUrl, $('#layout')); r.historyUrl = url; if (parts.length) { r.loadNext = parts; } else if (!! document.location.hash) { // Retain detail URL if the layout is rerendered parts = document.location.hash.split('#!').splice(1); if (parts.length) { 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; }); } } } else { if (url.match(/#!/)) { var parts = url.split(/#!/); icinga.ui.layout2col(); this.loadUrl(parts.shift(), $('#col1')); this.loadUrl(parts.shift(), $('#col2')); } else { if ($target.attr('id') === 'col2') { // TODO: multicol if ($('#col1').data('icingaUrl').split('?')[0] === url.split('?')[0]) { icinga.ui.layout1col(); $target = $('#col1'); delete(this.requests['col2']); } } var req = this.loadUrl(url, $target); req.forceFocus = url === origin ? forceFocus : null; req.autoRefreshInterval = autoRefreshInterval; req.referrer = referrer; } } }, cacheLoadedIcons: function($container) { // TODO: this is just a prototype, disabled for now return; var _this = this; $('img.icon', $container).each(function(idx, img) { var src = $(img).attr('src'); if (typeof _this.iconCache[src] !== 'undefined') { return; } var cache = new Image(); cache.src = src _this.iconCache[src] = cache; }); }, /** * Handle successful XHR response */ onResponse: function (data, textStatus, req) { var _this = this; if (this.failureNotice !== null) { if (! this.failureNotice.hasClass('fading-out')) { this.failureNotice.remove(); } this.failureNotice = null; } this.icinga.logger.debug( 'Got response for ', req.$target, ', URL was ' + req.url ); this.processNotificationHeader(req); var cssreload = req.getResponseHeader('X-Icinga-Reload-Css'); if (cssreload) { this.icinga.ui.reloadCss(); } if (req.getResponseHeader('X-Icinga-Redirect')) { return; } if (req.getResponseHeader('X-Icinga-Announcements') === 'refresh') { _this.loadUrl(_this.url('/layout/announcements'), $('#announcements')); } // div helps getting an XML tree var $resp = $('