mirror of
https://github.com/status-im/consul.git
synced 2025-01-25 05:00:32 +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>
374 lines
11 KiB
JavaScript
374 lines
11 KiB
JavaScript
/**
|
|
* Copyright (c) HashiCorp, Inc.
|
|
* SPDX-License-Identifier: BUSL-1.1
|
|
*/
|
|
|
|
import { env } from 'consul-ui/env';
|
|
const OPTIONAL = {};
|
|
if (env('CONSUL_PARTITIONS_ENABLED')) {
|
|
OPTIONAL.partition = /^_([a-zA-Z0-9]([a-zA-Z0-9-]{0,62}[a-zA-Z0-9])?)$/;
|
|
}
|
|
|
|
if (env('CONSUL_NSPACES_ENABLED')) {
|
|
OPTIONAL.nspace = /^~([a-zA-Z0-9]([a-zA-Z0-9-]{0,62}[a-zA-Z0-9])?)$/;
|
|
}
|
|
|
|
OPTIONAL.peer = /^:([a-zA-Z0-9]([a-zA-Z0-9-]{0,62}[a-zA-Z0-9])?)$/;
|
|
|
|
const trailingSlashRe = /\/$/;
|
|
|
|
// see below re: ember double slashes
|
|
// const moreThan1SlashRe = /\/{2,}/g;
|
|
|
|
const _uuid = function () {
|
|
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, (c) => {
|
|
const r = (Math.random() * 16) | 0;
|
|
return (c === 'x' ? r : (r & 3) | 8).toString(16);
|
|
});
|
|
};
|
|
|
|
// let popstateFired = false;
|
|
/**
|
|
* Register a callback to be invoked whenever the browser history changes,
|
|
* including using forward and back buttons.
|
|
*/
|
|
const route = function (e) {
|
|
const path = e.state.path;
|
|
const url = this.getURLForTransition(path);
|
|
// Ignore initial page load popstate event in Chrome
|
|
// if (!popstateFired) {
|
|
// popstateFired = true;
|
|
// if (url === this._previousURL) {
|
|
// return;
|
|
// }
|
|
// }
|
|
if (url === this._previousURL) {
|
|
if (path === this._previousPath) {
|
|
return;
|
|
}
|
|
this._previousPath = e.state.path;
|
|
// async
|
|
this.container.lookup('route:application').refresh();
|
|
}
|
|
if (typeof this.callback === 'function') {
|
|
// TODO: Can we use `settled` or similar to make this `route` method async?
|
|
// not async
|
|
this.callback(url);
|
|
}
|
|
// used for webkit workaround
|
|
this._previousURL = url;
|
|
this._previousPath = e.state.path;
|
|
};
|
|
export default class FSMWithOptionalLocation {
|
|
// extend FSMLocation
|
|
implementation = 'fsm-with-optional';
|
|
|
|
baseURL = '';
|
|
/**
|
|
* Set from router:main._setupLocation (-internals/routing/lib/system/router)
|
|
* Will be pre-pended to path upon state change
|
|
*/
|
|
rootURL = '/';
|
|
|
|
/**
|
|
* Path is the 'application path' i.e. the path/URL with no root/base URLs
|
|
* but potentially with optional parameters (these are remove when getURL is called)
|
|
*/
|
|
path = '/';
|
|
|
|
/**
|
|
* Sneaky undocumented property used in ember's main router used to skip any
|
|
* setup of location from the main router. We currently don't need this but
|
|
* document it here incase we ever do.
|
|
*/
|
|
cancelRouterSetup = false;
|
|
|
|
/**
|
|
* Used to store our 'optional' segments should we have any
|
|
*/
|
|
optional = {};
|
|
|
|
static create() {
|
|
return new this(...arguments);
|
|
}
|
|
|
|
constructor(owner, doc, env) {
|
|
this.container = Object.entries(owner)[0][1];
|
|
|
|
// add the route/state change handler
|
|
this.route = route.bind(this);
|
|
|
|
this.doc = typeof doc === 'undefined' ? this.container.lookup('service:-document') : doc;
|
|
this.env = typeof env === 'undefined' ? this.container.lookup('service:env') : env;
|
|
|
|
const base = this.doc.querySelector('base[href]');
|
|
if (base !== null) {
|
|
this.baseURL = base.getAttribute('href');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
* Called from router:main._setupLocation (-internals/routing/lib/system/router)
|
|
* Used to set state on first call to setURL
|
|
*/
|
|
initState() {
|
|
this.location = this.location || this.doc.defaultView.location;
|
|
this.machine = this.machine || this.doc.defaultView.history;
|
|
this.doc.defaultView.addEventListener('popstate', this.route);
|
|
|
|
const state = this.machine.state;
|
|
const url = this.getURL();
|
|
const href = this.formatURL(url);
|
|
|
|
if (state && state.path === href) {
|
|
// preserve existing state
|
|
// used for webkit workaround, since there will be no initial popstate event
|
|
this._previousPath = href;
|
|
this._previousURL = url;
|
|
} else {
|
|
this.dispatch('replace', href);
|
|
}
|
|
}
|
|
|
|
getURLFrom(url) {
|
|
// remove trailing slashes if they exist
|
|
url = url || this.location.pathname;
|
|
this.rootURL = this.rootURL.replace(trailingSlashRe, '');
|
|
this.baseURL = this.baseURL.replace(trailingSlashRe, '');
|
|
// remove baseURL and rootURL from start of path
|
|
return url
|
|
.replace(new RegExp(`^${this.baseURL}(?=/|$)`), '')
|
|
.replace(new RegExp(`^${this.rootURL}(?=/|$)`), '');
|
|
// ember default locations remove double slashes here e.g. '//'
|
|
// .replace(moreThan1SlashRe, '/'); // remove extra slashes
|
|
}
|
|
|
|
getURLForTransition(url) {
|
|
this.optional = {};
|
|
url = this.getURLFrom(url)
|
|
.split('/')
|
|
.filter((item, i) => {
|
|
if (i < 3) {
|
|
let found = false;
|
|
Object.entries(OPTIONAL).reduce((prev, [key, re]) => {
|
|
const res = re.exec(item);
|
|
if (res !== null) {
|
|
prev[key] = {
|
|
value: item,
|
|
match: res[1],
|
|
};
|
|
found = true;
|
|
}
|
|
return prev;
|
|
}, this.optional);
|
|
return !found;
|
|
}
|
|
return true;
|
|
})
|
|
.join('/');
|
|
return url;
|
|
}
|
|
|
|
optionalParams() {
|
|
let optional = this.optional || {};
|
|
return ['partition', 'nspace', 'peer'].reduce((prev, item) => {
|
|
let value = '';
|
|
if (typeof optional[item] !== 'undefined') {
|
|
value = optional[item].match;
|
|
}
|
|
prev[item] = value;
|
|
return prev;
|
|
}, {});
|
|
}
|
|
|
|
// public entrypoints for app hrefs/URLs
|
|
|
|
// visit and transitionTo can't be async/await as they return promise-like
|
|
// non-promises that get re-wrapped by the addition of async/await
|
|
visit() {
|
|
return this.transitionTo(...arguments);
|
|
}
|
|
|
|
/**
|
|
* Turns a routeName into a full URL string for anchor hrefs etc.
|
|
*/
|
|
hrefTo(routeName, params, _hash) {
|
|
// copy to always work with a new hash even when helper persists hash
|
|
const hash = { ..._hash };
|
|
|
|
if (typeof hash.dc !== 'undefined') {
|
|
delete hash.dc;
|
|
}
|
|
|
|
if (typeof hash.nspace !== 'undefined') {
|
|
hash.nspace = `~${hash.nspace}`;
|
|
}
|
|
if (typeof hash.partition !== 'undefined') {
|
|
hash.partition = `_${hash.partition}`;
|
|
}
|
|
if (typeof hash.peer !== 'undefined') {
|
|
hash.peer = `:${hash.peer}`;
|
|
}
|
|
|
|
if (typeof this.router === 'undefined') {
|
|
this.router = this.container.lookup('router:main');
|
|
}
|
|
let withOptional = true;
|
|
switch (true) {
|
|
case routeName === 'settings':
|
|
case routeName.startsWith('docs.'):
|
|
withOptional = false;
|
|
break;
|
|
}
|
|
if (this.router.currentRouteName.startsWith('docs.')) {
|
|
// If we are in docs, then add a default dc as there won't be one in the
|
|
// URL
|
|
params.unshift(env('CONSUL_DATACENTER_PRIMARY'));
|
|
if (routeName.startsWith('dc')) {
|
|
// if its an app URL replace it with debugging instead of linking
|
|
return `console://${routeName} <= ${JSON.stringify(params)}`;
|
|
}
|
|
}
|
|
const router = this.router._routerMicrolib;
|
|
let url;
|
|
try {
|
|
url = router.generate(routeName, ...params, {
|
|
queryParams: {},
|
|
});
|
|
} catch (e) {
|
|
// if the previous generation throws due to params not being available
|
|
// its probably due to the view wanting to re-render even though we are
|
|
// leaving the view and the router has already moved the state to old
|
|
// state so try again with the old state to avoid errors
|
|
params = Object.values(router.oldState.params).reduce((prev, item) => {
|
|
return prev.concat(Object.keys(item).length > 0 ? item : []);
|
|
}, []);
|
|
url = router.generate(routeName, ...params);
|
|
}
|
|
return this.formatURL(url, hash, withOptional);
|
|
}
|
|
|
|
/**
|
|
* Takes a full browser URL including rootURL and optional (a full href) and
|
|
* performs an ember transition/refresh and browser location update using that
|
|
*/
|
|
transitionTo(url) {
|
|
if (this.router.currentRouteName.startsWith('docs') && url.startsWith('console://')) {
|
|
console.info(`location.transitionTo: ${url.substr(10)}`);
|
|
return true;
|
|
}
|
|
const previousOptional = Object.entries(this.optionalParams());
|
|
const transitionURL = this.getURLForTransition(url);
|
|
if (this._previousURL === transitionURL) {
|
|
// probably an optional parameter change as the Ember URLs are the same
|
|
// whereas the entire URL is different
|
|
this.dispatch('push', url);
|
|
return Promise.resolve();
|
|
// this.setURL(url);
|
|
} else {
|
|
const currentOptional = this.optionalParams();
|
|
if (previousOptional.some(([key, value]) => currentOptional[key] !== value)) {
|
|
// an optional parameter change and a normal param change as the Ember
|
|
// URLs are different and we know the optional params changed
|
|
// TODO: Consider changing the above previousURL === transitionURL to
|
|
// use the same 'check the optionalParams' approach
|
|
this.dispatch('push', url);
|
|
}
|
|
// use ember to transition, which will eventually come around to use location.setURL
|
|
return this.container.lookup('router:main').transitionTo(transitionURL);
|
|
}
|
|
}
|
|
|
|
//
|
|
|
|
// Ember location interface
|
|
|
|
/**
|
|
* Returns the current `location.pathname` without `rootURL` or `baseURL`
|
|
*/
|
|
getURL() {
|
|
const search = this.location.search || '';
|
|
let hash = '';
|
|
if (typeof this.location.hash !== 'undefined') {
|
|
hash = this.location.hash.substr(0);
|
|
}
|
|
const url = this.getURLForTransition(this.location.pathname);
|
|
return `${url}${search}${hash}`;
|
|
}
|
|
|
|
formatURL(url, optional, withOptional = true) {
|
|
if (url !== '') {
|
|
// remove trailing slashes if they exists
|
|
this.rootURL = this.rootURL.replace(trailingSlashRe, '');
|
|
this.baseURL = this.baseURL.replace(trailingSlashRe, '');
|
|
} else if (this.baseURL[0] === '/' && this.rootURL[0] === '/') {
|
|
// if baseURL and rootURL both start with a slash
|
|
// ... remove trailing slash from baseURL if it exists
|
|
this.baseURL = this.baseURL.replace(trailingSlashRe, '');
|
|
}
|
|
|
|
if (withOptional) {
|
|
const temp = url.split('/');
|
|
optional = {
|
|
...this.optional,
|
|
...(optional || {}),
|
|
};
|
|
optional = Object.values(optional)
|
|
.filter((item) => Boolean(item))
|
|
.map((item) => item.value || item, []);
|
|
temp.splice(...[1, 0].concat(optional));
|
|
url = temp.join('/');
|
|
}
|
|
|
|
return `${this.baseURL}${this.rootURL}${url}`;
|
|
}
|
|
/**
|
|
* Change URL takes an ember application URL
|
|
*/
|
|
changeURL(type, path) {
|
|
this.path = path;
|
|
const state = this.machine.state;
|
|
path = this.formatURL(path);
|
|
|
|
if (!state || state.path !== path) {
|
|
this.dispatch(type, path);
|
|
}
|
|
}
|
|
|
|
setURL(path) {
|
|
// this.optional = {};
|
|
this.changeURL('push', path);
|
|
}
|
|
|
|
replaceURL(path) {
|
|
this.changeURL('replace', path);
|
|
}
|
|
|
|
onUpdateURL(callback) {
|
|
this.callback = callback;
|
|
}
|
|
|
|
//
|
|
|
|
/**
|
|
* Dispatch takes a full actual browser URL with all the rootURL and optional
|
|
* params if they exist
|
|
*/
|
|
dispatch(event, path) {
|
|
const state = {
|
|
path: path,
|
|
uuid: _uuid(),
|
|
};
|
|
this.machine[`${event}State`](state, null, path);
|
|
// popstate listeners only run from a browser action not when a state change
|
|
// is called directly, so manually call the popstate listener.
|
|
// https://developer.mozilla.org/en-US/docs/Web/API/Window/popstate_event#the_history_stack
|
|
this.route({ state: state });
|
|
}
|
|
|
|
willDestroy() {
|
|
this.doc.defaultView.removeEventListener('popstate', this.route);
|
|
}
|
|
}
|