Skip to content

Commit

Permalink
[8.x] [ES|QL] Normalize multiplication by one when pretty-printing (e…
Browse files Browse the repository at this point in the history
…lastic#197182) (elastic#197318)

# Backport

This will backport the following commits from `main` to `8.x`:
- [[ES|QL] Normalize multiplication by one when pretty-printing
(elastic#197182)](elastic#197182)

<!--- Backport version: 9.4.3 -->

### Questions ?
Please refer to the [Backport tool
documentation](https://github.com/sqren/backport)

<!--BACKPORT [{"author":{"name":"Vadim
Kibana","email":"[email protected]"},"sourceCommit":{"committedDate":"2024-10-22T18:05:09Z","message":"[ES|QL]
Normalize multiplication by one when pretty-printing (elastic#197182)\n\n##
Summary\r\n\r\nPartially addresses
https://github.com/elastic/kibana/issues/189258\r\n\r\nThis change will
likely
supercede\r\nhttps://github.com/elastic/pull/196817\r\n\r\nWhen
parsing, currently ES|QL parsers adds extraneous multiply by 1
or\r\nmultiply by -1 nodes when parsing arithmetic unary
expressions.\r\n\r\nFor example, `-(x)` is parsed as `-1 *
x`.\r\n\r\nThis change, reverts these when pretty-printing using
the\r\n`BasicPrettyPrinter`: `-1 * x` is pretty printed as
`-x`.\r\n\r\n\r\n### Checklist\r\n\r\nDelete any items that are not
applicable to this PR.\r\n\r\n- [x] [Unit or
functional\r\ntests](https://www.elastic.co/guide/en/kibana/master/development-tests.html)\r\nwere
updated or added to match the most common scenarios\r\n\r\n\r\n### For
maintainers\r\n\r\n- [x] This was checked for breaking API changes and
was
[labeled\r\nappropriately](https://www.elastic.co/guide/en/kibana/master/contributing.html#_add_your_labels)","sha":"848c9f48dc101744c8fae69951916dc511dfdc8a","branchLabelMapping":{"^v9.0.0$":"main","^v8.17.0$":"8.x","^v(\\d+).(\\d+).\\d+$":"$1.$2"}},"sourcePullRequest":{"labels":["review","release_note:skip","v9.0.0","backport:prev-minor","Feature:ES|QL","Team:ESQL","v8.17.0"],"title":"[ES|QL]
Normalize multiplication by one when
pretty-printing","number":197182,"url":"https://github.com/elastic/kibana/pull/197182","mergeCommit":{"message":"[ES|QL]
Normalize multiplication by one when pretty-printing (elastic#197182)\n\n##
Summary\r\n\r\nPartially addresses
https://github.com/elastic/kibana/issues/189258\r\n\r\nThis change will
likely
supercede\r\nhttps://github.com/elastic/pull/196817\r\n\r\nWhen
parsing, currently ES|QL parsers adds extraneous multiply by 1
or\r\nmultiply by -1 nodes when parsing arithmetic unary
expressions.\r\n\r\nFor example, `-(x)` is parsed as `-1 *
x`.\r\n\r\nThis change, reverts these when pretty-printing using
the\r\n`BasicPrettyPrinter`: `-1 * x` is pretty printed as
`-x`.\r\n\r\n\r\n### Checklist\r\n\r\nDelete any items that are not
applicable to this PR.\r\n\r\n- [x] [Unit or
functional\r\ntests](https://www.elastic.co/guide/en/kibana/master/development-tests.html)\r\nwere
updated or added to match the most common scenarios\r\n\r\n\r\n### For
maintainers\r\n\r\n- [x] This was checked for breaking API changes and
was
[labeled\r\nappropriately](https://www.elastic.co/guide/en/kibana/master/contributing.html#_add_your_labels)","sha":"848c9f48dc101744c8fae69951916dc511dfdc8a"}},"sourceBranch":"main","suggestedTargetBranches":["8.x"],"targetPullRequestStates":[{"branch":"main","label":"v9.0.0","branchLabelMappingKey":"^v9.0.0$","isSourceBranch":true,"state":"MERGED","url":"https://github.com/elastic/kibana/pull/197182","number":197182,"mergeCommit":{"message":"[ES|QL]
Normalize multiplication by one when pretty-printing (elastic#197182)\n\n##
Summary\r\n\r\nPartially addresses
https://github.com/elastic/kibana/issues/189258\r\n\r\nThis change will
likely
supercede\r\nhttps://github.com/elastic/pull/196817\r\n\r\nWhen
parsing, currently ES|QL parsers adds extraneous multiply by 1
or\r\nmultiply by -1 nodes when parsing arithmetic unary
expressions.\r\n\r\nFor example, `-(x)` is parsed as `-1 *
x`.\r\n\r\nThis change, reverts these when pretty-printing using
the\r\n`BasicPrettyPrinter`: `-1 * x` is pretty printed as
`-x`.\r\n\r\n\r\n### Checklist\r\n\r\nDelete any items that are not
applicable to this PR.\r\n\r\n- [x] [Unit or
functional\r\ntests](https://www.elastic.co/guide/en/kibana/master/development-tests.html)\r\nwere
updated or added to match the most common scenarios\r\n\r\n\r\n### For
maintainers\r\n\r\n- [x] This was checked for breaking API changes and
was
[labeled\r\nappropriately](https://www.elastic.co/guide/en/kibana/master/contributing.html#_add_your_labels)","sha":"848c9f48dc101744c8fae69951916dc511dfdc8a"}},{"branch":"8.x","label":"v8.17.0","branchLabelMappingKey":"^v8.17.0$","isSourceBranch":false,"state":"NOT_CREATED"}]}]
BACKPORT-->

Co-authored-by: Vadim Kibana <[email protected]>
  • Loading branch information
kibanamachine and vadimkibana authored Oct 23, 2024
1 parent 72774b1 commit b3fef2a
Show file tree
Hide file tree
Showing 5 changed files with 169 additions and 5 deletions.
27 changes: 25 additions & 2 deletions packages/kbn-esql-ast/src/ast/helpers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,22 @@
* License v3.0 only", or the "Server Side Public License, v 1".
*/

import { ESQLAstNode, ESQLBinaryExpression, ESQLFunction } from '../types';
import type {
ESQLAstNode,
ESQLBinaryExpression,
ESQLColumn,
ESQLFunction,
ESQLIntegerLiteral,
ESQLLiteral,
ESQLProperNode,
} from '../types';
import { BinaryExpressionGroup } from './constants';

export const isProperNode = (node: unknown): node is ESQLProperNode =>
!!node && typeof node === 'object' && !Array.isArray(node);

export const isFunctionExpression = (node: unknown): node is ESQLFunction =>
!!node && typeof node === 'object' && !Array.isArray(node) && (node as any).type === 'function';
isProperNode(node) && node.type === 'function';

/**
* Returns true if the given node is a binary expression, i.e. an operator
Expand All @@ -28,6 +39,18 @@ export const isFunctionExpression = (node: unknown): node is ESQLFunction =>
export const isBinaryExpression = (node: unknown): node is ESQLBinaryExpression =>
isFunctionExpression(node) && node.subtype === 'binary-expression';

export const isLiteral = (node: unknown): node is ESQLLiteral =>
isProperNode(node) && node.type === 'literal';

export const isIntegerLiteral = (node: unknown): node is ESQLIntegerLiteral =>
isLiteral(node) && node.literalType === 'integer';

export const isDoubleLiteral = (node: unknown): node is ESQLIntegerLiteral =>
isLiteral(node) && node.literalType === 'double';

export const isColumn = (node: unknown): node is ESQLColumn =>
isProperNode(node) && node.type === 'column';

/**
* Returns the group of a binary expression:
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ const reprint = (src: string) => {
const { root } = parse(src);
const text = BasicPrettyPrinter.print(root);

// console.log(JSON.stringify(ast, null, 2));
// console.log(JSON.stringify(root, null, 2));

return { text };
};
Expand Down Expand Up @@ -194,6 +194,66 @@ describe('single line query', () => {

expect(text).toBe('ROW NOT a');
});

test('negative numbers', () => {
const { text } = reprint('ROW -1');

expect(text).toBe('ROW -1');
});

test('negative numbers in brackets', () => {
const { text } = reprint('ROW -(1)');

expect(text).toBe('ROW -1');
});

test('negative column names', () => {
const { text } = reprint('ROW -col');

expect(text).toBe('ROW -col');
});

test('plus unary expression', () => {
const { text } = reprint('ROW +(23)');

expect(text).toBe('ROW 23');
});

test('chained multiple unary expressions', () => {
const { text } = reprint('ROW ----+-+(23)');

expect(text).toBe('ROW -23');
});

test('before another expression', () => {
const { text } = reprint('ROW ----+-+(1 + 1)');

expect(text).toBe('ROW -(1 + 1)');
});

test('negative one from the right side', () => {
const { text } = reprint('ROW 2 * -1');

expect(text).toBe('ROW -2');
});

test('two minuses is plus', () => {
const { text } = reprint('ROW --123');

expect(text).toBe('ROW 123');
});

test('two minuses is plus (float)', () => {
const { text } = reprint('ROW --1.23');

expect(text).toBe('ROW 1.23');
});

test('two minuses is plus (with brackets)', () => {
const { text } = reprint('ROW --(123)');

expect(text).toBe('ROW 123');
});
});

describe('postfix unary expression', () => {
Expand Down
79 changes: 78 additions & 1 deletion packages/kbn-esql-ast/src/pretty_print/basic_pretty_printer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,9 +7,18 @@
* License v3.0 only", or the "Server Side Public License, v 1".
*/

import { binaryExpressionGroup } from '../ast/helpers';
import {
binaryExpressionGroup,
isBinaryExpression,
isColumn,
isDoubleLiteral,
isIntegerLiteral,
isLiteral,
isProperNode,
} from '../ast/helpers';
import { ESQLAstBaseItem, ESQLAstCommand, ESQLAstQueryExpression } from '../types';
import { ESQLAstExpressionNode, Visitor } from '../visitor';
import { resolveItem } from '../visitor/utils';
import { LeafPrinter } from './leaf_printer';

export interface BasicPrettyPrinterOptions {
Expand Down Expand Up @@ -152,6 +161,62 @@ export class BasicPrettyPrinter {
return formatted;
}

protected simplifyMultiplicationByOne(
node: ESQLAstExpressionNode,
minusCount: number = 0
): string | undefined {
if (isBinaryExpression(node) && node.name === '*') {
let [left, right] = node.args;
left = resolveItem(left);
right = resolveItem(right);

if (isProperNode(left) && isProperNode(right)) {
if (!!left.formatting || !!right.formatting) {
return undefined;
}
if (isIntegerLiteral(left)) {
if (left.value === 1) {
return this.simplifyMultiplicationByOne(right, minusCount);
} else if (left.value === -1) {
return this.simplifyMultiplicationByOne(right, minusCount + 1);
}
}
if (isIntegerLiteral(right)) {
if (right.value === 1) {
return this.simplifyMultiplicationByOne(left, minusCount);
} else if (right.value === -1) {
return this.simplifyMultiplicationByOne(left, minusCount + 1);
}
}
return undefined;
} else {
return undefined;
}
}

const isNegative = minusCount % 2 === 1;

if (isNegative && (isIntegerLiteral(node) || isDoubleLiteral(node)) && node.value < 0) {
return BasicPrettyPrinter.expression(
{
...node,
value: Math.abs(node.value),
},
this.opts
);
}

let expression = BasicPrettyPrinter.expression(node, this.opts);
const sign = isNegative ? '-' : '';
const needsBrackets = !!sign && !isColumn(node) && !isLiteral(node);

if (needsBrackets) {
expression = `(${expression})`;
}

return sign ? `${sign}${expression}` : expression;
}

protected readonly visitor: Visitor<any> = new Visitor()
.on('visitExpression', (ctx) => {
return '<EXPRESSION>';
Expand Down Expand Up @@ -237,6 +302,18 @@ export class BasicPrettyPrinter {
const groupLeft = binaryExpressionGroup(left);
const groupRight = binaryExpressionGroup(right);

if (
node.name === '*' &&
((isIntegerLiteral(left) && Math.abs(left.value) === 1) ||
(isIntegerLiteral(right) && Math.abs(right.value) === 1))
) {
const formatted = this.simplifyMultiplicationByOne(node);

if (formatted) {
return formatted;
}
}

let leftFormatted = ctx.visitArgument(0);
let rightFormatted = ctx.visitArgument(1);

Expand Down
2 changes: 1 addition & 1 deletion packages/kbn-esql-ast/src/pretty_print/helpers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* License v3.0 only", or the "Server Side Public License, v 1".
*/

import { ESQLAstBaseItem, ESQLProperNode } from '../types';
import type { ESQLAstBaseItem, ESQLProperNode } from '../types';
import { Walker } from '../walker';

export interface QueryPrettyPrintStats {
Expand Down
4 changes: 4 additions & 0 deletions packages/kbn-esql-ast/src/visitor/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,10 @@ export const firstItem = (items: ESQLAstItem[]): ESQLSingleAstItem | undefined =
}
};

export const resolveItem = (items: ESQLAstItem | ESQLAstItem[]): ESQLAstItem => {
return Array.isArray(items) ? resolveItem(items[0]) : items;
};

/**
* Returns the last normalized "single item" from the "item" list.
*
Expand Down

0 comments on commit b3fef2a

Please sign in to comment.