Skip to content

Commit

Permalink
[8.x] [ES|QL] detect the type of `COUNT(*)` (#197914) (#198029
Browse files Browse the repository at this point in the history
)

# Backport

This will backport the following commits from `main` to `8.x`:
- [[ES|QL] detect the type of `COUNT(*)`
(#197914)](#197914)

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

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

<!--BACKPORT [{"author":{"name":"Drew
Tate","email":"[email protected]"},"sourceCommit":{"committedDate":"2024-10-28T15:31:48Z","message":"[ES|QL]
detect the type of `COUNT(*)` (#197914)\n\n## Summary\r\n\r\nWe weren't
properly detecting the type of the expression `COUNT(*)`. Now\r\nwe
are!\r\n\r\nBefore:\r\n<img width=\"950\" alt=\"Screenshot 2024-10-25 at
4 38
08 PM\"\r\nsrc=\"https://github.com/user-attachments/assets/e9bd8d78-d0c8-4069-a818-5bf3486b925b\">\r\n\r\nAfter:\r\n<img
width=\"1093\" alt=\"Screenshot 2024-10-25 at 4 35
44 PM\"\r\nsrc=\"https://github.com/user-attachments/assets/235c63dc-7d6c-49df-9adf-e225c4550a42\">\r\n\r\n\r\n###
Checklist\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\nCo-authored-by: Stratoula Kalafateli
<[email protected]>","sha":"a5f0c1916e6348d82306e14c772c66f195ae781e","branchLabelMapping":{"^v9.0.0$":"main","^v8.17.0$":"8.x","^v(\\d+).(\\d+).\\d+$":"$1.$2"}},"sourcePullRequest":{"labels":["release_note:skip","v7.9.0","v9.0.0","Feature:ES|QL","Team:ESQL","v8.16.0","backport:version","v8.17.0"],"title":"[ES|QL]
detect the type of
`COUNT(*)`","number":197914,"url":"https://github.com/elastic/kibana/pull/197914","mergeCommit":{"message":"[ES|QL]
detect the type of `COUNT(*)` (#197914)\n\n## Summary\r\n\r\nWe weren't
properly detecting the type of the expression `COUNT(*)`. Now\r\nwe
are!\r\n\r\nBefore:\r\n<img width=\"950\" alt=\"Screenshot 2024-10-25 at
4 38
08 PM\"\r\nsrc=\"https://github.com/user-attachments/assets/e9bd8d78-d0c8-4069-a818-5bf3486b925b\">\r\n\r\nAfter:\r\n<img
width=\"1093\" alt=\"Screenshot 2024-10-25 at 4 35
44 PM\"\r\nsrc=\"https://github.com/user-attachments/assets/235c63dc-7d6c-49df-9adf-e225c4550a42\">\r\n\r\n\r\n###
Checklist\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\nCo-authored-by: Stratoula Kalafateli
<[email protected]>","sha":"a5f0c1916e6348d82306e14c772c66f195ae781e"}},"sourceBranch":"main","suggestedTargetBranches":["7.9","8.16","8.x"],"targetPullRequestStates":[{"branch":"7.9","label":"v7.9.0","branchLabelMappingKey":"^v(\\d+).(\\d+).\\d+$","isSourceBranch":false,"state":"NOT_CREATED"},{"branch":"main","label":"v9.0.0","branchLabelMappingKey":"^v9.0.0$","isSourceBranch":true,"state":"MERGED","url":"https://github.com/elastic/kibana/pull/197914","number":197914,"mergeCommit":{"message":"[ES|QL]
detect the type of `COUNT(*)` (#197914)\n\n## Summary\r\n\r\nWe weren't
properly detecting the type of the expression `COUNT(*)`. Now\r\nwe
are!\r\n\r\nBefore:\r\n<img width=\"950\" alt=\"Screenshot 2024-10-25 at
4 38
08 PM\"\r\nsrc=\"https://github.com/user-attachments/assets/e9bd8d78-d0c8-4069-a818-5bf3486b925b\">\r\n\r\nAfter:\r\n<img
width=\"1093\" alt=\"Screenshot 2024-10-25 at 4 35
44 PM\"\r\nsrc=\"https://github.com/user-attachments/assets/235c63dc-7d6c-49df-9adf-e225c4550a42\">\r\n\r\n\r\n###
Checklist\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\nCo-authored-by: Stratoula Kalafateli
<[email protected]>","sha":"a5f0c1916e6348d82306e14c772c66f195ae781e"}},{"branch":"8.16","label":"v8.16.0","branchLabelMappingKey":"^v(\\d+).(\\d+).\\d+$","isSourceBranch":false,"state":"NOT_CREATED"},{"branch":"8.x","label":"v8.17.0","branchLabelMappingKey":"^v8.17.0$","isSourceBranch":false,"state":"NOT_CREATED"}]}]
BACKPORT-->

Co-authored-by: Drew Tate <[email protected]>
  • Loading branch information
kibanamachine and drewdaemon authored Oct 28, 2024
1 parent 9cc1459 commit 038fece
Show file tree
Hide file tree
Showing 2 changed files with 53 additions and 55 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -225,79 +225,47 @@ describe('getExpressionType', () => {
});

it('detects the return type of a function', () => {
expect(
getExpressionType(getASTForExpression('returns_keyword()'), new Map(), new Map())
).toBe('keyword');
expect(getExpressionType(getASTForExpression('returns_keyword()'))).toBe('keyword');
});

it('selects the correct signature based on the arguments', () => {
expect(getExpressionType(getASTForExpression('test("foo")'), new Map(), new Map())).toBe(
'keyword'
);
expect(getExpressionType(getASTForExpression('test(1.)'), new Map(), new Map())).toBe(
'double'
);
expect(getExpressionType(getASTForExpression('test(1., "foo")'), new Map(), new Map())).toBe(
'long'
);
expect(getExpressionType(getASTForExpression('test("foo")'))).toBe('keyword');
expect(getExpressionType(getASTForExpression('test(1.)'))).toBe('double');
expect(getExpressionType(getASTForExpression('test(1., "foo")'))).toBe('long');
});

it('supports nested functions', () => {
expect(
getExpressionType(
getASTForExpression('test(1., test(test(test(returns_keyword()))))'),
new Map(),
new Map()
)
getExpressionType(getASTForExpression('test(1., test(test(test(returns_keyword()))))'))
).toBe('long');
});

it('supports functions with casted results', () => {
expect(
getExpressionType(getASTForExpression('test(1.)::keyword'), new Map(), new Map())
).toBe('keyword');
expect(getExpressionType(getASTForExpression('test(1.)::keyword'))).toBe('keyword');
});

it('handles nulls and string-date casting', () => {
expect(getExpressionType(getASTForExpression('test(NULL)'), new Map(), new Map())).toBe(
'null'
);
expect(getExpressionType(getASTForExpression('test(NULL, NULL)'), new Map(), new Map())).toBe(
'null'
);
expect(
getExpressionType(getASTForExpression('accepts_dates("", "")'), new Map(), new Map())
).toBe('keyword');
expect(getExpressionType(getASTForExpression('test(NULL)'))).toBe('null');
expect(getExpressionType(getASTForExpression('test(NULL, NULL)'))).toBe('null');
expect(getExpressionType(getASTForExpression('accepts_dates("", "")'))).toBe('keyword');
});

it('deals with functions that do not exist', () => {
expect(getExpressionType(getASTForExpression('does_not_exist()'), new Map(), new Map())).toBe(
'unknown'
);
expect(getExpressionType(getASTForExpression('does_not_exist()'))).toBe('unknown');
});

it('deals with bad function invocations', () => {
expect(
getExpressionType(getASTForExpression('test(1., "foo", "bar")'), new Map(), new Map())
).toBe('unknown');
expect(getExpressionType(getASTForExpression('test(1., "foo", "bar")'))).toBe('unknown');

expect(getExpressionType(getASTForExpression('test()'), new Map(), new Map())).toBe(
'unknown'
);
expect(getExpressionType(getASTForExpression('test()'))).toBe('unknown');

expect(getExpressionType(getASTForExpression('test("foo", 1.)'), new Map(), new Map())).toBe(
'unknown'
);
expect(getExpressionType(getASTForExpression('test("foo", 1.)'))).toBe('unknown');
});

it('deals with the CASE function', () => {
expect(getExpressionType(getASTForExpression('CASE(true, 1, 2)'), new Map(), new Map())).toBe(
'integer'
);
expect(getExpressionType(getASTForExpression('CASE(true, 1, 2)'))).toBe('integer');

expect(
getExpressionType(getASTForExpression('CASE(true, 1., true, 1., 2.)'), new Map(), new Map())
).toBe('double');
expect(getExpressionType(getASTForExpression('CASE(true, 1., true, 1., 2.)'))).toBe('double');

expect(
getExpressionType(
Expand All @@ -306,6 +274,20 @@ describe('getExpressionType', () => {
new Map()
)
).toBe('keyword');

expect(
getExpressionType(
getASTForExpression('CASE(true, "", true, "", keywordVar)'),
new Map(),
new Map([
[`keywordVar`, [{ name: 'keywordVar', type: 'keyword', location: { min: 0, max: 0 } }]],
])
)
).toBe('keyword');
});

it('supports COUNT(*)', () => {
expect(getExpressionType(getASTForExpression('COUNT(*)'))).toBe<SupportedDataType>('long');
});
});

Expand Down
32 changes: 24 additions & 8 deletions packages/kbn-esql-validation-autocomplete/src/shared/helpers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -815,15 +815,31 @@ export function getExpressionType(
return 'unknown';
}

/**
* Special case for COUNT(*) because
* the "*" column doesn't match any
* of COUNT's function definitions
*/
if (
fnDefinition.name === 'count' &&
root.args[0] &&
isColumnItem(root.args[0]) &&
root.args[0].name === '*'
) {
return 'long';
}

if (fnDefinition.name === 'case' && root.args.length) {
// The CASE function doesn't fit our system of function definitions
// and needs special handling. This is imperfect, but it's a start because
// at least we know that the final argument to case will never be a conditional
// expression, always a result expression.
//
// One problem with this is that if a false case is not provided, the return type
// will be null, which we aren't detecting. But this is ok because we consider
// variables and fields to be nullable anyways and account for that during validation.
/**
* The CASE function doesn't fit our system of function definitions
* and needs special handling. This is imperfect, but it's a start because
* at least we know that the final argument to case will never be a conditional
* expression, always a result expression.
*
* One problem with this is that if a false case is not provided, the return type
* will be null, which we aren't detecting. But this is ok because we consider
* variables and fields to be nullable anyways and account for that during validation.
*/
return getExpressionType(root.args[root.args.length - 1], fields, variables);
}

Expand Down

0 comments on commit 038fece

Please sign in to comment.