mirror of
https://github.com/prometheus/prometheus.git
synced 2025-03-05 20:59:13 -08:00
ui: heatmap visualization for histogram buckets Signed-off-by: Yury Moladau <yurymolodov@gmail.com> --------- Signed-off-by: Yury Moladau <yurymolodov@gmail.com>
313 lines
9 KiB
TypeScript
313 lines
9 KiB
TypeScript
import moment from 'moment-timezone';
|
|
|
|
import { GraphDisplayMode, PanelDefaultOptions, PanelOptions, PanelType } from '../pages/graph/Panel';
|
|
import { PanelMeta } from '../pages/graph/PanelList';
|
|
|
|
export const generateID = (): string => {
|
|
return `_${Math.random().toString(36).substr(2, 9)}`;
|
|
};
|
|
|
|
export const byEmptyString = (p: string): boolean => p.length > 0;
|
|
|
|
export const isPresent = <T>(obj: T): obj is NonNullable<T> => obj !== null && obj !== undefined;
|
|
|
|
export const escapeHTML = (str: string): string => {
|
|
const entityMap: { [key: string]: string } = {
|
|
'&': '&',
|
|
'<': '<',
|
|
'>': '>',
|
|
'"': '"',
|
|
"'": ''',
|
|
'/': '/',
|
|
};
|
|
|
|
return String(str).replace(/[&<>"'/]/g, function (s) {
|
|
return entityMap[s];
|
|
});
|
|
};
|
|
|
|
export const metricToSeriesName = (labels: { [key: string]: string }): string => {
|
|
if (labels === null) {
|
|
return 'scalar';
|
|
}
|
|
let tsName = (labels.__name__ || '') + '{';
|
|
const labelStrings: string[] = [];
|
|
for (const label in labels) {
|
|
if (label !== '__name__') {
|
|
labelStrings.push(label + '="' + labels[label] + '"');
|
|
}
|
|
}
|
|
tsName += labelStrings.join(', ') + '}';
|
|
return tsName;
|
|
};
|
|
|
|
export const parseDuration = (durationStr: string): number | null => {
|
|
if (durationStr === '') {
|
|
return null;
|
|
}
|
|
if (durationStr === '0') {
|
|
// Allow 0 without a unit.
|
|
return 0;
|
|
}
|
|
|
|
const durationRE = new RegExp('^(([0-9]+)y)?(([0-9]+)w)?(([0-9]+)d)?(([0-9]+)h)?(([0-9]+)m)?(([0-9]+)s)?(([0-9]+)ms)?$');
|
|
const matches = durationStr.match(durationRE);
|
|
if (!matches) {
|
|
return null;
|
|
}
|
|
|
|
let dur = 0;
|
|
|
|
// Parse the match at pos `pos` in the regex and use `mult` to turn that
|
|
// into ms, then add that value to the total parsed duration.
|
|
const m = (pos: number, mult: number) => {
|
|
if (matches[pos] === undefined) {
|
|
return;
|
|
}
|
|
const n = parseInt(matches[pos]);
|
|
dur += n * mult;
|
|
};
|
|
|
|
m(2, 1000 * 60 * 60 * 24 * 365); // y
|
|
m(4, 1000 * 60 * 60 * 24 * 7); // w
|
|
m(6, 1000 * 60 * 60 * 24); // d
|
|
m(8, 1000 * 60 * 60); // h
|
|
m(10, 1000 * 60); // m
|
|
m(12, 1000); // s
|
|
m(14, 1); // ms
|
|
|
|
return dur;
|
|
};
|
|
|
|
export const formatDuration = (d: number): string => {
|
|
let ms = d;
|
|
let r = '';
|
|
if (ms === 0) {
|
|
return '0s';
|
|
}
|
|
|
|
const f = (unit: string, mult: number, exact: boolean) => {
|
|
if (exact && ms % mult !== 0) {
|
|
return;
|
|
}
|
|
const v = Math.floor(ms / mult);
|
|
if (v > 0) {
|
|
r += `${v}${unit}`;
|
|
ms -= v * mult;
|
|
}
|
|
};
|
|
|
|
// Only format years and weeks if the remainder is zero, as it is often
|
|
// easier to read 90d than 12w6d.
|
|
f('y', 1000 * 60 * 60 * 24 * 365, true);
|
|
f('w', 1000 * 60 * 60 * 24 * 7, true);
|
|
|
|
f('d', 1000 * 60 * 60 * 24, false);
|
|
f('h', 1000 * 60 * 60, false);
|
|
f('m', 1000 * 60, false);
|
|
f('s', 1000, false);
|
|
f('ms', 1, false);
|
|
|
|
return r;
|
|
};
|
|
|
|
export function parseTime(timeText: string): number {
|
|
return moment.utc(timeText).valueOf();
|
|
}
|
|
|
|
export function formatTime(time: number): string {
|
|
return moment.utc(time).format('YYYY-MM-DD HH:mm:ss');
|
|
}
|
|
|
|
export const now = (): number => moment().valueOf();
|
|
|
|
export const humanizeDuration = (milliseconds: number): string => {
|
|
const sign = milliseconds < 0 ? '-' : '';
|
|
const unsignedMillis = milliseconds < 0 ? -1 * milliseconds : milliseconds;
|
|
const duration = moment.duration(unsignedMillis, 'ms');
|
|
const ms = Math.floor(duration.milliseconds());
|
|
const s = Math.floor(duration.seconds());
|
|
const m = Math.floor(duration.minutes());
|
|
const h = Math.floor(duration.hours());
|
|
const d = Math.floor(duration.asDays());
|
|
if (d !== 0) {
|
|
return `${sign}${d}d ${h}h ${m}m ${s}s`;
|
|
}
|
|
if (h !== 0) {
|
|
return `${sign}${h}h ${m}m ${s}s`;
|
|
}
|
|
if (m !== 0) {
|
|
return `${sign}${m}m ${s}s`;
|
|
}
|
|
if (s !== 0) {
|
|
return `${sign}${s}.${ms}s`;
|
|
}
|
|
if (unsignedMillis > 0) {
|
|
return `${sign}${unsignedMillis.toFixed(3)}ms`;
|
|
}
|
|
return '0s';
|
|
};
|
|
|
|
export const formatRelative = (startStr: string, end: number): string => {
|
|
const start = parseTime(startStr);
|
|
if (start < 0) {
|
|
return 'Never';
|
|
}
|
|
return humanizeDuration(end - start) + ' ago';
|
|
};
|
|
|
|
const paramFormat = /^g\d+\..+=.+$/;
|
|
|
|
export const decodePanelOptionsFromQueryString = (query: string): PanelMeta[] => {
|
|
if (query === '') {
|
|
return [];
|
|
}
|
|
const urlParams = query.substring(1).split('&');
|
|
|
|
return urlParams.reduce<PanelMeta[]>((panels, urlParam, i) => {
|
|
const panelsCount = panels.length;
|
|
const prefix = `g${panelsCount}.`;
|
|
if (urlParam.startsWith(`${prefix}expr=`)) {
|
|
const prefixLen = prefix.length;
|
|
return [
|
|
...panels,
|
|
{
|
|
id: generateID(),
|
|
key: `${panelsCount}`,
|
|
options: urlParams.slice(i).reduce((opts, param) => {
|
|
return param.startsWith(prefix) && paramFormat.test(param)
|
|
? { ...opts, ...parseOption(param.substring(prefixLen)) }
|
|
: opts;
|
|
}, PanelDefaultOptions),
|
|
},
|
|
];
|
|
}
|
|
return panels;
|
|
}, []);
|
|
};
|
|
|
|
export const parseOption = (param: string): Partial<PanelOptions> => {
|
|
const [opt, val] = param.split('=');
|
|
const decodedValue = decodeURIComponent(val.replace(/\+/g, ' '));
|
|
switch (opt) {
|
|
case 'expr':
|
|
return { expr: decodedValue };
|
|
|
|
case 'tab':
|
|
return { type: decodedValue === '0' ? PanelType.Graph : PanelType.Table };
|
|
|
|
case 'display_mode':
|
|
const validKey = Object.values(GraphDisplayMode).includes(decodedValue as GraphDisplayMode);
|
|
return { displayMode: validKey ? (decodedValue as GraphDisplayMode) : GraphDisplayMode.Lines };
|
|
|
|
case 'stacked':
|
|
return { displayMode: decodedValue === '1' ? GraphDisplayMode.Stacked : GraphDisplayMode.Lines };
|
|
|
|
case 'show_exemplars':
|
|
return { showExemplars: decodedValue === '1' };
|
|
|
|
case 'range_input':
|
|
const range = parseDuration(decodedValue);
|
|
return isPresent(range) ? { range } : {};
|
|
|
|
case 'end_input':
|
|
case 'moment_input':
|
|
return { endTime: parseTime(decodedValue) };
|
|
|
|
case 'step_input':
|
|
const resolution = parseInt(decodedValue);
|
|
return resolution > 0 ? { resolution } : {};
|
|
}
|
|
return {};
|
|
};
|
|
|
|
export const formatParam =
|
|
(key: string) =>
|
|
(paramName: string, value: number | string | boolean): string => {
|
|
return `g${key}.${paramName}=${encodeURIComponent(value)}`;
|
|
};
|
|
|
|
export const toQueryString = ({ key, options }: PanelMeta): string => {
|
|
const formatWithKey = formatParam(key);
|
|
const { expr, type, displayMode, range, endTime, resolution, showExemplars } = options;
|
|
const time = isPresent(endTime) ? formatTime(endTime) : false;
|
|
const urlParams = [
|
|
formatWithKey('expr', expr),
|
|
formatWithKey('tab', type === PanelType.Graph ? 0 : 1),
|
|
formatWithKey('display_mode', displayMode),
|
|
formatWithKey('show_exemplars', showExemplars ? 1 : 0),
|
|
formatWithKey('range_input', formatDuration(range)),
|
|
time ? `${formatWithKey('end_input', time)}&${formatWithKey('moment_input', time)}` : '',
|
|
isPresent(resolution) ? formatWithKey('step_input', resolution) : '',
|
|
];
|
|
return urlParams.filter(byEmptyString).join('&');
|
|
};
|
|
|
|
export const encodePanelOptionsToQueryString = (panels: PanelMeta[]): string => {
|
|
return `?${panels.map(toQueryString).join('&')}`;
|
|
};
|
|
|
|
export const setQuerySearchFilter = (search: string) => {
|
|
setQueryParam('search', search);
|
|
};
|
|
|
|
export const getQuerySearchFilter = (): string => {
|
|
return getQueryParam('search');
|
|
};
|
|
|
|
export const setQueryParam = (key: string, value: string) => {
|
|
const params = new URLSearchParams(window.location.search);
|
|
params.set(key, value);
|
|
window.history.pushState({}, '', '?' + params.toString());
|
|
};
|
|
|
|
export const getQueryParam = (key: string): string => {
|
|
const locationSearch = window.location.search;
|
|
const params = new URLSearchParams(locationSearch);
|
|
return params.get(key) || '';
|
|
};
|
|
|
|
export const createExpressionLink = (expr: string): string => {
|
|
return `../graph?g0.expr=${encodeURIComponent(expr)}&g0.tab=1&g0.display_mode=${
|
|
GraphDisplayMode.Lines
|
|
}&g0.show_exemplars=0.g0.range_input=1h.`;
|
|
};
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any,
|
|
export const mapObjEntries = <T extends { [s: string]: any }, key extends keyof T, Z>(
|
|
o: T,
|
|
cb: ([k, v]: [string, T[key]], i: number, arr: [string, T[key]][]) => Z
|
|
): Z[] => Object.entries(o).map(cb);
|
|
|
|
export const callAll =
|
|
(
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
...fns: Array<(...args: any) => void>
|
|
) =>
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/explicit-module-boundary-types
|
|
(...args: any): void => {
|
|
// eslint-disable-next-line prefer-spread
|
|
fns.filter(Boolean).forEach((fn) => fn.apply(null, args));
|
|
};
|
|
|
|
export const parsePrometheusFloat = (value: string): string | number => {
|
|
if (isNaN(Number(value))) {
|
|
return value;
|
|
} else {
|
|
return Number(value);
|
|
}
|
|
};
|
|
|
|
export function debounce<Params extends unknown[]>(
|
|
func: (...args: Params) => unknown,
|
|
timeout: number
|
|
): (...args: Params) => void {
|
|
let timer: NodeJS.Timeout;
|
|
return (...args: Params) => {
|
|
clearTimeout(timer);
|
|
timer = setTimeout(() => {
|
|
func(...args);
|
|
}, timeout);
|
|
};
|
|
}
|