2014-04-25 13:49:36 -04:00
|
|
|
//
|
|
|
|
// A Consul service.
|
|
|
|
//
|
|
|
|
App.Service = Ember.Object.extend({
|
|
|
|
//
|
|
|
|
// The number of failing checks within the service.
|
|
|
|
//
|
|
|
|
failingChecks: function() {
|
2014-04-30 17:31:40 -04:00
|
|
|
// If the service was returned from `/v1/internal/ui/services`
|
|
|
|
// then we have a aggregated value which we can just grab
|
2014-04-30 14:02:20 -04:00
|
|
|
if (this.get('ChecksCritical') != undefined) {
|
|
|
|
return (this.get('ChecksCritical') + this.get('ChecksWarning'))
|
2014-04-30 17:31:40 -04:00
|
|
|
// Otherwise, we need to filter the child checks by both failing
|
|
|
|
// states
|
2014-04-30 14:02:20 -04:00
|
|
|
} else {
|
2014-06-03 13:53:22 -04:00
|
|
|
var checks = this.get('Checks');
|
|
|
|
return (checks.filterBy('Status', 'critical').get('length') +
|
|
|
|
checks.filterBy('Status', 'warning').get('length'))
|
2014-04-30 14:02:20 -04:00
|
|
|
}
|
2014-04-29 15:24:32 -04:00
|
|
|
}.property('Checks'),
|
2014-04-25 13:49:36 -04:00
|
|
|
|
|
|
|
//
|
|
|
|
// The number of passing checks within the service.
|
|
|
|
//
|
|
|
|
passingChecks: function() {
|
2014-04-30 17:31:40 -04:00
|
|
|
// If the service was returned from `/v1/internal/ui/services`
|
|
|
|
// then we have a aggregated value which we can just grab
|
2014-04-30 14:02:20 -04:00
|
|
|
if (this.get('ChecksPassing') != undefined) {
|
|
|
|
return this.get('ChecksPassing')
|
2014-04-30 17:31:40 -04:00
|
|
|
// Otherwise, we need to filter the child checks by both failing
|
|
|
|
// states
|
2014-04-30 14:02:20 -04:00
|
|
|
} else {
|
|
|
|
return this.get('Checks').filterBy('Status', 'passing').get('length');
|
|
|
|
}
|
2014-04-29 15:24:32 -04:00
|
|
|
}.property('Checks'),
|
2014-04-25 13:49:36 -04:00
|
|
|
|
|
|
|
//
|
|
|
|
// The formatted message returned for the user which represents the
|
|
|
|
// number of checks failing or passing. Returns `1 passing` or `2 failing`
|
|
|
|
//
|
|
|
|
checkMessage: function() {
|
|
|
|
if (this.get('hasFailingChecks') === false) {
|
|
|
|
return this.get('passingChecks') + ' passing';
|
|
|
|
} else {
|
|
|
|
return this.get('failingChecks') + ' failing';
|
|
|
|
}
|
2014-04-29 15:24:32 -04:00
|
|
|
}.property('Checks'),
|
2014-04-25 13:49:36 -04:00
|
|
|
|
2014-06-04 15:44:17 -04:00
|
|
|
nodes: function() {
|
|
|
|
return (this.get('Nodes'))
|
|
|
|
}.property('Nodes'),
|
|
|
|
|
2014-04-25 13:49:36 -04:00
|
|
|
//
|
|
|
|
// Boolean of whether or not there are failing checks in the service.
|
|
|
|
// This is used to set color backgrounds and so on.
|
|
|
|
//
|
|
|
|
hasFailingChecks: function() {
|
|
|
|
return (this.get('failingChecks') > 0);
|
2014-06-02 10:35:46 -04:00
|
|
|
}.property('Checks'),
|
|
|
|
|
2014-06-03 13:53:22 -04:00
|
|
|
//
|
|
|
|
// Key used for filtering through an array of this model, i.e s
|
|
|
|
// searching
|
|
|
|
//
|
2014-06-02 10:35:46 -04:00
|
|
|
filterKey: function() {
|
|
|
|
return this.get('Name')
|
2014-06-03 13:53:22 -04:00
|
|
|
}.property('Name'),
|
2014-04-25 13:49:36 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
//
|
|
|
|
// A Consul Node
|
|
|
|
//
|
|
|
|
App.Node = Ember.Object.extend({
|
|
|
|
//
|
|
|
|
// The number of failing checks within the service.
|
|
|
|
//
|
|
|
|
failingChecks: function() {
|
2014-04-30 17:31:40 -04:00
|
|
|
var checks = this.get('Checks');
|
|
|
|
// We view both warning and critical as failing
|
|
|
|
return (checks.filterBy('Status', 'critical').get('length') +
|
|
|
|
checks.filterBy('Status', 'warning').get('length'))
|
2014-04-29 15:30:00 -04:00
|
|
|
}.property('Checks'),
|
2014-04-25 13:49:36 -04:00
|
|
|
|
|
|
|
//
|
|
|
|
// The number of passing checks within the service.
|
|
|
|
//
|
|
|
|
passingChecks: function() {
|
|
|
|
return this.get('Checks').filterBy('Status', 'passing').get('length');
|
2014-04-29 15:24:32 -04:00
|
|
|
}.property('Checks'),
|
2014-04-25 13:49:36 -04:00
|
|
|
|
|
|
|
//
|
|
|
|
// The formatted message returned for the user which represents the
|
|
|
|
// number of checks failing or passing. Returns `1 passing` or `2 failing`
|
|
|
|
//
|
|
|
|
checkMessage: function() {
|
|
|
|
if (this.get('hasFailingChecks') === false) {
|
|
|
|
return this.get('passingChecks') + ' passing';
|
|
|
|
} else {
|
|
|
|
return this.get('failingChecks') + ' failing';
|
|
|
|
}
|
2014-04-29 15:24:32 -04:00
|
|
|
}.property('Checks'),
|
2014-04-25 13:49:36 -04:00
|
|
|
|
|
|
|
//
|
|
|
|
// Boolean of whether or not there are failing checks in the service.
|
|
|
|
// This is used to set color backgrounds and so on.
|
|
|
|
//
|
|
|
|
hasFailingChecks: function() {
|
|
|
|
return (this.get('failingChecks') > 0);
|
2014-06-02 10:35:46 -04:00
|
|
|
}.property('Checks'),
|
|
|
|
|
|
|
|
//
|
|
|
|
// The number of services on the node
|
|
|
|
//
|
|
|
|
numServices: function() {
|
|
|
|
return (this.get('Services').length)
|
|
|
|
}.property('Services'),
|
2014-06-04 15:44:17 -04:00
|
|
|
// The number of services on the node
|
|
|
|
//
|
|
|
|
|
|
|
|
services: function() {
|
|
|
|
return (this.get('Services'))
|
|
|
|
}.property('Services'),
|
2014-06-02 10:35:46 -04:00
|
|
|
|
|
|
|
filterKey: function() {
|
|
|
|
return this.get('Node')
|
2014-06-02 11:49:01 -04:00
|
|
|
}.property('Node'),
|
2014-06-03 13:53:22 -04:00
|
|
|
|
|
|
|
//
|
|
|
|
// Returns a combined and distinct list of the tags on the services
|
|
|
|
// running on the node
|
|
|
|
//
|
|
|
|
nodeTags: function() {
|
|
|
|
var tags = [];
|
|
|
|
|
|
|
|
// Collect the services tags
|
|
|
|
this.get('Services').map(function(Service){
|
|
|
|
tags.push(Service.Tags)
|
|
|
|
})
|
|
|
|
|
|
|
|
// strip nulls
|
|
|
|
tags = tags.filter(function(n){ return n != undefined });
|
|
|
|
|
|
|
|
// only keep unique tags and convert to comma sep
|
|
|
|
return tags.uniq().join(', ')
|
|
|
|
}.property('Services')
|
2014-04-25 13:49:36 -04:00
|
|
|
});
|
|
|
|
|
2014-04-29 13:06:26 -04:00
|
|
|
|
|
|
|
//
|
|
|
|
// A key/value object
|
|
|
|
//
|
2014-04-30 10:09:41 -04:00
|
|
|
App.Key = Ember.Object.extend(Ember.Validations.Mixin, {
|
2014-04-30 17:31:40 -04:00
|
|
|
// Validates using the Ember.Valdiations library
|
2014-04-30 10:09:41 -04:00
|
|
|
validations: {
|
2014-04-30 19:22:07 -04:00
|
|
|
Key: { presence: true }
|
2014-04-30 10:09:41 -04:00
|
|
|
},
|
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// Boolean if the key is valid
|
2014-04-30 15:02:31 -04:00
|
|
|
keyValid: Ember.computed.empty('errors.Key'),
|
2014-04-30 17:31:40 -04:00
|
|
|
// Boolean if the value is valid
|
2014-04-30 15:02:31 -04:00
|
|
|
valueValid: Ember.computed.empty('errors.Value'),
|
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// The key with the parent removed.
|
|
|
|
// This is only for display purposes, and used for
|
|
|
|
// showing the key name inside of a nested key.
|
2014-04-30 15:02:31 -04:00
|
|
|
keyWithoutParent: function() {
|
|
|
|
return (this.get('Key').replace(this.get('parentKey'), ''));
|
|
|
|
}.property('Key'),
|
2014-04-30 12:15:54 -04:00
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// Boolean if the key is a "folder" or not, i.e is a nested key
|
|
|
|
// that feels like a folder. Used for UI
|
2014-04-29 13:06:26 -04:00
|
|
|
isFolder: function() {
|
2014-04-30 19:22:07 -04:00
|
|
|
if (this.get('Key') === undefined) {
|
|
|
|
return false;
|
|
|
|
};
|
2014-05-04 23:05:00 +02:00
|
|
|
return (this.get('Key').slice(-1) === '/')
|
2014-04-30 16:30:14 -04:00
|
|
|
}.property('Key'),
|
|
|
|
|
2014-06-06 12:21:36 -04:00
|
|
|
// Boolean if the key is locked or now
|
|
|
|
isLocked: function() {
|
|
|
|
if (!this.get('Session')) {
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}.property('Session'),
|
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// Determines what route to link to. If it's a folder,
|
|
|
|
// it will link to kv.show. Otherwise, kv.edit
|
2014-04-29 16:32:38 -04:00
|
|
|
linkToRoute: function() {
|
2014-05-04 23:05:00 +02:00
|
|
|
if (this.get('Key').slice(-1) === '/') {
|
2014-04-29 16:32:38 -04:00
|
|
|
return 'kv.show'
|
|
|
|
} else {
|
|
|
|
return 'kv.edit'
|
|
|
|
}
|
2014-04-30 15:02:31 -04:00
|
|
|
}.property('Key'),
|
2014-04-29 16:32:38 -04:00
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// The base64 decoded value of the key.
|
|
|
|
// if you set on this key, it will update
|
|
|
|
// the key.Value
|
2014-04-30 16:30:14 -04:00
|
|
|
valueDecoded: function(key, value) {
|
2014-04-30 17:31:40 -04:00
|
|
|
|
|
|
|
// setter
|
2014-04-30 16:30:14 -04:00
|
|
|
if (arguments.length > 1) {
|
2014-04-30 17:31:40 -04:00
|
|
|
this.set('Value', value);
|
2014-04-30 18:31:45 -04:00
|
|
|
return value;
|
2014-04-30 16:30:14 -04:00
|
|
|
}
|
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// getter
|
|
|
|
|
|
|
|
// If the value is null, we don't
|
|
|
|
// want to try and base64 decode it, so just return
|
2014-04-30 16:30:14 -04:00
|
|
|
if (this.get('Value') === null) {
|
|
|
|
return "";
|
|
|
|
}
|
2014-04-30 17:31:40 -04:00
|
|
|
|
|
|
|
// base64 decode the value
|
2014-04-30 16:30:14 -04:00
|
|
|
return window.atob(this.get('Value'));
|
|
|
|
}.property('Value'),
|
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
|
|
|
|
// An array of the key broken up by the /
|
2014-04-29 14:49:07 -04:00
|
|
|
keyParts: function() {
|
2014-04-30 15:02:31 -04:00
|
|
|
var key = this.get('Key');
|
2014-04-29 13:34:13 -04:00
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// If the key is a folder, remove the last
|
|
|
|
// slash to split properly
|
2014-04-29 14:49:07 -04:00
|
|
|
if (key.slice(-1) == "/") {
|
|
|
|
key = key.substring(0, key.length - 1);
|
|
|
|
}
|
2014-04-30 17:31:40 -04:00
|
|
|
|
2014-04-29 14:49:07 -04:00
|
|
|
return key.split('/');
|
2014-04-30 15:02:31 -04:00
|
|
|
}.property('Key'),
|
2014-04-29 14:49:07 -04:00
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// The parent Key is the key one level above this.Key
|
|
|
|
// key: baz/bar/foobar/
|
|
|
|
// grandParent: baz/bar/
|
2014-04-29 14:49:07 -04:00
|
|
|
parentKey: function() {
|
2014-04-29 15:24:32 -04:00
|
|
|
var parts = this.get('keyParts').toArray();
|
2014-04-29 14:49:07 -04:00
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// Remove the last item, essentially going up a level
|
|
|
|
// in hiearchy
|
2014-04-29 14:49:07 -04:00
|
|
|
parts.pop();
|
|
|
|
|
|
|
|
return parts.join("/") + "/";
|
2014-04-30 15:02:31 -04:00
|
|
|
}.property('Key'),
|
2014-04-29 14:49:07 -04:00
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// The grandParent Key is the key two levels above this.Key
|
|
|
|
// key: baz/bar/foobar/
|
|
|
|
// grandParent: baz/
|
2014-04-29 14:49:07 -04:00
|
|
|
grandParentKey: function() {
|
2014-04-29 15:24:32 -04:00
|
|
|
var parts = this.get('keyParts').toArray();
|
2014-04-29 13:34:13 -04:00
|
|
|
|
2014-04-30 17:31:40 -04:00
|
|
|
// Remove the last two items, jumping two levels back
|
2014-04-29 14:49:07 -04:00
|
|
|
parts.pop();
|
|
|
|
parts.pop();
|
2014-04-29 13:06:26 -04:00
|
|
|
|
2014-04-29 14:49:07 -04:00
|
|
|
return parts.join("/") + "/";
|
2014-04-30 15:02:31 -04:00
|
|
|
}.property('Key')
|
2014-04-29 13:06:26 -04:00
|
|
|
});
|