e9e52e0dfe
This doesn’t include Ember Data, as we are still back on 3.12. Most changes are deprecation updates, linting fixes, and dependencies. It can be read commit-by-commit, though many of them are mechanical and skimmable. For the new linting exclusions, I’ve added them to the Tech Debt list. The decrease in test count is because linting is no longer included in ember test. There’s a new deprecation warning in the logs that can be fixed by updating Ember Power Select but when I tried that it caused it to render incorrectly, so I decided to ignore it for now and address it separately.
167 lines
4.5 KiB
JavaScript
167 lines
4.5 KiB
JavaScript
import Service, { inject as service } from '@ember/service';
|
|
import { computed } from '@ember/object';
|
|
import { alias } from '@ember/object/computed';
|
|
import PromiseObject from '../utils/classes/promise-object';
|
|
import PromiseArray from '../utils/classes/promise-array';
|
|
import { namespace } from '../adapters/application';
|
|
import jsonWithDefault from '../utils/json-with-default';
|
|
import classic from 'ember-classic-decorator';
|
|
import { task } from 'ember-concurrency';
|
|
|
|
@classic
|
|
export default class SystemService extends Service {
|
|
@service token;
|
|
@service store;
|
|
|
|
@computed('activeRegion')
|
|
get leader() {
|
|
const token = this.token;
|
|
|
|
return PromiseObject.create({
|
|
promise: token
|
|
.authorizedRequest(`/${namespace}/status/leader`)
|
|
.then(res => res.json())
|
|
.then(rpcAddr => ({ rpcAddr }))
|
|
.then(leader => {
|
|
// Dirty self so leader can be used as a dependent key
|
|
this.notifyPropertyChange('leader.rpcAddr');
|
|
return leader;
|
|
}),
|
|
});
|
|
}
|
|
|
|
@computed
|
|
get agent() {
|
|
const token = this.token;
|
|
return PromiseObject.create({
|
|
promise: token
|
|
.authorizedRawRequest(`/${namespace}/agent/self`)
|
|
.then(jsonWithDefault({}))
|
|
.then(agent => {
|
|
agent.version = agent.member?.Tags?.build || 'Unknown';
|
|
return agent;
|
|
}),
|
|
});
|
|
}
|
|
|
|
@computed
|
|
get defaultRegion() {
|
|
const token = this.token;
|
|
return PromiseObject.create({
|
|
promise: token
|
|
.authorizedRawRequest(`/${namespace}/agent/members`)
|
|
.then(jsonWithDefault({}))
|
|
.then(json => {
|
|
return { region: json.ServerRegion };
|
|
}),
|
|
});
|
|
}
|
|
|
|
@computed
|
|
get regions() {
|
|
const token = this.token;
|
|
|
|
return PromiseArray.create({
|
|
promise: token.authorizedRawRequest(`/${namespace}/regions`).then(jsonWithDefault([])),
|
|
});
|
|
}
|
|
|
|
@computed('regions.[]')
|
|
get activeRegion() {
|
|
const regions = this.regions;
|
|
const region = window.localStorage.nomadActiveRegion;
|
|
|
|
if (regions.includes(region)) {
|
|
return region;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
set activeRegion(value) {
|
|
if (value == null) {
|
|
window.localStorage.removeItem('nomadActiveRegion');
|
|
return;
|
|
} else {
|
|
// All localStorage values are strings. Stringify first so
|
|
// the return value is consistent with what is persisted.
|
|
const strValue = value + '';
|
|
window.localStorage.nomadActiveRegion = strValue;
|
|
}
|
|
}
|
|
|
|
@computed('regions.[]')
|
|
get shouldShowRegions() {
|
|
return this.get('regions.length') > 1;
|
|
}
|
|
|
|
@computed('activeRegion', 'defaultRegion.region', 'shouldShowRegions')
|
|
get shouldIncludeRegion() {
|
|
return this.shouldShowRegions && this.activeRegion !== this.get('defaultRegion.region');
|
|
}
|
|
|
|
@computed('activeRegion')
|
|
get namespaces() {
|
|
return PromiseArray.create({
|
|
promise: this.store.findAll('namespace').then(namespaces => namespaces.compact()),
|
|
});
|
|
}
|
|
|
|
@computed('namespaces.[]')
|
|
get shouldShowNamespaces() {
|
|
const namespaces = this.namespaces.toArray();
|
|
return namespaces.length && namespaces.some(namespace => namespace.get('id') !== 'default');
|
|
}
|
|
|
|
@computed('namespaces.[]')
|
|
get activeNamespace() {
|
|
const namespaceId = window.localStorage.nomadActiveNamespace || 'default';
|
|
const namespace = this.namespaces.findBy('id', namespaceId);
|
|
|
|
if (namespace) {
|
|
return namespace;
|
|
}
|
|
|
|
// If the namespace in localStorage is no longer in the cluster, it needs to
|
|
// be cleared from localStorage
|
|
window.localStorage.removeItem('nomadActiveNamespace');
|
|
return this.namespaces.findBy('id', 'default');
|
|
}
|
|
|
|
set activeNamespace(value) {
|
|
if (value == null) {
|
|
window.localStorage.removeItem('nomadActiveNamespace');
|
|
return;
|
|
} else if (typeof value === 'string') {
|
|
window.localStorage.nomadActiveNamespace = value;
|
|
} else {
|
|
window.localStorage.nomadActiveNamespace = value.get('name');
|
|
}
|
|
}
|
|
|
|
reset() {
|
|
this.set('activeNamespace', null);
|
|
this.notifyPropertyChange('namespaces');
|
|
}
|
|
|
|
@task(function*() {
|
|
const emptyLicense = { License: { Features: [] } };
|
|
|
|
try {
|
|
return yield this.token
|
|
.authorizedRawRequest(`/${namespace}/operator/license`)
|
|
.then(jsonWithDefault(emptyLicense));
|
|
} catch (e) {
|
|
return emptyLicense;
|
|
}
|
|
})
|
|
fetchLicense;
|
|
|
|
@alias('fetchLicense.lastSuccessful.value') license;
|
|
|
|
@computed('license.License.Features.[]')
|
|
get features() {
|
|
return this.get('license.License.Features') || [];
|
|
}
|
|
}
|