mirror of
https://github.com/status-im/consul.git
synced 2025-01-10 05:45:46 +00:00
5fb9df1640
* Adding explicit MPL license for sub-package This directory and its subdirectories (packages) contain files licensed with the MPLv2 `LICENSE` file in this directory and are intentionally licensed separately from the BSL `LICENSE` file at the root of this repository. * Adding explicit MPL license for sub-package This directory and its subdirectories (packages) contain files licensed with the MPLv2 `LICENSE` file in this directory and are intentionally licensed separately from the BSL `LICENSE` file at the root of this repository. * Updating the license from MPL to Business Source License Going forward, this project will be licensed under the Business Source License v1.1. Please see our blog post for more details at <Blog URL>, FAQ at www.hashicorp.com/licensing-faq, and details of the license at www.hashicorp.com/bsl. * add missing license headers * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 --------- Co-authored-by: hashicorp-copywrite[bot] <110428419+hashicorp-copywrite[bot]@users.noreply.github.com>
172 lines
4.9 KiB
JavaScript
172 lines
4.9 KiB
JavaScript
/**
|
|
* Copyright (c) HashiCorp, Inc.
|
|
* SPDX-License-Identifier: BUSL-1.1
|
|
*/
|
|
|
|
import Model, { attr, belongsTo } from '@ember-data/model';
|
|
import { computed } from '@ember/object';
|
|
import { tracked } from '@glimmer/tracking';
|
|
import { fragment } from 'ember-data-model-fragments/attributes';
|
|
import replace, { nullValue } from 'consul-ui/decorators/replace';
|
|
|
|
export const PRIMARY_KEY = 'uid';
|
|
export const SLUG_KEY = 'Name,PeerName';
|
|
|
|
export const Collection = class Collection {
|
|
@tracked items;
|
|
|
|
constructor(items) {
|
|
this.items = items;
|
|
}
|
|
|
|
get ExternalSources() {
|
|
const items = this.items.reduce(function (prev, item) {
|
|
return prev.concat(item.ExternalSources || []);
|
|
}, []);
|
|
// unique, non-empty values, alpha sort
|
|
return [...new Set(items)].filter(Boolean).sort();
|
|
}
|
|
// TODO: Think about when this/collections is worthwhile using and explain
|
|
// when and when not somewhere in the docs
|
|
get Partitions() {
|
|
// unique, non-empty values, alpha sort
|
|
return [...new Set(this.items.map((item) => item.Partition))].sort();
|
|
}
|
|
};
|
|
export default class Service extends Model {
|
|
@attr('string') uid;
|
|
@attr('string') Name;
|
|
|
|
@attr('string') Datacenter;
|
|
@attr('string') Namespace;
|
|
@attr('string') Partition;
|
|
@attr('string') Kind;
|
|
@replace('', undefined) @attr('string') PeerName;
|
|
@attr('number') ChecksPassing;
|
|
@attr('number') ChecksCritical;
|
|
@attr('number') ChecksWarning;
|
|
@attr('number') InstanceCount;
|
|
@attr('boolean') ConnectedWithGateway;
|
|
@attr('boolean') ConnectedWithProxy;
|
|
@attr({ defaultValue: () => [] }) Resources; // []
|
|
@attr('number') SyncTime;
|
|
@attr('number') CreateIndex;
|
|
@attr('number') ModifyIndex;
|
|
|
|
@nullValue([]) @attr({ defaultValue: () => [] }) Tags;
|
|
|
|
@attr() Nodes; // array
|
|
@attr() Proxy; // Service
|
|
@fragment('gateway-config') GatewayConfig;
|
|
@nullValue([]) @attr() ExternalSources; // array
|
|
@attr() Meta; // {}
|
|
|
|
@attr() meta; // {}
|
|
|
|
@belongsTo({ async: false }) peer;
|
|
|
|
@computed('peer', 'InstanceCount')
|
|
get isZeroCountButPeered() {
|
|
return this.peer && this.InstanceCount === 0;
|
|
}
|
|
|
|
@computed('peer.State')
|
|
get peerIsFailing() {
|
|
return this.peer && this.peer.State === 'FAILING';
|
|
}
|
|
|
|
@computed('ChecksPassing', 'ChecksWarning', 'ChecksCritical')
|
|
get ChecksTotal() {
|
|
return this.ChecksPassing + this.ChecksWarning + this.ChecksCritical;
|
|
}
|
|
|
|
@computed('MeshChecksPassing', 'MeshChecksWarning', 'MeshChecksCritical')
|
|
get MeshChecksTotal() {
|
|
return this.MeshChecksPassing + this.MeshChecksWarning + this.MeshChecksCritical;
|
|
}
|
|
|
|
/* Mesh properties involve both the service and the associated proxy */
|
|
@computed('ConnectedWithProxy', 'ConnectedWithGateway')
|
|
get MeshEnabled() {
|
|
return this.ConnectedWithProxy || this.ConnectedWithGateway;
|
|
}
|
|
|
|
@computed('MeshEnabled', 'Kind')
|
|
get InMesh() {
|
|
return this.MeshEnabled || (this.Kind || '').length > 0;
|
|
}
|
|
|
|
@computed(
|
|
'MeshChecksPassing',
|
|
'MeshChecksWarning',
|
|
'MeshChecksCritical',
|
|
'isZeroCountButPeered',
|
|
'peerIsFailing'
|
|
)
|
|
get MeshStatus() {
|
|
switch (true) {
|
|
case this.isZeroCountButPeered:
|
|
return 'unknown';
|
|
case this.peerIsFailing:
|
|
return 'unknown';
|
|
case this.MeshChecksCritical !== 0:
|
|
return 'critical';
|
|
case this.MeshChecksWarning !== 0:
|
|
return 'warning';
|
|
case this.MeshChecksPassing !== 0:
|
|
return 'passing';
|
|
default:
|
|
return 'empty';
|
|
}
|
|
}
|
|
|
|
@computed('isZeroCountButPeered', 'peerIsFailing', 'MeshStatus')
|
|
get healthTooltipText() {
|
|
const { MeshStatus, isZeroCountButPeered, peerIsFailing } = this;
|
|
if (isZeroCountButPeered) {
|
|
return 'This service currently has 0 instances. Check with the operator of its peer to make sure this is expected behavior.';
|
|
}
|
|
if (peerIsFailing) {
|
|
return 'This peer is out of sync, so the current health statuses of its services are unknown.';
|
|
}
|
|
if (MeshStatus === 'critical') {
|
|
return 'At least one health check on one instance is failing.';
|
|
}
|
|
if (MeshStatus === 'warning') {
|
|
return 'At least one health check on one instance has a warning.';
|
|
}
|
|
if (MeshStatus == 'passing') {
|
|
return 'All health checks are passing.';
|
|
}
|
|
return 'There are no health checks';
|
|
}
|
|
|
|
@computed('ChecksPassing', 'Proxy.ChecksPassing')
|
|
get MeshChecksPassing() {
|
|
let proxyCount = 0;
|
|
if (typeof this.Proxy !== 'undefined') {
|
|
proxyCount = this.Proxy.ChecksPassing;
|
|
}
|
|
return this.ChecksPassing + proxyCount;
|
|
}
|
|
|
|
@computed('ChecksWarning', 'Proxy.ChecksWarning')
|
|
get MeshChecksWarning() {
|
|
let proxyCount = 0;
|
|
if (typeof this.Proxy !== 'undefined') {
|
|
proxyCount = this.Proxy.ChecksWarning;
|
|
}
|
|
return this.ChecksWarning + proxyCount;
|
|
}
|
|
|
|
@computed('ChecksCritical', 'Proxy.ChecksCritical')
|
|
get MeshChecksCritical() {
|
|
let proxyCount = 0;
|
|
if (typeof this.Proxy !== 'undefined') {
|
|
proxyCount = this.Proxy.ChecksCritical;
|
|
}
|
|
return this.ChecksCritical + proxyCount;
|
|
}
|
|
/**/
|
|
}
|