vmui: change query params compatible with prometheus (#1619)

* feat: change url params for compatible prometheus

* style: add comment for TimeParams

* fix: change get default server for single version

* fix: change function for get query string value
This commit is contained in:
Yury Molodov 2021-09-15 09:42:49 +03:00 committed by GitHub
parent cf9efde50c
commit 777ff75874
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 98 additions and 21147 deletions

File diff suppressed because it is too large Load Diff

View File

@ -15,6 +15,7 @@
"@testing-library/user-event": "^12.2.2", "@testing-library/user-event": "^12.2.2",
"@types/d3": "^6.1.0", "@types/d3": "^6.1.0",
"@types/jest": "^26.0.15", "@types/jest": "^26.0.15",
"@types/lodash.get": "^4.4.6",
"@types/node": "^12.19.4", "@types/node": "^12.19.4",
"@types/qs": "^6.9.6", "@types/qs": "^6.9.6",
"@types/react": "^16.9.56", "@types/react": "^16.9.56",
@ -23,6 +24,7 @@
"codemirror-promql": "^0.10.2", "codemirror-promql": "^0.10.2",
"d3": "^6.2.0", "d3": "^6.2.0",
"dayjs": "^1.10.4", "dayjs": "^1.10.4",
"lodash.get": "^4.4.2",
"qs": "^6.5.2", "qs": "^6.5.2",
"react": "^17.0.1", "react": "^17.0.1",
"react-dom": "^17.0.1", "react-dom": "^17.0.1",

View File

@ -3,6 +3,7 @@ import {TimeParams, TimePeriod} from "../../types";
import {dateFromSeconds, getDurationFromPeriod, getTimeperiodForDuration} from "../../utils/time"; import {dateFromSeconds, getDurationFromPeriod, getTimeperiodForDuration} from "../../utils/time";
import {getFromStorage} from "../../utils/storage"; import {getFromStorage} from "../../utils/storage";
import {getDefaultServer} from "../../utils/default-server-url"; import {getDefaultServer} from "../../utils/default-server-url";
import {getQueryStringValue} from "../../utils/query-string";
export interface TimeState { export interface TimeState {
duration: string; duration: string;
@ -32,14 +33,16 @@ export type Action =
| { type: "TOGGLE_AUTOREFRESH"} | { type: "TOGGLE_AUTOREFRESH"}
| { type: "TOGGLE_AUTOCOMPLETE"} | { type: "TOGGLE_AUTOCOMPLETE"}
const duration = getQueryStringValue("g0.range_input", "1h");
const endInput = getQueryStringValue("g0.end_input", undefined);
export const initialState: AppState = { export const initialState: AppState = {
serverUrl: getFromStorage("PREFERRED_URL") as string || getDefaultServer(), // https://demo.promlabs.com or https://play.victoriametrics.com/select/accounting/1/6a716b0f-38bc-4856-90ce-448fd713e3fe/prometheus", serverUrl: getFromStorage("PREFERRED_URL") as string || getDefaultServer(), // https://demo.promlabs.com or https://play.victoriametrics.com/select/accounting/1/6a716b0f-38bc-4856-90ce-448fd713e3fe/prometheus",
displayType: "chart", displayType: "chart",
query: getFromStorage("LAST_QUERY") as string || "\n", // demo_memory_usage_bytes query: getQueryStringValue("g0.expr", getFromStorage("LAST_QUERY") as string || "\n"), // demo_memory_usage_bytes
time: { time: {
duration: "1h", duration,
period: getTimeperiodForDuration("1h") period: getTimeperiodForDuration(duration, endInput && new Date(endInput))
}, },
queryControls: { queryControls: {
autoRefresh: false, autoRefresh: false,

View File

@ -4,6 +4,7 @@ export interface TimeParams {
start: number; // timestamp in seconds start: number; // timestamp in seconds
end: number; // timestamp in seconds end: number; // timestamp in seconds
step?: number; // seconds step?: number; // seconds
date: string; // end input date
} }
export interface TimePeriod { export interface TimePeriod {

View File

@ -1,6 +1,6 @@
export const getDefaultServer = (): string => { export const getDefaultServer = (): string => {
const {href} = window.location; const {href, protocol, host, pathname} = window.location;
const regexp = /^http.+\/vmui/; const regexp = /^http.+\/vmui/;
const [result] = href.match(regexp) || ["https://"]; const [result] = href.match(regexp) || [`${protocol}//${host}${pathname}prometheus`];
return result.replace("vmui", "prometheus"); return result.replace("vmui", "prometheus");
}; };

View File

@ -1,49 +1,59 @@
import qs from "qs"; import qs from "qs";
import get from "lodash.get";
const decoder = (value: string) => { const stateToUrlParams = {
if (/^(\d+|\d*\.\d+)$/.test(value)) { "query": "g0.expr",
return parseFloat(value); "time.duration": "g0.range_input",
} "time.period.date": "g0.end_input",
"time.period.step": "g0.step_input",
const keywords = { "stacked": "g0.stacked",
true: true,
false: false,
null: null,
undefined: undefined,
};
if (value in keywords) {
return keywords[value as keyof typeof keywords];
}
return decodeURI(value);
}; };
// TODO need function for detect types.
// const decoder = (value: string) => {
// This function does not parse dates
// if (/^(\d+|\d*\.\d+)$/.test(value)) {
// return parseFloat(value);
// }
//
// const keywords = {
// true: true,
// false: false,
// null: null,
// undefined: undefined,
// };
// if (value in keywords) {
// return keywords[value as keyof typeof keywords];
// }
//
// return decodeURI(value);
// };
export const setQueryStringWithoutPageReload = (qsValue: string): void => { export const setQueryStringWithoutPageReload = (qsValue: string): void => {
const w = window; const w = window;
if (w) { if (w) {
const newurl = w.location.protocol + const newurl = `${w.location.protocol}//${w.location.host}${w.location.pathname}?${qsValue}`;
"//" +
w.location.host +
w.location.pathname +
"?" +
qsValue;
w.history.pushState({ path: newurl }, "", newurl); w.history.pushState({ path: newurl }, "", newurl);
} }
}; };
export const setQueryStringValue = ( export const setQueryStringValue = (newValue: Record<string, unknown>): void => {
newValue: Record<string, unknown>, const queryMap = new Map(Object.entries(stateToUrlParams));
queryString = window.location.search const newQsValue: string[] = [];
): void => { queryMap.forEach((queryKey, stateKey) => {
const values = qs.parse(queryString, { ignoreQueryPrefix: true, decoder }); const value = get(newValue, stateKey, "");
const newQsValue = qs.stringify({ ...values, ...newValue }, { encode: false }); if (value) {
setQueryStringWithoutPageReload(newQsValue); newQsValue.push(`${queryKey}=${value}`);
}
});
setQueryStringWithoutPageReload(newQsValue.join("&"));
}; };
export const getQueryStringValue = ( export const getQueryStringValue = (
key: string, key: string,
defaultValue?: any,
queryString = window.location.search queryString = window.location.search
): unknown => { ) => {
const values = qs.parse(queryString, { ignoreQueryPrefix: true, decoder }); const values = qs.parse(queryString, { ignoreQueryPrefix: true });
return values[key]; return get(values, key, defaultValue || "");
}; };

View File

@ -55,14 +55,12 @@ export const getTimeperiodForDuration = (dur: string, date?: Date): TimeParams =
return { return {
start: n - delta, start: n - delta,
end: n, end: n,
step: delta / MAX_ITEMS_PER_CHART step: delta / MAX_ITEMS_PER_CHART,
date: formatDateForNativeInput((date || new Date()))
}; };
}; };
export const formatDateForNativeInput = (date: Date): string => { export const formatDateForNativeInput = (date: Date): string => dayjs(date).format("YYYY-MM-DD[T]HH:mm:ss");
const isoString = dayjs(date).format("YYYY-MM-DD[T]HH:mm:ss");
return isoString;
};
export const getDurationFromPeriod = (p: TimePeriod): string => { export const getDurationFromPeriod = (p: TimePeriod): string => {
const dur = dayjs.duration(p.to.valueOf() - p.from.valueOf()); const dur = dayjs.duration(p.to.valueOf() - p.from.valueOf());