mirror of
https://github.com/grafana/grafana.git
synced 2025-09-28 07:43:56 +08:00

* Prometheus: Fix label value suggestion - remove quotes from typeahead input to suggest correct label values - fix acceptance of partial label values * Disable mid-word suggestions * Fix test
114 lines
4.1 KiB
TypeScript
114 lines
4.1 KiB
TypeScript
export const RATE_RANGES = ['1m', '5m', '10m', '30m', '1h'];
|
|
|
|
export const processHistogramLabels = (labels: string[]) => {
|
|
const result = [];
|
|
const regexp = new RegExp('_bucket($|:)');
|
|
for (let index = 0; index < labels.length; index++) {
|
|
const label = labels[index];
|
|
const isHistogramValue = regexp.test(label);
|
|
if (isHistogramValue) {
|
|
if (result.indexOf(label) === -1) {
|
|
result.push(label);
|
|
}
|
|
}
|
|
}
|
|
|
|
return { values: { __name__: result } };
|
|
};
|
|
|
|
export function processLabels(labels: Array<{ [key: string]: string }>, withName = false) {
|
|
const values: { [key: string]: string[] } = {};
|
|
labels.forEach(l => {
|
|
const { __name__, ...rest } = l;
|
|
if (withName) {
|
|
values['__name__'] = values['__name__'] || [];
|
|
if (!values['__name__'].includes(__name__)) {
|
|
values['__name__'].push(__name__);
|
|
}
|
|
}
|
|
|
|
Object.keys(rest).forEach(key => {
|
|
if (!values[key]) {
|
|
values[key] = [];
|
|
}
|
|
if (!values[key].includes(rest[key])) {
|
|
values[key].push(rest[key]);
|
|
}
|
|
});
|
|
});
|
|
return { values, keys: Object.keys(values) };
|
|
}
|
|
|
|
// const cleanSelectorRegexp = /\{(\w+="[^"\n]*?")(,\w+="[^"\n]*?")*\}/;
|
|
export const selectorRegexp = /\{[^}]*?\}/;
|
|
export const labelRegexp = /\b(\w+)(!?=~?)("[^"\n]*?")/g;
|
|
export function parseSelector(query: string, cursorOffset = 1): { labelKeys: any[]; selector: string } {
|
|
if (!query.match(selectorRegexp)) {
|
|
// Special matcher for metrics
|
|
if (query.match(/^[A-Za-z:][\w:]*$/)) {
|
|
return {
|
|
selector: `{__name__="${query}"}`,
|
|
labelKeys: ['__name__'],
|
|
};
|
|
}
|
|
throw new Error('Query must contain a selector: ' + query);
|
|
}
|
|
|
|
// Check if inside a selector
|
|
const prefix = query.slice(0, cursorOffset);
|
|
const prefixOpen = prefix.lastIndexOf('{');
|
|
const prefixClose = prefix.lastIndexOf('}');
|
|
if (prefixOpen === -1) {
|
|
throw new Error('Not inside selector, missing open brace: ' + prefix);
|
|
}
|
|
if (prefixClose > -1 && prefixClose > prefixOpen) {
|
|
throw new Error('Not inside selector, previous selector already closed: ' + prefix);
|
|
}
|
|
const suffix = query.slice(cursorOffset);
|
|
const suffixCloseIndex = suffix.indexOf('}');
|
|
const suffixClose = suffixCloseIndex + cursorOffset;
|
|
const suffixOpenIndex = suffix.indexOf('{');
|
|
const suffixOpen = suffixOpenIndex + cursorOffset;
|
|
if (suffixClose === -1) {
|
|
throw new Error('Not inside selector, missing closing brace in suffix: ' + suffix);
|
|
}
|
|
if (suffixOpenIndex > -1 && suffixOpen < suffixClose) {
|
|
throw new Error('Not inside selector, next selector opens before this one closed: ' + suffix);
|
|
}
|
|
|
|
// Extract clean labels to form clean selector, incomplete labels are dropped
|
|
const selector = query.slice(prefixOpen, suffixClose);
|
|
const labels: { [key: string]: { value: string; operator: string } } = {};
|
|
selector.replace(labelRegexp, (label, key, operator, value) => {
|
|
const labelOffset = query.indexOf(label);
|
|
const valueStart = labelOffset + key.length + operator.length + 1;
|
|
const valueEnd = labelOffset + key.length + operator.length + value.length - 1;
|
|
// Skip label if cursor is in value
|
|
if (cursorOffset < valueStart || cursorOffset > valueEnd) {
|
|
labels[key] = { value, operator };
|
|
}
|
|
return '';
|
|
});
|
|
|
|
// Add metric if there is one before the selector
|
|
const metricPrefix = query.slice(0, prefixOpen);
|
|
const metricMatch = metricPrefix.match(/[A-Za-z:][\w:]*$/);
|
|
if (metricMatch) {
|
|
labels['__name__'] = { value: `"${metricMatch[0]}"`, operator: '=' };
|
|
}
|
|
|
|
// Build sorted selector
|
|
const labelKeys = Object.keys(labels).sort();
|
|
const cleanSelector = labelKeys.map(key => `${key}${labels[key].operator}${labels[key].value}`).join(',');
|
|
|
|
const selectorString = ['{', cleanSelector, '}'].join('');
|
|
|
|
return { labelKeys, selector: selectorString };
|
|
}
|
|
|
|
export function expandRecordingRules(query: string, mapping: { [name: string]: string }): string {
|
|
const ruleNames = Object.keys(mapping);
|
|
const rulesRegex = new RegExp(`(\\s|^)(${ruleNames.join('|')})(\\s|$|\\(|\\[|\\{)`, 'ig');
|
|
return query.replace(rulesRegex, (match, pre, name, post) => `${pre}${mapping[name]}${post}`);
|
|
}
|