| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277 |
- import _ from 'lodash';
- import gfunc from './gfunc';
- import {Parser} from './parser';
- export default class GraphiteQuery {
- target: any;
- functions: any[];
- segments: any[];
- tags: any[];
- error: any;
- seriesByTagUsed: boolean;
- checkOtherSegmentsIndex: number;
- removeTagValue: string;
- templateSrv: any;
- scopedVars: any;
- /** @ngInject */
- constructor(target, templateSrv?, scopedVars?) {
- this.target = target;
- this.parseTarget();
- this.removeTagValue = '-- remove tag --';
- }
- parseTarget() {
- this.functions = [];
- this.segments = [];
- this.tags = [];
- this.error = null;
- if (this.target.textEditor) {
- return;
- }
- var parser = new Parser(this.target.target);
- var astNode = parser.getAst();
- if (astNode === null) {
- this.checkOtherSegmentsIndex = 0;
- return;
- }
- if (astNode.type === 'error') {
- this.error = astNode.message + " at position: " + astNode.pos;
- this.target.textEditor = true;
- return;
- }
- try {
- this.parseTargetRecursive(astNode, null);
- } catch (err) {
- console.log('error parsing target:', err.message);
- this.error = err.message;
- this.target.textEditor = true;
- }
- this.checkOtherSegmentsIndex = this.segments.length - 1;
- this.checkForSeriesByTag();
- }
- checkForSeriesByTag() {
- let seriesByTagFunc = _.find(this.functions, (func) => func.def.name === 'seriesByTag');
- if (seriesByTagFunc) {
- this.seriesByTagUsed = true;
- seriesByTagFunc.hidden = true;
- let tags = this.splitSeriesByTagParams(seriesByTagFunc);
- this.tags = tags;
- }
- }
- getSegmentPathUpTo(index) {
- var arr = this.segments.slice(0, index);
- return _.reduce(arr, function(result, segment) {
- return result ? (result + "." + segment.value) : segment.value;
- }, "");
- }
- parseTargetRecursive(astNode, func) {
- if (astNode === null) {
- return null;
- }
- switch (astNode.type) {
- case 'function':
- var innerFunc = gfunc.createFuncInstance(astNode.name, { withDefaultParams: false });
- _.each(astNode.params, param => {
- this.parseTargetRecursive(param, innerFunc);
- });
- innerFunc.updateText();
- this.functions.push(innerFunc);
- break;
- case 'series-ref':
- if (this.segments.length > 0) {
- this.addFunctionParameter(func, astNode.value);
- } else {
- this.segments.push(astNode);
- }
- break;
- case 'bool':
- case 'string':
- case 'number':
- this.addFunctionParameter(func, astNode.value);
- break;
- case 'metric':
- if (this.segments.length > 0) {
- this.addFunctionParameter(func, _.join(_.map(astNode.segments, 'value'), '.'));
- } else {
- this.segments = astNode.segments;
- }
- break;
- }
- }
- updateSegmentValue(segment, index) {
- this.segments[index].value = segment.value;
- }
- addSelectMetricSegment() {
- this.segments.push({value: "select metric"});
- }
- addFunction(newFunc) {
- this.functions.push(newFunc);
- this.moveAliasFuncLast();
- }
- moveAliasFuncLast() {
- var aliasFunc = _.find(this.functions, function(func) {
- return func.def.name === 'alias' ||
- func.def.name === 'aliasByNode' ||
- func.def.name === 'aliasByMetric';
- });
- if (aliasFunc) {
- this.functions = _.without(this.functions, aliasFunc);
- this.functions.push(aliasFunc);
- }
- }
- addFunctionParameter(func, value) {
- if (func.params.length >= func.def.params.length) {
- throw { message: 'too many parameters for function ' + func.def.name };
- }
- func.params.push(value);
- }
- removeFunction(func) {
- this.functions = _.without(this.functions, func);
- }
- updateModelTarget(targets) {
- // render query
- if (!this.target.textEditor) {
- var metricPath = this.getSegmentPathUpTo(this.segments.length).replace(/\.select metric$/, '');
- this.target.target = _.reduce(this.functions, wrapFunction, metricPath);
- }
- this.updateRenderedTarget(this.target, targets);
- // loop through other queries and update targetFull as needed
- for (const target of targets || []) {
- if (target.refId !== this.target.refId) {
- this.updateRenderedTarget(target, targets);
- }
- }
- }
- updateRenderedTarget(target, targets) {
- // render nested query
- var targetsByRefId = _.keyBy(targets, 'refId');
- // no references to self
- delete targetsByRefId[target.refId];
- var nestedSeriesRefRegex = /\#([A-Z])/g;
- var targetWithNestedQueries = target.target;
- // Keep interpolating until there are no query references
- // The reason for the loop is that the referenced query might contain another reference to another query
- while (targetWithNestedQueries.match(nestedSeriesRefRegex)) {
- var updated = targetWithNestedQueries.replace(nestedSeriesRefRegex, (match, g1) => {
- var t = targetsByRefId[g1];
- if (!t) {
- return match;
- }
- // no circular references
- delete targetsByRefId[g1];
- return t.target;
- });
- if (updated === targetWithNestedQueries) {
- break;
- }
- targetWithNestedQueries = updated;
- }
- delete target.targetFull;
- if (target.target !== targetWithNestedQueries) {
- target.targetFull = targetWithNestedQueries;
- }
- }
- splitSeriesByTagParams(func) {
- const tagPattern = /([^\!=~]+)([\!=~]+)([^\!=~]+)/;
- return _.flatten(_.map(func.params, (param: string) => {
- let matches = tagPattern.exec(param);
- if (matches) {
- let tag = matches.slice(1);
- if (tag.length === 3) {
- return {
- key: tag[0],
- operator: tag[1],
- value: tag[2]
- };
- }
- }
- return [];
- }));
- }
- getSeriesByTagFuncIndex() {
- return _.findIndex(this.functions, (func) => func.def.name === 'seriesByTag');
- }
- getSeriesByTagFunc() {
- let seriesByTagFuncIndex = this.getSeriesByTagFuncIndex();
- if (seriesByTagFuncIndex >= 0) {
- return this.functions[seriesByTagFuncIndex];
- } else {
- return undefined;
- }
- }
- addTag(tag) {
- let newTagParam = renderTagString(tag);
- this.getSeriesByTagFunc().params.push(newTagParam);
- this.tags.push(tag);
- }
- removeTag(index) {
- this.getSeriesByTagFunc().params.splice(index, 1);
- this.tags.splice(index, 1);
- }
- updateTag(tag, tagIndex) {
- this.error = null;
- if (tag.key === this.removeTagValue) {
- this.removeTag(tagIndex);
- return;
- }
- let newTagParam = renderTagString(tag);
- this.getSeriesByTagFunc().params[tagIndex] = newTagParam;
- this.tags[tagIndex] = tag;
- }
- renderTagExpressions(excludeIndex = -1) {
- return _.compact(_.map(this.tags, (tagExpr, index) => {
- // Don't render tag that we want to lookup
- if (index !== excludeIndex) {
- return tagExpr.key + tagExpr.operator + tagExpr.value;
- }
- }));
- }
- }
- function wrapFunction(target, func) {
- return func.render(target);
- }
- function renderTagString(tag) {
- return tag.key + tag.operator + tag.value;
- }
|