open-nomad/ui/app/services/system.js

166 lines
4.6 KiB
JavaScript
Raw Normal View History

import Service, { inject as service } from '@ember/service';
import { computed } from '@ember/object';
import { tracked } from '@glimmer/tracking';
import { alias } from '@ember/object/computed';
2017-09-19 14:47:10 +00:00
import PromiseObject from '../utils/classes/promise-object';
import PromiseArray from '../utils/classes/promise-array';
2017-09-19 14:47:10 +00:00
import { namespace } from '../adapters/application';
2018-11-02 05:07:58 +00:00
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() {
2019-03-26 07:46:44 +00:00
const token = this.token;
2017-09-19 14:47:10 +00:00
return PromiseObject.create({
promise: token
.authorizedRequest(`/${namespace}/status/leader`)
2017-09-19 14:47:10 +00:00
.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 => {
if (agent?.config?.Version) {
const { Version, VersionPrerelease, VersionMetadata } = agent.config.Version;
agent.version = Version;
if (VersionPrerelease) agent.version = `${agent.version}-${VersionPrerelease}`;
if (VersionMetadata) agent.version = `${agent.version}+${VersionMetadata}`;
}
return agent;
}),
});
}
@computed
get defaultRegion() {
2019-03-26 07:46:44 +00:00
const token = this.token;
return PromiseObject.create({
promise: token
.authorizedRawRequest(`/${namespace}/agent/members`)
.then(jsonWithDefault({}))
.then(json => {
return { region: json.ServerRegion };
}),
});
}
@computed
get regions() {
2019-03-26 07:46:44 +00:00
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() {
2019-03-26 07:46:44 +00:00
const namespaces = this.namespaces.toArray();
return namespaces.length && namespaces.some(namespace => namespace.get('id') !== 'default');
}
// The cachedNamespace is set on pages that have a namespaces filter.
// It is set so other pages that have a namespaces filter can default to
// what the previous namespaces filter page used rather than defaulting
// to 'default' or '*'.
@tracked cachedNamespace = null;
@task(function*() {
const emptyLicense = { License: { Features: [] } };
try {
return yield this.token
.authorizedRawRequest(`/${namespace}/operator/license`)
.then(jsonWithDefault(emptyLicense));
} catch (e) {
return emptyLicense;
}
})
fetchLicense;
ui: Change global search to use fuzzy search API (#10412) This updates the UI to use the new fuzzy search API. It’s a drop-in replacement so the / shortcut to jump to search is preserved, and results can be cycled through and chosen via arrow keys and the enter key. It doesn’t use everything returned by the API: * deployments and evaluations: these match by id, doesn’t seem like people would know those or benefit from quick navigation to them * namespaces: doesn’t seem useful as they currently function * scaling policies * tasks: the response doesn’t include an allocation id, which means they can’t be navigated to in the UI without an additional query * CSI volumes: aren’t actually returned by the API Since there’s no API to check the server configuration and know whether the feature has been disabled, this adds another query in route:application#beforeModel that acts as feature detection: if the attempt to query fails (500), the global search field is hidden. Upon having added another query on load, I realised that beforeModel was being triggered any time service:router#transitionTo was being called, which happens upon navigating to a search result, for instance, because of refreshModel being present on the region query parameter. This PR adds a check for transition.queryParamsOnly and skips rerunning the onload queries (token permissions check, license check, fuzzy search feature detection). Implementation notes: * there are changes to unrelated tests to ignore the on-load feature detection query * some lifecycle-related guards against undefined were required to address failures when navigating to an allocation * the minimum search length of 2 characters is hard-coded as there’s currently no way to determine min_term_length in the UI
2021-04-28 18:31:05 +00:00
@task(function*() {
try {
const request = yield this.token.authorizedRequest('/v1/search/fuzzy', {
method: 'POST',
body: JSON.stringify({
Text: 'feature-detection-query',
Context: 'namespaces',
}),
});
return request.ok;
} catch (e) {
return false;
}
})
checkFuzzySearchPresence;
@alias('fetchLicense.lastSuccessful.value') license;
ui: Change global search to use fuzzy search API (#10412) This updates the UI to use the new fuzzy search API. It’s a drop-in replacement so the / shortcut to jump to search is preserved, and results can be cycled through and chosen via arrow keys and the enter key. It doesn’t use everything returned by the API: * deployments and evaluations: these match by id, doesn’t seem like people would know those or benefit from quick navigation to them * namespaces: doesn’t seem useful as they currently function * scaling policies * tasks: the response doesn’t include an allocation id, which means they can’t be navigated to in the UI without an additional query * CSI volumes: aren’t actually returned by the API Since there’s no API to check the server configuration and know whether the feature has been disabled, this adds another query in route:application#beforeModel that acts as feature detection: if the attempt to query fails (500), the global search field is hidden. Upon having added another query on load, I realised that beforeModel was being triggered any time service:router#transitionTo was being called, which happens upon navigating to a search result, for instance, because of refreshModel being present on the region query parameter. This PR adds a check for transition.queryParamsOnly and skips rerunning the onload queries (token permissions check, license check, fuzzy search feature detection). Implementation notes: * there are changes to unrelated tests to ignore the on-load feature detection query * some lifecycle-related guards against undefined were required to address failures when navigating to an allocation * the minimum search length of 2 characters is hard-coded as there’s currently no way to determine min_term_length in the UI
2021-04-28 18:31:05 +00:00
@alias('checkFuzzySearchPresence.last.value') fuzzySearchEnabled;
@computed('license.License.Features.[]')
get features() {
return this.get('license.License.Features') || [];
}
}