diff --git a/GitVersion.yml b/GitVersion.yml index dea097f8d..d296a4ce2 100644 --- a/GitVersion.yml +++ b/GitVersion.yml @@ -1,5 +1,5 @@ assembly-versioning-scheme: Major -next-version: 1.19 +next-version: 1.20 branches: release: tag: rc diff --git a/src/ServicePulse.Host.Tests/ApprovalFiles/APIApprovals.PlatformSampleApprovals.approved.txt b/src/ServicePulse.Host.Tests/ApprovalFiles/APIApprovals.PlatformSampleApprovals.approved.txt index 76600dd73..87aac507b 100644 --- a/src/ServicePulse.Host.Tests/ApprovalFiles/APIApprovals.PlatformSampleApprovals.approved.txt +++ b/src/ServicePulse.Host.Tests/ApprovalFiles/APIApprovals.PlatformSampleApprovals.approved.txt @@ -1,14 +1,6 @@ -;(function (window, angular, undefined) { 'use strict'; - - window.config = { - default_route: '/dashboard', - service_control_url: 'http://localhost:33333/api/', - monitoring_urls: ['http://localhost:33633/'] - }; - - angular.module('sc') - .constant('version', '1.2.0') - .constant('showPendingRetry', false) - .constant('scConfig', window.config); - -}(window, window.angular)); +window.defaultConfig = { + default_route: '/dashboard', + version: '1.2.0', + service_control_url: 'http://localhost:33333/api/', + monitoring_urls: ['http://localhost:33633/'] +}; diff --git a/src/ServicePulse.Host.Tests/ServicePulse.Host.Tests.csproj b/src/ServicePulse.Host.Tests/ServicePulse.Host.Tests.csproj index d7a07f38b..9c735cec4 100644 --- a/src/ServicePulse.Host.Tests/ServicePulse.Host.Tests.csproj +++ b/src/ServicePulse.Host.Tests/ServicePulse.Host.Tests.csproj @@ -13,6 +13,7 @@ + diff --git a/src/ServicePulse.Host.Tests/VerifyAppConstantsJSTextReplacement.cs b/src/ServicePulse.Host.Tests/VerifyAppConstantsJSTextReplacement.cs index 1c192e2fa..20162f1d0 100644 --- a/src/ServicePulse.Host.Tests/VerifyAppConstantsJSTextReplacement.cs +++ b/src/ServicePulse.Host.Tests/VerifyAppConstantsJSTextReplacement.cs @@ -1,13 +1,10 @@ -using System.Reflection; - -namespace ServicePulse.Host.Tests +namespace ServicePulse.Host.Tests { + using NUnit.Framework; using System; using System.Collections.Generic; using System.IO; - using System.Text.RegularExpressions; - using NUnit.Framework; [TestFixture] public class VerifyAppConstantsJSTextReplacement @@ -16,14 +13,14 @@ public class VerifyAppConstantsJSTextReplacement // this both user configurable and updated during installation Regex sc_url_regex = new Regex(@"(service_control_url\s*\:\s*['""])(.*?)(['""])"); - Regex version_regex = new Regex(@"(constant\s*\(\s*'version'\s*,\s*['""])(.*?)(['""])"); + Regex version_regex = new Regex(@"(version\s*\:\s*['""])(.*?)(['""])"); [Test] public void app_constants_js_validation() { var pathToConfig = Path.Combine(TestContext.CurrentContext.TestDirectory, "app.constants.js"); Assert.IsTrue(File.Exists(pathToConfig), "app.constants.js does not exist - this will break installation code"); - + var config = File.ReadAllText(pathToConfig); var matchUrl = sc_url_regex.Match(config); Assert.IsTrue(matchUrl.Success, "regex failed to match app.constant.js for SC URI update"); @@ -37,35 +34,71 @@ public void app_constants_js_validation() [Test] public void replace_version_regex_tests() { - var configSnippets = new Dictionary + var configSnippets = new Dictionary() { - {"1.3.0", @"angular.module('sc') - .constant('version', '1.3.0') - .constant('scConfig';"}, - {"1.3.0-beta1", @"angular.module('sc') - .constant ( 'version' , '1.3.0-beta1') - .constant('scConfig';"}, - {"", @"angular.module('sc') - .constant('version' , '' ) - .constant('scConfig';"} + { + "1.3.0", + ( + ConfigSnippet: @"angular.module('sc') + .constant('version', '1.3.0') + .constant('scConfig';", + VersionRegex: new Regex(@"(constant\s*\(\s*'version'\s*,\s*['""])(.*?)(['""])") + ) + }, + { + "1.3.0-beta1", + ( + ConfigSnippet: @"angular.module('sc') + .constant ( 'version' , '1.3.0-beta1') + .constant('scConfig';", + VersionRegex: new Regex(@"(constant\s*\(\s*'version'\s*,\s*['""])(.*?)(['""])") + ) + }, + { + "", + ( + ConfigSnippet: @"angular.module('sc') + .constant('version' , '' ) + .constant('scConfig';", + VersionRegex: new Regex(@"(constant\s*\(\s*'version'\s*,\s*['""])(.*?)(['""])") + ) + }, + { + "1.20.0", + ( + ConfigSnippet: @"window.defaultConfig = { + version: '1.20.0', + service_control_url: ' + };", + VersionRegex: new Regex(@"(version\s*\:\s*['""])(.*?)(['""])") + ) + }, }; foreach (var config in configSnippets) { var expectedResult = config.Key; - var text = config.Value; + var text = config.Value.ConfigSnippet; - var match = version_regex.Match(text); + var match = config.Value.VersionRegex.Match(text); Assert.IsTrue(match.Success, "regex failed to match version string"); Assert.IsTrue(match.Groups[2].Value.Equals(expectedResult), string.Format("Version regex did not return expected value which was {0}", expectedResult)); } } - + [Test] - public void test_regex_match_against_config_variants () + public void test_regex_match_against_config_variants() { var configVariations = new[] { + // Standard 1.20.0 config + @"window.defaultConfig = { + default_route: '/dashboard', + version: '1.20.0', + service_control_url: 'http://localhost:33333/api/', + monitoring_urls: ['http://localhost:33633/'] + }; + ", // Standard 1.3 config @"angular.module('sc') .constant('version', '1.3.0') diff --git a/src/ServicePulse.Host.Tests/karma.conf.js b/src/ServicePulse.Host.Tests/karma.conf.js index 3b7ba3376..16e95eb80 100644 --- a/src/ServicePulse.Host.Tests/karma.conf.js +++ b/src/ServicePulse.Host.Tests/karma.conf.js @@ -4,17 +4,21 @@ module.exports = function(config) { browsers: ['PhantomJS', 'PhantomJS_custom', 'Chrome'], basePath: '../ServicePulse.Host/app', files: [ - './modules/dist/shell.dist.js', - '../../ServicePulse.Host.Tests/tests/js/angular-mocks.js', - './js/app.js', + './modules/dist/shell.dist.js', + '../../ServicePulse.Host.Tests/tests/js/angular-mocks.js', './js/**/*.html', './js/app.constants.js', + './modules/dist/configuration.dist.js', + './js/app.js', + './js/app.bootstrap.js', './js/**/*.module.js', './js/**/*.tabset.js', - './js/**/*.js', - './modules/dist/configuration.dist.js', - './modules/dist/monitoring.dist.js', - '../../ServicePulse.Host.Tests/tests/**/*.spec.js'], + './js/directives/**/*.js', + './js/polyfill/**/*.js', + './js/services/**/*.js', + './js/views/**/*.js', + './modules/dist/monitoring.dist.js', + '../../ServicePulse.Host.Tests/tests/**/*.spec.js'], frameworks: ['jasmine'], // you can define custom flags customLaunchers: { @@ -36,12 +40,12 @@ module.exports = function(config) { }, proxies: { - '/js/views/dashboard/dashboard.html': '/base/js/views/dashboard/dashboard.html' + '/js/views/dashboard/dashboard.html': '/base/js/views/dashboard/dashboard.html' }, phantomjsLauncher: { // Have phantomjs exit if a ResourceError is encountered (useful if karma exits without killing phantom) exitOnResourceError: true - } + } }) } \ No newline at end of file diff --git a/src/ServicePulse.Host.Tests/package.json b/src/ServicePulse.Host.Tests/package.json index 764d0400c..b6bb982aa 100644 --- a/src/ServicePulse.Host.Tests/package.json +++ b/src/ServicePulse.Host.Tests/package.json @@ -1,9 +1,7 @@ { "name": "ServicePulse.Host.Tests", "version": "1.0.0", - "dependencies": { - - }, + "dependencies": {}, "devDependencies": { "karma-chrome-launcher": "^2.2.0", "karma-firefox-launcher": "^1.1.0", diff --git a/src/ServicePulse.Host.Tests/tests/js/services/services.monitoring.spec.js b/src/ServicePulse.Host.Tests/tests/js/services/services.monitoring.spec.js index 8af317602..573ec8205 100644 --- a/src/ServicePulse.Host.Tests/tests/js/services/services.monitoring.spec.js +++ b/src/ServicePulse.Host.Tests/tests/js/services/services.monitoring.spec.js @@ -22,14 +22,14 @@ var monitoringService, $httpBackend, scConfig; - beforeEach(inject(function (_monitoringService_, _$httpBackend_, _scConfig_) { + beforeEach(inject(function (_monitoringService_, _$httpBackend_) { monitoringService = _monitoringService_; $httpBackend = _$httpBackend_; - scConfig = _scConfig_; + scConfig = window.defaultConfig; })); it('should push endpoints retrieved from monitoring server', function (done) { - scConfig.monitoring_urls = ['http://localhost:33633/']; + window.defaultConfig.monitoring_urls = ['http://localhost:33633/']; $httpBackend.whenGET('http://localhost:33633/monitored-endpoints?history=5').respond(monitoredEndpointWithData); @@ -37,7 +37,7 @@ var subscription = monitoringService.createEndpointsSource(5).subscribe(function (response) { monitoredEndpoints.push(response); - if (monitoredEndpoints.length == 2) { + if (monitoredEndpoints.length === 2) { expect(monitoredEndpoints[0].name).toEqual("Samples.Metrics.Tracing.Endpoint1"); expect(monitoredEndpoints[0].data.timestamps).toEqual([]); expect(monitoredEndpoints[0].data.criticalTime).toEqual([]); @@ -56,26 +56,4 @@ $httpBackend.flush(); }, 0); }); - - it('should push endpoints retrieved from multiple monitoring servers', function (done) { - $httpBackend.whenGET('http://localhost:1234/monitored-endpoints?history=5').respond(monitoredEndpointWithData); - $httpBackend.whenGET('http://localhost:5678/monitored-endpoints?history=5').respond(monitoredEndpointWithData); - - scConfig.monitoring_urls = ['http://localhost:1234/', 'http://localhost:5678/']; - - var monitoredEndpoints = []; - var subscription = monitoringService.createEndpointsSource(5).subscribe(function (response) { - monitoredEndpoints.push(response); - - if (monitoredEndpoints.length == 4) { - expect(monitoredEndpoints.length).toEqual(4); - subscription.dispose(); - done(); - } - }); - - setTimeout(function () { - $httpBackend.flush(); - }, 0); - }); }); \ No newline at end of file diff --git a/src/ServicePulse.Host.Tests/tests/js/services/services.spec.js b/src/ServicePulse.Host.Tests/tests/js/services/services.spec.js index 5aa46fca1..214d8c523 100644 --- a/src/ServicePulse.Host.Tests/tests/js/services/services.spec.js +++ b/src/ServicePulse.Host.Tests/tests/js/services/services.spec.js @@ -3,7 +3,7 @@ describe("Unit: Uri Service ", function() { it('should load angular', function() { - expect(typeof (angular) == typeof (undefined)).toEqual(false); + expect(typeof (angular) === typeof (undefined)).toEqual(false); }); describe("Uri Joins:", function() { @@ -12,7 +12,7 @@ describe("Unit: Uri Service ", function() { it('should contain a uri service', inject(function(uri) { - expect(typeof (uri) == typeof(undefined)).toEqual(false); + expect(typeof (uri) === typeof(undefined)).toEqual(false); })); it('should prevent double slashes for leading and trailing slashes', inject(function (uri) { @@ -25,39 +25,39 @@ describe("Unit: Uri Service ", function() { expect(url).toEqual('http://localhost:33333/api/eventlogitems'); })); - it('should make a url for get Failed Messages For Exception Group', inject(function (uri, scConfig) { + it('should make a url for get Failed Messages For Exception Group', inject(function (uri) { var groupId = '85147b12-458c-431d-a389-35ea53abc9e1'; var page = 1; var sortBy = 'time_of_failure'; - var url = uri.join(scConfig.service_control_url, 'recoverability', 'groups', groupId, 'errors?page=' + page + '&sort=' + sortBy + '&status=unresolved'); + var url = uri.join(window.defaultConfig.service_control_url, 'recoverability', 'groups', groupId, 'errors?page=' + page + '&sort=' + sortBy + '&status=unresolved'); expect(url).toEqual('http://localhost:33333/api/recoverability/groups/85147b12-458c-431d-a389-35ea53abc9e1/errors?page=1&sort=time_of_failure&status=unresolved'); })); - it('should make a url for get Failed Messages For Exception Group', inject(function (uri, scConfig) { + it('should make a url for get Failed Messages For Exception Group', inject(function (uri) { var messageId = '85147b12-458c-431d-a389-35ea53abc9e1'; - var url = uri.join(scConfig.service_control_url, 'messages', messageId, 'body'); + var url = uri.join(window.defaultConfig.service_control_url, 'messages', messageId, 'body'); expect(url).toEqual('http://localhost:33333/api/messages/85147b12-458c-431d-a389-35ea53abc9e1/body'); })); - it('should make a url for getMessageBody', inject(function (uri, scConfig) { + it('should make a url for getMessageBody', inject(function (uri) { var messageId = '85147b12-458c-431d-a389-35ea53abc9e1'; - var url = uri.join(scConfig.service_control_url, 'messages', messageId, 'body'); + var url = uri.join(window.defaultConfig.service_control_url, 'messages', messageId, 'body'); expect(url).toEqual('http://localhost:33333/api/messages/85147b12-458c-431d-a389-35ea53abc9e1/body'); })); - it('should make a url for getMessageHeaders', inject(function (uri, scConfig) { + it('should make a url for getMessageHeaders', inject(function (uri) { var messageId = '85147b12-458c-431d-a389-35ea53abc9e1'; - var url = uri.join(scConfig.service_control_url, 'messages', 'search', messageId); + var url = uri.join(window.defaultConfig.service_control_url, 'messages', 'search', messageId); expect(url).toEqual('http://localhost:33333/api/messages/search/85147b12-458c-431d-a389-35ea53abc9e1'); })); - it('should make a url for getTotalFailedMessages', inject(function (uri, scConfig) { - var url = uri.join(scConfig.service_control_url, 'errors?status=unresolved'); + it('should make a url for getTotalFailedMessages', inject(function (uri) { + var url = uri.join(window.defaultConfig.service_control_url, 'errors?status=unresolved'); expect(url).toEqual('http://localhost:33333/api/errors?status=unresolved'); })); - it('should make a url for retry errors', inject(function (uri, scConfig) { - var url = uri.join(scConfig.service_control_url, 'errors', 'retry'); + it('should make a url for retry errors', inject(function (uri) { + var url = uri.join(window.defaultConfig.service_control_url, 'errors', 'retry'); expect(url).toEqual('http://localhost:33333/api/errors/retry'); })); diff --git a/src/ServicePulse.Host/Commands/ExtractAndUpdateConstantsCommand.cs b/src/ServicePulse.Host/Commands/ExtractAndUpdateConstantsCommand.cs index e850ffb55..86bd35daf 100644 --- a/src/ServicePulse.Host/Commands/ExtractAndUpdateConstantsCommand.cs +++ b/src/ServicePulse.Host/Commands/ExtractAndUpdateConstantsCommand.cs @@ -59,7 +59,7 @@ public static void UpdateVersion(string directoryPath) { var appJsPath = Path.Combine(directoryPath, "js/app.constants.js"); var appJsCode = File.ReadAllText(appJsPath); - var updatedContent = Regex.Replace(appJsCode, @"(constant\(\s*'version'\s*,\s*')(.*?)(')", "${1}" + GetFileVersion() + "$3"); + var updatedContent = Regex.Replace(appJsCode, @"(version\s*\:\s*['""])(.*?)(['""])", "${1}" + GetFileVersion() + "$3"); File.WriteAllText(appJsPath, updatedContent); } @@ -79,4 +79,4 @@ static string GetFileVersion() return typeof(AbstractCommand).Assembly.GetName().Version.ToString(4); } } -} \ No newline at end of file +} diff --git a/src/ServicePulse.Host/ServicePulse.Host.csproj b/src/ServicePulse.Host/ServicePulse.Host.csproj index c6115c258..4f1dded06 100644 --- a/src/ServicePulse.Host/ServicePulse.Host.csproj +++ b/src/ServicePulse.Host/ServicePulse.Host.csproj @@ -15,6 +15,12 @@ + + + + + + diff --git a/src/ServicePulse.Host/app/css/particular.css b/src/ServicePulse.Host/app/css/particular.css index 5105a68ef..dd0eae90d 100644 --- a/src/ServicePulse.Host/app/css/particular.css +++ b/src/ServicePulse.Host/app/css/particular.css @@ -40,6 +40,10 @@ a { margin-left: 4px; } +.navbar-nav > li > a > span.no-margin { + margin: 0; +} + .navbar-toggle { margin-top: 13px; } @@ -545,6 +549,16 @@ h6 { text-decoration: none; } + .tabs h5.disabled > a { + color: #aaa; + cursor: default; + text-decoration: none; + } + + .tabs h5.disabled > a:hover { + cursor: not-allowed; + } + .system-status:hover { background-color: #fff; border-color: #eee !important; @@ -1849,9 +1863,22 @@ hr.top-separator { padding-right: 0; } +.btn-default { + padding: 8px 16px; +} + +.btn.btn-default.disabled { + opacity: 0.4; +} + +.btn.btn-default.disabled:hover { + border-color: #00A3C4; +} + .btn-primary { background-color: #00A3C4; border-color: #0686AA; + color: #fff; } .btn-primary:hover { @@ -1865,7 +1892,7 @@ hr.top-separator { } .btn-secondary { - background-color: transparent; + background-color: #fff; color: #00A3C4; border-color: #00A3C4; } @@ -1873,6 +1900,12 @@ hr.top-separator { .btn-secondary:hover { color: #00A3C4; border-color: #00A3C4; + background-color: #fff; + opacity: 0.7; +} + +.btn-secondary.disabled { + border-color: #00A3C4; } .table-head-row { @@ -2419,6 +2452,12 @@ div.alert.alert-warning strong { } } +@media (max-width: 1439px) and screen and (-ms-high-contrast: active), (-ms-high-contrast: none) { + nav.navbar { + margin-top: 0; + } +} + @media (max-width: 1550px) { footer { @@ -2428,10 +2467,6 @@ div.alert.alert-warning strong { } -#toast-container > div.toast-warning { - -} - .license-warning, .license-warning h5 { color: black; @@ -2442,12 +2477,6 @@ div.alert.alert-warning strong { background: black; } -btn-license-warning-light { - border: black; - color: black; - background: #f89406; -} - span.fa-exclamation-triangle.warning { color: #F3BC52; background: linear-gradient(black,black) center/20% 72% no-repeat; @@ -2497,6 +2526,12 @@ span.fa-exclamation-triangle.danger { padding: 24px 24px 24px 56px; } +#toast-container > div:hover { + box-shadow: 0 0 12px #999999; + opacity: 0.8; + cursor: default; +} + .toast-close-button { right: -10px; top: -16px; @@ -2504,14 +2539,14 @@ span.fa-exclamation-triangle.danger { #toast-container > .toast-error { background-image: url('../img/toast-danger.svg') !important; - background-position-y: 22px; + background-position-y: 30px; background-size: 28px 28px; } #toast-container > .toast-warning { background-color: #F3BC52; background-image: url('../img/toast-warning.svg') !important; - background-position-y: 22px; + background-position-y: 30px; background-size: 28px 28px; } @@ -2524,8 +2559,16 @@ span.fa-exclamation-triangle.danger { color: #F3BC52; } -.toast-message a.btn.btn-license-warning:hover { - opacity: 0.75; +.toast-message a.btn.btn-license-warning:hover, .toast-error a.btn.btn-default:hover { + opacity: 0.85; +} + +.toast-error a.btn.btn-default { + color: #bd362f; + padding-right: 20px; + padding-left: 20px; + border: none; + margin-top: 16px; } .toast-message a.btn.btn-license-warning-light { @@ -2608,4 +2651,97 @@ span.fa-exclamation-triangle.danger { div[content="Unable to connect to instance"], div[content="Unable to connect to monitoring server"] { z-index: 99999; +} + +.rotate { + -webkit-animation: glyphicon-spin-r 2s infinite linear; + animation: glyphicon-spin-r 2s infinite linear; +} + +@-webkit-keyframes glyphicon-spin-r { + 0% { + -webkit-transform: rotate(0deg); + transform: rotate(0deg); + } + + 100% { + -webkit-transform: rotate(359deg); + transform: rotate(359deg); + } +} + +@keyframes glyphicon-spin-r { + 0% { + -webkit-transform: rotate(0deg); + transform: rotate(0deg); + } + + 100% { + -webkit-transform: rotate(359deg); + transform: rotate(359deg); + } +} + +section[name="connections"] .box { + padding-bottom: 50px; +} + +form .connection h3 { + margin-bottom: 16px; +} + +form .connection .form-group { + padding-left: 0; +} + +.connection:nth-child(2) h3 { + margin-top: 40px; +} + +form .connection .form-group input { + font-size: 16px; + height: 44px; +} + +button[ng-click="vm.save()"] { + padding-right: 24px; + padding-left: 24px; + margin-top: 30px; +} + +span.connection-test { + position: relative; + top: 14px; + left: 10px; + text-transform: uppercase; + font-weight: bold; +} + +span.failed-validation { + text-transform: uppercase; + font-weight: bold; + margin-left: 10px; +} + +span.connection-successful, span.connection-successful i { + color: #00C468 !important; +} + +span.connection-failed, span.connection-failed i, .failed-validation, .failed-validation i { + color: #CE4844 !important; +} + +.form-control:focus { + border-color: #00a3c4; + box-shadow: 0 0 2px rgb(0, 163, 196); +} + +span.auxilliary-label { + color: #aaa; +} + +.btn-connection-test { + margin-top: 25px; + padding-top: 11px; + padding-bottom: 11px; } \ No newline at end of file diff --git a/src/ServicePulse.Host/app/index.html b/src/ServicePulse.Host/app/index.html index 97e9013b0..501862062 100644 --- a/src/ServicePulse.Host/app/index.html +++ b/src/ServicePulse.Host/app/index.html @@ -48,7 +48,7 @@

Warning!

- +
@@ -57,21 +57,10 @@

Warning!

-
-
-

Cannot connect to ServiceControl

-

- ServicePulse is unable to connect to the ServiceControl instance. Please ensure that ServiceControl is running and accesible from your machine. -

- -
-
-
+
- + @@ -151,6 +140,8 @@

Cannot connect to ServiceControl

} + + @@ -159,7 +150,6 @@

Cannot connect to ServiceControl

- diff --git a/src/ServicePulse.Host/app/js/app.bootstrap.js b/src/ServicePulse.Host/app/js/app.bootstrap.js index f7dc7a7d4..6e75cccef 100644 --- a/src/ServicePulse.Host/app/js/app.bootstrap.js +++ b/src/ServicePulse.Host/app/js/app.bootstrap.js @@ -5,17 +5,23 @@ var injector = angular.injector(['ng']); var $http = injector.get('$http'); - var scConfig = window.config; - $http.get(scConfig.service_control_url + '/license').then(function (response) { + var scUrl = window.connectionsManager.getServiceControlUrl(); + console.debug('Retrieving license from ServiceControl at: ', scUrl); + + $http.get(scUrl + '/license').then(function (response) { + serviceControlApp.constant('license', response.data); + }, function () { + + serviceControlApp.constant('license', { 'license_status': 'Unavailable' }); + + }).then(function () { + angular.element(document).ready(function () { angular.bootstrap(document, ['sc']); }); - }, function () { - window.document.getElementById('cantConnectMessage').style.display = 'block'; - window.document.getElementById('connectingToServiceControl').style.display = 'none'; - window.document.getElementById('serviceControlUrl').innerHTML = ' hosted at ' + scConfig.service_control_url + ''; + }); -}(window, window.angular, window.jQuery)); \ No newline at end of file +}(window, window.angular, window.jQuery)); diff --git a/src/ServicePulse.Host/app/js/app.constants.js b/src/ServicePulse.Host/app/js/app.constants.js index 76600dd73..87aac507b 100644 --- a/src/ServicePulse.Host/app/js/app.constants.js +++ b/src/ServicePulse.Host/app/js/app.constants.js @@ -1,14 +1,6 @@ -;(function (window, angular, undefined) { 'use strict'; - - window.config = { - default_route: '/dashboard', - service_control_url: 'http://localhost:33333/api/', - monitoring_urls: ['http://localhost:33633/'] - }; - - angular.module('sc') - .constant('version', '1.2.0') - .constant('showPendingRetry', false) - .constant('scConfig', window.config); - -}(window, window.angular)); +window.defaultConfig = { + default_route: '/dashboard', + version: '1.2.0', + service_control_url: 'http://localhost:33333/api/', + monitoring_urls: ['http://localhost:33633/'] +}; diff --git a/src/ServicePulse.Host/app/js/app.controller.js b/src/ServicePulse.Host/app/js/app.controller.js index 7f5c24cd7..76cc65764 100644 --- a/src/ServicePulse.Host/app/js/app.controller.js +++ b/src/ServicePulse.Host/app/js/app.controller.js @@ -15,25 +15,43 @@ signalRListener, notifyService, semverService, - scConfig, + connectionsManager, uriService, reindexingChecker, licenseNotifierService, licenseService, - license + license, + $route ) { - $scope.isMonitoringEnabled = scConfig.monitoring_urls && scConfig.monitoring_urls.reduce(function (currentlyEnabled, url) { - return currentlyEnabled || url; - }, false); + var notifier = notifyService(); + + var mu = connectionsManager.getMonitoringUrl(); + var scu = connectionsManager.getServiceControlUrl(); + + $scope.isMonitoringEnabled = connectionsManager.getIsMonitoringEnabled(); $scope.loadingInitialData = true; - $scope.isRecoverabilityEnabled = scConfig.service_control_url; + $scope.scConnectedAtLeastOnce = false; + $scope.isRecoverabilityEnabled = scu !== null && scu !== undefined; + $scope.serviceControlUrl = scu; $scope.SCVersion = ''; $scope.is_compatible_with_sc = true; $scope.Version = version; $scope.isSCConnecting = true; + $scope.$on('$locationChangeStart', function(event, next, current) { + if (!$scope.isSCConnected && !$scope.scConnectedAtLeastOnce) { + var routeData = $route.routes[$location.path()].data; + + if (routeData && routeData.redirectWhenNotConnected) { + $log.debug('not connected, and never connected once. Current route is a configuration route that requires redirect to: ', routeData.redirectWhenNotConnected); + event.preventDefault(); + $location.path(routeData.redirectWhenNotConnected); + } + } + }); + setTimeout(function () { // This delay needs to be here for the toastr service to be ready. licenseNotifierService.warnOfLicenseProblem(license.license_status); @@ -98,7 +116,21 @@ $log.debug(data); } - var notifier = notifyService(); + notifier.subscribe($scope, function(event, data) { + if (connectionsManager.getIsMonitoringEnabled()) { + if ((data.status.isSCConnected || data.status.isSCConnecting) && (data.status.isMonitoringConnected || data.status.isMonitoringConnecting || data.status.isMonitoringConnecting === undefined)) { + $scope.connectionswarning = undefined; + } else if (!data.status.isSCConnected || !data.status.isMonitoringConnected) { + $scope.connectionswarning = 'danger'; + } + } else { + if (data.status.isSCConnected || data.status.isSCConnecting) { + $scope.connectionswarning = undefined; + } else if (!data.status.isSCConnected) { + $scope.connectionswarning = 'danger'; + } + } + }, 'ConnectionsStatusChanged'); notifier.subscribe($scope, customChecksUpdated, 'CustomChecksUpdated'); notifier.subscribe($scope, messageFailuresUpdated, 'MessageFailuresUpdated'); @@ -139,28 +171,49 @@ logit(event, data); switch(data) { + case 'SignalR starting': + $scope.isSCConnected = false; + $scope.isSCConnecting = true; + break; case 'SignalR started': $scope.isSCConnected = true; $scope.isSCConnecting = false; + $scope.scConnectedAtLeastOnce = true; break; case 'Reconnected': $scope.isSCConnected = true; $scope.isSCConnecting = false; + $scope.scConnectedAtLeastOnce = true; + + if ($scope.signalRConnectionErrorToast) { + toastService.clear($scope.signalRConnectionErrorToast); + $scope.signalRConnectionErrorToast = undefined; + } break; default: toastService.showWarning(data); break; } + + notifier.notify('ServiceControlConnectionStatusChanged', { + isSCConnected : $scope.isSCConnected, + isSCConnecting: $scope.isSCConnecting, + scConnectedAtLeastOnce: $scope.scConnectedAtLeastOnce + }); }, 'SignalREvent'); notifier.subscribe($scope, function(event, data) { logit(event, data); - if ($scope.isSCConnected) { - toastService.showError(data); - } + $scope.isSCConnected = false; $scope.isSCConnecting = false; + + notifier.notify('ServiceControlConnectionStatusChanged', { + isSCConnected : $scope.isSCConnected, + isSCConnecting : $scope.isSCConnecting, + scConnectedAtLeastOnce : $scope.scConnectedAtLeastOnce + }); }, 'SignalRError'); notifier.subscribe($scope, function(event, data) { @@ -193,7 +246,7 @@ }, 'reindexing'); // signalR Listener - var listener = signalRListener(uriService.join(scConfig.service_control_url, 'messagestream')); + var listener = signalRListener(uriService.join(scu, 'messagestream')); listener.subscribe($scope, function(message) { notifier.notify('SignalREvent', message); @@ -311,12 +364,13 @@ 'signalRListener', 'notifyService', 'semverService', - 'scConfig', + 'connectionsManager', 'uri', 'reindexingChecker', 'licenseNotifierService', 'licenseService', 'license', + '$route' ]; angular.module('sc').controller('AppCtrl', controller); diff --git a/src/ServicePulse.Host/app/js/app.js b/src/ServicePulse.Host/app/js/app.js index 2a5987989..eab61c582 100644 --- a/src/ServicePulse.Host/app/js/app.js +++ b/src/ServicePulse.Host/app/js/app.js @@ -36,7 +36,11 @@ $rootScope.$log = $log; }]); - angular.module('sc').value('$jquery', $); + angular.module('sc') + .value('$jquery', $) + .constant('version', window.defaultConfig.version) + .constant('showPendingRetry', false) + .constant('scConfig', window.defaultConfig); angular.module('sc').config(['$locationProvider', function ($locationProvider) { $locationProvider.hashPrefix(''); diff --git a/src/ServicePulse.Host/app/js/app.route.js b/src/ServicePulse.Host/app/js/app.route.js index 0519cef58..ec4dc4e75 100644 --- a/src/ServicePulse.Host/app/js/app.route.js +++ b/src/ServicePulse.Host/app/js/app.route.js @@ -1,13 +1,12 @@ ; (function (window, angular, undefined) { 'use strict'; - function routeProvider($routeProvider, scConfig) { - $routeProvider.otherwise({ redirectTo: scConfig.default_route }); + function routeProvider($routeProvider) { + $routeProvider.otherwise({ redirectTo: window.defaultConfig.default_route }); } routeProvider.$inject = [ - '$routeProvider', - 'scConfig' + '$routeProvider' ]; angular.module('sc') diff --git a/src/ServicePulse.Host/app/js/directives/ui.particular.exclamation.js b/src/ServicePulse.Host/app/js/directives/ui.particular.exclamation.js index 6c1f88dbf..88692c3a8 100644 --- a/src/ServicePulse.Host/app/js/directives/ui.particular.exclamation.js +++ b/src/ServicePulse.Host/app/js/directives/ui.particular.exclamation.js @@ -6,7 +6,7 @@ scope: { type: '@' }, - restrict: 'E', + restrict: 'EA', replace: true, templateUrl: 'js/directives/ui.particular.exclamation.tpl.html', link: function (scope, element) { } diff --git a/src/ServicePulse.Host/app/js/services/factory.listener.js b/src/ServicePulse.Host/app/js/services/factory.listener.js index 369e59017..96ac2ab37 100644 --- a/src/ServicePulse.Host/app/js/services/factory.listener.js +++ b/src/ServicePulse.Host/app/js/services/factory.listener.js @@ -1,7 +1,7 @@ ; (function (window, angular, undefined) { 'use strict'; - function factory($rootScope, $jquery, notifyService) { + function factory($rootScope, $jquery, notifyService, toastService, $window) { function listener(msgUrl) { @@ -39,7 +39,6 @@ connection.start() .done(function () { - notifier.notify('SignalREvent', 'SignalR started'); connection.error(function (error) { @@ -51,15 +50,21 @@ }); connection.stateChanged(function (change) { - if (change.newState === $jquery.signalR.connectionState.disconnected) { notifier.notify('SignalRError', 'The server is offline'); } }); }) .fail(function () { - notifier.notify('SignalRError', 'Can not connect to ServiceControl'); + //notification is needed for other part of Pulse that depend on the notifier to get connectivity status. + notifier.notify('SignalRError'); + if ($window.location.hash.indexOf('/configuration/connections') < 0) { + // Uses the toastService directly to avoid breaking the notifier class. The previous notifier calls should all be removed at some point too. + toastService.showError('Could not connect to ServiceControl. View connection settings', true, false); + } }); + + notifier.notify('SignalREvent', 'SignalR starting'); } @@ -84,7 +89,9 @@ factory.$inject = [ '$rootScope', '$jquery', - 'notifyService' + 'notifyService', + 'toastService', + '$window' ]; angular.module('sc') diff --git a/src/ServicePulse.Host/app/js/services/services.service-control.js b/src/ServicePulse.Host/app/js/services/services.service-control.js index 88caf9548..7dbacea00 100644 --- a/src/ServicePulse.Host/app/js/services/services.service-control.js +++ b/src/ServicePulse.Host/app/js/services/services.service-control.js @@ -2,17 +2,19 @@ (function(window, angular, $, undefined) { 'use strict'; - function Service($http, scConfig, notifications, uri) { + function Service($http, connectionsManager, notifications, uri) { + + var scu = connectionsManager.getServiceControlUrl(); function getVersion() { - var url = uri.join(scConfig.service_control_url); + var url = uri.join(scu); return $http.get(url).then(function(response) { return response.headers('X-Particular-Version'); }); } function checkLicense() { - var url = uri.join(scConfig.service_control_url); + var url = uri.join(scu); return $http.get(url).then(function(response) { if (response.data.license_status !== 'valid') { return false; @@ -22,14 +24,14 @@ } function getEventLogItems() { - var url = uri.join(scConfig.service_control_url, 'eventlogitems'); + var url = uri.join(scu, 'eventlogitems'); return $http.get(url).then(function(response) { return response.data; }); } function getFailedMessages(sortBy, page, direction) { - var url = uri.join(scConfig.service_control_url, 'errors?status=unresolved&page=' + page + '&sort=' + sortBy + '&direction=' + direction); + var url = uri.join(scu, 'errors?status=unresolved&page=' + page + '&sort=' + sortBy + '&direction=' + direction); return $http.get(url).then(function(response) { return { data: response.data, @@ -41,7 +43,7 @@ var previousExceptionGroupEtag; function getExceptionGroups(classifier, classifierFilter) { - var url = uri.join(scConfig.service_control_url, 'recoverability', 'groups', classifier) + '?classifierFilter=' + classifierFilter; + var url = uri.join(scu, 'recoverability', 'groups', classifier) + '?classifierFilter=' + classifierFilter; return $http.get(url).then(function (response) { var status = 200; if (previousExceptionGroupEtag === response.headers('etag')) { @@ -57,7 +59,7 @@ } function getExceptionGroup(groupId) { - var url = uri.join(scConfig.service_control_url, 'recoverability', 'groups', 'id', groupId); + var url = uri.join(scu, 'recoverability', 'groups', 'id', groupId); return $http.get(url).then(function (response) { var status = 200; if (previousExceptionGroupEtag === response.headers('etag')) { @@ -81,7 +83,7 @@ } function getHistoricGroups() { - var url = uri.join(scConfig.service_control_url, 'recoverability', 'history'); + var url = uri.join(scu, 'recoverability', 'history'); return $http.get(url).then(function (response) { return { data: response.data, @@ -91,12 +93,12 @@ } function getFailedMessageById(messageId) { - var url = uri.join(scConfig.service_control_url, 'errors', 'last', messageId); + var url = uri.join(scu, 'errors', 'last', messageId); return $http.get(url); } function getFailedMessagesForExceptionGroup(groupId, sortBy, page) { - var url = uri.join(scConfig.service_control_url, 'recoverability', 'groups', groupId, 'errors?page=' + page + '&sort=' + sortBy + '&status=unresolved'); + var url = uri.join(scu, 'recoverability', 'groups', groupId, 'errors?page=' + page + '&sort=' + sortBy + '&status=unresolved'); return $http.get(url).then(function(response) { return { data: response.data, @@ -106,7 +108,7 @@ } function getMessageBody(messageId) { - var url = uri.join(scConfig.service_control_url, 'messages', messageId, 'body'); + var url = uri.join(scu, 'messages', messageId, 'body'); return $http.get(url).then(function(response) { return { data: response.data @@ -115,7 +117,7 @@ } function getMessageHeaders(messageId) { - var url = uri.join(scConfig.service_control_url, 'messages', 'search', messageId); + var url = uri.join(scu, 'messages', 'search', messageId); return $http.get(url).then(function(response) { return { data: response.data @@ -124,49 +126,49 @@ } function getTotalFailedMessages() { - var url = uri.join(scConfig.service_control_url, 'errors?status=unresolved'); + var url = uri.join(scu, 'errors?status=unresolved'); return $http.head(url).then(function(response) { return response.headers('Total-Count'); }); } function getTotalArchivedMessages() { - var url = uri.join(scConfig.service_control_url, 'errors?status=archived'); + var url = uri.join(scu, 'errors?status=archived'); return $http.head(url).then(function(response) { return response.headers('Total-Count'); }); } function getExceptionGroupClassifiers() { - var url = uri.join(scConfig.service_control_url, 'recoverability', 'classifiers'); + var url = uri.join(scu, 'recoverability', 'classifiers'); return $http.get(url).then(function (response) { return response.data; }); } function getConfiguration() { - var url = uri.join(scConfig.service_control_url, 'configuration'); + var url = uri.join(scu, 'configuration'); return $http.get(url).then(function(response) { return response.data; }); } function getTotalFailingCustomChecks() { - var url = uri.join(scConfig.service_control_url, 'customchecks?status=fail'); + var url = uri.join(scu, 'customchecks?status=fail'); return $http.get(url).then(function(response) { return response.headers('Total-Count'); }); } function getTotalPendingRetries() { - var url = uri.join(scConfig.service_control_url, 'errors?status=retryissued'); + var url = uri.join(scu, 'errors?status=retryissued'); return $http.head(url).then(function(response) { return response.headers('Total-Count'); }); } function getFailingCustomChecks(page) { - var url = uri.join(scConfig.service_control_url, 'customchecks?status=fail&page=' + page); + var url = uri.join(scu, 'customchecks?status=fail&page=' + page); return $http.get(url).then(function(response) { return { data: response.data, @@ -176,20 +178,20 @@ } function getFailedMessageStats() { - var url = uri.join(scConfig.service_control_url, 'errors', 'summary'); + var url = uri.join(scu, 'errors', 'summary'); return $http.get(url).then(function(response) { return response.data; }); } function dismissCustomChecks(customCheck) { - var url = uri.join(scConfig.service_control_url, 'customchecks', customCheck.id); + var url = uri.join(scu, 'customchecks', customCheck.id); $http.delete(url); } function retryPendingMessagesForQueue(queueName) { - var url = uri.join(scConfig.service_control_url, 'errors', 'queues', queueName, 'retry'); + var url = uri.join(scu, 'errors', 'queues', queueName, 'retry'); $http.post(url) .then(function() { notifications.pushForCurrentRoute('Retrying all pending retry messages for queue ' + queueName, 'info'); @@ -199,7 +201,7 @@ } function retryAllFailedMessages() { - var url = uri.join(scConfig.service_control_url, 'errors', 'retry', 'all'); + var url = uri.join(scu, 'errors', 'retry', 'all'); return $http.post(url) .then(function() { notifications.pushForCurrentRoute('Retrying all messages...', 'info'); @@ -209,7 +211,7 @@ } function retryFailedMessages(selectedMessages) { - var url = uri.join(scConfig.service_control_url, 'errors', 'retry'); + var url = uri.join(scu, 'errors', 'retry'); return $http.post(url, selectedMessages) .then(function() { notifications.pushForCurrentRoute('Retrying {{num}} messages...', 'info', { num: selectedMessages.length }); @@ -219,7 +221,7 @@ } function archiveFailedMessages(selectedMessages) { - var url = uri.join(scConfig.service_control_url, 'errors', 'archive'); + var url = uri.join(scu, 'errors', 'archive'); return $http({ url: url, @@ -234,7 +236,7 @@ } function archiveExceptionGroup(id, successText) { - var url = uri.join(scConfig.service_control_url, 'recoverability', 'groups', id, 'errors', 'archive'); + var url = uri.join(scu, 'recoverability', 'groups', id, 'errors', 'archive'); return $http.post(url) .then(null, function() { notifications.pushForCurrentRoute('Archiving messages failed', 'danger'); @@ -242,14 +244,14 @@ } function acknowledgeArchiveGroup(groupId) { - var url = uri.join(scConfig.service_control_url, 'recoverability', 'unacknowledgedgroups', groupId); + var url = uri.join(scu, 'recoverability', 'unacknowledgedgroups', groupId); return $http.delete(url).then(null, function () { notifications.pushForCurrentRoute('Archive messages failed', 'danger'); }); } function acknowledgeGroup(id, successText, failureText) { - var url = uri.join(scConfig.service_control_url, 'recoverability', 'unacknowledgedgroups', id); + var url = uri.join(scu, 'recoverability', 'unacknowledgedgroups', id); return $http.delete(url).then(null, function () { notifications.pushForCurrentRoute('Retrying messages failed', 'danger'); }); @@ -257,7 +259,7 @@ function retryExceptionGroup(id, successText) { - var url = uri.join(scConfig.service_control_url, 'recoverability', 'groups', id, 'errors', 'retry'); + var url = uri.join(scu, 'recoverability', 'groups', id, 'errors', 'retry'); return $http.post(url) .then(null, function() { notifications.pushForCurrentRoute('Retrying messages failed', 'danger'); @@ -265,7 +267,7 @@ } function getHeartbeatStats() { - var url = uri.join(scConfig.service_control_url, 'heartbeats', 'stats'); + var url = uri.join(scu, 'heartbeats', 'stats'); return $http.get(url).then(function(response) { return response.data; }); @@ -277,7 +279,7 @@ .then(function(endpoints) { var results = []; endpoints.forEach(function(item) { - var url = uri.join(scConfig.service_control_url, 'endpoints', item.name, 'sla'); + var url = uri.join(scu, 'endpoints', item.name, 'sla'); $http.get(url).then(function(response) { angular.extend(item, { sla: response.data.current }); results.push(item); @@ -289,13 +291,13 @@ } function loadQueueNames() { - var url = uri.join(scConfig.service_control_url, 'errors', 'queues', 'addresses'); + var url = uri.join(scu, 'errors', 'queues', 'addresses'); return $http.get(url); } function isBusyUpgradingIndexes() { - var url = uri.join(scConfig.service_control_url, 'upgrade'); + var url = uri.join(scu, 'upgrade'); return $http.get(url); } @@ -339,7 +341,7 @@ return service; } - Service.$inject = ['$http', 'scConfig', 'notifications', 'uri']; + Service.$inject = ['$http', 'connectionsManager', 'notifications', 'uri']; angular.module('services.serviceControlService', []) .service('serviceControlService', Service); diff --git a/src/ServicePulse.Host/app/js/services/services.stream.js b/src/ServicePulse.Host/app/js/services/services.stream.js index 8f69fe3fa..746af0cb9 100644 --- a/src/ServicePulse.Host/app/js/services/services.stream.js +++ b/src/ServicePulse.Host/app/js/services/services.stream.js @@ -1,11 +1,12 @@ ; (function (window, angular, undefined) { 'use strict'; - function Service(notifications, $log, $rootScope, scConfig, $jquery, uri) { + function Service(notifications, $log, $rootScope, connectionsManager, $jquery, uri) { var subscriberRegistry = {}, registryKey = 1; - var url = uri.join(scConfig.service_control_url, 'messagestream'); + var scu = connectionsManager.getServiceControlUrl(); + var url = uri.join(scu, 'messagestream'); var connection = $jquery.connection(url); @@ -41,7 +42,7 @@ }) .fail(function () { - notifications.pushForCurrentRoute('Can\'t connect to ServiceControl ({{url}})', 'danger', { url: scConfig.service_control_url }); + notifications.pushForCurrentRoute('Can\'t connect to ServiceControl ({{url}})', 'danger', { url: scu }); }); function callSubscribers(messageType, message) { @@ -86,7 +87,7 @@ }; }; - Service.$inject = ['notifications', '$log', '$rootScope', 'scConfig', '$jquery', 'uri']; + Service.$inject = ['notifications', '$log', '$rootScope', 'connectionsManager', '$jquery', 'uri']; angular .module('services.streamService', []) diff --git a/src/ServicePulse.Host/app/js/views/archive/controller.js b/src/ServicePulse.Host/app/js/views/archive/controller.js index 62851502d..ab9ca14b8 100644 --- a/src/ServicePulse.Host/app/js/views/archive/controller.js +++ b/src/ServicePulse.Host/app/js/views/archive/controller.js @@ -9,7 +9,6 @@ moment, $location, $cookies, - scConfig, sharedDataService, notifyService, serviceControlService, @@ -285,7 +284,6 @@ "moment", "$location", "$cookies", - "scConfig", "sharedDataService", "notifyService", "serviceControlService", diff --git a/src/ServicePulse.Host/app/js/views/archive/service.js b/src/ServicePulse.Host/app/js/views/archive/service.js index d0df144d9..d76099e92 100644 --- a/src/ServicePulse.Host/app/js/views/archive/service.js +++ b/src/ServicePulse.Host/app/js/views/archive/service.js @@ -7,11 +7,12 @@ $timeout, $q, notifyService, - scConfig, + connectionsManager, uri ) { var notifier = notifyService(); + var scu = connectionsManager.getServiceControlUrl(); function patchPromise(url, success, error, ids) { @@ -35,9 +36,9 @@ getArchivedMessages: function (sort, page, direction, start, end) { var url = ''; if (start && end) { - url = uri.join(scConfig.service_control_url, 'errors?status=archived&page=' + page + '&sort=' + sort + '&direction=' + direction + '&modified=' + start + '...' + end); + url = uri.join(scu, 'errors?status=archived&page=' + page + '&sort=' + sort + '&direction=' + direction + '&modified=' + start + '...' + end); } else { - url = uri.join(scConfig.service_control_url, 'errors?status=archived&page=' + page + '&sort=' + sort + '&direction=' + direction); + url = uri.join(scu, 'errors?status=archived&page=' + page + '&sort=' + sort + '&direction=' + direction); } return $http.get(url).then(function (response) { @@ -51,7 +52,7 @@ getArchivedCount: function () { - var url = uri.join(scConfig.service_control_url, 'errors?status=archived'); + var url = uri.join(scu, 'errors?status=archived'); return $http.head(url).then(function (response) { @@ -61,19 +62,19 @@ restoreFromArchive: function (startdate, enddate, success, error) { - var url = uri.join(scConfig.service_control_url, 'errors', startdate.format('YYYY-MM-DDTHH:mm:ss') + '...' + enddate.format('YYYY-MM-DDTHH:mm:ss'), 'unarchive'); + var url = uri.join(scu, 'errors', startdate.format('YYYY-MM-DDTHH:mm:ss') + '...' + enddate.format('YYYY-MM-DDTHH:mm:ss'), 'unarchive'); return patchPromise(url, success, error); }, restoreMessageFromArchive: function (id, success, error) { - var url = uri.join(scConfig.service_control_url, 'errors', 'unarchive'); + var url = uri.join(scu, 'errors', 'unarchive'); return patchPromise(url, success, error, [id]); }, restoreMessagesFromArchive: function (ids, success, error) { - var url = uri.join(scConfig.service_control_url, 'errors', 'unarchive'); + var url = uri.join(scu, 'errors', 'unarchive'); return patchPromise(url, success, error, ids); } }; @@ -85,7 +86,7 @@ '$timeout', '$q', 'notifyService', - 'scConfig', + 'connectionsManager', 'uri' ]; diff --git a/src/ServicePulse.Host/app/js/views/archive/view.html b/src/ServicePulse.Host/app/js/views/archive/view.html index 43e2e7cb6..a90586864 100644 --- a/src/ServicePulse.Host/app/js/views/archive/view.html +++ b/src/ServicePulse.Host/app/js/views/archive/view.html @@ -3,9 +3,11 @@
+
+
-
+
diff --git a/src/ServicePulse.Host/app/js/views/custom_checks/customChecks.html b/src/ServicePulse.Host/app/js/views/custom_checks/customChecks.html index 985ad88f1..069440604 100644 --- a/src/ServicePulse.Host/app/js/views/custom_checks/customChecks.html +++ b/src/ServicePulse.Host/app/js/views/custom_checks/customChecks.html @@ -3,46 +3,51 @@
- +
+
-
-
-

Custom checks

+
+ + +
+
+

Custom checks

+
-
-
- +
+ - + -
-
-
-
-
-
-
-
-
-

{{item.failure_reason}}

-

{{item.custom_check_id}}

-

- {{item.category}} reported by {{item.originating_endpoint.name}} on {{item.originating_endpoint.host}} -

+
+
+
+
+
+
+
+
+
+

{{item.failure_reason}}

+

{{item.custom_check_id}}

+

+ {{item.category}} reported by {{item.originating_endpoint.name}} on {{item.originating_endpoint.host}} +

+
-
-
- +
+ +
-
+
\ No newline at end of file diff --git a/src/ServicePulse.Host/app/js/views/dashboard/dashboard.html b/src/ServicePulse.Host/app/js/views/dashboard/dashboard.html index ad80d1af6..27e638e24 100644 --- a/src/ServicePulse.Host/app/js/views/dashboard/dashboard.html +++ b/src/ServicePulse.Host/app/js/views/dashboard/dashboard.html @@ -3,56 +3,61 @@
\ No newline at end of file diff --git a/src/ServicePulse.Host/app/js/views/endpoints/endpoints.html b/src/ServicePulse.Host/app/js/views/endpoints/endpoints.html index 242905b0b..720a1a5f8 100644 --- a/src/ServicePulse.Host/app/js/views/endpoints/endpoints.html +++ b/src/ServicePulse.Host/app/js/views/endpoints/endpoints.html @@ -3,61 +3,67 @@
- +
+
-
-
-

Endpoint heartbeats

-
-
+
- + -
-
-
-
-
+

Endpoint heartbeats

+
+
+ + + +
+ +
+
+
+
+
-
-
-
-

{{endpoint.name}}@{{endpoint.host_display_name}}

-

- latest heartbeat received -

+
+
+
+

{{endpoint.name}}@{{endpoint.host_display_name}}

+

+ latest heartbeat received +

+
-
+
-
+
-
-
+
-
- -
-
-
-
-
-
-
-
-

{{endpoint.name}}@{{endpoint.host_display_name}}

-

- latest heartbeat received -

-

No plugin installed

+
+ +
+
+
+
+
+
+
+
+

{{endpoint.name}}@{{endpoint.host_display_name}}

+

+ latest heartbeat received +

+

No plugin installed

+
@@ -65,6 +71,6 @@
- +
+
-
+
+ - +
-
- -
- -
-
-
-
-
-
-

{{group.originator}}

+ + +
+ +
+ +
+
+
+
+
+
+

{{group.originator}}

+
-
-
-
- +
+
+ +
+ There is only 1 completed group retry + There are only {{vm.historicGroups.length}} completed group retries
- There is only 1 completed group retry - There are only {{vm.historicGroups.length}} completed group retries
-
-
-
-
Failed message groups
-
+
+
+
Failed message groups
+
-
- - -
-
-
- - +
+
+ + +
-
-
-
+
+
-
+
-
-
-
-
-
-
-

{{group.title}}

-
+
+
+
+
+
+

{{group.title}}

+ + +

+
-
-
-
- - +
+
+ + +
-
-
-
-
-
-
    -
  • -
    Initialize retry request...
    -
  • -
  • - -
    -
    -
    Prepare messages...
    -
    +
    +
    +
    +
    +
      +
    • +
      Initialize retry request...
      +
    • +
    • + +
      +
      +
      Prepare messages...
      +
      -
      -
      -
      - {{group.workflow_state.total | number : 0}}% +
      +
      +
      + {{group.workflow_state.total | number : 0}}% +
      -
      -
    • -
    • +
    • +
    • -
      -
      -
      Send messages to retry...
      -
      -
      - (Queued) -
      -
      +
      +
      +
      Send messages to retry...
      +
      +
      + (Queued) +
      +
      -
      -
      - {{group.workflow_state.total | number : 0}}% +
      +
      + {{group.workflow_state.total | number : 0}}% +
      -
      -
    • -
    • -
      Retry request completed
      - -
      - WARNING: Not all messages will be retried because ServiceControl had to restart. You need to request retrying the remaining messages. -
      -
    • -
    - -
  • +
  • +
    Retry request completed
    + +
    + WARNING: Not all messages will be retried because ServiceControl had to restart. You need to request retrying the remaining messages. +
    +
  • +
+ +
-
-
-
-
-
-
    -
  • -
    Initialize archive request...
    -
  • -
  • -
    -
    -
    Archive request in progress...
    -
    +
    +
    +
    +
    +
      +
    • +
      Initialize archive request...
      +
    • +
    • +
      +
      +
      Archive request in progress...
      +
      -
      -
      -
      - {{group.workflow_state.total | number : 0}}% +
      +
      +
      + {{group.workflow_state.total | number : 0}}% +
      -
      -
    • -
    • -
      -
      -
      Cleaning up...
      +
    • +
    • +
      +
      +
      Cleaning up...
      +
      -
    -
  • -
  • -
    Archive request completed
    - -
  • -
- -
@@ -224,10 +229,10 @@
Failed message groups
-
+
-
+
\ No newline at end of file diff --git a/src/ServicePulse.Host/app/js/views/failed_messages/view.html b/src/ServicePulse.Host/app/js/views/failed_messages/view.html index fe6e51230..64bb7a241 100644 --- a/src/ServicePulse.Host/app/js/views/failed_messages/view.html +++ b/src/ServicePulse.Host/app/js/views/failed_messages/view.html @@ -3,105 +3,109 @@
- - -
-
- -
- -
-
- -
All failed messages ({{vm.failedMessages.length}} / {{vm.selectedExceptionGroup.count}} | number)
-

- {{vm.selectedExceptionGroup.parentTitle}} - - {{vm.selectedExceptionGroup.title}} -

-

{{vm.selectedExceptionGroup.count | number}} messages in group

+
+
+ +
+ + +
+
+
-
- - - - - -
-
-
- - - - - - -
-
-
+
+
+
- + -
+
-
-
-
-
-

{{message.message_type || 'Message Type Unknown - missing metadata EnclosedMessageTypes'}}

-
+
+
+
+

{{message.message_type || 'Message Type Unknown - missing metadata EnclosedMessageTypes'}}

+ + + + + + +

-
{{ message.exception.message }}
+
{{ message.exception.message }}
+
-
+
-
-
- +
+
\ No newline at end of file diff --git a/src/ServicePulse.Host/app/js/views/message/controller.js b/src/ServicePulse.Host/app/js/views/message/controller.js index e69e2e389..9650ccb54 100644 --- a/src/ServicePulse.Host/app/js/views/message/controller.js +++ b/src/ServicePulse.Host/app/js/views/message/controller.js @@ -7,7 +7,7 @@ $routeParams, moment, $window, - scConfig, + connectionsManager, toastService, serviceControlService, archivedMessageService, @@ -136,7 +136,7 @@ vm.debugInServiceInsight = function () { var messageId = vm.message.message_id; - var dnsName = scConfig.service_control_url.toLowerCase(); + var dnsName = connectionsManager.getServiceControlUrl().toLowerCase(); if (dnsName.indexOf("https") === 0) { dnsName = dnsName.replace("https://", ""); @@ -173,7 +173,7 @@ '$routeParams', 'moment', '$window', - 'scConfig', + 'connectionsManager', 'toastService', 'serviceControlService', 'archivedMessageService', diff --git a/src/ServicePulse.Host/app/js/views/message/messages-view.html b/src/ServicePulse.Host/app/js/views/message/messages-view.html index 28574865f..37ecfc805 100644 --- a/src/ServicePulse.Host/app/js/views/message/messages-view.html +++ b/src/ServicePulse.Host/app/js/views/message/messages-view.html @@ -3,84 +3,89 @@
- +
+
-
-
-
-
BACK
-

{{vm.message.message_type}}

+
+ + +
+
+
+
BACK
+

{{vm.message.message_type}}

+
-
- + - + -
-
-
-
\ No newline at end of file diff --git a/src/ServicePulse.Host/app/js/views/pending_retries/controller.js b/src/ServicePulse.Host/app/js/views/pending_retries/controller.js index 4cff6b26d..62a9a9c3b 100644 --- a/src/ServicePulse.Host/app/js/views/pending_retries/controller.js +++ b/src/ServicePulse.Host/app/js/views/pending_retries/controller.js @@ -9,7 +9,6 @@ $location, moment, $filter, - scConfig, toastService, sharedDataService, notifyService, @@ -346,7 +345,6 @@ "$location", "moment", "$filter", - "scConfig", "toastService", "sharedDataService", "notifyService", diff --git a/src/ServicePulse.Host/app/js/views/pending_retries/service.js b/src/ServicePulse.Host/app/js/views/pending_retries/service.js index ee33e3c1b..0f021d18d 100644 --- a/src/ServicePulse.Host/app/js/views/pending_retries/service.js +++ b/src/ServicePulse.Host/app/js/views/pending_retries/service.js @@ -3,10 +3,12 @@ - function service($http, moment, scConfig, notifications, uri) { + function service($http, moment, connectionsManager, notifications, uri) { + + var scu = connectionsManager.getServiceControlUrl(); function getPendingRetryMessages(searchPhrase, sortBy, page, direction, start, end) { - var url = uri.join(scConfig.service_control_url, 'errors?status=retryissued&page=' + page + '&sort=' + sortBy + '&direction=' + direction); + var url = uri.join(scu, 'errors?status=retryissued&page=' + page + '&sort=' + sortBy + '&direction=' + direction); if (start && end) { url = url + '&modified=' + start + '...' + end; @@ -26,7 +28,7 @@ function getTotalPendingRetryMessages(searchPhrase, start, end) { - var url = uri.join(scConfig.service_control_url, 'errors?status=retryissued'); + var url = uri.join(scu, 'errors?status=retryissued'); if (start && end) { url = url + '&modified=' + start + '...' + end; @@ -52,10 +54,10 @@ var url = null; var data = {}; if (searchPhrase) { - url = uri.join(scConfig.service_control_url, 'pendingretries', 'queues', 'retry'); + url = uri.join(scu, 'pendingretries', 'queues', 'retry'); data.queueaddress = searchPhrase; } else { - url = uri.join(scConfig.service_control_url, 'pendingretries', 'retry'); + url = uri.join(scu, 'pendingretries', 'retry'); } data.from = start; data.to = end; @@ -72,7 +74,7 @@ } function retryPendingRetriedMessages(selectedMessages) { - var url = uri.join(scConfig.service_control_url, 'pendingretries', 'retry'); + var url = uri.join(scu, 'pendingretries', 'retry'); return $http.post(url, selectedMessages) .then(function () { notifications.pushForCurrentRoute('Retrying {{num}} pending retried messages...', 'info', { num: selectedMessages.length }); @@ -89,10 +91,10 @@ var data = {}; var url = null; if (searchPhrase) { - url = uri.join(scConfig.service_control_url, 'pendingretries', 'queues', 'resolve'); + url = uri.join(scu, 'pendingretries', 'queues', 'resolve'); data.queueaddress = searchPhrase; } else { - url = uri.join(scConfig.service_control_url, 'pendingretries', 'resolve'); + url = uri.join(scu, 'pendingretries', 'resolve'); } data.from = start; data.to = end; @@ -109,7 +111,7 @@ } function markAsResolvedMessages(selectedMessages) { - var url = uri.join(scConfig.service_control_url, 'pendingretries', 'resolve'); + var url = uri.join(scu, 'pendingretries', 'resolve'); return $http({ url: url, @@ -133,7 +135,7 @@ }; } - service.$inject = ['$http', 'moment', 'scConfig', 'notifications', 'uri']; + service.$inject = ['$http', 'moment', 'connectionsManager', 'notifications', 'uri']; angular.module('sc') .service('pendingRetryService', service); diff --git a/src/ServicePulse.Host/app/js/views/pending_retries/view.html b/src/ServicePulse.Host/app/js/views/pending_retries/view.html index a076aceac..a0abfe53c 100644 --- a/src/ServicePulse.Host/app/js/views/pending_retries/view.html +++ b/src/ServicePulse.Host/app/js/views/pending_retries/view.html @@ -3,7 +3,10 @@
-
+
+
+ +
diff --git a/src/ServicePulse.Host/app/js/views/sc_not_available.html b/src/ServicePulse.Host/app/js/views/sc_not_available.html new file mode 100644 index 000000000..decf500c3 --- /dev/null +++ b/src/ServicePulse.Host/app/js/views/sc_not_available.html @@ -0,0 +1,9 @@ +
+

Cannot connect to ServiceControl

+

+ ServicePulse is unable to connect to the ServiceControl instanceput here the url we tried to connect to. Please ensure that ServiceControl is running and accesible from your machine. +

+ +
diff --git a/src/ServicePulse.Host/app/layout/footer.html b/src/ServicePulse.Host/app/layout/footer.html index 2505d88a2..1d9efc579 100644 --- a/src/ServicePulse.Host/app/layout/footer.html +++ b/src/ServicePulse.Host/app/layout/footer.html @@ -3,14 +3,15 @@
- + + support@particular.net Documentation
-
+
Service Control:
Connected @@ -24,6 +25,6 @@
+
-
diff --git a/src/ServicePulse.Host/app/layout/navbar.html b/src/ServicePulse.Host/app/layout/navbar.html index 83dac63cb..3af48a9e4 100644 --- a/src/ServicePulse.Host/app/layout/navbar.html +++ b/src/ServicePulse.Host/app/layout/navbar.html @@ -46,7 +46,8 @@ Configuration - + +
  • diff --git a/src/ServicePulse.Host/app/modules/configuration/configuration.js b/src/ServicePulse.Host/app/modules/configuration/configuration.js index 102cd832e..f44f133aa 100644 --- a/src/ServicePulse.Host/app/modules/configuration/configuration.js +++ b/src/ServicePulse.Host/app/modules/configuration/configuration.js @@ -1 +1,3 @@ require('./js/configuration.module'); +require('./connectionsManager'); +require('./connectionsStatus'); \ No newline at end of file diff --git a/src/ServicePulse.Host/app/modules/configuration/connectionsManager.js b/src/ServicePulse.Host/app/modules/configuration/connectionsManager.js new file mode 100644 index 000000000..4d3f25cd5 --- /dev/null +++ b/src/ServicePulse.Host/app/modules/configuration/connectionsManager.js @@ -0,0 +1,81 @@ +require('url-search-params-polyfill'); + +class ConnectionsManager { + constructor() { + const urlParams = new URLSearchParams(window.location.search); + + if (urlParams.has('scu')) { + this.serviceControlUrl = urlParams.get('scu'); + window.localStorage.setItem('scu', this.serviceControlUrl); + console.debug(`ServiceControl Url found in QS and stored in local storage: ${this.serviceControlUrl}`); + } else if (window.localStorage.getItem('scu')) { + this.serviceControlUrl = window.localStorage.getItem('scu'); + console.debug(`ServiceControl Url, not in QS, found in local storage: ${this.serviceControlUrl}`); + } else if (window.defaultConfig && window.defaultConfig.service_control_url) { + this.serviceControlUrl = window.defaultConfig.service_control_url; + console.debug(`setting ServiceControl Url to its default value: ${window.defaultConfig.service_control_url}`); + } else { + console.warn('ServiceControl Url is not defined.'); + } + + if (urlParams.has('mu')) { + this.monitoringUrl = urlParams.get('mu'); + window.localStorage.setItem('mu', this.monitoringUrl); + console.debug(`Monitoring Url found in QS and stored in local storage: ${this.monitoringUrl}`); + } else if (window.localStorage.getItem('mu')) { + this.monitoringUrl = window.localStorage.getItem('mu'); + console.debug(`Monitoring Url, not in QS, found in local storage: ${this.monitoringUrl}`); + } else if (window.defaultConfig && window.defaultConfig.monitoring_urls && window.defaultConfig.monitoring_urls.length) { + this.monitoringUrl = window.defaultConfig.monitoring_urls[0]; + console.debug(`setting Monitoring Url to its default value: ${window.defaultConfig.monitoring_urls[0]}`); + } else { + console.warn('Monitoring Url is not defined.'); + } + } + + getIsMonitoringEnabled() { + return this.monitoringUrl !== '!' + && this.monitoringUrl !== '' + && this.monitoringUrl !== null + && this.monitoringUrl !== undefined; + } + + getMonitoringUrl() { + if (this.getIsMonitoringEnabled()) { + return this.monitoringUrl; + } + return null; + } + + getServiceControlUrl() { return this.serviceControlUrl; } + + updateConnections(serviceControlUrl, monitoringUrl) { + + const urlParams = new URLSearchParams(window.location.search); + + if (!serviceControlUrl) { + throw 'ServiceControl URL is mandatory'; + } + + urlParams.set('scu', serviceControlUrl); + + if (!monitoringUrl) { + monitoringUrl = '!'; //disabled + } + + urlParams.set('mu', monitoringUrl); + + //values have changed. They'll be reset after page reloads + window.localStorage.removeItem('scu'); + window.localStorage.removeItem('mu'); + + let newSearch = urlParams.toString(); + console.debug('updateConnections - new query string: ', newSearch); + window.location.search = newSearch; + } +} + +window.connectionsManager = new ConnectionsManager(); + +angular.module('configuration') + .service('connectionsManager', function () { return window.connectionsManager; }); diff --git a/src/ServicePulse.Host/app/modules/configuration/connectionsStatus.js b/src/ServicePulse.Host/app/modules/configuration/connectionsStatus.js new file mode 100644 index 000000000..4e6e759db --- /dev/null +++ b/src/ServicePulse.Host/app/modules/configuration/connectionsStatus.js @@ -0,0 +1,39 @@ +import { timingSafeEqual } from "crypto"; + +class ConnectionsStatus { + constructor(notifyService, $rootScope) { + + var notifier = notifyService(); + + notifier.subscribe($rootScope, (event, data) => { + if (data.isSCConnected !== this.isSCConnected + || data.isSCConnecting !== this.isSCConnecting + || data.scConnectedAtLeastOnce !== this.scConnectedAtLeastOnce) { + + this.isSCConnected = data.isSCConnected; + this.isSCConnecting = data.isSCConnecting; + this.scConnectedAtLeastOnce = data.scConnectedAtLeastOnce; + + notifier.notify('ConnectionsStatusChanged', { status: this }); + } + }, 'ServiceControlConnectionStatusChanged'); + + notifier.subscribe($rootScope, (event, data) => { + if (data.isMonitoringConnected !== this.isMonitoringConnected + || data.isMonitoringConnecting !== this.isMonitoringConnecting) { + + this.isMonitoringConnected = data.isMonitoringConnected; + this.isMonitoringConnecting = data.isMonitoringConnecting; + + notifier.notify('ConnectionsStatusChanged', { status: this }); + } + }, 'MonitoringConnectionStatusChanged'); + } +} + +angular.module('configuration') + .service('connectionsStatus', ['notifyService', '$rootScope', function (notifyService, $rootScope) { + return new ConnectionsStatus(notifyService, $rootScope); + }]).run(['connectionsStatus', function(connectionsStatus) { + //make sure the service is initialized as the app starts, otherwise it won't raise notifications unless it's required as dependency + }]); diff --git a/src/ServicePulse.Host/app/modules/configuration/js/configuration.module.js b/src/ServicePulse.Host/app/modules/configuration/js/configuration.module.js index 6d6765eb3..db706959c 100644 --- a/src/ServicePulse.Host/app/modules/configuration/js/configuration.module.js +++ b/src/ServicePulse.Host/app/modules/configuration/js/configuration.module.js @@ -10,12 +10,12 @@ require('./directives/ui.particular.configurationTabs'); require('./directives/ui.particular.redirectLink'); - require('./redirect/redirect.module'); - require('./license/license.module'); require('./redirect/redirect.module'); + require('./connections/connections.module'); + angular.module('configuration', [ 'ui.bootstrap', 'configuration.route', @@ -24,6 +24,7 @@ 'configuration.tabs', 'configuration.redirect', 'configuration.license', + 'configuration.connections', ]); } (window, window.angular)); \ No newline at end of file diff --git a/src/ServicePulse.Host/app/modules/configuration/js/configuration.route.js b/src/ServicePulse.Host/app/modules/configuration/js/configuration.route.js index 7563d287a..b2c8c5229 100644 --- a/src/ServicePulse.Host/app/modules/configuration/js/configuration.route.js +++ b/src/ServicePulse.Host/app/modules/configuration/js/configuration.route.js @@ -8,7 +8,8 @@ redirectTo: "/configuration/license" }).when('/configuration/endpoints', { data: { - pageTitle: 'Monitored endpoints - Configuration' + pageTitle: 'Monitored endpoints - Configuration', + redirectWhenNotConnected: '/configuration/connections' }, template: template, controller: 'ConfigurationCtrl', diff --git a/src/ServicePulse.Host/app/modules/configuration/js/configuration.service.js b/src/ServicePulse.Host/app/modules/configuration/js/configuration.service.js index 34726b799..a35cc3fca 100644 --- a/src/ServicePulse.Host/app/modules/configuration/js/configuration.service.js +++ b/src/ServicePulse.Host/app/modules/configuration/js/configuration.service.js @@ -2,7 +2,9 @@ (function (window, angular, undefined) { 'use strict'; - function Service($http, $q, scConfig, uri) { + function Service($http, $q, connectionsManager, uri) { + + var scu = connectionsManager.getServiceControlUrl(); function patchPromise(url, data, success, error) { @@ -30,7 +32,7 @@ } function getData() { - var url = uri.join(scConfig.service_control_url, 'endpoints'); + var url = uri.join(scu, 'endpoints'); return $http.get(url).then(function (response) { return { data: response.data @@ -41,7 +43,7 @@ var service = { getData: getData, update: function (id, newState, success, error) { - var url = uri.join(scConfig.service_control_url, 'endpoints', id); + var url = uri.join(scu, 'endpoints', id); return patchPromise(url, { "monitor_heartbeat": newState }, success, error); } }; @@ -50,7 +52,7 @@ } - Service.$inject = ['$http', '$q', 'scConfig', 'uri']; + Service.$inject = ['$http', '$q', 'connectionsManager', 'uri']; angular.module('configuration.service', []) .factory('configurationService', Service); diff --git a/src/ServicePulse.Host/app/modules/configuration/js/connections/connections.controller.js b/src/ServicePulse.Host/app/modules/configuration/js/connections/connections.controller.js new file mode 100644 index 000000000..6be749088 --- /dev/null +++ b/src/ServicePulse.Host/app/modules/configuration/js/connections/connections.controller.js @@ -0,0 +1,106 @@ +; (function (window, angular, undefined) { + 'use strict'; + + function controller( + $scope, + connectionsManager, + $http, + notifyService, + connectionsStatus, + uri) { + + var vm = this; + var notifier = notifyService(); + + var initialServiceControlUrl = connectionsManager.getServiceControlUrl(); + var initialMonitoringUrl = connectionsManager.getMonitoringUrl(); + var isMonitoringEnabled = connectionsManager.getIsMonitoringEnabled(); + + vm.loadingData = false; + vm.configuredServiceControlUrl = initialServiceControlUrl; + vm.configuredMonitoringUrl = initialMonitoringUrl; + + vm.unableToConnectToServiceControl = false; + vm.unableToConnectToMonitoring = false; + + var evalConnectionsStatus = function () { + if (connectionsStatus.isSCConnecting) { + vm.unableToConnectToServiceControl = false; + } else { + vm.unableToConnectToServiceControl = !connectionsStatus.isSCConnected; + } + + if (!isMonitoringEnabled || connectionsStatus.isMonitoringConnecting || connectionsStatus.isMonitoringConnecting === undefined) { + vm.unableToConnectToMonitoring = false; + } else { + vm.unableToConnectToMonitoring = !connectionsStatus.isMonitoringConnected; + } + } + + notifier.subscribe($scope, (event, data) => { + evalConnectionsStatus(); + }, 'ConnectionsStatusChanged'); + + function prependSchemeIfMissing(userUrl) { + var url = userUrl.toLowerCase(); + if (url.startsWith('http://') || url.startsWith('https://')) { + return userUrl; + } + + return 'http://' + userUrl; + } + + vm.testServiceControlUrl = () => { + if (vm.configuredServiceControlUrl) { + vm.configuredServiceControlUrl = prependSchemeIfMissing(vm.configuredServiceControlUrl); + vm.testingServiceControl = true; + $http.get(vm.configuredServiceControlUrl).then(() => { + vm.serviceControlValid = true; + }, (error) => { + vm.serviceControlValid = false; + }).then(() => { + vm.testingServiceControl = false; + }); + } + }; + + vm.testMonitoringUrl = () => { + if (vm.configuredMonitoringUrl) { + vm.configuredMonitoringUrl = prependSchemeIfMissing(vm.configuredMonitoringUrl); + vm.testingMonitoring = true; + /* + Monitoring root URL doesn't support CORS, + so to test connectivity we need to hit one + of the Monitoring API URLs that are CORS enabled. + */ + var urlToTest = uri.join(vm.configuredMonitoringUrl, '/monitored-endpoints'); + $http.get(urlToTest).then(() => { + vm.monitoringValid = true; + }, (error) => { + vm.monitoringValid = false; + }).then(() => { + vm.testingMonitoring = false; + }); + } + }; + + vm.save = () => { + connectionsManager.updateConnections(vm.configuredServiceControlUrl, vm.configuredMonitoringUrl); + }; + + evalConnectionsStatus(); + } + + controller.$inject = [ + '$scope', + 'connectionsManager', + '$http', + 'notifyService', + 'connectionsStatus', + 'uri', + ]; + + angular.module('configuration.connections') + .controller('connectionsController', controller); + +})(window, window.angular); \ No newline at end of file diff --git a/src/ServicePulse.Host/app/modules/configuration/js/connections/connections.module.js b/src/ServicePulse.Host/app/modules/configuration/js/connections/connections.module.js new file mode 100644 index 000000000..4f215b805 --- /dev/null +++ b/src/ServicePulse.Host/app/modules/configuration/js/connections/connections.module.js @@ -0,0 +1,9 @@ +; (function (window, angular, undefined) { + 'use strict'; + + angular.module('configuration.connections', []); + + require('./connections.route'); + require('./connections.controller'); + +}(window, window.angular)); \ No newline at end of file diff --git a/src/ServicePulse.Host/app/modules/configuration/js/connections/connections.route.js b/src/ServicePulse.Host/app/modules/configuration/js/connections/connections.route.js new file mode 100644 index 000000000..8a3e52b68 --- /dev/null +++ b/src/ServicePulse.Host/app/modules/configuration/js/connections/connections.route.js @@ -0,0 +1,26 @@ +; (function (window, angular, undefined) { + 'use strict'; + + function routeProvider($routeProvider) { + const template = require('../../views/connections.html'); + + $routeProvider.when('/connections', { + redirectTo: "/configuration/connections" + }).when('/configuration/connections', { + data: { + pageTitle: 'Connections - Configuration' + }, + template: template, + controller: 'connectionsController', + controllerAs: 'vm' + }); + } + + routeProvider.$inject = [ + '$routeProvider' + ]; + + angular.module('configuration.connections', []) + .config(routeProvider); + +}(window, window.angular)); diff --git a/src/ServicePulse.Host/app/modules/configuration/js/directives/ui.particular.configurationTabs.js b/src/ServicePulse.Host/app/modules/configuration/js/directives/ui.particular.configurationTabs.js index 1b72df2c4..f45608108 100644 --- a/src/ServicePulse.Host/app/modules/configuration/js/directives/ui.particular.configurationTabs.js +++ b/src/ServicePulse.Host/app/modules/configuration/js/directives/ui.particular.configurationTabs.js @@ -3,11 +3,35 @@ angular.module('configuration', []); - function controller($scope, $location, redirectService, notifyService, sharedDataService, licenseService, licenseNotifierService) { + function controller($scope, $location, redirectService, notifyService, sharedDataService, licenseService, licenseNotifierService, connectionsStatus, connectionsManager) { var notifier = notifyService(); + var isMonitoringEnabled = connectionsManager.getIsMonitoringEnabled(); + $scope.isActive = (viewLocation) => viewLocation === $location.path(); + $scope.connectionsStatus = connectionsStatus; + $scope.unableToConnectToServiceControl = undefined; + $scope.unableToConnectToMonitoring = undefined; + + var evalConnectionsStatus = function() { + if (connectionsStatus.isSCConnecting) { + $scope.unableToConnectToServiceControl = false; + } else { + $scope.unableToConnectToServiceControl = !connectionsStatus.isSCConnected; + } + + if (!isMonitoringEnabled || connectionsStatus.isMonitoringConnecting || connectionsStatus.isMonitoringConnecting === undefined) { + $scope.unableToConnectToMonitoring = false; + } else { + $scope.unableToConnectToMonitoring = !connectionsStatus.isMonitoringConnected; + } + } + + notifier.subscribe($scope, (event, data) => { + evalConnectionsStatus(); + }, 'ConnectionsStatusChanged'); + var stats = sharedDataService.getstats(); $scope.counters = { @@ -40,10 +64,19 @@ $scope.licensewarning = 'danger'; } }); + + evalConnectionsStatus(); } - controller.$inject = ['$scope', '$location', 'redirectService', 'notifyService', 'sharedDataService', - 'licenseService', 'licenseNotifierService']; + controller.$inject = ['$scope', + '$location', + 'redirectService', + 'notifyService', + 'sharedDataService', + 'licenseService', + 'licenseNotifierService', + 'connectionsStatus', + 'connectionsManager']; function directive() { const template = require('./ui.particular.configurationTabs.tpl.html'); diff --git a/src/ServicePulse.Host/app/modules/configuration/js/directives/ui.particular.configurationTabs.tpl.html b/src/ServicePulse.Host/app/modules/configuration/js/directives/ui.particular.configurationTabs.tpl.html index b8cf70f64..2980cf2db 100644 --- a/src/ServicePulse.Host/app/modules/configuration/js/directives/ui.particular.configurationTabs.tpl.html +++ b/src/ServicePulse.Host/app/modules/configuration/js/directives/ui.particular.configurationTabs.tpl.html @@ -1,9 +1,22 @@  + \ No newline at end of file diff --git a/src/ServicePulse.Host/app/modules/configuration/js/license/license.route.js b/src/ServicePulse.Host/app/modules/configuration/js/license/license.route.js index 1195eb059..f14a46d88 100644 --- a/src/ServicePulse.Host/app/modules/configuration/js/license/license.route.js +++ b/src/ServicePulse.Host/app/modules/configuration/js/license/license.route.js @@ -6,7 +6,8 @@ $routeProvider.when('/configuration/license', { data: { - pageTitle: 'License - Configuration' + pageTitle: 'License - Configuration', + redirectWhenNotConnected: '/configuration/connections' }, template: template, controller: 'LicenseController', diff --git a/src/ServicePulse.Host/app/modules/configuration/js/license/license.service.js b/src/ServicePulse.Host/app/modules/configuration/js/license/license.service.js index 8afcc42d8..c32f1ec76 100644 --- a/src/ServicePulse.Host/app/modules/configuration/js/license/license.service.js +++ b/src/ServicePulse.Host/app/modules/configuration/js/license/license.service.js @@ -2,7 +2,7 @@ (function (window, angular, undefined) { 'use strict'; - function service($http, $q, scConfig, uri, notifyService) { + function service($http, $q, connectionsManager, uri, notifyService) { var notifier = notifyService(); var license = { @@ -15,7 +15,7 @@ }; function getData() { - var url = uri.join(scConfig.service_control_url, 'license'); + var url = uri.join(connectionsManager.getServiceControlUrl(), 'license'); return $http.get(url).then(function (response) { license = response.data; @@ -33,7 +33,7 @@ }; } - service.$inject = ['$http', '$q', 'scConfig', 'uri', 'notifyService']; + service.$inject = ['$http', '$q', 'connectionsManager', 'uri', 'notifyService']; angular.module('configuration.license') .service('licenseService', service); diff --git a/src/ServicePulse.Host/app/modules/configuration/js/redirect/redirect.route.js b/src/ServicePulse.Host/app/modules/configuration/js/redirect/redirect.route.js index 892e4e662..336a12459 100644 --- a/src/ServicePulse.Host/app/modules/configuration/js/redirect/redirect.route.js +++ b/src/ServicePulse.Host/app/modules/configuration/js/redirect/redirect.route.js @@ -8,7 +8,8 @@ redirectTo: "/configuration/redirects" }).when('/configuration/redirects', { data: { - pageTitle: 'Retry Redirects - Configuration' + pageTitle: 'Retry Redirects - Configuration', + redirectWhenNotConnected: '/configuration/connections' }, template: template, controller: 'redirectController', diff --git a/src/ServicePulse.Host/app/modules/configuration/js/redirect/redirect.service.js b/src/ServicePulse.Host/app/modules/configuration/js/redirect/redirect.service.js index 33dc2dd53..267c5b0d4 100644 --- a/src/ServicePulse.Host/app/modules/configuration/js/redirect/redirect.service.js +++ b/src/ServicePulse.Host/app/modules/configuration/js/redirect/redirect.service.js @@ -1,8 +1,9 @@ ; (function (window, angular, undefined) { 'use strict'; - function service($http, $timeout, $q, $rootScope, $interval, moment, scConfig, uri, notifications, notifyService) { + function service($http, $timeout, $q, $rootScope, $interval, moment, connectionsManager, uri, notifications, notifyService) { var notifier = notifyService(); + var scu = connectionsManager.getServiceControlUrl(); var redirects = { total :0, @@ -10,7 +11,7 @@ }; function getData() { - var url = uri.join(scConfig.service_control_url, 'redirects'); + var url = uri.join(scu, 'redirects'); return $http.get(url).then(function (response) { redirects.data = response.data; redirects.data.forEach(function(item) { @@ -81,7 +82,7 @@ return { createRedirect: function(sourceEndpoint, targetEndpoint, success, error) { - var url = uri.join(scConfig.service_control_url, 'redirects'); + var url = uri.join(scu, 'redirects'); var promise = sendPromise(url, 'POST', { "fromphysicaladdress": sourceEndpoint, "tophysicaladdress": targetEndpoint }, @@ -91,7 +92,7 @@ return promise; }, updateRedirect: function(redirectId, sourceEndpoint, targetEndpoint, success, error) { - var url = uri.join(scConfig.service_control_url, 'redirects', redirectId); + var url = uri.join(scu, 'redirects', redirectId); var promise = sendPromise(url, 'PUT', { "id": redirectId, "fromphysicaladdress": sourceEndpoint, "tophysicaladdress": targetEndpoint }, @@ -101,7 +102,7 @@ return promise; }, deleteRedirect: function(id, success, error) { - var url = uri.join(scConfig.service_control_url, 'redirects', id); + var url = uri.join(scu, 'redirects', id); return $http.delete(url) .then(function() { notifications.pushForCurrentRoute(success, 'info'); @@ -119,7 +120,7 @@ }; } - service.$inject = ['$http', '$timeout', '$q', '$rootScope', '$interval', 'moment', 'scConfig', 'uri', 'notifications', 'notifyService']; + service.$inject = ['$http', '$timeout', '$q', '$rootScope', '$interval', 'moment', 'connectionsManager', 'uri', 'notifications', 'notifyService']; angular.module('configuration.redirect') .service('redirectService', service); diff --git a/src/ServicePulse.Host/app/modules/configuration/views/configuration.html b/src/ServicePulse.Host/app/modules/configuration/views/configuration.html index f286147c1..c3660b290 100644 --- a/src/ServicePulse.Host/app/modules/configuration/views/configuration.html +++ b/src/ServicePulse.Host/app/modules/configuration/views/configuration.html @@ -16,66 +16,72 @@

    Configuration

    -
    -
    +
    +
    -
    - Warning: The list of endpoints below only contains endpoints with the heartbeats plug-in installed. Toggling heartbeat monitoring won't toggle performance monitoring -
    +
    + +
    +
    + +
    + Warning: The list of endpoints below only contains endpoints with the heartbeats plug-in installed. Toggling heartbeat monitoring won't toggle performance monitoring +
    -
    -
    -
    -
    -
    - - +
    +
    +
    +
    +
    + + +
    +
    - -
    -
    -
    -
    -

    - {{e.name}}@{{e.host_display_name}} -

    - - - +
    +
    +
    +

    + {{e.name}}@{{e.host_display_name}} +

    + + + +
    -
    -
    -
    -
    -
    - -
    -
    -
    -
    -

    Nothing to configure

    +
    +
    +
    +
    + +
    +
    +
    +
    +

    Nothing to configure

    +
    -
    +
    -
    +
    diff --git a/src/ServicePulse.Host/app/modules/configuration/views/connections.html b/src/ServicePulse.Host/app/modules/configuration/views/connections.html new file mode 100644 index 000000000..cc8a89088 --- /dev/null +++ b/src/ServicePulse.Host/app/modules/configuration/views/connections.html @@ -0,0 +1,88 @@ + + + + +
    +
    +
    +

    Configuration

    +
    +
    +
    + +
    +
    + + + +
    +
    +
    +
    + +
    +
    +

    ServiceControl

    + +
    + + +
    + +
    + + Testing + Connection successful + Connection failed +
    + +
    + +
    +

    ServiceControl Monitoring

    +
    + + +
    + +
    + + Testing + Connection successful + Connection failed +
    + +
    + + + + Connection saved + Unable to save +
    + +
    +
    +
    +
    + +
    +
    \ No newline at end of file diff --git a/src/ServicePulse.Host/app/modules/configuration/views/license.html b/src/ServicePulse.Host/app/modules/configuration/views/license.html index 3ecaba737..5bed23bde 100644 --- a/src/ServicePulse.Host/app/modules/configuration/views/license.html +++ b/src/ServicePulse.Host/app/modules/configuration/views/license.html @@ -10,60 +10,67 @@

    Configuration

    - -
    -
    -
    -
    +
    +
    -
    Platform license type: {{vm.licenseType}}{{vm.licenseEdition}}
    +
    -
    - License expiry date: - - {{vm.formattedExpirationDate}} {{vm.expirationDaysLeft}} - - -
    - Your license expired. Please update the license to continue using the Particular Service Platform. -
    -
    - Your trial period has expired. To continue using the Particular Service Platform you'll need to extend your trial or purchase a license. -
    - -
    + + +
    +
    +
    +
    + +
    Platform license type: {{vm.licenseType}}{{vm.licenseEdition}}
    -
    - - Upgrade protection expiry date: - - {{vm.formattedUpgradeProtectionExpiration}} {{vm.upgradeDaysLeft}} +
    + License expiry date: + + {{vm.formattedExpirationDate}} {{vm.expirationDaysLeft}} - -
    - Warning: Once upgrade protection expires, you'll no longer have access to support or new product versions. +
    + Your license expired. Please update the license to continue using the Particular Service Platform. +
    +
    + Your trial period has expired. To continue using the Particular Service Platform you'll need to extend your trial or purchase a license. +
    +
    -
    - Your license upgrade protection expired before this version of ServicePulse was released. + +
    + + Upgrade protection expiry date: + + {{vm.formattedUpgradeProtectionExpiration}} {{vm.upgradeDaysLeft}} + + + +
    + Warning: Once upgrade protection expires, you'll no longer have access to support or new product versions. +
    +
    + Your license upgrade protection expired before this version of ServicePulse was released. +
    -
    -
    ServiceControl instance: {{vm.scInstanceName}}
    - +
    ServiceControl instance: {{vm.scInstanceName}}
    + -
    - Need help? Contact us +
    + Need help? Contact us +
    -
    +
    \ No newline at end of file diff --git a/src/ServicePulse.Host/app/modules/configuration/views/redirect.html b/src/ServicePulse.Host/app/modules/configuration/views/redirect.html index 413fb67e4..8b95d9004 100644 --- a/src/ServicePulse.Host/app/modules/configuration/views/redirect.html +++ b/src/ServicePulse.Host/app/modules/configuration/views/redirect.html @@ -15,57 +15,63 @@

    Configuration

    - +
    +
    -
    -
    -
    - - +
    + + + +
    +
    +
    + + +
    -
    - + -
    -
    +
    +
    -
    -
    -
    -

    - - {{redirect.from_physical_address}} -

    -

    - - {{redirect.to_physical_address}} -

    - +
    +
    +
    +

    + + {{redirect.from_physical_address}} +

    +

    + + {{redirect.to_physical_address}} +

    + +
    -
    -
    -
    -

    +

    +
    +

    - - -

    -
    + + +

    +
    +
    -
    +
    diff --git a/src/ServicePulse.Host/app/modules/monitoring/dist/monitoring.dist.js b/src/ServicePulse.Host/app/modules/monitoring/dist/monitoring.dist.js deleted file mode 100644 index 634b49e92..000000000 --- a/src/ServicePulse.Host/app/modules/monitoring/dist/monitoring.dist.js +++ /dev/null @@ -1,191 +0,0 @@ -/******/ (function(modules) { // webpackBootstrap -/******/ // The module cache -/******/ var installedModules = {}; -/******/ -/******/ // The require function -/******/ function __webpack_require__(moduleId) { -/******/ -/******/ // Check if module is in cache -/******/ if(installedModules[moduleId]) { -/******/ return installedModules[moduleId].exports; -/******/ } -/******/ // Create a new module (and put it into the cache) -/******/ var module = installedModules[moduleId] = { -/******/ i: moduleId, -/******/ l: false, -/******/ exports: {} -/******/ }; -/******/ -/******/ // Execute the module function -/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__); -/******/ -/******/ // Flag the module as loaded -/******/ module.l = true; -/******/ -/******/ // Return the exports of the module -/******/ return module.exports; -/******/ } -/******/ -/******/ -/******/ // expose the modules object (__webpack_modules__) -/******/ __webpack_require__.m = modules; -/******/ -/******/ // expose the module cache -/******/ __webpack_require__.c = installedModules; -/******/ -/******/ // define getter function for harmony exports -/******/ __webpack_require__.d = function(exports, name, getter) { -/******/ if(!__webpack_require__.o(exports, name)) { -/******/ Object.defineProperty(exports, name, { -/******/ configurable: false, -/******/ enumerable: true, -/******/ get: getter -/******/ }); -/******/ } -/******/ }; -/******/ -/******/ // getDefaultExport function for compatibility with non-harmony modules -/******/ __webpack_require__.n = function(module) { -/******/ var getter = module && module.__esModule ? -/******/ function getDefault() { return module['default']; } : -/******/ function getModuleExports() { return module; }; -/******/ __webpack_require__.d(getter, 'a', getter); -/******/ return getter; -/******/ }; -/******/ -/******/ // Object.prototype.hasOwnProperty.call -/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); }; -/******/ -/******/ // __webpack_public_path__ -/******/ __webpack_require__.p = ""; -/******/ -/******/ // Load entry module and return exports -/******/ return __webpack_require__(__webpack_require__.s = 6); -/******/ }) -/************************************************************************/ -/******/ ([ -/* 0 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n;\n(function (window, angular, $, undefined) {\n 'use strict';\n\n function Service(toastService, scConfig) {\n\n var isConnectedToSourceIndex = Array(scConfig.monitoring_urls.length).fill(true);\n\n function reportFailedConnection(sourceIndex) {\n\n if (isConnectedToSourceIndex[sourceIndex]) {\n var message = 'Could not connect to the ServiceControl Monitoring service.';\n if (scConfig.monitoring_urls.length > 1) {\n message = 'Could not connect to the ServiceControl Monitoring service at' + scConfig.monitoring_urls[sourceIndex] + '.';\n }\n toastService.showError(message);\n }\n isConnectedToSourceIndex[sourceIndex] = false;\n }\n\n function reportSuccessfulConnection(sourceIndex) {\n if (!isConnectedToSourceIndex[sourceIndex]) {\n var message = 'Connection to ServiceControl Monitoring service was successful.';\n if (scConfig.monitoring_urls.length > 1) {\n message = 'Connection to ServiceControl Monitoring service was successful ' + scConfig.monitoring_urls[sourceIndex] + '.';\n }\n toastService.showInfo(message, 'Info', true);\n }\n isConnectedToSourceIndex[sourceIndex] = true;\n }\n\n var service = {\n reportFailedConnection: reportFailedConnection,\n reportSuccessfulConnection: reportSuccessfulConnection\n };\n\n return service;\n }\n\n Service.$inject = ['toastService', 'scConfig'];\n\n angular.module('services.connectivityNotifier', ['sc']).service('connectivityNotifier', Service);\n})(window, window.angular, window.jQuery);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL3NlcnZpY2VzL3NlcnZpY2VzLmNvbm5lY3Rpdml0eU5vdGlmaWVyLmpzP2RjZjYiXSwibmFtZXMiOlsid2luZG93IiwiYW5ndWxhciIsIiQiLCJ1bmRlZmluZWQiLCJTZXJ2aWNlIiwidG9hc3RTZXJ2aWNlIiwic2NDb25maWciLCJpc0Nvbm5lY3RlZFRvU291cmNlSW5kZXgiLCJBcnJheSIsIm1vbml0b3JpbmdfdXJscyIsImxlbmd0aCIsImZpbGwiLCJyZXBvcnRGYWlsZWRDb25uZWN0aW9uIiwic291cmNlSW5kZXgiLCJtZXNzYWdlIiwic2hvd0Vycm9yIiwicmVwb3J0U3VjY2Vzc2Z1bENvbm5lY3Rpb24iLCJzaG93SW5mbyIsInNlcnZpY2UiLCIkaW5qZWN0IiwibW9kdWxlIiwialF1ZXJ5Il0sIm1hcHBpbmdzIjoiOztBQUFBO0FBQ0MsV0FBVUEsTUFBVixFQUFrQkMsT0FBbEIsRUFBMkJDLENBQTNCLEVBQThCQyxTQUE5QixFQUF5QztBQUN0Qzs7QUFFQSxhQUFTQyxPQUFULENBQWlCQyxZQUFqQixFQUErQkMsUUFBL0IsRUFBeUM7O0FBRXJDLFlBQUlDLDJCQUEyQkMsTUFBTUYsU0FBU0csZUFBVCxDQUF5QkMsTUFBL0IsRUFBdUNDLElBQXZDLENBQTRDLElBQTVDLENBQS9COztBQUVBLGlCQUFTQyxzQkFBVCxDQUFnQ0MsV0FBaEMsRUFBNkM7O0FBRXpDLGdCQUFJTix5QkFBeUJNLFdBQXpCLENBQUosRUFBMkM7QUFDdkMsb0JBQUlDLFVBQVUsNkRBQWQ7QUFDQSxvQkFBSVIsU0FBU0csZUFBVCxDQUF5QkMsTUFBekIsR0FBa0MsQ0FBdEMsRUFBeUM7QUFDckNJLDhCQUFVLGtFQUFrRVIsU0FBU0csZUFBVCxDQUF5QkksV0FBekIsQ0FBbEUsR0FBMEcsR0FBcEg7QUFDSDtBQUNEUiw2QkFBYVUsU0FBYixDQUF1QkQsT0FBdkI7QUFDSDtBQUNEUCxxQ0FBeUJNLFdBQXpCLElBQXdDLEtBQXhDO0FBQ0g7O0FBRUQsaUJBQVNHLDBCQUFULENBQW9DSCxXQUFwQyxFQUFpRDtBQUM3QyxnQkFBSSxDQUFDTix5QkFBeUJNLFdBQXpCLENBQUwsRUFBNEM7QUFDeEMsb0JBQUlDLFVBQVUsaUVBQWQ7QUFDQSxvQkFBSVIsU0FBU0csZUFBVCxDQUF5QkMsTUFBekIsR0FBa0MsQ0FBdEMsRUFBeUM7QUFDckNJLDhCQUFVLG9FQUFvRVIsU0FBU0csZUFBVCxDQUF5QkksV0FBekIsQ0FBcEUsR0FBMkcsR0FBckg7QUFDSDtBQUNEUiw2QkFBYVksUUFBYixDQUFzQkgsT0FBdEIsRUFBK0IsTUFBL0IsRUFBdUMsSUFBdkM7QUFDSDtBQUNEUCxxQ0FBeUJNLFdBQXpCLElBQXdDLElBQXhDO0FBQ0g7O0FBRUQsWUFBSUssVUFBVTtBQUNWTixvQ0FBd0JBLHNCQURkO0FBRVZJLHdDQUE0QkE7QUFGbEIsU0FBZDs7QUFLQSxlQUFPRSxPQUFQO0FBQ0g7O0FBRURkLFlBQVFlLE9BQVIsR0FBa0IsQ0FBQyxjQUFELEVBQWlCLFVBQWpCLENBQWxCOztBQUVBbEIsWUFBUW1CLE1BQVIsQ0FBZSwrQkFBZixFQUFnRCxDQUFDLElBQUQsQ0FBaEQsRUFDS0YsT0FETCxDQUNhLHNCQURiLEVBQ3FDZCxPQURyQztBQUVILENBMUNBLEVBMENDSixNQTFDRCxFQTBDU0EsT0FBT0MsT0ExQ2hCLEVBMEN5QkQsT0FBT3FCLE1BMUNoQyxDQUFEIiwiZmlsZSI6IjAuanMiLCJzb3VyY2VzQ29udGVudCI6WyI7XHJcbihmdW5jdGlvbiAod2luZG93LCBhbmd1bGFyLCAkLCB1bmRlZmluZWQpIHtcclxuICAgICd1c2Ugc3RyaWN0JztcclxuXHJcbiAgICBmdW5jdGlvbiBTZXJ2aWNlKHRvYXN0U2VydmljZSwgc2NDb25maWcpIHtcclxuXHJcbiAgICAgICAgdmFyIGlzQ29ubmVjdGVkVG9Tb3VyY2VJbmRleCA9IEFycmF5KHNjQ29uZmlnLm1vbml0b3JpbmdfdXJscy5sZW5ndGgpLmZpbGwodHJ1ZSk7XHJcbiAgICAgICAgXHJcbiAgICAgICAgZnVuY3Rpb24gcmVwb3J0RmFpbGVkQ29ubmVjdGlvbihzb3VyY2VJbmRleCkge1xyXG5cclxuICAgICAgICAgICAgaWYgKGlzQ29ubmVjdGVkVG9Tb3VyY2VJbmRleFtzb3VyY2VJbmRleF0pIHtcclxuICAgICAgICAgICAgICAgIHZhciBtZXNzYWdlID0gJ0NvdWxkIG5vdCBjb25uZWN0IHRvIHRoZSBTZXJ2aWNlQ29udHJvbCBNb25pdG9yaW5nIHNlcnZpY2UuJztcclxuICAgICAgICAgICAgICAgIGlmIChzY0NvbmZpZy5tb25pdG9yaW5nX3VybHMubGVuZ3RoID4gMSkge1xyXG4gICAgICAgICAgICAgICAgICAgIG1lc3NhZ2UgPSAnQ291bGQgbm90IGNvbm5lY3QgdG8gdGhlIFNlcnZpY2VDb250cm9sIE1vbml0b3Jpbmcgc2VydmljZSBhdCcgKyBzY0NvbmZpZy5tb25pdG9yaW5nX3VybHNbc291cmNlSW5kZXhdICsgJy4nO1xyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgdG9hc3RTZXJ2aWNlLnNob3dFcnJvcihtZXNzYWdlKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICBpc0Nvbm5lY3RlZFRvU291cmNlSW5kZXhbc291cmNlSW5kZXhdID0gZmFsc2U7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBmdW5jdGlvbiByZXBvcnRTdWNjZXNzZnVsQ29ubmVjdGlvbihzb3VyY2VJbmRleCkge1xyXG4gICAgICAgICAgICBpZiAoIWlzQ29ubmVjdGVkVG9Tb3VyY2VJbmRleFtzb3VyY2VJbmRleF0pIHtcclxuICAgICAgICAgICAgICAgIHZhciBtZXNzYWdlID0gJ0Nvbm5lY3Rpb24gdG8gU2VydmljZUNvbnRyb2wgTW9uaXRvcmluZyBzZXJ2aWNlIHdhcyBzdWNjZXNzZnVsLic7XHJcbiAgICAgICAgICAgICAgICBpZiAoc2NDb25maWcubW9uaXRvcmluZ191cmxzLmxlbmd0aCA+IDEpIHtcclxuICAgICAgICAgICAgICAgICAgICBtZXNzYWdlID0gJ0Nvbm5lY3Rpb24gdG8gU2VydmljZUNvbnRyb2wgTW9uaXRvcmluZyBzZXJ2aWNlIHdhcyBzdWNjZXNzZnVsICcgKyBzY0NvbmZpZy5tb25pdG9yaW5nX3VybHNbc291cmNlSW5kZXhdICsnLic7XHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgICAgICB0b2FzdFNlcnZpY2Uuc2hvd0luZm8obWVzc2FnZSwgJ0luZm8nLCB0cnVlKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICBpc0Nvbm5lY3RlZFRvU291cmNlSW5kZXhbc291cmNlSW5kZXhdID0gdHJ1ZTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHZhciBzZXJ2aWNlID0ge1xyXG4gICAgICAgICAgICByZXBvcnRGYWlsZWRDb25uZWN0aW9uOiByZXBvcnRGYWlsZWRDb25uZWN0aW9uLFxyXG4gICAgICAgICAgICByZXBvcnRTdWNjZXNzZnVsQ29ubmVjdGlvbjogcmVwb3J0U3VjY2Vzc2Z1bENvbm5lY3Rpb25cclxuICAgICAgICB9O1xyXG5cclxuICAgICAgICByZXR1cm4gc2VydmljZTtcclxuICAgIH1cclxuXHJcbiAgICBTZXJ2aWNlLiRpbmplY3QgPSBbJ3RvYXN0U2VydmljZScsICdzY0NvbmZpZyddO1xyXG5cclxuICAgIGFuZ3VsYXIubW9kdWxlKCdzZXJ2aWNlcy5jb25uZWN0aXZpdHlOb3RpZmllcicsIFsnc2MnXSlcclxuICAgICAgICAuc2VydmljZSgnY29ubmVjdGl2aXR5Tm90aWZpZXInLCBTZXJ2aWNlKTtcclxufSh3aW5kb3csIHdpbmRvdy5hbmd1bGFyLCB3aW5kb3cualF1ZXJ5KSk7XG5cblxuLy8gV0VCUEFDSyBGT09URVIgLy9cbi8vIC4vYXBwL21vZHVsZXMvbW9uaXRvcmluZy9qcy9zZXJ2aWNlcy9zZXJ2aWNlcy5jb25uZWN0aXZpdHlOb3RpZmllci5qcyJdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///0\n"); - -/***/ }), -/* 1 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n;(function (window, angular, undefined) {\n 'use strict';\n\n angular.module('monitored_endpoints').constant('largeGraphsMinimumYAxis', {\n 'queueLength': 10,\n 'throughputRetries': 10,\n 'processingCritical': 10\n }).constant('smallGraphsMinimumYAxis', {\n 'queueLength': 10,\n 'throughput': 10,\n 'retries': 10,\n 'processingTime': 10,\n 'criticalTime': 10\n });\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL2NvbnN0YW50LmRpYWdyYW1zLmpzPzhjMjkiXSwibmFtZXMiOlsid2luZG93IiwiYW5ndWxhciIsInVuZGVmaW5lZCIsIm1vZHVsZSIsImNvbnN0YW50Il0sIm1hcHBpbmdzIjoiOztBQUFBLENBQUcsV0FBVUEsTUFBVixFQUFrQkMsT0FBbEIsRUFBMkJDLFNBQTNCLEVBQXNDO0FBQ3JDOztBQUVBRCxZQUFRRSxNQUFSLENBQWUscUJBQWYsRUFDS0MsUUFETCxDQUNjLHlCQURkLEVBQ3lDO0FBQzdCLHVCQUFlLEVBRGM7QUFFN0IsNkJBQXFCLEVBRlE7QUFHN0IsOEJBQXNCO0FBSE8sS0FEekMsRUFNS0EsUUFOTCxDQU1jLHlCQU5kLEVBTXlDO0FBQ2pDLHVCQUFlLEVBRGtCO0FBRWpDLHNCQUFjLEVBRm1CO0FBR2pDLG1CQUFXLEVBSHNCO0FBSWpDLDBCQUFrQixFQUplO0FBS2pDLHdCQUFnQjtBQUxpQixLQU56QztBQWNILENBakJFLEVBaUJESixNQWpCQyxFQWlCT0EsT0FBT0MsT0FqQmQsQ0FBRCIsImZpbGUiOiIxLmpzIiwic291cmNlc0NvbnRlbnQiOlsiOyAoZnVuY3Rpb24gKHdpbmRvdywgYW5ndWxhciwgdW5kZWZpbmVkKSB7XHJcbiAgICAndXNlIHN0cmljdCc7XHJcblxyXG4gICAgYW5ndWxhci5tb2R1bGUoJ21vbml0b3JlZF9lbmRwb2ludHMnKVxyXG4gICAgICAgIC5jb25zdGFudCgnbGFyZ2VHcmFwaHNNaW5pbXVtWUF4aXMnLCB7XHJcbiAgICAgICAgICAgICAgICAncXVldWVMZW5ndGgnOiAxMCxcclxuICAgICAgICAgICAgICAgICd0aHJvdWdocHV0UmV0cmllcyc6IDEwLFxyXG4gICAgICAgICAgICAgICAgJ3Byb2Nlc3NpbmdDcml0aWNhbCc6IDEwLFxyXG4gICAgICAgICAgICB9KVxyXG4gICAgICAgIC5jb25zdGFudCgnc21hbGxHcmFwaHNNaW5pbXVtWUF4aXMnLCB7XHJcbiAgICAgICAgICAgICdxdWV1ZUxlbmd0aCc6IDEwLFxyXG4gICAgICAgICAgICAndGhyb3VnaHB1dCc6IDEwLFxyXG4gICAgICAgICAgICAncmV0cmllcyc6IDEwLFxyXG4gICAgICAgICAgICAncHJvY2Vzc2luZ1RpbWUnOiAxMCxcclxuICAgICAgICAgICAgJ2NyaXRpY2FsVGltZSc6IDEwLFxyXG4gICAgICAgIH0pO1xyXG5cclxufSh3aW5kb3csIHdpbmRvdy5hbmd1bGFyKSk7XHJcblxuXG5cbi8vIFdFQlBBQ0sgRk9PVEVSIC8vXG4vLyAuL2FwcC9tb2R1bGVzL21vbml0b3JpbmcvanMvY29uc3RhbnQuZGlhZ3JhbXMuanMiXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///1\n"); - -/***/ }), -/* 2 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular, undefined) {\n 'use strict';\n\n angular.module('ui.particular.graph', []).directive('graph', function () {\n return {\n restrict: 'E',\n scope: {\n plotData: '=',\n formatter: '&',\n minimumYaxis: '@'\n },\n template: '',\n link: function link(scope, element, attrs) {\n scope.plotData = scope.plotData || { points: [], average: 0 };\n\n scope.$watch('plotData', function () {\n var svg = element.find('svg')[0];\n\n var width = svg.clientWidth;\n var height = svg.clientHeight;\n\n //HINT: This is workaround for Firefox\n if (width === 0) {\n var box = svg.getBoundingClientRect();\n\n width = box.right - box.left;\n height = box.bottom - box.top;\n }\n\n var verticalMargin = 6;\n var horizontalMargin = 2;\n\n var points = scope.plotData.points;\n var average = scope.plotData.average || 0;\n var minimumYaxis = !isNaN(scope.minimumYaxis) ? Number(scope.minimumYaxis) : 10;\n var max = points && points.length ? Math.max(average * 1.5, d3.max(points), minimumYaxis) : 1;\n var numberOfPoints = points && points.length ? points.length : 2;\n\n var scaleY = d3.scaleLinear().domain([0, max]).range([height - verticalMargin, verticalMargin]);\n\n var scaleX = d3.scaleLinear().domain([0, numberOfPoints - 1]).range([horizontalMargin, width - horizontalMargin]);\n\n var area = d3.area().x(function (d, i) {\n return scaleX(i);\n }).y(function (d, i) {\n return scaleY(d);\n }).y1(function (d) {\n return scaleY(0);\n }).curve(d3.curveLinear);\n\n var line = d3.line().x(function (d, i) {\n return scaleX(i);\n }).y(function (d, i) {\n return scaleY(d);\n }).curve(d3.curveLinear);\n\n d3.select(svg).selectAll(\"*\").remove();\n\n var chart = d3.select(svg).attr('width', width).attr('height', height);\n\n chart.append('rect').attr('width', width - 2 * horizontalMargin).attr('height', height - 2 * verticalMargin).attr('transform', 'translate(' + horizontalMargin + ',' + verticalMargin + ')').attr('fill', '#F2F6F7');\n\n if (points) {\n chart.append('path').datum(points).attr('d', area).attr('class', 'graph-data-fill');\n\n chart.append('path').datum(points).attr('d', line).attr('class', 'graph-data-line');\n }\n\n chart.append('path').datum(Array(numberOfPoints).fill(average)).attr('d', line).attr('class', 'graph-avg-line');\n });\n }\n };\n });\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./app/modules/monitoring/js/directives/ui.particular.graph.js?c1b8"],"names":["window","angular","undefined","module","directive","restrict","scope","plotData","formatter","minimumYaxis","template","link","element","attrs","points","average","$watch","svg","find","width","clientWidth","height","clientHeight","box","getBoundingClientRect","right","left","bottom","top","verticalMargin","horizontalMargin","isNaN","Number","max","length","Math","d3","numberOfPoints","scaleY","scaleLinear","domain","range","scaleX","area","x","d","i","y","y1","curve","curveLinear","line","select","selectAll","remove","chart","attr","append","datum","Array","fill"],"mappings":"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","file":"2.js","sourcesContent":["(function(window, angular, undefined) {\r\n    'use strict';\r\n\r\n    angular.module('ui.particular.graph', [])\r\n        .directive('graph',\r\n            function() {\r\n                return {\r\n                    restrict: 'E',\r\n                    scope: {\r\n                        plotData: '=',\r\n                        formatter: '&',\r\n                        minimumYaxis: '@'\r\n                    },\r\n                    template: '<svg></svg>',\r\n                    link: function link(scope, element, attrs) {\r\n                        scope.plotData = scope.plotData || { points: [], average: 0 };\r\n\r\n                        scope.$watch('plotData',\r\n                            function() {\r\n                                var svg = element.find('svg')[0];\r\n\r\n                                var width = svg.clientWidth;\r\n                                var height = svg.clientHeight;\r\n\r\n                                //HINT: This is workaround for Firefox\r\n                                if (width === 0) {\r\n                                    var box = svg.getBoundingClientRect();\r\n\r\n                                    width = box.right - box.left;\r\n                                    height = box.bottom - box.top;\r\n                                }\r\n\r\n                                var verticalMargin = 6;\r\n                                var horizontalMargin = 2;\r\n\r\n                                var points = scope.plotData.points;\r\n                                var average = scope.plotData.average || 0;\r\n                                var minimumYaxis = !isNaN(scope.minimumYaxis) ? Number(scope.minimumYaxis) : 10;\r\n                                var max = points && points.length ? Math.max(average * 1.5, d3.max(points), minimumYaxis) : 1;\r\n                                var numberOfPoints = points && points.length ? points.length : 2;\r\n\r\n                                var scaleY = d3.scaleLinear()\r\n                                    .domain([0, max])\r\n                                    .range([height - verticalMargin, verticalMargin]);\r\n\r\n                                var scaleX = d3.scaleLinear()\r\n                                    .domain([0, numberOfPoints - 1])\r\n                                    .range([horizontalMargin, width - horizontalMargin]);\r\n\r\n                                var area = d3.area()\r\n                                    .x(function (d, i) {\r\n                                        return scaleX(i);\r\n                                    })\r\n                                    .y(function (d, i) { return scaleY(d); })\r\n                                    .y1(function (d) { return scaleY(0); })\r\n                                    .curve(d3.curveLinear);\r\n\r\n                                var line = d3.line()\r\n                                    .x(function (d, i) {\r\n                                        return scaleX(i);\r\n                                    })\r\n                                    .y(function (d, i) {\r\n                                        return scaleY(d);\r\n                                    })\r\n                                    .curve(d3.curveLinear);\r\n\r\n                                d3.select(svg).selectAll(\"*\").remove();\r\n\r\n                                var chart = d3.select(svg)\r\n                                    .attr('width', width)\r\n                                    .attr('height', height);\r\n\r\n                                chart.append('rect')\r\n                                    .attr('width', width - 2 * horizontalMargin)\r\n                                    .attr('height', height - 2 * verticalMargin)\r\n                                    .attr('transform', 'translate(' + horizontalMargin + ',' + verticalMargin + ')')\r\n                                    .attr('fill', '#F2F6F7');\r\n\r\n                                if (points) {\r\n                                    chart.append('path')\r\n                                        .datum(points)\r\n                                        .attr('d', area)\r\n                                        .attr('class', 'graph-data-fill');\r\n\r\n\r\n                                    chart.append('path')\r\n                                        .datum(points)\r\n                                        .attr('d', line)\r\n                                        .attr('class', 'graph-data-line');\r\n                                }\r\n\r\n                                chart.append('path')\r\n                                    .datum(Array(numberOfPoints).fill(average))\r\n                                    .attr('d', line)\r\n                                    .attr('class', 'graph-avg-line');\r\n                            });\r\n                    }\r\n                };\r\n            });\r\n\r\n}(window, window.angular));\n\n\n// WEBPACK FOOTER //\n// ./app/modules/monitoring/js/directives/ui.particular.graph.js"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///2\n"); - -/***/ }), -/* 3 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular, undefined) {\n 'use strict';\n\n angular.module('ui.particular.graphdecimal', []).filter('graphdecimal', ['$filter', function ($filter) {\n return function (input, decimals) {\n if (input) {\n var lastValue = input.points.length > 0 ? input.points[input.points.length - 1] : 0;\n input.displayValue = $filter(\"metricslargenumber\")(lastValue, decimals);\n } else {\n input = {\n points: [],\n average: 0,\n displayValue: 0\n };\n }\n\n return input;\n };\n }]);\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5ncmFwaGRlY2ltYWwuanM/NTFkMyJdLCJuYW1lcyI6WyJ3aW5kb3ciLCJhbmd1bGFyIiwidW5kZWZpbmVkIiwibW9kdWxlIiwiZmlsdGVyIiwiJGZpbHRlciIsImlucHV0IiwiZGVjaW1hbHMiLCJsYXN0VmFsdWUiLCJwb2ludHMiLCJsZW5ndGgiLCJkaXNwbGF5VmFsdWUiLCJhdmVyYWdlIl0sIm1hcHBpbmdzIjoiOztBQUFDLFdBQVVBLE1BQVYsRUFBa0JDLE9BQWxCLEVBQTJCQyxTQUEzQixFQUFzQztBQUNuQzs7QUFFQUQsWUFBUUUsTUFBUixDQUFlLDRCQUFmLEVBQTZDLEVBQTdDLEVBQ0tDLE1BREwsQ0FDWSxjQURaLEVBQzRCLENBQUMsU0FBRCxFQUFZLFVBQVVDLE9BQVYsRUFBbUI7QUFDbkQsZUFBTyxVQUFVQyxLQUFWLEVBQWlCQyxRQUFqQixFQUEyQjtBQUM5QixnQkFBSUQsS0FBSixFQUFXO0FBQ1Asb0JBQUlFLFlBQVlGLE1BQU1HLE1BQU4sQ0FBYUMsTUFBYixHQUFzQixDQUF0QixHQUEwQkosTUFBTUcsTUFBTixDQUFhSCxNQUFNRyxNQUFOLENBQWFDLE1BQWIsR0FBc0IsQ0FBbkMsQ0FBMUIsR0FBa0UsQ0FBbEY7QUFDQUosc0JBQU1LLFlBQU4sR0FBcUJOLFFBQVEsb0JBQVIsRUFBOEJHLFNBQTlCLEVBQXlDRCxRQUF6QyxDQUFyQjtBQUNILGFBSEQsTUFHTztBQUNIRCx3QkFBUTtBQUNKRyw0QkFBUSxFQURKO0FBRUpHLDZCQUFTLENBRkw7QUFHSkQsa0NBQWM7QUFIVixpQkFBUjtBQUtIOztBQUVELG1CQUFPTCxLQUFQO0FBQ0gsU0FiRDtBQWNILEtBZnVCLENBRDVCO0FBaUJILENBcEJBLEVBb0JDTixNQXBCRCxFQW9CU0EsT0FBT0MsT0FwQmhCLENBQUQiLCJmaWxlIjoiMy5qcyIsInNvdXJjZXNDb250ZW50IjpbIihmdW5jdGlvbiAod2luZG93LCBhbmd1bGFyLCB1bmRlZmluZWQpIHtcclxuICAgICd1c2Ugc3RyaWN0JztcclxuXHJcbiAgICBhbmd1bGFyLm1vZHVsZSgndWkucGFydGljdWxhci5ncmFwaGRlY2ltYWwnLCBbXSlcclxuICAgICAgICAuZmlsdGVyKCdncmFwaGRlY2ltYWwnLCBbJyRmaWx0ZXInLCBmdW5jdGlvbiAoJGZpbHRlcikge1xyXG4gICAgICAgICAgICByZXR1cm4gZnVuY3Rpb24gKGlucHV0LCBkZWNpbWFscykge1xyXG4gICAgICAgICAgICAgICAgaWYgKGlucHV0KSB7XHJcbiAgICAgICAgICAgICAgICAgICAgdmFyIGxhc3RWYWx1ZSA9IGlucHV0LnBvaW50cy5sZW5ndGggPiAwID8gaW5wdXQucG9pbnRzW2lucHV0LnBvaW50cy5sZW5ndGggLSAxXSA6IDA7XHJcbiAgICAgICAgICAgICAgICAgICAgaW5wdXQuZGlzcGxheVZhbHVlID0gJGZpbHRlcihcIm1ldHJpY3NsYXJnZW51bWJlclwiKShsYXN0VmFsdWUsIGRlY2ltYWxzKTtcclxuICAgICAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgICAgICAgaW5wdXQgPSB7XHJcbiAgICAgICAgICAgICAgICAgICAgICAgIHBvaW50czogW10sXHJcbiAgICAgICAgICAgICAgICAgICAgICAgIGF2ZXJhZ2U6IDAsXHJcbiAgICAgICAgICAgICAgICAgICAgICAgIGRpc3BsYXlWYWx1ZTogMFxyXG4gICAgICAgICAgICAgICAgICAgIH07XHJcbiAgICAgICAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgICAgICAgcmV0dXJuIGlucHV0O1xyXG4gICAgICAgICAgICB9O1xyXG4gICAgICAgIH1dKTtcclxufSh3aW5kb3csIHdpbmRvdy5hbmd1bGFyKSk7XHJcblxuXG5cbi8vIFdFQlBBQ0sgRk9PVEVSIC8vXG4vLyAuL2FwcC9tb2R1bGVzL21vbml0b3JpbmcvanMvZGlyZWN0aXZlcy91aS5wYXJ0aWN1bGFyLmdyYXBoZGVjaW1hbC5qcyJdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///3\n"); - -/***/ }), -/* 4 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular, undefined) {\n 'use strict';\n\n angular.module('ui.particular.graphduration', []).filter('graphduration', ['formatter', function (formatter) {\n return function (input) {\n if (input) {\n var lastValue = input.points.length > 0 ? input.points[input.points.length - 1] : 0;\n input.displayValue = formatter.formatTime(lastValue);\n }\n\n return input;\n };\n }]);\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5ncmFwaGR1cmF0aW9uLmpzP2Y3NWIiXSwibmFtZXMiOlsid2luZG93IiwiYW5ndWxhciIsInVuZGVmaW5lZCIsIm1vZHVsZSIsImZpbHRlciIsImZvcm1hdHRlciIsImlucHV0IiwibGFzdFZhbHVlIiwicG9pbnRzIiwibGVuZ3RoIiwiZGlzcGxheVZhbHVlIiwiZm9ybWF0VGltZSJdLCJtYXBwaW5ncyI6Ijs7QUFBQyxXQUFVQSxNQUFWLEVBQWtCQyxPQUFsQixFQUEyQkMsU0FBM0IsRUFBc0M7QUFDbkM7O0FBRUFELFlBQVFFLE1BQVIsQ0FBZSw2QkFBZixFQUE4QyxFQUE5QyxFQUNLQyxNQURMLENBQ1ksZUFEWixFQUM2QixDQUFDLFdBQUQsRUFBYyxVQUFVQyxTQUFWLEVBQXFCO0FBQ3hELGVBQU8sVUFBVUMsS0FBVixFQUFpQjtBQUNwQixnQkFBSUEsS0FBSixFQUFXO0FBQ1Asb0JBQUlDLFlBQVlELE1BQU1FLE1BQU4sQ0FBYUMsTUFBYixHQUFzQixDQUF0QixHQUEwQkgsTUFBTUUsTUFBTixDQUFhRixNQUFNRSxNQUFOLENBQWFDLE1BQWIsR0FBc0IsQ0FBbkMsQ0FBMUIsR0FBa0UsQ0FBbEY7QUFDQUgsc0JBQU1JLFlBQU4sR0FBcUJMLFVBQVVNLFVBQVYsQ0FBcUJKLFNBQXJCLENBQXJCO0FBQ0g7O0FBRUQsbUJBQU9ELEtBQVA7QUFDSCxTQVBEO0FBUUgsS0FUd0IsQ0FEN0I7QUFXSCxDQWRBLEVBY0NOLE1BZEQsRUFjU0EsT0FBT0MsT0FkaEIsQ0FBRCIsImZpbGUiOiI0LmpzIiwic291cmNlc0NvbnRlbnQiOlsiKGZ1bmN0aW9uICh3aW5kb3csIGFuZ3VsYXIsIHVuZGVmaW5lZCkge1xyXG4gICAgJ3VzZSBzdHJpY3QnO1xyXG5cclxuICAgIGFuZ3VsYXIubW9kdWxlKCd1aS5wYXJ0aWN1bGFyLmdyYXBoZHVyYXRpb24nLCBbXSlcclxuICAgICAgICAuZmlsdGVyKCdncmFwaGR1cmF0aW9uJywgWydmb3JtYXR0ZXInLCBmdW5jdGlvbiAoZm9ybWF0dGVyKSB7XHJcbiAgICAgICAgICAgIHJldHVybiBmdW5jdGlvbiAoaW5wdXQpIHtcclxuICAgICAgICAgICAgICAgIGlmIChpbnB1dCkge1xyXG4gICAgICAgICAgICAgICAgICAgIHZhciBsYXN0VmFsdWUgPSBpbnB1dC5wb2ludHMubGVuZ3RoID4gMCA/IGlucHV0LnBvaW50c1tpbnB1dC5wb2ludHMubGVuZ3RoIC0gMV0gOiAwO1xyXG4gICAgICAgICAgICAgICAgICAgIGlucHV0LmRpc3BsYXlWYWx1ZSA9IGZvcm1hdHRlci5mb3JtYXRUaW1lKGxhc3RWYWx1ZSk7XHJcbiAgICAgICAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgICAgICAgcmV0dXJuIGlucHV0O1xyXG4gICAgICAgICAgICB9O1xyXG4gICAgICAgIH1dKTtcclxufSh3aW5kb3csIHdpbmRvdy5hbmd1bGFyKSk7XHJcblxuXG5cbi8vIFdFQlBBQ0sgRk9PVEVSIC8vXG4vLyAuL2FwcC9tb2R1bGVzL21vbml0b3JpbmcvanMvZGlyZWN0aXZlcy91aS5wYXJ0aWN1bGFyLmdyYXBoZHVyYXRpb24uanMiXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///4\n"); - -/***/ }), -/* 5 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular) {\n 'use strict';\n\n angular.module('ui.particular.metricslargenumber', []).filter('metricslargenumber', ['formatter', function (formatter) {\n return function (input, dec) {\n var decimals = 0;\n if (input < 10 || input > 1000000) {\n decimals = 2;\n }\n return formatter.formatLargeNumber(input, dec || decimals);\n };\n }]);\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5tZXRyaWNzbGFyZ2VudW1iZXIuanM/ZWQ2MSJdLCJuYW1lcyI6WyJ3aW5kb3ciLCJhbmd1bGFyIiwibW9kdWxlIiwiZmlsdGVyIiwiZm9ybWF0dGVyIiwiaW5wdXQiLCJkZWMiLCJkZWNpbWFscyIsImZvcm1hdExhcmdlTnVtYmVyIl0sIm1hcHBpbmdzIjoiOztBQUFDLFdBQVNBLE1BQVQsRUFBaUJDLE9BQWpCLEVBQTBCO0FBQzFCOztBQUVHQSxZQUFRQyxNQUFSLENBQWUsa0NBQWYsRUFBbUQsRUFBbkQsRUFDS0MsTUFETCxDQUNZLG9CQURaLEVBQ2tDLENBQUMsV0FBRCxFQUFjLFVBQVVDLFNBQVYsRUFBcUI7QUFDN0QsZUFBTyxVQUFVQyxLQUFWLEVBQWlCQyxHQUFqQixFQUFzQjtBQUN6QixnQkFBSUMsV0FBVyxDQUFmO0FBQ0EsZ0JBQUlGLFFBQVEsRUFBUixJQUFjQSxRQUFRLE9BQTFCLEVBQW1DO0FBQy9CRSwyQkFBVyxDQUFYO0FBQ0g7QUFDRCxtQkFBT0gsVUFBVUksaUJBQVYsQ0FBNEJILEtBQTVCLEVBQW1DQyxPQUFPQyxRQUExQyxDQUFQO0FBQ0gsU0FORDtBQU9ILEtBUjZCLENBRGxDO0FBVUgsQ0FiQSxFQWFDUCxNQWJELEVBYVNBLE9BQU9DLE9BYmhCLENBQUQiLCJmaWxlIjoiNS5qcyIsInNvdXJjZXNDb250ZW50IjpbIihmdW5jdGlvbih3aW5kb3csIGFuZ3VsYXIpIHtcclxuXHQndXNlIHN0cmljdCc7XHJcblxyXG4gICAgYW5ndWxhci5tb2R1bGUoJ3VpLnBhcnRpY3VsYXIubWV0cmljc2xhcmdlbnVtYmVyJywgW10pXHJcbiAgICAgICAgLmZpbHRlcignbWV0cmljc2xhcmdlbnVtYmVyJywgWydmb3JtYXR0ZXInLCBmdW5jdGlvbiAoZm9ybWF0dGVyKSB7XHJcbiAgICAgICAgICAgIHJldHVybiBmdW5jdGlvbiAoaW5wdXQsIGRlYykge1xyXG4gICAgICAgICAgICAgICAgdmFyIGRlY2ltYWxzID0gMDtcclxuICAgICAgICAgICAgICAgIGlmIChpbnB1dCA8IDEwIHx8IGlucHV0ID4gMTAwMDAwMCkge1xyXG4gICAgICAgICAgICAgICAgICAgIGRlY2ltYWxzID0gMjtcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgIHJldHVybiBmb3JtYXR0ZXIuZm9ybWF0TGFyZ2VOdW1iZXIoaW5wdXQsIGRlYyB8fCBkZWNpbWFscyk7XHJcbiAgICAgICAgICAgIH07XHJcbiAgICAgICAgfV0pO1xyXG59KHdpbmRvdywgd2luZG93LmFuZ3VsYXIpKTtcblxuXG4vLyBXRUJQQUNLIEZPT1RFUiAvL1xuLy8gLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5tZXRyaWNzbGFyZ2VudW1iZXIuanMiXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///5\n"); - -/***/ }), -/* 6 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n__webpack_require__(7);\n__webpack_require__(12);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL21vbml0b3JpbmcuanM/ZTJhZCJdLCJuYW1lcyI6WyJyZXF1aXJlIl0sIm1hcHBpbmdzIjoiOztBQUFBLG1CQUFBQSxDQUFRLENBQVI7QUFDQSxtQkFBQUEsQ0FBUSxFQUFSIiwiZmlsZSI6IjYuanMiLCJzb3VyY2VzQ29udGVudCI6WyJyZXF1aXJlKCcuL2pzL21vbml0b3JlZF9lbmRwb2ludHMubW9kdWxlJyk7XHJcbnJlcXVpcmUoJy4vanMvZW5kcG9pbnRfZGV0YWlscy5tb2R1bGUnKTtcblxuXG4vLyBXRUJQQUNLIEZPT1RFUiAvL1xuLy8gLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL21vbml0b3JpbmcuanMiXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///6\n"); - -/***/ }), -/* 7 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular, undefined) {\n 'use strict';\n\n angular.module('monitored_endpoints', []);\n\n __webpack_require__(8);\n __webpack_require__(0);\n __webpack_require__(9);\n __webpack_require__(10);\n __webpack_require__(1);\n\n __webpack_require__(2);\n __webpack_require__(3);\n __webpack_require__(4);\n __webpack_require__(5);\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL21vbml0b3JlZF9lbmRwb2ludHMubW9kdWxlLmpzPzA3YmQiXSwibmFtZXMiOlsid2luZG93IiwiYW5ndWxhciIsInVuZGVmaW5lZCIsIm1vZHVsZSIsInJlcXVpcmUiXSwibWFwcGluZ3MiOiI7O0FBQUMsV0FBVUEsTUFBVixFQUFrQkMsT0FBbEIsRUFBMkJDLFNBQTNCLEVBQXNDO0FBQ25DOztBQUNBRCxZQUFRRSxNQUFSLENBQWUscUJBQWYsRUFBc0MsRUFBdEM7O0FBRUFDLElBQUEsbUJBQUFBLENBQVEsQ0FBUjtBQUNBQSxJQUFBLG1CQUFBQSxDQUFRLENBQVI7QUFDQUEsSUFBQSxtQkFBQUEsQ0FBUSxDQUFSO0FBQ0FBLElBQUEsbUJBQUFBLENBQVEsRUFBUjtBQUNBQSxJQUFBLG1CQUFBQSxDQUFRLENBQVI7O0FBRUFBLElBQUEsbUJBQUFBLENBQVEsQ0FBUjtBQUNBQSxJQUFBLG1CQUFBQSxDQUFRLENBQVI7QUFDQUEsSUFBQSxtQkFBQUEsQ0FBUSxDQUFSO0FBQ0FBLElBQUEsbUJBQUFBLENBQVEsQ0FBUjtBQUNILENBZEEsRUFjQ0osTUFkRCxFQWNTQSxPQUFPQyxPQWRoQixDQUFEIiwiZmlsZSI6IjcuanMiLCJzb3VyY2VzQ29udGVudCI6WyIoZnVuY3Rpb24gKHdpbmRvdywgYW5ndWxhciwgdW5kZWZpbmVkKSB7XHJcbiAgICAndXNlIHN0cmljdCc7XHJcbiAgICBhbmd1bGFyLm1vZHVsZSgnbW9uaXRvcmVkX2VuZHBvaW50cycsIFtdKTtcclxuXHJcbiAgICByZXF1aXJlKCcuL3NlcnZpY2VzL3NlcnZpY2VzLm1vbml0b3JpbmcnKTtcclxuICAgIHJlcXVpcmUoJy4vc2VydmljZXMvc2VydmljZXMuY29ubmVjdGl2aXR5Tm90aWZpZXInKTtcclxuICAgIHJlcXVpcmUoJy4vbW9uaXRvcmVkX2VuZHBvaW50cy5jb250cm9sbGVyJyk7XHJcbiAgICByZXF1aXJlKCcuL21vbml0b3JlZF9lbmRwb2ludHMucm91dGUuanMnKTtcclxuICAgIHJlcXVpcmUoJy4vY29uc3RhbnQuZGlhZ3JhbXMuanMnKTtcclxuXHJcbiAgICByZXF1aXJlKCcuL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5ncmFwaC5qcycpO1xyXG4gICAgcmVxdWlyZSgnLi9kaXJlY3RpdmVzL3VpLnBhcnRpY3VsYXIuZ3JhcGhkZWNpbWFsLmpzJyk7XHJcbiAgICByZXF1aXJlKCcuL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5ncmFwaGR1cmF0aW9uLmpzJyk7XHJcbiAgICByZXF1aXJlKCcuL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5tZXRyaWNzbGFyZ2VudW1iZXIuanMnKTtcclxufSh3aW5kb3csIHdpbmRvdy5hbmd1bGFyKSk7XG5cblxuLy8gV0VCUEFDSyBGT09URVIgLy9cbi8vIC4vYXBwL21vZHVsZXMvbW9uaXRvcmluZy9qcy9tb25pdG9yZWRfZW5kcG9pbnRzLm1vZHVsZS5qcyJdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///7\n"); - -/***/ }), -/* 8 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n;\n(function (window, angular, $, undefined) {\n 'use strict';\n\n function Service($http, rx, scConfig, uri, $q) {\n\n function createEndpointsSource(historyPeriod, refreshInterval) {\n return Rx.Observable.interval(refreshInterval).startWith(0).flatMap(function (i) {\n return Rx.Observable.fromArray(loadEndpointDataFromMonitoringService(historyPeriod)).flatMap(function (p) {\n var o = Rx.Observable.fromPromise(p);\n o = o.catch(Rx.Observable.empty());\n return o;\n });\n }).selectMany(function (endpoints) {\n return endpoints;\n });\n }\n\n function loadEndpointDataFromMonitoringService(historyPeriod) {\n return scConfig.monitoring_urls.map(function (url) {\n return $http.get(uri.join(url, 'monitored-endpoints') + '?history=' + historyPeriod).then(function (result) {\n var sourceIndex = scConfig.monitoring_urls.indexOf(url);\n\n result.data.forEach(function (endpoint) {\n endpoint.sourceIndex = sourceIndex;\n });\n\n return result.data;\n }, function (error) {\n var sourceIndex = scConfig.monitoring_urls.indexOf(url);\n return [{ error: error, sourceIndex: sourceIndex }];\n });\n });\n }\n\n function loadEndpointDetailsFromMonitoringService(endpointName, sourceIndex, historyPeriod) {\n return $http.get(uri.join(scConfig.monitoring_urls[sourceIndex], 'monitored-endpoints', endpointName) + \"?history=\" + historyPeriod).then(function (result) {\n return result.data;\n }, function (error) {\n return { error: error };\n });\n }\n\n function createEndpointDetailsSource(endpointName, sourceIndex, historyPeriod, refreshInterval) {\n return Rx.Observable.interval(refreshInterval).startWith(0).flatMap(function (i) {\n return Rx.Observable.fromPromise(loadEndpointDetailsFromMonitoringService(endpointName, sourceIndex, historyPeriod));\n });\n }\n\n var service = {\n createEndpointsSource: createEndpointsSource,\n createEndpointDetailsSource: createEndpointDetailsSource\n };\n\n return service;\n }\n\n Service.$inject = ['$http', 'rx', 'scConfig', 'uri', '$q', 'toastService'];\n\n angular.module('services.monitoringService', ['sc']).service('monitoringService', Service);\n})(window, window.angular, window.jQuery);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./app/modules/monitoring/js/services/services.monitoring.js?fa65"],"names":["window","angular","$","undefined","Service","$http","rx","scConfig","uri","$q","createEndpointsSource","historyPeriod","refreshInterval","Rx","Observable","interval","startWith","flatMap","i","fromArray","loadEndpointDataFromMonitoringService","p","o","fromPromise","catch","empty","selectMany","endpoints","monitoring_urls","map","url","get","join","then","result","sourceIndex","indexOf","data","forEach","endpoint","error","loadEndpointDetailsFromMonitoringService","endpointName","createEndpointDetailsSource","service","$inject","module","jQuery"],"mappings":"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","file":"8.js","sourcesContent":[";\r\n(function (window, angular, $, undefined) {\r\n    'use strict';\r\n\r\n    function Service($http, rx, scConfig, uri, $q) {\r\n\r\n        function createEndpointsSource(historyPeriod, refreshInterval) {\r\n            return Rx.Observable.interval(refreshInterval).startWith(0)\r\n                .flatMap(function (i) {\r\n                    return Rx.Observable.fromArray(loadEndpointDataFromMonitoringService(historyPeriod))\r\n                        .flatMap(function (p) {\r\n                            var o = Rx.Observable.fromPromise(p);\r\n                            o = o.catch(Rx.Observable.empty());\r\n                            return o;\r\n                        });\r\n                }).selectMany(function (endpoints) {\r\n                    return endpoints;\r\n                });\r\n        }\r\n\r\n        function loadEndpointDataFromMonitoringService(historyPeriod) {\r\n            return scConfig.monitoring_urls.map(function (url) {\r\n                return $http.get(uri.join(url, 'monitored-endpoints') + '?history=' + historyPeriod)\r\n                    .then(function (result) {\r\n                        var sourceIndex = scConfig.monitoring_urls.indexOf(url);\r\n\r\n                        result.data.forEach(function (endpoint) {\r\n                            endpoint.sourceIndex = sourceIndex;\r\n                        });\r\n\r\n                        return result.data;\r\n                    },\r\n                    (error) => {\r\n                        var sourceIndex = scConfig.monitoring_urls.indexOf(url);\r\n                        return [{ error: error, sourceIndex: sourceIndex }];\r\n                        }\r\n                    );\r\n            });\r\n        }\r\n\r\n        function loadEndpointDetailsFromMonitoringService(endpointName, sourceIndex, historyPeriod) {\r\n            return $http.get(uri.join(scConfig.monitoring_urls[sourceIndex], 'monitored-endpoints', endpointName) + \"?history=\" + historyPeriod)\r\n                .then(function (result) {\r\n                    return result.data;\r\n                }, function (error) {\r\n                    return { error: error };\r\n                });\r\n        }\r\n\r\n        function createEndpointDetailsSource(endpointName, sourceIndex, historyPeriod, refreshInterval) {\r\n            return Rx.Observable.interval(refreshInterval).startWith(0)\r\n                .flatMap(function (i) {\r\n                    return Rx.Observable.fromPromise(loadEndpointDetailsFromMonitoringService(endpointName, sourceIndex, historyPeriod));\r\n                });\r\n        }\r\n\r\n        var service = {\r\n            createEndpointsSource: createEndpointsSource,\r\n            createEndpointDetailsSource: createEndpointDetailsSource\r\n        };\r\n\r\n        return service;\r\n    }\r\n\r\n    Service.$inject = ['$http', 'rx', 'scConfig', 'uri', '$q', 'toastService'];\r\n\r\n    angular.module('services.monitoringService', ['sc'])\r\n        .service('monitoringService', Service);\r\n}(window, window.angular, window.jQuery));\r\n\n\n\n// WEBPACK FOOTER //\n// ./app/modules/monitoring/js/services/services.monitoring.js"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///8\n"); - -/***/ }), -/* 9 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular, undefined) {\n 'use strict';\n\n function controller($scope, $location, monitoringService, serviceControlService, toastService, historyPeriods, rx, $filter, smallGraphsMinimumYAxis, connectivityNotifier) {\n\n var subscription, endpointsFromScSubscription;\n\n $scope.periods = historyPeriods;\n $scope.selectedPeriod = $scope.periods[0];\n $scope.smallGraphsMinimumYAxis = smallGraphsMinimumYAxis;\n\n if ($location.$$search.historyPeriod) {\n $scope.selectedPeriod = $scope.periods[$scope.periods.findIndex(function (period) {\n return period.value == $location.$$search.historyPeriod;\n })];\n }\n\n $scope.endpoints = [];\n\n $scope.selectPeriod = function (period) {\n $scope.selectedPeriod = period;\n\n updateUI();\n };\n\n $scope.getDetailsUrl = function (endpoint) {\n return '#/endpoint_details/' + endpoint.name + '/' + (endpoint.sourceIndex | 0) + '?historyPeriod=' + $scope.selectedPeriod.value;\n };\n\n function fillDisplayValuesForEndpoint(endpoint) {\n\n $filter('graphduration')(endpoint.metrics.processingTime);\n $filter('graphduration')(endpoint.metrics.criticalTime);\n $filter('graphdecimal')(endpoint.metrics.queueLength, 0);\n $filter('graphdecimal')(endpoint.metrics.throughput, 2);\n $filter('graphdecimal')(endpoint.metrics.retries, 2);\n }\n\n function mergeIn(destination, source) {\n for (var propName in source) {\n if (source.hasOwnProperty(propName)) {\n destination[propName] = source[propName];\n }\n }\n }\n\n function updateUI() {\n if (subscription) {\n subscription.dispose();\n }\n\n if (endpointsFromScSubscription) {\n endpointsFromScSubscription.dispose();\n }\n\n var selectedPeriod = $scope.selectedPeriod;\n\n subscription = monitoringService.createEndpointsSource(selectedPeriod.value, selectedPeriod.refreshInterval).subscribe(function (endpoint) {\n if (endpoint.error) {\n connectivityNotifier.reportFailedConnection(endpoint.sourceIndex);\n if ($scope.endpoints) {\n $scope.endpoints.filter(function (item) {\n return item.sourceIndex === endpoint.sourceIndex;\n }).forEach(function (item) {\n return item.isScMonitoringDisconnected = true;\n });\n }\n } else {\n connectivityNotifier.reportSuccessfulConnection(endpoint.sourceIndex);\n var index = $scope.endpoints.findIndex(function (item) {\n return item.name === endpoint.name;\n });\n\n endpoint.isScMonitoringDisconnected = false;\n fillDisplayValuesForEndpoint(endpoint);\n if (index >= 0) {\n mergeIn($scope.endpoints[index], endpoint);\n } else {\n $scope.endpoints.push(endpoint);\n\n $scope.endpoints.sort(function (first, second) {\n if (first.name < second.name) {\n return -1;\n }\n\n if (first.name > second.name) {\n return 1;\n }\n\n return 0;\n });\n }\n }\n\n $scope.$apply();\n });\n\n endpointsFromScSubscription = Rx.Observable.interval(5000).startWith(0).flatMap(function (i) {\n return Rx.Observable.fromPromise(serviceControlService.getExceptionGroups('Endpoint Name', ''));\n }).selectMany(function (endpoints) {\n return endpoints.data;\n }).subscribe(function (endpoint) {\n var index = $scope.endpoints.findIndex(function (item) {\n return item.name === endpoint.title;\n });\n if (index >= 0) {\n $scope.endpoints[index].serviceControlId = endpoint.id;\n $scope.endpoints[index].errorCount = endpoint.count;\n } else {\n $scope.endpoints.push({ name: endpoint.title, errorCount: endpoint.count, serviceControlId: endpoint.id, isScMonitoringDisconnected: true });\n }\n });\n }\n\n updateUI();\n\n $scope.$on(\"$destroy\", function handler() {\n subscription.dispose();\n endpointsFromScSubscription.dispose();\n });\n };\n\n controller.$inject = ['$scope', '$location', 'monitoringService', 'serviceControlService', 'toastService', 'historyPeriods', 'rx', '$filter', 'smallGraphsMinimumYAxis', 'connectivityNotifier'];\n\n angular.module('monitored_endpoints').controller('monitoredEndpointsCtrl', controller);\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./app/modules/monitoring/js/monitored_endpoints.controller.js?06ce"],"names":["window","angular","undefined","controller","$scope","$location","monitoringService","serviceControlService","toastService","historyPeriods","rx","$filter","smallGraphsMinimumYAxis","connectivityNotifier","subscription","endpointsFromScSubscription","periods","selectedPeriod","$$search","historyPeriod","findIndex","period","value","endpoints","selectPeriod","updateUI","getDetailsUrl","endpoint","name","sourceIndex","fillDisplayValuesForEndpoint","metrics","processingTime","criticalTime","queueLength","throughput","retries","mergeIn","destination","source","propName","hasOwnProperty","dispose","createEndpointsSource","refreshInterval","subscribe","error","reportFailedConnection","filter","item","forEach","isScMonitoringDisconnected","reportSuccessfulConnection","index","push","sort","first","second","$apply","Rx","Observable","interval","startWith","flatMap","i","fromPromise","getExceptionGroups","selectMany","data","title","serviceControlId","id","errorCount","count","$on","handler","$inject","module"],"mappings":"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","file":"9.js","sourcesContent":["(function(window, angular, undefined) {\r\n    'use strict';\r\n\r\n    function controller(\r\n        $scope,\r\n        $location,\r\n        monitoringService,\r\n        serviceControlService,\r\n        toastService,\r\n        historyPeriods,\r\n        rx,\r\n        $filter,\r\n        smallGraphsMinimumYAxis,\r\n        connectivityNotifier) {\r\n\r\n        var subscription, endpointsFromScSubscription;\r\n\r\n        $scope.periods = historyPeriods;\r\n        $scope.selectedPeriod = $scope.periods[0];\r\n        $scope.smallGraphsMinimumYAxis = smallGraphsMinimumYAxis;\r\n\r\n        if ($location.$$search.historyPeriod) {\r\n            $scope.selectedPeriod = $scope.periods[$scope.periods.findIndex(function (period) {\r\n                return period.value == $location.$$search.historyPeriod;\r\n            })];\r\n        }\r\n\r\n        $scope.endpoints = [];\r\n\r\n        $scope.selectPeriod = function (period) {\r\n            $scope.selectedPeriod = period;\r\n\r\n            updateUI();\r\n        };\r\n\r\n        $scope.getDetailsUrl = endpoint => {\r\n            return '#/endpoint_details/' + endpoint.name + '/' + (endpoint.sourceIndex | 0) + '?historyPeriod=' + $scope.selectedPeriod.value;\r\n        };\r\n\r\n        function fillDisplayValuesForEndpoint(endpoint) {\r\n\r\n            $filter('graphduration')(endpoint.metrics.processingTime);\r\n            $filter('graphduration')(endpoint.metrics.criticalTime);\r\n            $filter('graphdecimal')(endpoint.metrics.queueLength, 0);\r\n            $filter('graphdecimal')(endpoint.metrics.throughput, 2);\r\n            $filter('graphdecimal')(endpoint.metrics.retries, 2);\r\n        }\r\n\r\n        function mergeIn(destination, source) {\r\n            for (var propName in source) {\r\n                if (source.hasOwnProperty(propName)) {\r\n                    destination[propName] = source[propName];\r\n                }\r\n            }\r\n        }\r\n\r\n        function updateUI() {\r\n            if (subscription) {\r\n                subscription.dispose();\r\n            }\r\n\r\n            if (endpointsFromScSubscription) {\r\n                endpointsFromScSubscription.dispose();\r\n            }\r\n\r\n            var selectedPeriod = $scope.selectedPeriod;\r\n\r\n            subscription = monitoringService.createEndpointsSource(selectedPeriod.value, selectedPeriod.refreshInterval)\r\n                .subscribe(function(endpoint) {\r\n                    if (endpoint.error) {\r\n                        connectivityNotifier.reportFailedConnection(endpoint.sourceIndex);\r\n                        if ($scope.endpoints) {\r\n                            $scope.endpoints.filter((item) => item.sourceIndex === endpoint.sourceIndex)\r\n                                .forEach((item) => item.isScMonitoringDisconnected = true);\r\n                        }\r\n                    } else {\r\n                        connectivityNotifier.reportSuccessfulConnection(endpoint.sourceIndex);\r\n                        var index = $scope.endpoints.findIndex(function(item) { return item.name === endpoint.name });\r\n\r\n                        endpoint.isScMonitoringDisconnected = false;\r\n                        fillDisplayValuesForEndpoint(endpoint);\r\n                        if (index >= 0) {\r\n                          mergeIn($scope.endpoints[index], endpoint);\r\n                        } else {\r\n                            $scope.endpoints.push(endpoint);\r\n\r\n                            $scope.endpoints.sort(function(first, second) {\r\n                                if (first.name < second.name) {\r\n                                    return -1;\r\n                                }\r\n\r\n                                if (first.name > second.name) {\r\n                                    return 1;\r\n                                }\r\n\r\n                                return 0;\r\n                            });\r\n                        }\r\n                    }\r\n\r\n                    $scope.$apply();\r\n                });\r\n\r\n            endpointsFromScSubscription =\r\n                Rx.Observable.interval(5000).startWith(0)\r\n                .flatMap(function(i) {\r\n                    return Rx.Observable.fromPromise(serviceControlService.getExceptionGroups('Endpoint Name', ''));\r\n                }).selectMany(function(endpoints) {\r\n                    return endpoints.data;\r\n                }).subscribe(function (endpoint) {\r\n                    var index = $scope.endpoints.findIndex(function(item) { return item.name === endpoint.title });\r\n                    if (index >= 0) {\r\n                        $scope.endpoints[index].serviceControlId = endpoint.id;\r\n                        $scope.endpoints[index].errorCount = endpoint.count;\r\n                    } else {\r\n                        $scope.endpoints.push({ name: endpoint.title, errorCount: endpoint.count, serviceControlId: endpoint.id, isScMonitoringDisconnected : true });\r\n                    }\r\n                });\r\n        }\r\n\r\n        updateUI();\r\n\r\n        $scope.$on(\"$destroy\", function handler() {\r\n            subscription.dispose();\r\n            endpointsFromScSubscription.dispose();\r\n        });\r\n    };\r\n\r\n    controller.$inject = [\r\n        '$scope',\r\n        '$location',\r\n        'monitoringService',\r\n        'serviceControlService',\r\n        'toastService',\r\n        'historyPeriods',\r\n        'rx',\r\n        '$filter',\r\n        'smallGraphsMinimumYAxis',\r\n        'connectivityNotifier'\r\n    ];\r\n\r\n    angular.module('monitored_endpoints')\r\n        .controller('monitoredEndpointsCtrl', controller);\r\n\r\n}(window, window.angular));\n\n\n// WEBPACK FOOTER //\n// ./app/modules/monitoring/js/monitored_endpoints.controller.js"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///9\n"); - -/***/ }), -/* 10 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular, undefined) {\n 'use strict';\n\n function routeProvider($routeProvider) {\n var template = __webpack_require__(11);\n\n $routeProvider.when('/monitored_endpoints', {\n data: {\n pageTitle: 'Monitored Endpoints'\n },\n template: template,\n controller: 'monitoredEndpointsCtrl',\n controllerAs: 'vm',\n reloadOnSearch: false\n });\n };\n\n routeProvider.$inject = ['$routeProvider'];\n\n angular.module('monitored_endpoints').config(routeProvider);\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL21vbml0b3JlZF9lbmRwb2ludHMucm91dGUuanM/MjcwNiJdLCJuYW1lcyI6WyJ3aW5kb3ciLCJhbmd1bGFyIiwidW5kZWZpbmVkIiwicm91dGVQcm92aWRlciIsIiRyb3V0ZVByb3ZpZGVyIiwidGVtcGxhdGUiLCJyZXF1aXJlIiwid2hlbiIsImRhdGEiLCJwYWdlVGl0bGUiLCJjb250cm9sbGVyIiwiY29udHJvbGxlckFzIiwicmVsb2FkT25TZWFyY2giLCIkaW5qZWN0IiwibW9kdWxlIiwiY29uZmlnIl0sIm1hcHBpbmdzIjoiOztBQUFDLFdBQVVBLE1BQVYsRUFBa0JDLE9BQWxCLEVBQTJCQyxTQUEzQixFQUFzQztBQUNuQzs7QUFFQSxhQUFTQyxhQUFULENBQXVCQyxjQUF2QixFQUF1QztBQUNuQyxZQUFJQyxXQUFXLG1CQUFBQyxDQUFRLEVBQVIsQ0FBZjs7QUFFQUYsdUJBQWVHLElBQWYsQ0FBb0Isc0JBQXBCLEVBQTRDO0FBQ3hDQyxrQkFBTTtBQUNGQywyQkFBVztBQURULGFBRGtDO0FBSXhDSixzQkFBVUEsUUFKOEI7QUFLeENLLHdCQUFZLHdCQUw0QjtBQU14Q0MsMEJBQWMsSUFOMEI7QUFPeENDLDRCQUFnQjtBQVB3QixTQUE1QztBQVNIOztBQUVEVCxrQkFBY1UsT0FBZCxHQUF3QixDQUNwQixnQkFEb0IsQ0FBeEI7O0FBSUFaLFlBQVFhLE1BQVIsQ0FBZSxxQkFBZixFQUNLQyxNQURMLENBQ1laLGFBRFo7QUFFSCxDQXZCQSxFQXVCRUgsTUF2QkYsRUF1QlVBLE9BQU9DLE9BdkJqQixDQUFEIiwiZmlsZSI6IjEwLmpzIiwic291cmNlc0NvbnRlbnQiOlsiKGZ1bmN0aW9uICh3aW5kb3csIGFuZ3VsYXIsIHVuZGVmaW5lZCkge1xyXG4gICAgJ3VzZSBzdHJpY3QnO1xyXG5cclxuICAgIGZ1bmN0aW9uIHJvdXRlUHJvdmlkZXIoJHJvdXRlUHJvdmlkZXIpIHtcclxuICAgICAgICBsZXQgdGVtcGxhdGUgPSByZXF1aXJlKCcuLy4uL3ZpZXdzL21vbml0b3JlZF9lbmRwb2ludHMuaHRtbCcpO1xyXG5cclxuICAgICAgICAkcm91dGVQcm92aWRlci53aGVuKCcvbW9uaXRvcmVkX2VuZHBvaW50cycsIHtcclxuICAgICAgICAgICAgZGF0YToge1xyXG4gICAgICAgICAgICAgICAgcGFnZVRpdGxlOiAnTW9uaXRvcmVkIEVuZHBvaW50cydcclxuICAgICAgICAgICAgfSxcclxuICAgICAgICAgICAgdGVtcGxhdGU6IHRlbXBsYXRlLFxyXG4gICAgICAgICAgICBjb250cm9sbGVyOiAnbW9uaXRvcmVkRW5kcG9pbnRzQ3RybCcsXHJcbiAgICAgICAgICAgIGNvbnRyb2xsZXJBczogJ3ZtJyxcclxuICAgICAgICAgICAgcmVsb2FkT25TZWFyY2g6IGZhbHNlXHJcbiAgICAgICAgfSk7XHJcbiAgICB9O1xyXG5cclxuICAgIHJvdXRlUHJvdmlkZXIuJGluamVjdCA9IFtcclxuICAgICAgICAnJHJvdXRlUHJvdmlkZXInXHJcbiAgICBdO1xyXG5cclxuICAgIGFuZ3VsYXIubW9kdWxlKCdtb25pdG9yZWRfZW5kcG9pbnRzJylcclxuICAgICAgICAuY29uZmlnKHJvdXRlUHJvdmlkZXIpO1xyXG59ICh3aW5kb3csIHdpbmRvdy5hbmd1bGFyKSk7XG5cblxuLy8gV0VCUEFDSyBGT09URVIgLy9cbi8vIC4vYXBwL21vZHVsZXMvbW9uaXRvcmluZy9qcy9tb25pdG9yZWRfZW5kcG9pbnRzLnJvdXRlLmpzIl0sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///10\n"); - -/***/ }), -/* 11 */ -/***/ (function(module, exports) { - -eval("module.exports = \"

    Endpoints overview

    Endpoint name
    Queue Length (msgs)

    Queue length: The estimated number of messages in an endpoint's queue.

    WARNING: This is an experimental feature. Learn more

    Throughput (msgs/s)
    Scheduled retry rate (msgs/s)
    Processing Time (t)
    Critical Time (t)
    {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : endpoint.metrics.queueLength.displayValue}} ?
    {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : endpoint.metrics.throughput.displayValue}} ?
    {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : endpoint.metrics.retries.displayValue}} ?
    {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : endpoint.metrics.processingTime.displayValue.value}} ? {{endpoint.metrics.processingTime.displayValue.unit}}
    {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : endpoint.metrics.criticalTime.displayValue.value}} ? {{endpoint.metrics.criticalTime.displayValue.unit}}
    \";//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./app/modules/monitoring/views/monitored_endpoints.html?3a87"],"names":[],"mappings":"AAAA,sfAAsf,cAAc,KAAK,aAAa,mwFAAmwF,yBAAyB,GAAG,eAAe,2hBAA2hB,2BAA2B,8CAA8C,0CAA0C,6HAA6H,0CAA0C,qNAAqN,qCAAqC,0GAA0G,8HAA8H,qRAAqR,oCAAoC,wGAAwG,6HAA6H,kRAAkR,iCAAiC,qGAAqG,0HAA0H,yRAAyR,wCAAwC,6GAA6G,uIAAuI,mLAAmL,mDAAmD,yMAAyM,sCAAsC,2GAA2G,qIAAqI,mLAAmL,iDAAiD","file":"11.js","sourcesContent":["module.exports = \"<div class=container ng-show=endpoints.length> <div class=\\\"row monitoring-head\\\"> <div class=\\\"col-sm-9 no-side-padding list-section\\\"> <h1>Endpoints overview</h1> </div> <div class=\\\"col-sm-3 no-side-padding toolbar-menus\\\"> <ul class=\\\"nav nav-pills period-selector\\\"> <li role=presentation ng-repeat=\\\"period in periods\\\" ng-class=\\\"(period.value == selectedPeriod.value ? 'active' : 'notselected')\\\"> <a ng-click=selectPeriod(period) href=\\\"#/monitored_endpoints?historyPeriod={{period.value}}\\\">{{period.text}}</a> </li> </ul> </div> </div> </div> <div class=container> <section ng-show=true> <div ng-include=\\\"'modules/monitoring/views/monitoring_not_available.html'\\\" ng-show=!endpoints.length></div> <div ng-show=endpoints.length class=\\\"row box box-no-click table-head-row\\\"> <div class=\\\"col-sm-2 col-xl-7\\\"> <div class=\\\"row box-header\\\"> <div class=col-sm-12> Endpoint name </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\"> <div class=tooltip-trigger> Queue Length <span class=table-header-unit>(msgs)</span> <i class=\\\"fa fa-flask fake-link\\\"></i> <div class=interactive-tooltip> <div class=tooltip-contents> <p>Queue length: The estimated number of messages in an endpoint's queue.</p> <p>WARNING: This is an experimental feature. <a href=\\\"https://docs.particular.net/search?q=nservicebus+queue+length+metric+%2bexperimental\\\" target=_blank>Learn more</a> <i class=\\\"fa fa-external-link fake-link\\\"></i> </p> </div> </div> </div> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Throughput: The number of messages per second successfully processed by a receiving endpoint.\\\"> Throughput <span class=table-header-unit>(msgs/s)</span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Scheduled retry rate: The number of messages per second scheduled for retries (immediate or delayed).\\\"> Scheduled retry rate <span class=table-header-unit>(msgs/s)</span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Processing time: The time taken for a receiving endpoint to successfully process a message.\\\"> Processing Time <span class=table-header-unit>(t)</span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Critical time: The elapsed time from when a message was sent, until it was successfully processed by a receiving endpoint.\\\"> Critical Time <span class=table-header-unit>(t)</span> </div> </div> </div> </div> <div class=row> <div class=\\\"col-sm-12 no-side-padding\\\"> <div class=\\\"row box endpoint-row\\\" ng-repeat=\\\"endpoint in endpoints\\\" ng-mouseenter=\\\"endpoint.hover1=true\\\" ng-mouseleave=\\\"endpoint.hover1=false\\\"> <div class=\\\"col-sm-12 no-side-padding\\\"> <div class=row> <div class=\\\"col-sm-2 col-xl-7 endpoint-name name-overview\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-lg-max-8 no-side-padding lead\\\"> <a ng-click=\\\"endpoint.isExpanded = !endpoint.isExpanded\\\" ng-href={{getDetailsUrl(endpoint)}}>{{endpoint.name}}</a> </div> <div class=\\\"col-lg-5 no-side-padding endpoint-status\\\"> <span class=warning ng-if=endpoint.isScMonitoringDisconnected> <i class=\\\"fa pa-monitoring-lost endpoints-overview\\\" uib-tooltip=\\\"Unable to connect to monitoring server\\\"></i> </span> <span class=warning ng-if=endpoint.isStale> <i class=\\\"fa pa-endpoint-lost endpoints-overview\\\" uib-tooltip=\\\"Unable to connect to instance\\\"></i> </span> <span class=warning ng-if=endpoint.errorCount> <a ng-if=endpoint.errorCount class=\\\"warning btn\\\" href=#/failed-messages/groups/{{endpoint.serviceControlId}}> <i class=\\\"fa fa-envelope\\\" uib-tooltip=\\\"{{endpoint.errorCount | metricslargenumber}} failed messages associated with this endpoint. Click to see list.\\\"></i> <span class=\\\"badge badge-important ng-binding\\\">{{endpoint.errorCount | metricslargenumber}}</span> </a> </span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=endpoint.metrics.queueLength minimum-yaxis={{smallGraphsMinimumYAxis.queueLength}} class=\\\"graph queue-length pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : endpoint.metrics.queueLength.displayValue}} <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=endpoint.metrics.throughput minimum-yaxis={{smallGraphsMinimumYAxis.throughput}} class=\\\"graph throughput pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : endpoint.metrics.throughput.displayValue}} <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=endpoint.metrics.retries minimum-yaxis={{smallGraphsMinimumYAxis.retries}} class=\\\"graph retries pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : endpoint.metrics.retries.displayValue}} <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=endpoint.metrics.processingTime minimum-yaxis={{smallGraphsMinimumYAxis.processingTime}} class=\\\"graph processing-time pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : endpoint.metrics.processingTime.displayValue.value}} <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> <span ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.metrics.processingTime.displayValue.unit}}</span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=endpoint.metrics.criticalTime minimum-yaxis={{smallGraphsMinimumYAxis.criticalTime}} class=\\\"graph critical-time pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : endpoint.metrics.criticalTime.displayValue.value}} <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> <span ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.metrics.criticalTime.displayValue.unit}}</span> </div> </div> </div> </div> </div> </div> </div> </div> </section> </div> \";\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./app/modules/monitoring/views/monitored_endpoints.html\n// module id = 11\n// module chunks = 0"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///11\n"); - -/***/ }), -/* 12 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular, undefined) {\n 'use strict';\n\n angular.module('endpoint_details', []);\n\n __webpack_require__(0);\n __webpack_require__(13);\n __webpack_require__(14);\n __webpack_require__(1);\n\n __webpack_require__(2);\n __webpack_require__(3);\n __webpack_require__(16);\n __webpack_require__(4);\n __webpack_require__(17);\n __webpack_require__(5);\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL2VuZHBvaW50X2RldGFpbHMubW9kdWxlLmpzPzRmMGIiXSwibmFtZXMiOlsid2luZG93IiwiYW5ndWxhciIsInVuZGVmaW5lZCIsIm1vZHVsZSIsInJlcXVpcmUiXSwibWFwcGluZ3MiOiI7O0FBQUMsV0FBU0EsTUFBVCxFQUFpQkMsT0FBakIsRUFBMEJDLFNBQTFCLEVBQXFDO0FBQ2xDOztBQUVBRCxZQUFRRSxNQUFSLENBQWUsa0JBQWYsRUFBbUMsRUFBbkM7O0FBRUFDLElBQUEsbUJBQUFBLENBQVEsQ0FBUjtBQUNBQSxJQUFBLG1CQUFBQSxDQUFRLEVBQVI7QUFDQUEsSUFBQSxtQkFBQUEsQ0FBUSxFQUFSO0FBQ0FBLElBQUEsbUJBQUFBLENBQVEsQ0FBUjs7QUFFQUEsSUFBQSxtQkFBQUEsQ0FBUSxDQUFSO0FBQ0FBLElBQUEsbUJBQUFBLENBQVEsQ0FBUjtBQUNBQSxJQUFBLG1CQUFBQSxDQUFRLEVBQVI7QUFDQUEsSUFBQSxtQkFBQUEsQ0FBUSxDQUFSO0FBQ0FBLElBQUEsbUJBQUFBLENBQVEsRUFBUjtBQUNBQSxJQUFBLG1CQUFBQSxDQUFRLENBQVI7QUFDSCxDQWhCQSxFQWdCQ0osTUFoQkQsRUFnQlNBLE9BQU9DLE9BaEJoQixDQUFEIiwiZmlsZSI6IjEyLmpzIiwic291cmNlc0NvbnRlbnQiOlsiKGZ1bmN0aW9uKHdpbmRvdywgYW5ndWxhciwgdW5kZWZpbmVkKSB7XHJcbiAgICAndXNlIHN0cmljdCc7XHJcblxyXG4gICAgYW5ndWxhci5tb2R1bGUoJ2VuZHBvaW50X2RldGFpbHMnLCBbXSk7XHJcblxyXG4gICAgcmVxdWlyZSgnLi9zZXJ2aWNlcy9zZXJ2aWNlcy5jb25uZWN0aXZpdHlOb3RpZmllcicpO1xyXG4gICAgcmVxdWlyZSgnLi9lbmRwb2ludF9kZXRhaWxzLmNvbnRyb2xsZXInKTtcclxuICAgIHJlcXVpcmUoJy4vZW5kcG9pbnRfZGV0YWlscy5yb3V0ZS5qcycpO1xyXG4gICAgcmVxdWlyZSgnLi9jb25zdGFudC5kaWFncmFtcy5qcycpO1xyXG5cclxuICAgIHJlcXVpcmUoJy4vZGlyZWN0aXZlcy91aS5wYXJ0aWN1bGFyLmdyYXBoLmpzJyk7XHJcbiAgICByZXF1aXJlKCcuL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5ncmFwaGRlY2ltYWwuanMnKTtcclxuICAgIHJlcXVpcmUoJy4vZGlyZWN0aXZlcy91aS5wYXJ0aWN1bGFyLmR1cmF0aW9uLmpzJyk7XHJcbiAgICByZXF1aXJlKCcuL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5ncmFwaGR1cmF0aW9uLmpzJyk7XHJcbiAgICByZXF1aXJlKCcuL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5sYXJnZUdyYXBoLmpzJyk7XHJcbiAgICByZXF1aXJlKCcuL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5tZXRyaWNzbGFyZ2VudW1iZXIuanMnKTtcclxufSh3aW5kb3csIHdpbmRvdy5hbmd1bGFyKSk7XG5cblxuLy8gV0VCUEFDSyBGT09URVIgLy9cbi8vIC4vYXBwL21vZHVsZXMvbW9uaXRvcmluZy9qcy9lbmRwb2ludF9kZXRhaWxzLm1vZHVsZS5qcyJdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///12\n"); - -/***/ }), -/* 13 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular, undefined) {\n 'use strict';\n\n function controller($scope, $routeParams, $location, toastService, serviceControlService, monitoringService, historyPeriods, $filter, smallGraphsMinimumYAxis, largeGraphsMinimumYAxis, connectivityNotifier) {\n\n $scope.endpointName = $routeParams.endpointName;\n $scope.sourceIndex = $routeParams.sourceIndex;\n $scope.loading = true;\n $scope.showInstancesBreakdown = false;\n $scope.largeGraphsMinimumYAxis = largeGraphsMinimumYAxis;\n $scope.smallGraphsMinimumYAxis = smallGraphsMinimumYAxis;\n\n var subscription;\n\n $scope.periods = historyPeriods;\n $scope.selectedPeriod = $scope.periods[0];\n\n if ($location.$$search.historyPeriod) {\n $scope.selectedPeriod = $scope.periods[$scope.periods.findIndex(function (period) {\n return period.value == $location.$$search.historyPeriod;\n })];\n }\n\n $scope.selectPeriod = function (period) {\n $scope.selectedPeriod = period;\n\n updateUI();\n };\n\n function mergeIn(destination, source) {\n for (var propName in source) {\n if (source.hasOwnProperty(propName)) {\n destination[propName] = source[propName];\n }\n }\n }\n\n function updateUI() {\n if (subscription) {\n subscription.dispose();\n }\n\n var selectedPeriod = $scope.selectedPeriod;\n\n $scope.endpoint = {};\n\n subscription = monitoringService.createEndpointDetailsSource($routeParams.endpointName, $routeParams.sourceIndex, selectedPeriod.value, selectedPeriod.refreshInterval).subscribe(function (endpoint) {\n if (endpoint.error) {\n connectivityNotifier.reportFailedConnection($routeParams.sourceIndex);\n if ($scope.endpoint && $scope.endpoint.instances) {\n $scope.endpoint.instances.forEach(function (item) {\n return item.isScMonitoringDisconnected = true;\n });\n }\n\n $scope.endpoint.isScMonitoringDisconnected = true;\n } else {\n\n mergeIn($scope.endpoint, endpoint);\n\n connectivityNotifier.reportSuccessfulConnection($routeParams.sourceIndex);\n\n $scope.endpoint.instances.sort(function (first, second) {\n if (first.id < second.id) {\n return -1;\n }\n\n if (first.id > second.id) {\n return 1;\n }\n\n return 0;\n });\n\n $scope.loading = false;\n $scope.endpoint.messageTypes.forEach(function (messageType) {\n return fillDisplayValues(messageType);\n });\n\n $scope.endpoint.isStale = true;\n $scope.endpoint.isScMonitoringDisconnected = false;\n\n $scope.endpoint.instances.forEach(function (instance) {\n fillDisplayValues(instance);\n serviceControlService.getExceptionGroupsForEndpointInstance(instance.id).then(function (result) {\n if (result.data.length > 0) {\n instance.serviceControlId = result.data[0].id;\n instance.errorCount = result.data[0].count;\n }\n }, function (err) {\n // Warn user?\n });\n $scope.endpoint.isStale = $scope.endpoint.isStale && instance.isStale;\n });\n }\n\n serviceControlService.getExceptionGroupsForLogicalEndpoint($scope.endpointName).then(function (result) {\n if (result.data.length > 0) {\n $scope.endpoint.serviceControlId = result.data[0].id;\n $scope.endpoint.errorCount = result.data[0].count;\n }\n });\n });\n }\n\n function fillDisplayValues(instance) {\n $filter('graphduration')(instance.metrics.processingTime);\n $filter('graphduration')(instance.metrics.criticalTime);\n $filter('graphdecimal')(instance.metrics.throughput, 2);\n $filter('graphdecimal')(instance.metrics.retries, 2);\n }\n\n $scope.$on(\"$destroy\", function handler() {\n subscription.dispose();\n });\n\n updateUI();\n }\n\n controller.$inject = ['$scope', '$routeParams', '$location', 'toastService', 'serviceControlService', 'monitoringService', 'historyPeriods', '$filter', 'smallGraphsMinimumYAxis', 'largeGraphsMinimumYAxis', 'connectivityNotifier'];\n\n angular.module('endpoint_details').controller('endpointDetailsCtrl', controller);\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./app/modules/monitoring/js/endpoint_details.controller.js?df84"],"names":["window","angular","undefined","controller","$scope","$routeParams","$location","toastService","serviceControlService","monitoringService","historyPeriods","$filter","smallGraphsMinimumYAxis","largeGraphsMinimumYAxis","connectivityNotifier","endpointName","sourceIndex","loading","showInstancesBreakdown","subscription","periods","selectedPeriod","$$search","historyPeriod","findIndex","period","value","selectPeriod","updateUI","mergeIn","destination","source","propName","hasOwnProperty","dispose","endpoint","createEndpointDetailsSource","refreshInterval","subscribe","error","reportFailedConnection","instances","forEach","item","isScMonitoringDisconnected","reportSuccessfulConnection","sort","first","second","id","messageTypes","messageType","fillDisplayValues","isStale","instance","getExceptionGroupsForEndpointInstance","then","result","data","length","serviceControlId","errorCount","count","err","getExceptionGroupsForLogicalEndpoint","metrics","processingTime","criticalTime","throughput","retries","$on","handler","$inject","module"],"mappings":"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","file":"13.js","sourcesContent":["(function(window, angular, undefined) {\r\n    'use strict';\r\n\r\n    function controller(\r\n        $scope,\r\n        $routeParams,\r\n        $location,\r\n        toastService,\r\n        serviceControlService,\r\n        monitoringService,\r\n        historyPeriods,\r\n        $filter,\r\n        smallGraphsMinimumYAxis,\r\n        largeGraphsMinimumYAxis,\r\n        connectivityNotifier\r\n    ) {\r\n\r\n        $scope.endpointName = $routeParams.endpointName;\r\n        $scope.sourceIndex = $routeParams.sourceIndex;\r\n        $scope.loading = true;\r\n        $scope.showInstancesBreakdown = false;\r\n        $scope.largeGraphsMinimumYAxis = largeGraphsMinimumYAxis;\r\n        $scope.smallGraphsMinimumYAxis = smallGraphsMinimumYAxis;\r\n\r\n        var subscription;\r\n\r\n        $scope.periods = historyPeriods;\r\n        $scope.selectedPeriod = $scope.periods[0];\r\n\r\n        if ($location.$$search.historyPeriod) {\r\n            $scope.selectedPeriod = $scope.periods[$scope.periods.findIndex(function (period) {\r\n                return period.value == $location.$$search.historyPeriod;\r\n            })];\r\n        }\r\n\r\n        $scope.selectPeriod = function (period) {\r\n            $scope.selectedPeriod = period;\r\n\r\n            updateUI();\r\n        };\r\n\r\n\r\n        function mergeIn(destination, source) {\r\n            for (var propName in source) {\r\n                if (source.hasOwnProperty(propName)) {\r\n                    destination[propName] = source[propName];\r\n                }\r\n            }\r\n        }\r\n\r\n        function updateUI() {\r\n            if (subscription) {\r\n                subscription.dispose();\r\n            }\r\n\r\n            var selectedPeriod = $scope.selectedPeriod;\r\n\r\n            $scope.endpoint = {};\r\n\r\n            subscription = monitoringService.createEndpointDetailsSource($routeParams.endpointName, $routeParams.sourceIndex, selectedPeriod.value, selectedPeriod.refreshInterval).subscribe(function (endpoint) {\r\n                if (endpoint.error) {\r\n                    connectivityNotifier.reportFailedConnection($routeParams.sourceIndex);\r\n                    if ($scope.endpoint && $scope.endpoint.instances) {\r\n                        $scope.endpoint.instances.forEach((item) => item.isScMonitoringDisconnected = true);\r\n                    }\r\n\r\n                    $scope.endpoint.isScMonitoringDisconnected = true;\r\n\r\n                } else {\r\n\r\n                    mergeIn($scope.endpoint, endpoint);\r\n\r\n                    connectivityNotifier.reportSuccessfulConnection($routeParams.sourceIndex);\r\n\r\n                    $scope.endpoint.instances.sort(function (first, second) {\r\n                        if (first.id < second.id) {\r\n                            return -1;\r\n                        }\r\n\r\n                        if (first.id > second.id) {\r\n                            return 1;\r\n                        }\r\n\r\n                        return 0;\r\n                    });\r\n\r\n                    $scope.loading = false;\r\n                    $scope.endpoint.messageTypes.forEach((messageType) => fillDisplayValues(messageType));\r\n\r\n                    $scope.endpoint.isStale = true;\r\n                    $scope.endpoint.isScMonitoringDisconnected = false;\r\n\r\n                    $scope.endpoint.instances.forEach(function (instance) {\r\n                        fillDisplayValues(instance);\r\n                        serviceControlService.getExceptionGroupsForEndpointInstance(instance.id).then(function (result) {\r\n                            if (result.data.length > 0) {\r\n                                instance.serviceControlId = result.data[0].id;\r\n                                instance.errorCount = result.data[0].count;\r\n                            }\r\n                        }, function (err) {\r\n                            // Warn user?\r\n                    });\r\n                        $scope.endpoint.isStale = $scope.endpoint.isStale && instance.isStale;\r\n                    });\r\n                }\r\n\r\n                serviceControlService.getExceptionGroupsForLogicalEndpoint($scope.endpointName).then(function(result) {\r\n                    if (result.data.length > 0) {\r\n                        $scope.endpoint.serviceControlId = result.data[0].id;\r\n                        $scope.endpoint.errorCount = result.data[0].count;\r\n                    }\r\n                });\r\n            });\r\n        }\r\n\r\n        function fillDisplayValues(instance) {\r\n            $filter('graphduration')(instance.metrics.processingTime);\r\n            $filter('graphduration')(instance.metrics.criticalTime);\r\n            $filter('graphdecimal')(instance.metrics.throughput, 2);\r\n            $filter('graphdecimal')(instance.metrics.retries, 2);\r\n        }\r\n\r\n        $scope.$on(\"$destroy\", function handler() {\r\n            subscription.dispose();\r\n        });\r\n\r\n        updateUI();\r\n    }\r\n\r\n    controller.$inject = [\r\n        '$scope',\r\n        '$routeParams',\r\n        '$location',\r\n        'toastService',\r\n        'serviceControlService',\r\n        'monitoringService',\r\n        'historyPeriods',\r\n        '$filter',\r\n        'smallGraphsMinimumYAxis',\r\n        'largeGraphsMinimumYAxis',\r\n        'connectivityNotifier'\r\n    ];\r\n\r\n    angular.module('endpoint_details')\r\n        .controller('endpointDetailsCtrl', controller);\r\n\r\n}(window, window.angular));\n\n\n// WEBPACK FOOTER //\n// ./app/modules/monitoring/js/endpoint_details.controller.js"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///13\n"); - -/***/ }), -/* 14 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular, undefined) {\n 'use strict';\n\n function routeProvider($routeProvider) {\n var template = __webpack_require__(15);\n\n $routeProvider.when('/endpoint_details/:endpointName/:sourceIndex', {\n data: {\n pageTitle: 'Endpoint Details'\n },\n template: template,\n controller: 'endpointDetailsCtrl',\n controllerAs: 'vm',\n reloadOnSearch: false\n });\n };\n\n routeProvider.$inject = ['$routeProvider'];\n\n angular.module('endpoint_details').config(routeProvider);\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL2VuZHBvaW50X2RldGFpbHMucm91dGUuanM/NjViNiJdLCJuYW1lcyI6WyJ3aW5kb3ciLCJhbmd1bGFyIiwidW5kZWZpbmVkIiwicm91dGVQcm92aWRlciIsIiRyb3V0ZVByb3ZpZGVyIiwidGVtcGxhdGUiLCJyZXF1aXJlIiwid2hlbiIsImRhdGEiLCJwYWdlVGl0bGUiLCJjb250cm9sbGVyIiwiY29udHJvbGxlckFzIiwicmVsb2FkT25TZWFyY2giLCIkaW5qZWN0IiwibW9kdWxlIiwiY29uZmlnIl0sIm1hcHBpbmdzIjoiOztBQUFDLFdBQVVBLE1BQVYsRUFBa0JDLE9BQWxCLEVBQTJCQyxTQUEzQixFQUFzQztBQUNuQzs7QUFFQSxhQUFTQyxhQUFULENBQXVCQyxjQUF2QixFQUF1QztBQUNuQyxZQUFJQyxXQUFXLG1CQUFBQyxDQUFRLEVBQVIsQ0FBZjs7QUFFQUYsdUJBQWVHLElBQWYsQ0FBb0IsOENBQXBCLEVBQW9FO0FBQ2hFQyxrQkFBTTtBQUNGQywyQkFBVztBQURULGFBRDBEO0FBSWhFSixzQkFBVUEsUUFKc0Q7QUFLaEVLLHdCQUFZLHFCQUxvRDtBQU1oRUMsMEJBQWMsSUFOa0Q7QUFPaEVDLDRCQUFnQjtBQVBnRCxTQUFwRTtBQVNIOztBQUVEVCxrQkFBY1UsT0FBZCxHQUF3QixDQUNwQixnQkFEb0IsQ0FBeEI7O0FBSUFaLFlBQVFhLE1BQVIsQ0FBZSxrQkFBZixFQUNLQyxNQURMLENBQ1laLGFBRFo7QUFFSCxDQXZCQSxFQXVCRUgsTUF2QkYsRUF1QlVBLE9BQU9DLE9BdkJqQixDQUFEIiwiZmlsZSI6IjE0LmpzIiwic291cmNlc0NvbnRlbnQiOlsiKGZ1bmN0aW9uICh3aW5kb3csIGFuZ3VsYXIsIHVuZGVmaW5lZCkge1xyXG4gICAgJ3VzZSBzdHJpY3QnO1xyXG5cclxuICAgIGZ1bmN0aW9uIHJvdXRlUHJvdmlkZXIoJHJvdXRlUHJvdmlkZXIpIHtcclxuICAgICAgICBsZXQgdGVtcGxhdGUgPSByZXF1aXJlKCcuLy4uL3ZpZXdzL2VuZHBvaW50X2RldGFpbHMuaHRtbCcpO1xyXG5cclxuICAgICAgICAkcm91dGVQcm92aWRlci53aGVuKCcvZW5kcG9pbnRfZGV0YWlscy86ZW5kcG9pbnROYW1lLzpzb3VyY2VJbmRleCcsIHtcclxuICAgICAgICAgICAgZGF0YToge1xyXG4gICAgICAgICAgICAgICAgcGFnZVRpdGxlOiAnRW5kcG9pbnQgRGV0YWlscydcclxuICAgICAgICAgICAgfSxcclxuICAgICAgICAgICAgdGVtcGxhdGU6IHRlbXBsYXRlLFxyXG4gICAgICAgICAgICBjb250cm9sbGVyOiAnZW5kcG9pbnREZXRhaWxzQ3RybCcsXHJcbiAgICAgICAgICAgIGNvbnRyb2xsZXJBczogJ3ZtJyxcclxuICAgICAgICAgICAgcmVsb2FkT25TZWFyY2g6IGZhbHNlXHJcbiAgICAgICAgfSk7XHJcbiAgICB9O1xyXG5cclxuICAgIHJvdXRlUHJvdmlkZXIuJGluamVjdCA9IFtcclxuICAgICAgICAnJHJvdXRlUHJvdmlkZXInXHJcbiAgICBdO1xyXG5cclxuICAgIGFuZ3VsYXIubW9kdWxlKCdlbmRwb2ludF9kZXRhaWxzJylcclxuICAgICAgICAuY29uZmlnKHJvdXRlUHJvdmlkZXIpO1xyXG59ICh3aW5kb3csIHdpbmRvdy5hbmd1bGFyKSk7XG5cblxuLy8gV0VCUEFDSyBGT09URVIgLy9cbi8vIC4vYXBwL21vZHVsZXMvbW9uaXRvcmluZy9qcy9lbmRwb2ludF9kZXRhaWxzLnJvdXRlLmpzIl0sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///14\n"); - -/***/ }), -/* 15 */ -/***/ (function(module, exports) { - -eval("module.exports = \"
    Queue Length

    Queue length: The estimated number of messages in an endpoint's queue.

    NOTE: This is an experimental feature. Learn more

    {{endpoint.digest.metrics.queueLength.latest | metricslargenumber:0}} MSGS
    ?
    {{endpoint.digest.metrics.queueLength.average | metricslargenumber:0}} MSGS
    ? AVG
    Throughput
    {{endpoint.digest.metrics.throughput.latest | metricslargenumber:2}} MSGS/S
    ?
    {{endpoint.digest.metrics.throughput.average | metricslargenumber:2}} MSGS/S
    ? AVG
    Scheduled Retries Rate
    {{endpoint.digest.metrics.retries.latest | metricslargenumber:2}} MSGS/S
    ?
    {{endpoint.digest.metrics.retries.average | metricslargenumber:2}} MSGS/S
    ? AVG
    Processing Time
    {{endpoint.digest.metrics.processingTime.latest | durationValue}} {{endpoint.digest.metrics.processingTime.latest | durationUnit}}
    ?
    {{endpoint.digest.metrics.processingTime.average | durationValue}} {{endpoint.digest.metrics.processingTime.average | durationUnit}} AVG
    ? AVG
    Critical Time
    {{endpoint.digest.metrics.criticalTime.latest | durationValue}} {{endpoint.digest.metrics.criticalTime.latest | durationUnit}}
    ?
    {{endpoint.digest.metrics.criticalTime.average | durationValue}} {{endpoint.digest.metrics.criticalTime.average | durationUnit}}
    ? AVG
    Instance Name
    Throughput (msgs/s)
    Scheduled retry rate (msgs/s)
    Processing Time (t)
    Critical Time (t)
    {{(instance.isStale == true || instance.isScMonitoringDisconnected == true) ? \\\"\\\" : instance.metrics.throughput.displayValue}} ?
    {{(instance.isStale == true || instance.isScMonitoringDisconnected == true) ? \\\"\\\" : instance.metrics.retries.displayValue}} ?
    {{(instance.isStale == true || instance.isScMonitoringDisconnected == true) ? \\\"\\\" : instance.metrics.processingTime.displayValue.value}} {{instance.metrics.processingTime.displayValue.unit}} ?
    {{(instance.isStale == true || instance.isScMonitoringDisconnected == true) ? \\\"\\\" : instance.metrics.criticalTime.displayValue.value}} {{instance.metrics.criticalTime.displayValue.unit}} ?
    Message type name
    Throughput (msgs/s)
    Scheduled retry rate (msgs/s)
    Processing Time (t)
    Critical Time (t)
    {{messageType.typeName ? messageType.typeName : 'Unknown'}}
    {{messageType.assemblyName + '-' + messageType.assemblyVersion}}
    {{'Culture=' + messageType.culture}}
    {{'PublicKeyToken=' + messageType.publicKeyToken}}
    {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : messageType.metrics.throughput.displayValue}} ?
    {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : messageType.metrics.retries.displayValue}} ?
    {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : messageType.metrics.processingTime.displayValue.value}} {{messageType.metrics.processingTime.displayValue.unit}} ?
    {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : messageType.metrics.criticalTime.displayValue.value}} {{messageType.metrics.criticalTime.displayValue.unit}} ?
    \";//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./app/modules/monitoring/views/endpoint_details.html?e863"],"names":[],"mappings":"AAAA,+IAA+I,6HAA6H,cAAc,wbAAwb,cAAc,GAAG,aAAa,GAAG,2BAA2B,8CAA8C,0CAA0C,6HAA6H,0CAA0C,wUAAwU,cAAc,GAAG,aAAa,iBAAiB,cAAc,KAAK,aAAa,yZAAyZ,qCAAqC,g3BAAg3B,mEAAmE,uXAAuX,oEAAoE,wpBAAwpB,2CAA2C,+nBAA+nB,kEAAkE,qSAAqS,mEAAmE,wsBAAwsB,+DAA+D,qSAAqS,gEAAgE,8qBAA8qB,4CAA4C,8pBAA8pB,+DAA+D,0CAA0C,8DAA8D,uPAAuP,gEAAgE,0CAA0C,+DAA+D,iqBAAiqB,6DAA6D,0CAA0C,4DAA4D,uPAAuP,8DAA8D,0CAA0C,6DAA6D,8YAA8Y,gCAAgC,4GAA4G,+BAA+B,wiEAAwiE,eAAe,mgBAAmgB,cAAc,GAAG,aAAa,GAAG,2BAA2B,8CAA8C,0CAA0C,6HAA6H,0CAA0C,oNAAoN,oCAAoC,wGAAwG,6HAA6H,kRAAkR,iCAAiC,qGAAqG,0HAA0H,yRAAyR,wCAAwC,6GAA6G,uIAAuI,6FAA6F,mDAAmD,8RAA8R,sCAAsC,2GAA2G,qIAAqI,6FAA6F,iDAAiD,2qEAA2qE,yDAAyD,ufAAuf,wDAAwD,6BAA6B,8DAA8D,wBAAwB,sDAAsD,6BAA6B,kCAAkC,wBAAwB,oEAAoE,6BAA6B,gDAAgD,kMAAkM,oCAAoC,wGAAwG,gIAAgI,qRAAqR,iCAAiC,qGAAqG,6HAA6H,4RAA4R,wCAAwC,6GAA6G,0IAA0I,6FAA6F,sDAAsD,iSAAiS,sCAAsC,2GAA2G,wIAAwI,6FAA6F,oDAAoD","file":"15.js","sourcesContent":["module.exports = \"<div class=container> <div class=\\\"row monitoring-head\\\"> <div class=back-nav> <span class=fake-link aria-hidden=true>&#9664;</span> <a href=/#/monitored_endpoints>All endpoints</a> </div> <div class=\\\"col-sm-9 no-side-padding list-section\\\"> <h1> {{endpointName}} <div class=endpoint-status> <span ng-if=endpoint.isStale class=warning> <i class=\\\"fa pa-endpoint-lost endpoint-details\\\" uib-tooltip=\\\"Unable to connect to endpoint\\\"></i> </span> <span class=warning ng-if=endpoint.isScMonitoringDisconnected> <i class=\\\"fa pa-monitoring-lost endpoint-details\\\" uib-tooltip=\\\"Unable to connect to monitoring server\\\"></i> </span> <a ng-if=endpoint.errorCount class=warning href=#/failed-messages/groups/{{endpointName}}/{{sourceIndex}}/{{endpoint.serviceControlId}}> <i class=\\\"fa fa-envelope\\\" uib-tooltip=\\\"{{endpoint.errorCount | metricslargenumber}} failed messages associated with this endpoint. Click to see list.\\\"></i> <span class=\\\"badge badge-important ng-binding\\\">{{endpoint.errorCount | metricslargenumber}}</span> </a> </div> </h1> </div> <div class=\\\"col-sm-3 no-side-padding toolbar-menus\\\"> <ul class=\\\"nav nav-pills period-selector\\\"> <li role=presentation ng-repeat=\\\"period in periods\\\" ng-class=\\\"(period.value == selectedPeriod.value ? 'active' : 'notselected')\\\"> <a ng-click=selectPeriod(period) href=\\\"#/endpoint_details/{{endpointName}}/{{sourceIndex}}?historyPeriod={{period.value}}\\\">{{period.text}}</a> </li> </ul> </div> </div> </div> <div class=\\\"container large-graphs\\\"> <div class=container> <div class=row> <div class=\\\"col-sm-4 no-side-padding list-section graph-area graph-queue-length\\\"> <large-graph ng-if=endpoint.metricDetails.metrics.queueLength first-data-series=endpoint.metricDetails.metrics.queueLength xaxis-points=endpoint.metricDetails.metrics.queueLength.timeAxisValues minimum-yaxis={{largeGraphsMinimumYAxis.queueLength}} plot-width=750 plot-height=200 first-series-color=#EA7E00 first-series-fill-color=#EADDCE class=\\\"large-graph pull-left\\\"></large-graph> <div class=\\\"col-sm-12 no-side-padding graph-values\\\"> <div class=\\\"row queue-length-values tooltip-trigger\\\"> <div class=metric-digest-header> Queue Length <i class=\\\"fa fa-flask fake-link\\\"></i> </div> <div class=interactive-tooltip> <div class=tooltip-contents> <p>Queue length: The estimated number of messages in an endpoint's queue.</p> <p>NOTE: This is an experimental feature. <a href=https://docs.particular.net/monitoring/metrics/definitions#metrics-captured-queue-length-known-limitations target=_blank>Learn more</a> <i class=\\\"fa fa-external-link fake-link\\\"></i></p> </div> </div> </div> <div class=\\\"row metric-digest-value current\\\"> <div ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.digest.metrics.queueLength.latest | metricslargenumber:0}} <span ng-if=\\\"endpoint.isStale == false || endpoint.isScMonitoringDisconnected == false\\\" class=metric-digest-value-suffix>MSGS</span> </div> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> <div class=\\\"row metric-digest-value average\\\"> <div ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.digest.metrics.queueLength.average | metricslargenumber:0}} <span class=metric-digest-value-suffix>MSGS</span> </div> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> <span ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\" class=metric-digest-value-suffix> AVG</span> </div> </div> </div> <div class=\\\"col-sm-4 no-side-padding list-section graph-area graph-message-retries-throughputs\\\"> <large-graph ng-if=endpoint.metricDetails.metrics.throughput first-data-series=endpoint.metricDetails.metrics.throughput second-data-series=endpoint.metricDetails.metrics.retries xaxis-points=endpoint.metricDetails.metrics.throughput.timeAxisValues minimum-yaxis={{largeGraphsMinimumYAxis.throughputRetries}} plot-width=750 plot-height=200 first-series-color=#176397 first-series-fill-color=#CADCE8 second-series-color=#CC1252 second-series-fill-color=#E9C4D1 class=\\\"large-graph pull-left\\\"></large-graph> <div class=\\\"col-sm-12 no-side-padding graph-values\\\"> <div class=\\\"col-sm-6 no-side-padding throughput-values\\\"> <div class=row> <span class=metric-digest-header uib-tooltip=\\\"Throughput: The number of messages per second successfully processed by a receiving endpoint.\\\"> Throughput </span> </div> <div class=\\\"row metric-digest-value current\\\"> <div ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.digest.metrics.throughput.latest | metricslargenumber:2}} <span class=metric-digest-value-suffix>MSGS/S</span> </div> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> <div class=\\\"row metric-digest-value average\\\"> <div ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.digest.metrics.throughput.average | metricslargenumber:2}} <span class=metric-digest-value-suffix>MSGS/S</span> </div> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> <span ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\" class=metric-digest-value-suffix> AVG</span> </div> </div> <div class=\\\"col-sm-6 no-side-padding scheduled-retries-rate-values\\\"> <div class=row> <span class=metric-digest-header uib-tooltip=\\\"Scheduled retry rate: The number of messages per second scheduled for retries (immediate or delayed).\\\"> Scheduled Retries Rate </span> </div> <div class=\\\"row metric-digest-value current\\\"> <div ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.digest.metrics.retries.latest | metricslargenumber:2}} <span class=metric-digest-value-suffix>MSGS/S</span> </div> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> <div class=\\\"row metric-digest-value average\\\"> <div ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.digest.metrics.retries.average | metricslargenumber:2}} <span class=metric-digest-value-suffix>MSGS/S</span> </div> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> <span ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\" class=metric-digest-value-suffix> AVG</span> </div> </div> </div> </div> <div class=\\\"col-sm-4 no-side-padding list-section graph-area graph-critical-processing-times\\\"> <large-graph ng-if=endpoint.metricDetails.metrics.processingTime first-data-series=endpoint.metricDetails.metrics.criticalTime second-data-series=endpoint.metricDetails.metrics.processingTime xaxis-points=endpoint.metricDetails.metrics.criticalTime.timeAxisValues minimum-yaxis={{largeGraphsMinimumYAxis.processingCritical}} plot-width=750 plot-height=200 first-series-color=#2700CB first-series-fill-color=#C4BCE5 second-series-color=#258135 second-series-fill-color=#BEE6C5 is-duration-graph=true class=\\\"large-graph pull-left\\\"></large-graph> <div class=\\\"col-sm-12 no-side-padding graph-values\\\"> <div class=\\\"col-sm-6 no-side-padding processing-time-values\\\"> <div class=row> <span class=metric-digest-header uib-tooltip=\\\"Processing time: The time taken for a receiving endpoint to successfully process a message.\\\"> Processing Time </span> </div> <div class=\\\"row metric-digest-value current\\\"> <div ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.digest.metrics.processingTime.latest | durationValue}} <span class=metric-digest-value-suffix>{{endpoint.digest.metrics.processingTime.latest | durationUnit}}</span> </div> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> <div class=\\\"row metric-digest-value average\\\"> <div ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.digest.metrics.processingTime.average | durationValue}} <span class=metric-digest-value-suffix>{{endpoint.digest.metrics.processingTime.average | durationUnit}} AVG</span> </div> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> <span ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\" class=metric-digest-value-suffix> AVG</span> </div> </div> <div class=\\\"col-sm-6 no-side-padding critical-time-values\\\"> <div class=row> <span class=metric-digest-header uib-tooltip=\\\"Critical time: The elapsed time from when a message was sent, until it was successfully processed by a receiving endpoint.\\\"> Critical Time </span> </div> <div class=\\\"row metric-digest-value current\\\"> <div ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.digest.metrics.criticalTime.latest | durationValue}} <span class=metric-digest-value-suffix>{{endpoint.digest.metrics.criticalTime.latest | durationUnit}}</span> </div> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> <div class=\\\"row metric-digest-value average\\\"> <div ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\"> {{endpoint.digest.metrics.criticalTime.average | durationValue}} <span class=metric-digest-value-suffix>{{endpoint.digest.metrics.criticalTime.average | durationUnit}} </span> </div> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> <span ng-if=\\\"endpoint.isStale == false && endpoint.isScMonitoringDisconnected == false\\\" class=metric-digest-value-suffix> AVG</span> </div> </div> </div> </div> </div> </div> </div> <div class=container> <busy ng-show=loading message=\\\"Loading details\\\"></busy> <div class=tabs> <h5 ng-class=\\\"{active: !showInstancesBreakdown}\\\"> <a ng-click=\\\"showInstancesBreakdown = false\\\" class=ng-binding>Message Types</a> </h5> <h5 ng-class=\\\"{active: showInstancesBreakdown}\\\"> <a ng-click=\\\"showInstancesBreakdown = true\\\" class=ng-binding>Instances</a> </h5> </div> <section ng-if=showInstancesBreakdown class=endpoint-instances> <div class=row> <div class=\\\"col-sm-12 no-side-padding\\\"> <busy ng-show=loading message=\\\"Loading details\\\"></busy> <div ng-show=!loading class=\\\"row box box-no-click table-head-row\\\"> <div class=\\\"col-sm-2 col-xl-8\\\"> <div class=\\\"row box-header\\\"> <div class=col-sm-12> Instance Name </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Throughput: The number of messages per second successfully processed by a receiving endpoint.\\\"> Throughput <span class=table-header-unit>(msgs/s)</span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Scheduled retry rate: The number of messages per second scheduled for retries (immediate or delayed).\\\"> Scheduled retry rate <span class=table-header-unit>(msgs/s)</span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Processing time: The time taken for a receiving endpoint to successfully process a message.\\\"> Processing Time <span class=table-header-unit>(t)</span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Critical time: The elapsed time from when a message was sent, until it was successfully processed by a receiving endpoint.\\\"> Critical Time <span class=table-header-unit>(t)</span> </div> </div> </div> </div> <div class=row> <div class=\\\"col-sm-12 no-side-padding\\\"> <div class=\\\"row box endpoint-row\\\" ng-repeat=\\\"instance in endpoint.instances\\\"> <div class=\\\"col-sm-12 no-side-padding\\\"> <div class=row> <div class=\\\"col-sm-2 col-xl-8 endpoint-name\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-lg-max-9 no-side-padding lead\\\"> {{instance.name}} </div> <div class=\\\"col-lg-4 endpoint-status\\\"> <span class=warning ng-if=instance.isScMonitoringDisconnected> <i class=\\\"fa pa-monitoring-lost endpoint-details\\\" uib-tooltip=\\\"Unable to connect to monitoring server\\\"></i> </span> <span class=warning ng-if=instance.isStale> <i class=\\\"fa pa-endpoint-lost endpoint-details\\\" uib-tooltip=\\\"Unable to connect to instance\\\"></i> </span> <span class=warning ng-if=instance.errorCount> <a ng-if=instance.errorCount class=\\\"warning btn\\\" href=#/failed-messages/groups/{{endpointName}}/{{sourceIndex}}/{{instance.serviceControlId}}> <i class=\\\"fa fa-envelope\\\" uib-tooltip=\\\"{{instance.errorCount | metricslargenumber}} failed messages associated with this endpoint. Click to see list.\\\"></i> <span class=\\\"badge badge-important ng-binding\\\">{{instance.errorCount | metricslargenumber}}</span> </a> </span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=instance.metrics.throughput minimum-yaxis={{smallGraphsMinimumYAxis.throughput}} class=\\\"graph throughput pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(instance.isStale == true || instance.isScMonitoringDisconnected == true) ? \\\"\\\" : instance.metrics.throughput.displayValue}} <strong ng-if=\\\"instance.isStale || instance.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=instance.metrics.retries minimum-yaxis={{smallGraphsMinimumYAxis.retries}} class=\\\"graph retries pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(instance.isStale == true || instance.isScMonitoringDisconnected == true) ? \\\"\\\" : instance.metrics.retries.displayValue}} <strong ng-if=\\\"instance.isStale || instance.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=instance.metrics.processingTime minimum-yaxis={{smallGraphsMinimumYAxis.processingTime}} class=\\\"graph processing-time pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(instance.isStale == true || instance.isScMonitoringDisconnected == true) ? \\\"\\\" : instance.metrics.processingTime.displayValue.value}} <span ng-if=\\\"instance.isStale == false || instance.isScMonitoringDisconnected == false\\\">{{instance.metrics.processingTime.displayValue.unit}}</span> <strong ng-if=\\\"instance.isStale || instance.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=instance.metrics.criticalTime minimum-yaxis={{smallGraphsMinimumYAxis.criticalTime}} class=\\\"graph critical-time pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(instance.isStale == true || instance.isScMonitoringDisconnected == true) ? \\\"\\\" : instance.metrics.criticalTime.displayValue.value}} <span ng-if=\\\"instance.isStale == false || instance.isScMonitoringDisconnected == false\\\">{{instance.metrics.criticalTime.displayValue.unit}}</span> <strong ng-if=\\\"instance.isStale || instance.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> </div> </div> </div> </div> </div> </div> </div> </section> <section ng-if=!showInstancesBreakdown class=endpoint-message-types> <div class=row> <div class=\\\"col-sm-12 no-side-padding\\\"> <busy ng-show=loading message=\\\"Loading details\\\"></busy> <div ng-show=!loading class=\\\"row box box-no-click table-head-row\\\"> <div class=\\\"col-sm-2 col-xl-8\\\"> <div class=\\\"row box-header\\\"> <div class=col-sm-12> Message type name </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Throughput: The number of messages per second successfully processed by a receiving endpoint.\\\"> Throughput <span class=table-header-unit>(msgs/s)</span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Scheduled retry rate: The number of messages per second scheduled for retries (immediate or delayed).\\\"> Scheduled retry rate <span class=table-header-unit>(msgs/s)</span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Processing time: The time taken for a receiving endpoint to successfully process a message.\\\"> Processing Time <span class=table-header-unit>(t)</span> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-sm-12 no-side-padding\\\" uib-tooltip=\\\"Critical time: The elapsed time from when a message was sent, until it was successfully processed by a receiving endpoint.\\\"> Critical Time <span class=table-header-unit>(t)</span> </div> </div> </div> </div> <div class=row> <div class=\\\"col-sm-12 no-side-padding\\\"> <div class=\\\"row box endpoint-row\\\" ng-repeat=\\\"messageType in endpoint.messageTypes\\\"> <div class=\\\"col-sm-12 no-side-padding\\\"> <div class=row> <div class=\\\"col-sm-2 col-xl-8 endpoint-name\\\"> <div class=\\\"row box-header\\\"> <div class=\\\"col-lg-max-9 no-side-padding lead message-type-label\\\"> <div class=lead> {{messageType.typeName ? messageType.typeName : 'Unknown'}} </div> </div> <div class=\\\"col-lg-4 no-side-padding endpoint-status message-type-status\\\"> <span class=warning ng-if=endpoint.isScMonitoringDisconnected> <i class=\\\"fa pa-monitoring-lost endpoint-details\\\" uib-tooltip=\\\"Unable to connect to monitoring server\\\"></i> </span> <span class=warning ng-if=endpoint.isStale> <i class=\\\"fa pa-endpoint-lost endpoint-details\\\" uib-tooltip=\\\"Unable to connect to instance\\\"></i> </span> </div> </div> <div class=\\\"row message-type-properties\\\"> <div ng-show=\\\"{{messageType.typeName && messageType.typeName != 'null'}}\\\" class=message-type-part>{{messageType.assemblyName + '-' + messageType.assemblyVersion}}</div> <div ng-show=\\\"{{messageType.culture && messageType.culture != 'null'}}\\\" class=message-type-part>{{'Culture=' + messageType.culture}}</div> <div ng-show=\\\"{{messageType.publicKeyToken && messageType.publicKeyToken != 'null'}}\\\" class=message-type-part>{{'PublicKeyToken=' + messageType.publicKeyToken}}</div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=messageType.metrics.throughput minimum-yaxis={{smallGraphsMinimumYAxis.throughput}} class=\\\"graph throughput pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : messageType.metrics.throughput.displayValue}} <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=messageType.metrics.retries minimum-yaxis={{smallGraphsMinimumYAxis.retries}} class=\\\"graph retries pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : messageType.metrics.retries.displayValue}} <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=messageType.metrics.processingTime minimum-yaxis={{smallGraphsMinimumYAxis.processingTime}} class=\\\"graph processing-time pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : messageType.metrics.processingTime.displayValue.value}} <span ng-if=\\\"instance.isStale == false || instance.isScMonitoringDisconnected == false\\\">{{messageType.metrics.processingTime.displayValue.unit}}</span> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> <div class=\\\"col-sm-2 col-xl-1 no-side-padding\\\"> <div class=\\\"row box-header\\\"> <div class=no-side-padding> <graph plot-data=messageType.metrics.criticalTime minimum-yaxis={{smallGraphsMinimumYAxis.criticalTime}} class=\\\"graph critical-time pull-left\\\"></graph> </div> <div class=\\\"no-side-padding sparkline-value\\\"> {{(endpoint.isStale == true || endpoint.isScMonitoringDisconnected == true) ? \\\"\\\" : messageType.metrics.criticalTime.displayValue.value}} <span ng-if=\\\"instance.isStale == false || instance.isScMonitoringDisconnected == false\\\">{{messageType.metrics.criticalTime.displayValue.unit}}</span> <strong ng-if=\\\"endpoint.isStale || endpoint.isScMonitoringDisconnected\\\">?</strong> </div> </div> </div> </div> </div> </div> </div> </div> </div> </div> </section> </div>\";\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./app/modules/monitoring/views/endpoint_details.html\n// module id = 15\n// module chunks = 0"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///15\n"); - -/***/ }), -/* 16 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular, undefined) {\n 'use strict';\n\n angular.module('ui.particular.duration', []).filter('duration', ['formatter', function (formatter) {\n return function (input) {\n var time = formatter.formatTime(input);\n return time.value + ' ' + time.unit;\n };\n }]).filter('durationValue', ['formatter', function (formatter) {\n return function (input) {\n return formatter.formatTime(input).value;\n };\n }]).filter('durationUnit', ['formatter', function (formatter) {\n return function (input) {\n return formatter.formatTime(input).unit;\n };\n }]);\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly8vLi9hcHAvbW9kdWxlcy9tb25pdG9yaW5nL2pzL2RpcmVjdGl2ZXMvdWkucGFydGljdWxhci5kdXJhdGlvbi5qcz85YjhlIl0sIm5hbWVzIjpbIndpbmRvdyIsImFuZ3VsYXIiLCJ1bmRlZmluZWQiLCJtb2R1bGUiLCJmaWx0ZXIiLCJmb3JtYXR0ZXIiLCJpbnB1dCIsInRpbWUiLCJmb3JtYXRUaW1lIiwidmFsdWUiLCJ1bml0Il0sIm1hcHBpbmdzIjoiOztBQUFDLFdBQVVBLE1BQVYsRUFBa0JDLE9BQWxCLEVBQTJCQyxTQUEzQixFQUFzQztBQUNuQzs7QUFFQUQsWUFBUUUsTUFBUixDQUFlLHdCQUFmLEVBQXlDLEVBQXpDLEVBQ0tDLE1BREwsQ0FDWSxVQURaLEVBQ3dCLENBQUMsV0FBRCxFQUFjLFVBQVVDLFNBQVYsRUFBcUI7QUFDbkQsZUFBTyxVQUFVQyxLQUFWLEVBQWlCO0FBQ3BCLGdCQUFJQyxPQUFPRixVQUFVRyxVQUFWLENBQXFCRixLQUFyQixDQUFYO0FBQ0EsbUJBQVVDLEtBQUtFLEtBQWYsU0FBd0JGLEtBQUtHLElBQTdCO0FBQ0gsU0FIRDtBQUlILEtBTG1CLENBRHhCLEVBT0tOLE1BUEwsQ0FPWSxlQVBaLEVBTzZCLENBQUMsV0FBRCxFQUFjLFVBQVVDLFNBQVYsRUFBcUI7QUFDeEQsZUFBTyxVQUFVQyxLQUFWLEVBQWlCO0FBQ3BCLG1CQUFPRCxVQUFVRyxVQUFWLENBQXFCRixLQUFyQixFQUE0QkcsS0FBbkM7QUFDSCxTQUZEO0FBR0gsS0FKd0IsQ0FQN0IsRUFZS0wsTUFaTCxDQVlZLGNBWlosRUFZNEIsQ0FBQyxXQUFELEVBQWMsVUFBVUMsU0FBVixFQUFxQjtBQUN2RCxlQUFPLFVBQVVDLEtBQVYsRUFBaUI7QUFDcEIsbUJBQU9ELFVBQVVHLFVBQVYsQ0FBcUJGLEtBQXJCLEVBQTRCSSxJQUFuQztBQUNILFNBRkQ7QUFHSCxLQUp1QixDQVo1QjtBQWlCSCxDQXBCQSxFQW9CQ1YsTUFwQkQsRUFvQlNBLE9BQU9DLE9BcEJoQixDQUFEIiwiZmlsZSI6IjE2LmpzIiwic291cmNlc0NvbnRlbnQiOlsiKGZ1bmN0aW9uICh3aW5kb3csIGFuZ3VsYXIsIHVuZGVmaW5lZCkge1xyXG4gICAgJ3VzZSBzdHJpY3QnO1xyXG5cclxuICAgIGFuZ3VsYXIubW9kdWxlKCd1aS5wYXJ0aWN1bGFyLmR1cmF0aW9uJywgW10pXHJcbiAgICAgICAgLmZpbHRlcignZHVyYXRpb24nLCBbJ2Zvcm1hdHRlcicsIGZ1bmN0aW9uIChmb3JtYXR0ZXIpIHtcclxuICAgICAgICAgICAgcmV0dXJuIGZ1bmN0aW9uIChpbnB1dCkge1xyXG4gICAgICAgICAgICAgICAgdmFyIHRpbWUgPSBmb3JtYXR0ZXIuZm9ybWF0VGltZShpbnB1dCk7XHJcbiAgICAgICAgICAgICAgICByZXR1cm4gYCR7dGltZS52YWx1ZX0gJHt0aW1lLnVuaXR9YDtcclxuICAgICAgICAgICAgfTtcclxuICAgICAgICB9XSlcclxuICAgICAgICAuZmlsdGVyKCdkdXJhdGlvblZhbHVlJywgWydmb3JtYXR0ZXInLCBmdW5jdGlvbiAoZm9ybWF0dGVyKSB7XHJcbiAgICAgICAgICAgIHJldHVybiBmdW5jdGlvbiAoaW5wdXQpIHtcclxuICAgICAgICAgICAgICAgIHJldHVybiBmb3JtYXR0ZXIuZm9ybWF0VGltZShpbnB1dCkudmFsdWU7XHJcbiAgICAgICAgICAgIH07XHJcbiAgICAgICAgfV0pXHJcbiAgICAgICAgLmZpbHRlcignZHVyYXRpb25Vbml0JywgWydmb3JtYXR0ZXInLCBmdW5jdGlvbiAoZm9ybWF0dGVyKSB7XHJcbiAgICAgICAgICAgIHJldHVybiBmdW5jdGlvbiAoaW5wdXQpIHtcclxuICAgICAgICAgICAgICAgIHJldHVybiBmb3JtYXR0ZXIuZm9ybWF0VGltZShpbnB1dCkudW5pdDtcclxuICAgICAgICAgICAgfTtcclxuICAgICAgICB9XSk7XHJcbn0od2luZG93LCB3aW5kb3cuYW5ndWxhcikpO1xuXG5cbi8vIFdFQlBBQ0sgRk9PVEVSIC8vXG4vLyAuL2FwcC9tb2R1bGVzL21vbml0b3JpbmcvanMvZGlyZWN0aXZlcy91aS5wYXJ0aWN1bGFyLmR1cmF0aW9uLmpzIl0sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///16\n"); - -/***/ }), -/* 17 */ -/***/ (function(module, exports, __webpack_require__) { - -"use strict"; -eval("\n\n(function (window, angular) {\n 'use strict';\n\n function drawDataSeries(chart, data, color, fillColor, scaleX, scaleY) {\n\n var area = d3.area().x(function (d, i) {\n return scaleX(i);\n }).y(function (d) {\n return scaleY(d);\n }).y1(function () {\n return scaleY(0);\n }).curve(d3.curveLinear);\n\n var line = d3.line().x(function (d, i) {\n return scaleX(i);\n }).y(function (d, i) {\n return scaleY(d);\n }).curve(d3.curveLinear);\n\n var group = chart.append('g').attr('class', 'dataSeries');\n\n group.append('path').datum(data.points).attr('d', area).attr('fill', fillColor).attr('opacity', 0.8).attr('stroke', fillColor);\n\n group.append('path').datum(data.points).attr('d', line).attr('stroke', color).attr('stroke-width', 2.75).attr('fill', 'none');\n }\n\n function drawAverageLine(chart, data, color, fillColor, scaleX, scaleY) {\n\n var line = d3.line().x(function (d, i) {\n return scaleX(i);\n }).y(function (d, i) {\n return scaleY(d);\n }).curve(d3.curveLinear);\n\n var group = chart.append('g').attr('class', 'dataAverage');\n\n group.append('path').datum(Array(data.points.length).fill(data.average)).attr('d', line).attr('stroke', color).attr('stroke-width', 1.5).attr('opacity', 0.5).attr('stroke-dasharray', '10,10');\n }\n\n function padToWholeValue(value) {\n var emptyDataSetyAxisMax = 10;\n\n if (!value) {\n return emptyDataSetyAxisMax;\n }\n\n var upperBound = 10;\n\n while (value > upperBound) {\n upperBound *= 10;\n }\n\n upperBound /= 10;\n\n return Math.floor(value / upperBound) * upperBound + upperBound;\n }\n\n angular.module('ui.particular.largeGraph', []).directive('largeGraph', function (formatter) {\n return {\n restrict: 'E',\n scope: {\n dates: '=xaxisPoints',\n firstDataSeries: '=firstDataSeries',\n secondDataSeries: '=secondDataSeries',\n isDurationGraph: '=isDurationGraph',\n minimumYaxis: '@',\n width: '=plotWidth',\n height: '=plotHeight'\n },\n template: '',\n link: function link(scope, element, attrs) {\n scope.$watch('firstDataSeries', function () {\n\n var svg = element.find('svg')[0];\n\n d3.select(svg).selectAll('*').remove();\n\n var topMargin = 10;\n var bottomMargin = 5;\n var leftMargin = 60;\n\n var chart = d3.select(svg).attr('width', scope.width).attr('height', scope.height);\n\n var width = svg.clientWidth;\n var height = svg.clientHeight;\n\n //HINT: This is workaround for Firefox\n if (width === 0) {\n var box = svg.getBoundingClientRect();\n\n width = box.right - box.left;\n height = box.bottom - box.top;\n }\n\n var firstSeries = scope.firstDataSeries;\n var secondSeries = scope.secondDataSeries;\n\n var scaleX = d3.scaleLinear().domain([0, firstSeries.points.length - 1]).range([leftMargin, width]);\n\n chart.append('rect').attr('width', width - leftMargin).attr('height', height - topMargin - bottomMargin).attr('transform', 'translate(' + leftMargin + ',' + topMargin + ')').attr('fill', '#F2F6F7');\n\n var minimumYaxis = !isNaN(scope.minimumYaxis) ? Number(scope.minimumYaxis) : 10;\n var max = Math.max(firstSeries.average, d3.max(firstSeries.points), minimumYaxis);\n\n if (secondSeries && secondSeries.points.length > 0) {\n max = Math.max(max, secondSeries.average, d3.max(secondSeries.points));\n }\n\n var max = padToWholeValue(max);\n\n var scaleY = d3.scaleLinear().domain([0, max]).range([height - bottomMargin, topMargin]);\n\n var yAxis = d3.axisLeft(scaleY).tickValues([0, max * 1 / 4, max * 1 / 2, max * 3 / 4, max]);\n\n if (scope.isDurationGraph) {\n yAxis = yAxis.tickFormat(function (v) {\n var formattedTime = formatter.formatTime(v);\n\n return formattedTime.value + ' ' + formattedTime.unit;\n });\n }\n\n chart.append('g').attr('class', 'y axis').attr('transform', 'translate(' + leftMargin + ', 0)').call(function (g) {\n g.call(yAxis);\n g.select('.domain').remove();\n g.selectAll('.tick line').attr('stroke', 'black').attr('stroke-width', '1.75').attr('opacity', 0.1).attr('x', 0).attr('x2', width - leftMargin);\n g.selectAll('.tick text').attr('x', -4).attr('fill', '#828282');\n });\n\n var drawSeries = function drawSeries(data, lineColor, fillColor) {\n drawDataSeries(chart, data, lineColor, fillColor, scaleX, scaleY);\n };\n\n var drawAverage = function drawAverage(data, lineColor, fillColor) {\n drawAverageLine(chart, data, lineColor, fillColor, scaleX, scaleY);\n };\n\n drawSeries(firstSeries, attrs.firstSeriesColor, attrs.firstSeriesFillColor);\n\n if (secondSeries) {\n drawSeries(secondSeries, attrs.secondSeriesColor, attrs.secondSeriesFillColor);\n }\n\n drawAverage(firstSeries, attrs.firstSeriesColor, attrs.firstSeriesFillColor);\n\n if (secondSeries) {\n drawAverage(secondSeries, attrs.secondSeriesColor, attrs.secondSeriesFillColor);\n }\n });\n }\n };\n });\n})(window, window.angular);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///./app/modules/monitoring/js/directives/ui.particular.largeGraph.js?070b"],"names":["window","angular","drawDataSeries","chart","data","color","fillColor","scaleX","scaleY","area","d3","x","d","i","y","y1","curve","curveLinear","line","group","append","attr","datum","points","drawAverageLine","Array","length","fill","average","padToWholeValue","value","emptyDataSetyAxisMax","upperBound","Math","floor","module","directive","formatter","restrict","scope","dates","firstDataSeries","secondDataSeries","isDurationGraph","minimumYaxis","width","height","template","link","element","attrs","$watch","svg","find","select","selectAll","remove","topMargin","bottomMargin","leftMargin","clientWidth","clientHeight","box","getBoundingClientRect","right","left","bottom","top","firstSeries","secondSeries","scaleLinear","domain","range","isNaN","Number","max","yAxis","axisLeft","tickValues","tickFormat","v","formattedTime","formatTime","unit","call","g","drawSeries","lineColor","drawAverage","firstSeriesColor","firstSeriesFillColor","secondSeriesColor","secondSeriesFillColor"],"mappings":"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","file":"17.js","sourcesContent":["(function(window, angular) {\r\n    'use strict';\r\n\r\n    function drawDataSeries(chart, data, color, fillColor, scaleX, scaleY) {\r\n\r\n        var area = d3.area()\r\n            .x(function (d, i) { return scaleX(i);})\r\n            .y(function (d)    { return scaleY(d); })\r\n            .y1(function ()    { return scaleY(0); })\r\n            .curve(d3.curveLinear);\r\n\r\n        var line = d3.line()\r\n            .x(function (d, i) { return scaleX(i);})\r\n            .y(function (d, i) { return scaleY(d);})\r\n            .curve(d3.curveLinear);\r\n\r\n        var group = chart.append('g').attr('class', 'dataSeries');\r\n\r\n        group.append('path')\r\n            .datum(data.points)\r\n            .attr('d', area)\r\n            .attr('fill', fillColor)\r\n            .attr('opacity', 0.8)\r\n            .attr('stroke', fillColor);\r\n\r\n        group.append('path')\r\n            .datum(data.points)\r\n            .attr('d', line)\r\n            .attr('stroke', color)\r\n            .attr('stroke-width', 2.75)\r\n            .attr('fill', 'none');\r\n    }\r\n\r\n    function drawAverageLine(chart, data, color, fillColor, scaleX, scaleY) {\r\n\r\n        var line = d3.line()\r\n            .x(function (d, i) { return scaleX(i); })\r\n            .y(function (d, i) { return scaleY(d); })\r\n            .curve(d3.curveLinear);\r\n\r\n        var group = chart.append('g').attr('class', 'dataAverage');\r\n\r\n        group.append('path')\r\n            .datum(Array(data.points.length).fill(data.average))\r\n            .attr('d', line)\r\n            .attr('stroke', color)\r\n            .attr('stroke-width', 1.5)\r\n            .attr('opacity', 0.5)\r\n            .attr('stroke-dasharray', '10,10');\r\n    }\r\n\r\n    function padToWholeValue(value) {\r\n        var emptyDataSetyAxisMax = 10;\r\n\r\n        if (!value) {\r\n            return emptyDataSetyAxisMax;\r\n        }\r\n\r\n        var upperBound = 10;\r\n\r\n        while (value > upperBound) {\r\n            upperBound *= 10;\r\n        }\r\n\r\n        upperBound /= 10;\r\n\r\n        return Math.floor(value / upperBound) * upperBound + upperBound;\r\n    }\r\n\r\n    angular.module('ui.particular.largeGraph', [])\r\n        .directive('largeGraph',\r\n            function(formatter) {\r\n                return {\r\n                    restrict: 'E',\r\n                    scope: {                        \r\n                        dates: '=xaxisPoints',\r\n                        firstDataSeries: '=firstDataSeries',\r\n                        secondDataSeries: '=secondDataSeries',\r\n                        isDurationGraph: '=isDurationGraph',\r\n                        minimumYaxis: '@',\r\n                        width: '=plotWidth',\r\n                        height: '=plotHeight'\r\n                    },\r\n                    template: '<svg></svg>',\r\n                    link: function link(scope, element, attrs) {\r\n                        scope.$watch('firstDataSeries', function () {\r\n\r\n                            var svg = element.find('svg')[0];\r\n\r\n                            d3.select(svg).selectAll('*').remove();\r\n\r\n                            var topMargin = 10;\r\n                            var bottomMargin = 5;\r\n                            var leftMargin = 60;\r\n\r\n                            var chart = d3.select(svg)\r\n                                .attr('width', scope.width)\r\n                                .attr('height', scope.height);\r\n\r\n                            var width = svg.clientWidth;\r\n                            var height = svg.clientHeight;\r\n\r\n                            //HINT: This is workaround for Firefox\r\n                            if (width === 0) {\r\n                                var box = svg.getBoundingClientRect();\r\n\r\n                                width = box.right - box.left;\r\n                                height = box.bottom - box.top;\r\n                            }\r\n\r\n                            var firstSeries = scope.firstDataSeries;\r\n                            var secondSeries = scope.secondDataSeries;\r\n\r\n                            var scaleX = d3.scaleLinear()\r\n                                .domain([0, firstSeries.points.length - 1])\r\n                                .range([leftMargin, width]);\r\n\r\n                            chart.append('rect')\r\n                                .attr('width', width - leftMargin)\r\n                                .attr('height', height - topMargin - bottomMargin)\r\n                                .attr('transform', 'translate(' + leftMargin + ',' + topMargin + ')')\r\n                                .attr('fill', '#F2F6F7');\r\n\r\n                            var minimumYaxis = !isNaN(scope.minimumYaxis) ? Number(scope.minimumYaxis) : 10;\r\n                            var max = Math.max(firstSeries.average, d3.max(firstSeries.points), minimumYaxis);\r\n\r\n                            if (secondSeries && secondSeries.points.length > 0) {\r\n                                max = Math.max(max, secondSeries.average, d3.max(secondSeries.points));\r\n                            }\r\n\r\n                            var max = padToWholeValue(max);\r\n\r\n                            var scaleY = d3.scaleLinear()\r\n                                .domain([0, max])\r\n                                .range([height - bottomMargin, topMargin]);\r\n\r\n                            var yAxis = d3.axisLeft(scaleY) \r\n                                .tickValues([0, max * 1/4, max * 1/2, max * 3/4, max]);\r\n\r\n                            if (scope.isDurationGraph) {\r\n                                yAxis = yAxis.tickFormat(function (v) {\r\n                                    var formattedTime = formatter.formatTime(v);\r\n\r\n                                    return formattedTime.value + '  ' + formattedTime.unit;\r\n                                });\r\n                            }\r\n\r\n                            chart.append('g')\r\n                                .attr('class', 'y axis')\r\n                                .attr('transform', 'translate(' + leftMargin + ', 0)')\r\n                                .call(function (g) {\r\n                                    g.call(yAxis);\r\n                                    g.select('.domain').remove();\r\n                                    g.selectAll('.tick line').attr('stroke', 'black').attr('stroke-width', '1.75').attr('opacity', 0.1).attr('x', 0).attr('x2', width - leftMargin);\r\n                                    g.selectAll('.tick text').attr('x', -4).attr('fill', '#828282');\r\n                                });\r\n\r\n                            var drawSeries = function(data, lineColor, fillColor) {\r\n                                drawDataSeries(chart, data, lineColor, fillColor, scaleX, scaleY);\r\n                            }\r\n\r\n                            var drawAverage = function(data, lineColor, fillColor) {\r\n                                drawAverageLine(chart, data, lineColor, fillColor, scaleX, scaleY);\r\n                            }\r\n\r\n                            drawSeries(firstSeries, attrs.firstSeriesColor, attrs.firstSeriesFillColor);\r\n\r\n                            if (secondSeries) {\r\n                                drawSeries(secondSeries, attrs.secondSeriesColor,attrs.secondSeriesFillColor);\r\n                            }\r\n\r\n                            drawAverage(firstSeries, attrs.firstSeriesColor, attrs.firstSeriesFillColor );\r\n\r\n                            if (secondSeries) {\r\n                                drawAverage(secondSeries, attrs.secondSeriesColor, attrs.secondSeriesFillColor);\r\n                            }\r\n                        });\r\n                    }\r\n                };\r\n            });\r\n\r\n}(window, window.angular));\r\n\n\n\n// WEBPACK FOOTER //\n// ./app/modules/monitoring/js/directives/ui.particular.largeGraph.js"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///17\n"); - -/***/ }) -/******/ ]); \ No newline at end of file diff --git a/src/ServicePulse.Host/app/modules/monitoring/js/directives/ui.particular.monitoringConnectivityStatus.js b/src/ServicePulse.Host/app/modules/monitoring/js/directives/ui.particular.monitoringConnectivityStatus.js index 3346142cf..21c6291e9 100644 --- a/src/ServicePulse.Host/app/modules/monitoring/js/directives/ui.particular.monitoringConnectivityStatus.js +++ b/src/ServicePulse.Host/app/modules/monitoring/js/directives/ui.particular.monitoringConnectivityStatus.js @@ -3,20 +3,35 @@ 'use strict'; - function controller($scope, connectivityNotifier, monitoringService, $interval) { - $scope.isSCMonitoringConnecting = true; + function controller($scope, connectivityNotifier, monitoringService, $interval, connectionsManager) { + $scope.isSCMonitoringConnecting = connectionsManager.getIsMonitoringEnabled() + + if ($scope.isSCMonitoringConnecting) { + connectivityNotifier.reportConnecting(); + } + + $scope.monitoringUrl = connectionsManager.getMonitoringUrl(); connectivityNotifier.getConnectionStatusSource().subscribe(value => { - $scope.isSCMonitoringConnected = value; - $scope.isSCMonitoringConnecting = false; + $scope.isSCMonitoringConnected = value.isConnected; + $scope.isSCMonitoringConnecting = value.isConnecting; }); + var lastReport = undefined; var scMonitoringConnectionPing = $interval(function () { - var promises = monitoringService.getMonitoredEndpoints().map((request, index) => { - request.then(r => { - connectivityNotifier.reportSuccessfulConnection(index); - }, e => { - connectivityNotifier.reportFailedConnection(index); - }); + var promise = monitoringService.getMonitoredEndpoints().then(r => { + if (lastReport === 'success') { + return; + } + + connectivityNotifier.reportSuccessfulConnection(); + lastReport = 'success'; + }, e => { + if (lastReport === 'failed') { + return; + } + + connectivityNotifier.reportFailedConnection(); + lastReport = 'failed'; }); }, 10000); @@ -33,7 +48,7 @@ - controller.$inject = ['$scope', 'connectivityNotifier', 'monitoringService', '$interval']; + controller.$inject = ['$scope', 'connectivityNotifier', 'monitoringService', '$interval', 'connectionsManager']; function directive() { return { diff --git a/src/ServicePulse.Host/app/modules/monitoring/js/directives/ui.particular.monitoringConnectivityStatus.tpl.html b/src/ServicePulse.Host/app/modules/monitoring/js/directives/ui.particular.monitoringConnectivityStatus.tpl.html index b9256ac10..8a7836166 100644 --- a/src/ServicePulse.Host/app/modules/monitoring/js/directives/ui.particular.monitoringConnectivityStatus.tpl.html +++ b/src/ServicePulse.Host/app/modules/monitoring/js/directives/ui.particular.monitoringConnectivityStatus.tpl.html @@ -1,4 +1,4 @@ - + SC Monitoring:
    Connected diff --git a/src/ServicePulse.Host/app/modules/monitoring/js/endpoint_details.controller.js b/src/ServicePulse.Host/app/modules/monitoring/js/endpoint_details.controller.js index 5109a99de..a475df4bf 100644 --- a/src/ServicePulse.Host/app/modules/monitoring/js/endpoint_details.controller.js +++ b/src/ServicePulse.Host/app/modules/monitoring/js/endpoint_details.controller.js @@ -18,7 +18,6 @@ ) { $scope.endpointName = $routeParams.endpointName; - $scope.sourceIndex = $routeParams.sourceIndex; $scope.showInstancesBreakdown = $routeParams.tab === 'instancesBreakdown'; $scope.loading = true; $scope.loadedSuccessfully = false; @@ -50,7 +49,7 @@ var breakdownTabName = showInstacesBreakdown ? 'instancesBreakdown' : 'messageTypeBreakdown'; - return `#/monitoring/endpoint/${$scope.endpointName}/${$scope.sourceIndex}?historyPeriod=${selectedPeriodValue}&tab=${breakdownTabName}&pageNo=${breakdownPageNo}`; + return `#/monitoring/endpoint/${$scope.endpointName}?historyPeriod=${selectedPeriodValue}&tab=${breakdownTabName}&pageNo=${breakdownPageNo}`; }; $scope.updateUrl = function () { @@ -101,12 +100,12 @@ var selectedPeriod = $scope.selectedPeriod; - subscription = monitoringService.createEndpointDetailsSource($routeParams.endpointName, $routeParams.sourceIndex, selectedPeriod.value, selectedPeriod.refreshInterval).subscribe(function (endpoint) { + subscription = monitoringService.createEndpointDetailsSource($routeParams.endpointName, selectedPeriod.value, selectedPeriod.refreshInterval).subscribe(function (endpoint) { $scope.loading = false; if (endpoint.error) { - connectivityNotifier.reportFailedConnection($routeParams.sourceIndex); + connectivityNotifier.reportFailedConnection(); if ($scope.endpoint && $scope.endpoint.instances) { $scope.endpoint.instances.forEach((item) => item.isScMonitoringDisconnected = true); } @@ -127,7 +126,7 @@ mergeIn($scope.endpoint, endpoint); } - connectivityNotifier.reportSuccessfulConnection($routeParams.sourceIndex); + connectivityNotifier.reportSuccessfulConnection(); $scope.endpoint.instances.sort(function (first, second) { if (first.id < second.id) { diff --git a/src/ServicePulse.Host/app/modules/monitoring/js/endpoint_details.route.js b/src/ServicePulse.Host/app/modules/monitoring/js/endpoint_details.route.js index 7c8ced8a4..e1cbc743c 100644 --- a/src/ServicePulse.Host/app/modules/monitoring/js/endpoint_details.route.js +++ b/src/ServicePulse.Host/app/modules/monitoring/js/endpoint_details.route.js @@ -4,7 +4,7 @@ function routeProvider($routeProvider) { let template = require('./../views/endpoint_details.html'); - $routeProvider.when('/monitoring/endpoint/:endpointName/:sourceIndex', { + $routeProvider.when('/monitoring/endpoint/:endpointName', { data: { pageTitle: 'Endpoint Details' }, diff --git a/src/ServicePulse.Host/app/modules/monitoring/js/monitored_endpoints.controller.js b/src/ServicePulse.Host/app/modules/monitoring/js/monitored_endpoints.controller.js index 145dd8629..fc2f75d0d 100644 --- a/src/ServicePulse.Host/app/modules/monitoring/js/monitored_endpoints.controller.js +++ b/src/ServicePulse.Host/app/modules/monitoring/js/monitored_endpoints.controller.js @@ -28,7 +28,7 @@ }; $scope.getDetailsUrl = endpoint => { - return '#/monitoring/endpoint/' + endpoint.name + '/' + (endpoint.sourceIndex | 0) + '?historyPeriod=' + $scope.selectedPeriod.value; + return '#/monitoring/endpoint/' + endpoint.name + '?historyPeriod=' + $scope.selectedPeriod.value; }; function fillDisplayValuesForEndpoint(endpoint) { @@ -69,14 +69,13 @@ } if (endpoint.error) { - connectivityNotifier.reportFailedConnection(endpoint.sourceIndex); + connectivityNotifier.reportFailedConnection(); if ($scope.endpoints) { - $scope.endpoints.filter((item) => item.sourceIndex === endpoint.sourceIndex) - .forEach((item) => item.isScMonitoringDisconnected = true); + $scope.endpoints.forEach((item) => item.isScMonitoringDisconnected = true); } } else { - connectivityNotifier.reportSuccessfulConnection(endpoint.sourceIndex); - var index = $scope.endpoints.findIndex(function(item) { return item.name === endpoint.name }); + connectivityNotifier.reportSuccessfulConnection(); + var index = $scope.endpoints.findIndex(function (item) { return item.name === endpoint.name; }); endpoint.isScMonitoringDisconnected = false; fillDisplayValuesForEndpoint(endpoint); diff --git a/src/ServicePulse.Host/app/modules/monitoring/js/services/services.connectivityNotifier.js b/src/ServicePulse.Host/app/modules/monitoring/js/services/services.connectivityNotifier.js index 6d35c1a25..b976adea2 100644 --- a/src/ServicePulse.Host/app/modules/monitoring/js/services/services.connectivityNotifier.js +++ b/src/ServicePulse.Host/app/modules/monitoring/js/services/services.connectivityNotifier.js @@ -2,45 +2,58 @@ (function (window, angular, $, undefined) { 'use strict'; - function Service(toastService, scConfig) { + function Service(toastService, connectionsManager, notifyService) { - var isConnectedToSourceIndex = Array(scConfig.monitoring_urls.length).fill(true); + var notifier = notifyService(); + var mu = connectionsManager.getMonitoringUrl(); + var isConnected = false; + var isConnecting = false; var connectivitySource = new Rx.Subject(); var shouldShowFailedMessage = true; - function reportFailedConnection(sourceIndex) { + function reportFailedConnection() { - if (isConnectedToSourceIndex[sourceIndex]) { - var message = 'Could not connect to the ServiceControl Monitoring service.'; - if (scConfig.monitoring_urls.length > 1) { - message = 'Could not connect to the ServiceControl Monitoring service at' + scConfig.monitoring_urls[sourceIndex] + '.'; - } + if (isConnected) { + var message = 'Could not connect to the ServiceControl Monitoring service at ' + mu + '. View connection settings'; console.log(message); if (shouldShowFailedMessage) { toastService.showError(message); shouldShowFailedMessage = false; } } - isConnectedToSourceIndex[sourceIndex] = false; - emitChange(isConnectedToSourceIndex); + isConnected = false; + isConnecting = false; + emitChange(); } - function reportSuccessfulConnection(sourceIndex) { - if (!isConnectedToSourceIndex[sourceIndex]) { - var message = 'Connection to ServiceControl Monitoring service was successful.'; - if (scConfig.monitoring_urls.length > 1) { - message = 'Connection to ServiceControl Monitoring service was successful ' + scConfig.monitoring_urls[sourceIndex] +'.'; - } + function reportSuccessfulConnection() { + if (!isConnected) { + var message = 'Connection to ServiceControl Monitoring service was successful ' + mu + '.'; console.log(message); shouldShowFailedMessage = true; } - isConnectedToSourceIndex[sourceIndex] = true; - emitChange(isConnectedToSourceIndex); + isConnected = true; + isConnecting = false; + emitChange(); + } + + function reportConnecting() { + isConnecting = true; + emitChange(); } - function emitChange(connectedToSourceIndex) { - var result = connectedToSourceIndex.every(item => item); + function emitChange() { + var result = { + isConnected: isConnected, + isConnecting: isConnecting + }; + connectivitySource.onNext(result); + + notifier.notify('MonitoringConnectionStatusChanged', { + isMonitoringConnected : isConnected, + isMonitoringConnecting : isConnecting + }); }; function getConnectionStatusSource() { @@ -48,6 +61,7 @@ } var service = { + reportConnecting: reportConnecting, reportFailedConnection: reportFailedConnection, reportSuccessfulConnection: reportSuccessfulConnection, getConnectionStatusSource: getConnectionStatusSource, @@ -57,8 +71,8 @@ return service; } - Service.$inject = ['toastService', 'scConfig']; + Service.$inject = ['toastService', 'connectionsManager', 'notifyService']; angular.module('services.connectivityNotifier', ['sc']) .service('connectivityNotifier', Service); -}(window, window.angular, window.jQuery)); \ No newline at end of file +}(window, window.angular, window.jQuery)); diff --git a/src/ServicePulse.Host/app/modules/monitoring/js/services/services.monitoring.js b/src/ServicePulse.Host/app/modules/monitoring/js/services/services.monitoring.js index 324510a6c..6369f3eb0 100644 --- a/src/ServicePulse.Host/app/modules/monitoring/js/services/services.monitoring.js +++ b/src/ServicePulse.Host/app/modules/monitoring/js/services/services.monitoring.js @@ -2,46 +2,34 @@ (function (window, angular, $, undefined) { 'use strict'; - function Service($http, rx, scConfig, uri, $q) { + function Service($http, rx, connectionsManager, uri, $q) { + + var mu = connectionsManager.getMonitoringUrl(); function createEndpointsSource(historyPeriod, refreshInterval) { return Rx.Observable.interval(refreshInterval).startWith(0) .flatMap(function (i) { - return Rx.Observable.fromArray(loadEndpointDataFromMonitoringService(historyPeriod)) - .flatMap(function (p) { - var o = Rx.Observable.fromPromise(p); - o = o.catch(Rx.Observable.empty()); - return o; - }); + return Rx.Observable.fromPromise(loadEndpointDataFromMonitoringService(historyPeriod)); }).selectMany(function (endpoints) { return endpoints; }); } function loadEndpointDataFromMonitoringService(historyPeriod) { - return scConfig.monitoring_urls.map(function (url) { - return $http.get(uri.join(url, 'monitored-endpoints') + '?history=' + historyPeriod) - .then(function (result) { - var sourceIndex = scConfig.monitoring_urls.indexOf(url); - - result.data.forEach(function (endpoint) { - endpoint.sourceIndex = sourceIndex; - }); - - return result.data.length !== 0 - ? result.data - : [{empty: true, sourceIndex: sourceIndex}]; - }, + return $http.get(uri.join(mu, 'monitored-endpoints') + '?history=' + historyPeriod) + .then(function (result) { + return result.data.length !== 0 + ? result.data + : [{ empty: true }]; + }, (error) => { - var sourceIndex = scConfig.monitoring_urls.indexOf(url); - return [{ error: error, sourceIndex: sourceIndex }]; - } - ); - }); + return [{ error: error }]; + } + ); } - function loadEndpointDetailsFromMonitoringService(endpointName, sourceIndex, historyPeriod) { - return $http.get(uri.join(scConfig.monitoring_urls[sourceIndex], 'monitored-endpoints', endpointName) + "?history=" + historyPeriod) + function loadEndpointDetailsFromMonitoringService(endpointName, historyPeriod) { + return $http.get(uri.join(mu, 'monitored-endpoints', endpointName) + "?history=" + historyPeriod) .then(function (result) { filterOutSystemMessage(result.data); return result.data; @@ -57,17 +45,15 @@ }); } - function createEndpointDetailsSource(endpointName, sourceIndex, historyPeriod, refreshInterval) { + function createEndpointDetailsSource(endpointName, historyPeriod, refreshInterval) { return Rx.Observable.interval(refreshInterval).startWith(0) .flatMap(function (i) { - return Rx.Observable.fromPromise(loadEndpointDetailsFromMonitoringService(endpointName, sourceIndex, historyPeriod)); + return Rx.Observable.fromPromise(loadEndpointDetailsFromMonitoringService(endpointName, historyPeriod)); }); } function getMonitoredEndpoints() { - return scConfig.monitoring_urls.map(function (url) { - return $http.get(uri.join(url, 'monitored-endpoints') + '?history=1'); - }); + return $http.get(uri.join(mu, 'monitored-endpoints') + '?history=1'); } var service = { @@ -79,7 +65,7 @@ return service; } - Service.$inject = ['$http', 'rx', 'scConfig', 'uri', '$q', 'toastService']; + Service.$inject = ['$http', 'rx', 'connectionsManager', 'uri', '$q', 'toastService']; angular.module('services.monitoringService', ['sc']) .service('monitoringService', Service); diff --git a/src/ServicePulse.Host/app/modules/monitoring/views/endpoint_details.html b/src/ServicePulse.Host/app/modules/monitoring/views/endpoint_details.html index c8ff06521..0e79603ee 100644 --- a/src/ServicePulse.Host/app/modules/monitoring/views/endpoint_details.html +++ b/src/ServicePulse.Host/app/modules/monitoring/views/endpoint_details.html @@ -22,7 +22,7 @@

    - + {{endpoint.errorCount | metricslargenumber}} @@ -275,7 +275,7 @@

    - + {{instance.errorCount | metricslargenumber}} diff --git a/src/ServicePulse.Host/app/modules/monitoring/views/monitoring_not_available.html b/src/ServicePulse.Host/app/modules/monitoring/views/monitoring_not_available.html index c13cbdf80..8c796a9c3 100644 --- a/src/ServicePulse.Host/app/modules/monitoring/views/monitoring_not_available.html +++ b/src/ServicePulse.Host/app/modules/monitoring/views/monitoring_not_available.html @@ -3,10 +3,10 @@

    Endpoint monitoring not available

    Monitoring is not available due to one or more of these reasons:

    • the monitoring server is not configured or is unavailable
    • -
    • the monitoring plugin is not installed on the endpoints to be monitored
    • +
    • no endpoints with monitoring plugin available yet
    • endpoints without the monitoring plugin do not have auditing enabled
    diff --git a/src/ServicePulse.Host/app/modules/shell/js/services/service.toast.js b/src/ServicePulse.Host/app/modules/shell/js/services/service.toast.js index e3fdfab05..e2c52fb41 100644 --- a/src/ServicePulse.Host/app/modules/shell/js/services/service.toast.js +++ b/src/ServicePulse.Host/app/modules/shell/js/services/service.toast.js @@ -4,33 +4,42 @@ function ToastService(toaster) { this.showToast = function(text, type, title, sticky) { sticky = sticky || false; + const toastId = Math.random(); + toaster.pop({ type: type || 'info', title: title, body: text, bodyOutputType: 'trustedHtml', timeout: sticky ? 0 : 5000, - showCloseButton: sticky + showCloseButton: sticky, + toastId }); + + return toastId; }; this.showInfo = function(text, title, sticky) { - this.showToast(text, 'info', title || 'Info', sticky); + return this.showToast(text, 'info', title || 'Info', sticky); }; this.showError = function(text, sticky) { if (sticky === undefined) { sticky = true; } - this.showToast(text, 'error', 'Error', sticky); + return this.showToast(text, 'error', 'Error', sticky); }; this.showWarning = function(text, sticky, showTitle = true) { if (sticky === undefined) { sticky = true; } - this.showToast(text, 'warning', showTitle ? 'Warning' : '', sticky); + return this.showToast(text, 'warning', showTitle ? 'Warning' : '', sticky); }; + + this.clear = function(toastInstance) { + toaster.clear('*', toastInstance); + } } ToastService.$inject = [ diff --git a/src/ServicePulse.Host/package.json b/src/ServicePulse.Host/package.json index 6059f09e2..3c6a59121 100644 --- a/src/ServicePulse.Host/package.json +++ b/src/ServicePulse.Host/package.json @@ -25,6 +25,7 @@ "rx": "^4.1.0", "signalr": "^2.2.1", "ui-select": "^0.18.1", + "url-search-params-polyfill": "^5.0.0", "zeroclipboard": "^2.2.0" }, "devDependencies": { @@ -38,6 +39,7 @@ "file-loader": "^1.1.6", "html-loader": "^0.5.1", "http-server": "^0.11.1", + "npm-run-all": "^4.1.5", "style-loader": "^0.19.1", "url-loader": "^0.6.2", "webpack": "^3.8.1" @@ -46,8 +48,9 @@ "test": "cd ../ServicePulse.Host.Tests && npm test", "load": "npm install && node ./node_modules/webpack/bin/webpack.js --config app/modules/modules.webpackconfig.builder.js", "setup": "npm install && node ./node_modules/webpack/bin/webpack.js --config app/modules/modules.webpackconfig.js", - "serve": "start http-server ./app", - "webpack": "node ./node_modules/webpack/bin/webpack.js --config app/modules/modules.webpackconfig.js" + "serve": "http-server ./app", + "webpack": "node ./node_modules/webpack/bin/webpack.js --config app/modules/modules.webpackconfig.js", + "dev": "npm-run-all -p serve webpack" }, "author": "Particular Software", "license": "RPL-1.5" diff --git a/src/Setup/ServicePulse.aip b/src/Setup/ServicePulse.aip index b4c370dc5..73bb5129d 100644 --- a/src/Setup/ServicePulse.aip +++ b/src/Setup/ServicePulse.aip @@ -211,22 +211,21 @@ - + - - + + - - - - - - - - - + + + + + + + +