make completions smarter (#13830)

This commit is contained in:
Vadim Ogievetsky 2023-02-23 10:17:10 -08:00 committed by GitHub
parent 70f9052f1d
commit e4e6c7ed01
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
1 changed files with 89 additions and 49 deletions

View File

@ -72,9 +72,10 @@ export interface FlexibleQueryInputProps {
export interface FlexibleQueryInputState { export interface FlexibleQueryInputState {
// For reasons (https://github.com/securingsincity/react-ace/issues/415) react ace editor needs an explicit height // For reasons (https://github.com/securingsincity/react-ace/issues/415) react ace editor needs an explicit height
// Since this component will grown and shrink dynamically we will measure its height and then set it. // Since this component will grow and shrink dynamically we will measure its height and then set it.
editorHeight: number; editorHeight: number;
completions: any[]; quotedCompletions: Ace.Completion[];
unquotedCompletions: Ace.Completion[];
prevColumnMetadata?: readonly ColumnMetadata[]; prevColumnMetadata?: readonly ColumnMetadata[];
prevCurrentTable?: string; prevCurrentTable?: string;
prevCurrentSchema?: string; prevCurrentSchema?: string;
@ -89,7 +90,7 @@ export class FlexibleQueryInput extends React.PureComponent<
static replaceDefaultAutoCompleter(): void { static replaceDefaultAutoCompleter(): void {
if (!langTools) return; if (!langTools) return;
const keywordList = ([] as any[]).concat( const keywordList = ([] as Ace.Completion[]).concat(
SQL_KEYWORDS.map(v => ({ name: v, value: v, score: 0, meta: 'keyword' })), SQL_KEYWORDS.map(v => ({ name: v, value: v, score: 0, meta: 'keyword' })),
SQL_EXPRESSION_PARTS.map(v => ({ name: v, value: v, score: 0, meta: 'keyword' })), SQL_EXPRESSION_PARTS.map(v => ({ name: v, value: v, score: 0, meta: 'keyword' })),
SQL_CONSTANTS.map(v => ({ name: v, value: v, score: 0, meta: 'constant' })), SQL_CONSTANTS.map(v => ({ name: v, value: v, score: 0, meta: 'constant' })),
@ -108,7 +109,13 @@ export class FlexibleQueryInput extends React.PureComponent<
langTools.snippetCompleter, langTools.snippetCompleter,
langTools.textCompleter, langTools.textCompleter,
{ {
getCompletions: (_editor: any, _session: any, _pos: any, _prefix: any, callback: any) => { getCompletions: (
_state: string,
_session: Ace.EditSession,
_pos: Ace.Point,
_prefix: string,
callback: any,
) => {
return callback(null, keywordList); return callback(null, keywordList);
}, },
getDocTooltip: (item: any) => { getDocTooltip: (item: any) => {
@ -123,7 +130,8 @@ export class FlexibleQueryInput extends React.PureComponent<
static addFunctionAutoCompleter(): void { static addFunctionAutoCompleter(): void {
if (!langTools) return; if (!langTools) return;
const functionList: any[] = Object.entries(SQL_FUNCTIONS).flatMap(([name, versions]) => { const functionList: Ace.Completion[] = Object.entries(SQL_FUNCTIONS).flatMap(
([name, versions]) => {
return versions.map(([args, description]) => ({ return versions.map(([args, description]) => ({
name: name, name: name,
value: versions.length > 1 ? `${name}(${args})` : name, value: versions.length > 1 ? `${name}(${args})` : name,
@ -133,7 +141,8 @@ export class FlexibleQueryInput extends React.PureComponent<
description, description,
completer: COMPLETER, completer: COMPLETER,
})); }));
}); },
);
langTools.addCompleter({ langTools.addCompleter({
getCompletions: (_editor: any, _session: any, _pos: any, _prefix: any, callback: any) => { getCompletions: (_editor: any, _session: any, _pos: any, _prefix: any, callback: any) => {
@ -154,18 +163,15 @@ export class FlexibleQueryInput extends React.PureComponent<
<div class="doc-description">${item.description}</div>`; <div class="doc-description">${item.description}</div>`;
} }
static getDerivedStateFromProps(props: FlexibleQueryInputProps, state: FlexibleQueryInputState) { static getCompletions(
const { columnMetadata, currentSchema, currentTable } = props; columnMetadata: readonly ColumnMetadata[],
currentSchema: string | undefined,
if ( currentTable: string | undefined,
columnMetadata && quote: boolean,
(columnMetadata !== state.prevColumnMetadata || ): Ace.Completion[] {
currentSchema !== state.prevCurrentSchema || return ([] as Ace.Completion[]).concat(
currentTable !== state.prevCurrentTable)
) {
const completions = ([] as any[]).concat(
uniq(columnMetadata.map(d => d.TABLE_SCHEMA)).map(v => ({ uniq(columnMetadata.map(d => d.TABLE_SCHEMA)).map(v => ({
value: String(T(v)), value: quote ? String(T(v)) : v,
score: 10, score: 10,
meta: 'schema', meta: 'schema',
})), })),
@ -174,7 +180,7 @@ export class FlexibleQueryInput extends React.PureComponent<
.filter(d => (currentSchema ? d.TABLE_SCHEMA === currentSchema : true)) .filter(d => (currentSchema ? d.TABLE_SCHEMA === currentSchema : true))
.map(d => d.TABLE_NAME), .map(d => d.TABLE_NAME),
).map(v => ({ ).map(v => ({
value: String(T(v)), value: quote ? String(T(v)) : v,
score: 49, score: 49,
meta: 'datasource', meta: 'datasource',
})), })),
@ -187,14 +193,35 @@ export class FlexibleQueryInput extends React.PureComponent<
) )
.map(d => d.COLUMN_NAME), .map(d => d.COLUMN_NAME),
).map(v => ({ ).map(v => ({
value: String(C(v)), value: quote ? String(C(v)) : v,
score: 50, score: 50,
meta: 'column', meta: 'column',
})), })),
); );
}
static getDerivedStateFromProps(props: FlexibleQueryInputProps, state: FlexibleQueryInputState) {
const { columnMetadata, currentSchema, currentTable } = props;
if (
columnMetadata &&
(columnMetadata !== state.prevColumnMetadata ||
currentSchema !== state.prevCurrentSchema ||
currentTable !== state.prevCurrentTable)
) {
return { return {
completions, quotedCompletions: FlexibleQueryInput.getCompletions(
columnMetadata,
currentSchema,
currentTable,
true,
),
unquotedCompletions: FlexibleQueryInput.getCompletions(
columnMetadata,
currentSchema,
currentTable,
false,
),
prevColumnMetadata: columnMetadata, prevColumnMetadata: columnMetadata,
prevCurrentSchema: currentSchema, prevCurrentSchema: currentSchema,
prevCurrentTable: currentTable, prevCurrentTable: currentTable,
@ -207,7 +234,8 @@ export class FlexibleQueryInput extends React.PureComponent<
super(props, context); super(props, context);
this.state = { this.state = {
editorHeight: 200, editorHeight: 200,
completions: [], quotedCompletions: [],
unquotedCompletions: [],
}; };
} }
@ -216,8 +244,20 @@ export class FlexibleQueryInput extends React.PureComponent<
FlexibleQueryInput.addFunctionAutoCompleter(); FlexibleQueryInput.addFunctionAutoCompleter();
if (langTools) { if (langTools) {
langTools.addCompleter({ langTools.addCompleter({
getCompletions: (_editor: any, _session: any, _pos: any, _prefix: any, callback: any) => { getCompletions: (
callback(null, this.state.completions); _state: string,
session: Ace.EditSession,
pos: Ace.Point,
prefix: string,
callback: any,
) => {
const charBeforePrefix = session.getLine(pos.row)[pos.column - prefix.length - 1];
callback(
null,
charBeforePrefix === '"'
? this.state.unquotedCompletions
: this.state.quotedCompletions,
);
}, },
}); });
} }