@neo4j-cypher/language-support
Advanced tools
Comparing version 2.0.0-canary-dbe560f to 2.0.0-canary-dcbe67d
# @neo4j-cypher/language-support | ||
## 2.0.0-next.8 | ||
### Patch Changes | ||
- 05663bd: Fixes bug using non language keywords (EXPLAIN, PROFILE, etc) as symbolic names | ||
## 2.0.0-next.7 | ||
### Patch Changes | ||
- 3661e9d: Fixes database completions for CREATE ALIAS commands | ||
- b76af58: Fixes bug in signature help of functions nested inside procedure calls | ||
- 21699b7: Updates the semantic analysis module to use the antlr parser instead of the javaCC one | ||
- 6afc0e3: Adds signature information on auto-completions | ||
- 39b924d: Fixes bug in labels completion | ||
## 2.0.0-next.6 | ||
@@ -4,0 +20,0 @@ |
@@ -1,3 +0,3 @@ | ||
import { CompletionItem } from 'vscode-languageserver-types'; | ||
import { DbSchema } from '../dbSchema'; | ||
import { CompletionItem } from '../types'; | ||
export declare function autocomplete(query: string, dbSchema: DbSchema, caretPosition?: number, manual?: boolean): CompletionItem[]; |
@@ -1,4 +0,5 @@ | ||
import { CompletionItem } from 'vscode-languageserver-types'; | ||
import { Token } from 'antlr4'; | ||
import { DbSchema } from '../dbSchema'; | ||
import { ParsedStatement } from '../parserWrapper'; | ||
export declare function completionCoreCompletion(parsingResult: ParsedStatement, dbSchema: DbSchema, manualTrigger?: boolean): CompletionItem[]; | ||
import { CompletionItem } from '../types'; | ||
export declare function completionCoreCompletion(parsingResult: ParsedStatement, dbSchema: DbSchema, caretToken: Token, manualTrigger?: boolean): CompletionItem[]; |
@@ -36,279 +36,287 @@ import { ATN, CharStream, DFA, Lexer } from "antlr4"; | ||
static readonly AT = 33; | ||
static readonly BAR = 34; | ||
static readonly BINDINGS = 35; | ||
static readonly BOOL = 36; | ||
static readonly BOOLEAN = 37; | ||
static readonly BOOSTED = 38; | ||
static readonly BOTH = 39; | ||
static readonly BREAK = 40; | ||
static readonly BRIEF = 41; | ||
static readonly BTREE = 42; | ||
static readonly BUILT = 43; | ||
static readonly BY = 44; | ||
static readonly CALL = 45; | ||
static readonly CASE = 46; | ||
static readonly CHANGE = 47; | ||
static readonly CIDR = 48; | ||
static readonly COLLECT = 49; | ||
static readonly COLON = 50; | ||
static readonly COLONCOLON = 51; | ||
static readonly COMMA = 52; | ||
static readonly COMMAND = 53; | ||
static readonly COMMANDS = 54; | ||
static readonly COMMIT = 55; | ||
static readonly COMPOSITE = 56; | ||
static readonly CONCURRENT = 57; | ||
static readonly CONSTRAINT = 58; | ||
static readonly CONSTRAINTS = 59; | ||
static readonly CONTAINS = 60; | ||
static readonly COPY = 61; | ||
static readonly CONTINUE = 62; | ||
static readonly COUNT = 63; | ||
static readonly CREATE = 64; | ||
static readonly CSV = 65; | ||
static readonly CURRENT = 66; | ||
static readonly DATA = 67; | ||
static readonly DATABASE = 68; | ||
static readonly DATABASES = 69; | ||
static readonly DATE = 70; | ||
static readonly DATETIME = 71; | ||
static readonly DBMS = 72; | ||
static readonly DEALLOCATE = 73; | ||
static readonly DEFAULT = 74; | ||
static readonly DEFINED = 75; | ||
static readonly DELETE = 76; | ||
static readonly DENY = 77; | ||
static readonly DESC = 78; | ||
static readonly DESCENDING = 79; | ||
static readonly DESTROY = 80; | ||
static readonly DETACH = 81; | ||
static readonly DIFFERENT = 82; | ||
static readonly DOLLAR = 83; | ||
static readonly DISTINCT = 84; | ||
static readonly DIVIDE = 85; | ||
static readonly DOT = 86; | ||
static readonly DOTDOT = 87; | ||
static readonly DOUBLEBAR = 88; | ||
static readonly DRIVER = 89; | ||
static readonly DROP = 90; | ||
static readonly DRYRUN = 91; | ||
static readonly DUMP = 92; | ||
static readonly DURATION = 93; | ||
static readonly EACH = 94; | ||
static readonly EDGE = 95; | ||
static readonly ENABLE = 96; | ||
static readonly ELEMENT = 97; | ||
static readonly ELEMENTS = 98; | ||
static readonly ELSE = 99; | ||
static readonly ENCRYPTED = 100; | ||
static readonly END = 101; | ||
static readonly ENDS = 102; | ||
static readonly EQ = 103; | ||
static readonly EXECUTABLE = 104; | ||
static readonly EXECUTE = 105; | ||
static readonly EXIST = 106; | ||
static readonly EXISTENCE = 107; | ||
static readonly EXISTS = 108; | ||
static readonly ERROR = 109; | ||
static readonly FAIL = 110; | ||
static readonly FALSE = 111; | ||
static readonly FIELDTERMINATOR = 112; | ||
static readonly FINISH = 113; | ||
static readonly FLOAT = 114; | ||
static readonly FOR = 115; | ||
static readonly FOREACH = 116; | ||
static readonly FROM = 117; | ||
static readonly FULLTEXT = 118; | ||
static readonly FUNCTION = 119; | ||
static readonly FUNCTIONS = 120; | ||
static readonly GE = 121; | ||
static readonly GRANT = 122; | ||
static readonly GRAPH = 123; | ||
static readonly GRAPHS = 124; | ||
static readonly GROUP = 125; | ||
static readonly GROUPS = 126; | ||
static readonly GT = 127; | ||
static readonly HEADERS = 128; | ||
static readonly HOME = 129; | ||
static readonly IF = 130; | ||
static readonly IMPERSONATE = 131; | ||
static readonly IMMUTABLE = 132; | ||
static readonly IN = 133; | ||
static readonly INDEX = 134; | ||
static readonly INDEXES = 135; | ||
static readonly INF = 136; | ||
static readonly INFINITY = 137; | ||
static readonly INSERT = 138; | ||
static readonly INT = 139; | ||
static readonly INTEGER = 140; | ||
static readonly IS = 141; | ||
static readonly JOIN = 142; | ||
static readonly KEY = 143; | ||
static readonly LABEL = 144; | ||
static readonly LABELS = 145; | ||
static readonly AMPERSAND = 146; | ||
static readonly EXCLAMATION_MARK = 147; | ||
static readonly LBRACKET = 148; | ||
static readonly LCURLY = 149; | ||
static readonly LE = 150; | ||
static readonly LEADING = 151; | ||
static readonly LIMITROWS = 152; | ||
static readonly LIST = 153; | ||
static readonly LOAD = 154; | ||
static readonly LOCAL = 155; | ||
static readonly LOOKUP = 156; | ||
static readonly LPAREN = 157; | ||
static readonly LT = 158; | ||
static readonly MANAGEMENT = 159; | ||
static readonly MAP = 160; | ||
static readonly MATCH = 161; | ||
static readonly MERGE = 162; | ||
static readonly MINUS = 163; | ||
static readonly PERCENT = 164; | ||
static readonly INVALID_NEQ = 165; | ||
static readonly NEQ = 166; | ||
static readonly NAME = 167; | ||
static readonly NAMES = 168; | ||
static readonly NAN = 169; | ||
static readonly NFC = 170; | ||
static readonly NFD = 171; | ||
static readonly NFKC = 172; | ||
static readonly NFKD = 173; | ||
static readonly NEW = 174; | ||
static readonly NODE = 175; | ||
static readonly NODETACH = 176; | ||
static readonly NODES = 177; | ||
static readonly NONE = 178; | ||
static readonly NORMALIZE = 179; | ||
static readonly NORMALIZED = 180; | ||
static readonly NOT = 181; | ||
static readonly NOTHING = 182; | ||
static readonly NOWAIT = 183; | ||
static readonly NULL = 184; | ||
static readonly OF = 185; | ||
static readonly ON = 186; | ||
static readonly ONLY = 187; | ||
static readonly OPTIONAL = 188; | ||
static readonly OPTIONS = 189; | ||
static readonly OPTION = 190; | ||
static readonly OR = 191; | ||
static readonly ORDER = 192; | ||
static readonly OUTPUT = 193; | ||
static readonly PASSWORD = 194; | ||
static readonly PASSWORDS = 195; | ||
static readonly PATH = 196; | ||
static readonly PATHS = 197; | ||
static readonly PERIODIC = 198; | ||
static readonly PLAINTEXT = 199; | ||
static readonly PLUS = 200; | ||
static readonly PLUSEQUAL = 201; | ||
static readonly POINT = 202; | ||
static readonly POPULATED = 203; | ||
static readonly POW = 204; | ||
static readonly PRIMARY = 205; | ||
static readonly PRIMARIES = 206; | ||
static readonly PRIVILEGE = 207; | ||
static readonly PRIVILEGES = 208; | ||
static readonly PROCEDURE = 209; | ||
static readonly PROCEDURES = 210; | ||
static readonly PROPERTIES = 211; | ||
static readonly PROPERTY = 212; | ||
static readonly QUESTION = 213; | ||
static readonly RANGE = 214; | ||
static readonly RBRACKET = 215; | ||
static readonly RCURLY = 216; | ||
static readonly READ = 217; | ||
static readonly REALLOCATE = 218; | ||
static readonly REDUCE = 219; | ||
static readonly RENAME = 220; | ||
static readonly REGEQ = 221; | ||
static readonly REL = 222; | ||
static readonly RELATIONSHIP = 223; | ||
static readonly RELATIONSHIPS = 224; | ||
static readonly REMOVE = 225; | ||
static readonly REPEATABLE = 226; | ||
static readonly REPLACE = 227; | ||
static readonly REPORT = 228; | ||
static readonly REQUIRE = 229; | ||
static readonly REQUIRED = 230; | ||
static readonly RETURN = 231; | ||
static readonly REVOKE = 232; | ||
static readonly ROLE = 233; | ||
static readonly ROLES = 234; | ||
static readonly ROW = 235; | ||
static readonly ROWS = 236; | ||
static readonly RPAREN = 237; | ||
static readonly SCAN = 238; | ||
static readonly SEC = 239; | ||
static readonly SECOND = 240; | ||
static readonly SECONDARY = 241; | ||
static readonly SECONDARIES = 242; | ||
static readonly SECONDS = 243; | ||
static readonly SEEK = 244; | ||
static readonly SEMICOLON = 245; | ||
static readonly SERVER = 246; | ||
static readonly SERVERS = 247; | ||
static readonly SET = 248; | ||
static readonly SETTING = 249; | ||
static readonly SETTINGS = 250; | ||
static readonly SHORTEST_PATH = 251; | ||
static readonly SHORTEST = 252; | ||
static readonly SHOW = 253; | ||
static readonly SIGNED = 254; | ||
static readonly SINGLE = 255; | ||
static readonly SKIPROWS = 256; | ||
static readonly START = 257; | ||
static readonly STARTS = 258; | ||
static readonly STATUS = 259; | ||
static readonly STOP = 260; | ||
static readonly STRING = 261; | ||
static readonly SUPPORTED = 262; | ||
static readonly SUSPENDED = 263; | ||
static readonly TARGET = 264; | ||
static readonly TERMINATE = 265; | ||
static readonly TEXT = 266; | ||
static readonly THEN = 267; | ||
static readonly TIME = 268; | ||
static readonly TIMES = 269; | ||
static readonly TIMESTAMP = 270; | ||
static readonly TIMEZONE = 271; | ||
static readonly TO = 272; | ||
static readonly TOPOLOGY = 273; | ||
static readonly TRAILING = 274; | ||
static readonly TRANSACTION = 275; | ||
static readonly TRANSACTIONS = 276; | ||
static readonly TRAVERSE = 277; | ||
static readonly TRIM = 278; | ||
static readonly TRUE = 279; | ||
static readonly TYPE = 280; | ||
static readonly TYPED = 281; | ||
static readonly TYPES = 282; | ||
static readonly UNION = 283; | ||
static readonly UNIQUE = 284; | ||
static readonly UNIQUENESS = 285; | ||
static readonly UNWIND = 286; | ||
static readonly URL = 287; | ||
static readonly USE = 288; | ||
static readonly USER = 289; | ||
static readonly USERS = 290; | ||
static readonly USING = 291; | ||
static readonly VALUE = 292; | ||
static readonly VARCHAR = 293; | ||
static readonly VECTOR = 294; | ||
static readonly VERBOSE = 295; | ||
static readonly VERTEX = 296; | ||
static readonly WAIT = 297; | ||
static readonly WHEN = 298; | ||
static readonly WHERE = 299; | ||
static readonly WITH = 300; | ||
static readonly WITHOUT = 301; | ||
static readonly WRITE = 302; | ||
static readonly XOR = 303; | ||
static readonly YIELD = 304; | ||
static readonly ZONED = 305; | ||
static readonly IDENTIFIER = 306; | ||
static readonly ARROW_LINE = 307; | ||
static readonly ARROW_LEFT_HEAD = 308; | ||
static readonly ARROW_RIGHT_HEAD = 309; | ||
static readonly ErrorChar = 310; | ||
static readonly AUTH = 34; | ||
static readonly BAR = 35; | ||
static readonly BINDINGS = 36; | ||
static readonly BOOL = 37; | ||
static readonly BOOLEAN = 38; | ||
static readonly BOOSTED = 39; | ||
static readonly BOTH = 40; | ||
static readonly BREAK = 41; | ||
static readonly BRIEF = 42; | ||
static readonly BTREE = 43; | ||
static readonly BUILT = 44; | ||
static readonly BY = 45; | ||
static readonly CALL = 46; | ||
static readonly CASCADE = 47; | ||
static readonly CASE = 48; | ||
static readonly CHANGE = 49; | ||
static readonly CIDR = 50; | ||
static readonly COLLECT = 51; | ||
static readonly COLON = 52; | ||
static readonly COLONCOLON = 53; | ||
static readonly COMMA = 54; | ||
static readonly COMMAND = 55; | ||
static readonly COMMANDS = 56; | ||
static readonly COMMIT = 57; | ||
static readonly COMPOSITE = 58; | ||
static readonly CONCURRENT = 59; | ||
static readonly CONSTRAINT = 60; | ||
static readonly CONSTRAINTS = 61; | ||
static readonly CONTAINS = 62; | ||
static readonly COPY = 63; | ||
static readonly CONTINUE = 64; | ||
static readonly COUNT = 65; | ||
static readonly CREATE = 66; | ||
static readonly CSV = 67; | ||
static readonly CURRENT = 68; | ||
static readonly DATA = 69; | ||
static readonly DATABASE = 70; | ||
static readonly DATABASES = 71; | ||
static readonly DATE = 72; | ||
static readonly DATETIME = 73; | ||
static readonly DBMS = 74; | ||
static readonly DEALLOCATE = 75; | ||
static readonly DEFAULT = 76; | ||
static readonly DEFINED = 77; | ||
static readonly DELETE = 78; | ||
static readonly DENY = 79; | ||
static readonly DESC = 80; | ||
static readonly DESCENDING = 81; | ||
static readonly DESTROY = 82; | ||
static readonly DETACH = 83; | ||
static readonly DIFFERENT = 84; | ||
static readonly DOLLAR = 85; | ||
static readonly DISTINCT = 86; | ||
static readonly DIVIDE = 87; | ||
static readonly DOT = 88; | ||
static readonly DOTDOT = 89; | ||
static readonly DOUBLEBAR = 90; | ||
static readonly DRIVER = 91; | ||
static readonly DROP = 92; | ||
static readonly DRYRUN = 93; | ||
static readonly DUMP = 94; | ||
static readonly DURATION = 95; | ||
static readonly EACH = 96; | ||
static readonly EDGE = 97; | ||
static readonly ENABLE = 98; | ||
static readonly ELEMENT = 99; | ||
static readonly ELEMENTS = 100; | ||
static readonly ELSE = 101; | ||
static readonly ENCRYPTED = 102; | ||
static readonly END = 103; | ||
static readonly ENDS = 104; | ||
static readonly EQ = 105; | ||
static readonly EXECUTABLE = 106; | ||
static readonly EXECUTE = 107; | ||
static readonly EXIST = 108; | ||
static readonly EXISTENCE = 109; | ||
static readonly EXISTS = 110; | ||
static readonly ERROR = 111; | ||
static readonly FAIL = 112; | ||
static readonly FALSE = 113; | ||
static readonly FIELDTERMINATOR = 114; | ||
static readonly FINISH = 115; | ||
static readonly FLOAT = 116; | ||
static readonly FOR = 117; | ||
static readonly FOREACH = 118; | ||
static readonly FROM = 119; | ||
static readonly FULLTEXT = 120; | ||
static readonly FUNCTION = 121; | ||
static readonly FUNCTIONS = 122; | ||
static readonly GE = 123; | ||
static readonly GRANT = 124; | ||
static readonly GRAPH = 125; | ||
static readonly GRAPHS = 126; | ||
static readonly GROUP = 127; | ||
static readonly GROUPS = 128; | ||
static readonly GT = 129; | ||
static readonly HEADERS = 130; | ||
static readonly HOME = 131; | ||
static readonly ID = 132; | ||
static readonly IF = 133; | ||
static readonly IMPERSONATE = 134; | ||
static readonly IMMUTABLE = 135; | ||
static readonly IN = 136; | ||
static readonly INDEX = 137; | ||
static readonly INDEXES = 138; | ||
static readonly INF = 139; | ||
static readonly INFINITY = 140; | ||
static readonly INSERT = 141; | ||
static readonly INT = 142; | ||
static readonly INTEGER = 143; | ||
static readonly IS = 144; | ||
static readonly JOIN = 145; | ||
static readonly KEY = 146; | ||
static readonly LABEL = 147; | ||
static readonly LABELS = 148; | ||
static readonly AMPERSAND = 149; | ||
static readonly EXCLAMATION_MARK = 150; | ||
static readonly LBRACKET = 151; | ||
static readonly LCURLY = 152; | ||
static readonly LE = 153; | ||
static readonly LEADING = 154; | ||
static readonly LIMITROWS = 155; | ||
static readonly LIST = 156; | ||
static readonly LOAD = 157; | ||
static readonly LOCAL = 158; | ||
static readonly LOOKUP = 159; | ||
static readonly LPAREN = 160; | ||
static readonly LT = 161; | ||
static readonly MANAGEMENT = 162; | ||
static readonly MAP = 163; | ||
static readonly MATCH = 164; | ||
static readonly MERGE = 165; | ||
static readonly MINUS = 166; | ||
static readonly PERCENT = 167; | ||
static readonly INVALID_NEQ = 168; | ||
static readonly NEQ = 169; | ||
static readonly NAME = 170; | ||
static readonly NAMES = 171; | ||
static readonly NAN = 172; | ||
static readonly NFC = 173; | ||
static readonly NFD = 174; | ||
static readonly NFKC = 175; | ||
static readonly NFKD = 176; | ||
static readonly NEW = 177; | ||
static readonly NODE = 178; | ||
static readonly NODETACH = 179; | ||
static readonly NODES = 180; | ||
static readonly NONE = 181; | ||
static readonly NORMALIZE = 182; | ||
static readonly NORMALIZED = 183; | ||
static readonly NOT = 184; | ||
static readonly NOTHING = 185; | ||
static readonly NOWAIT = 186; | ||
static readonly NULL = 187; | ||
static readonly OF = 188; | ||
static readonly OFFSET = 189; | ||
static readonly ON = 190; | ||
static readonly ONLY = 191; | ||
static readonly OPTIONAL = 192; | ||
static readonly OPTIONS = 193; | ||
static readonly OPTION = 194; | ||
static readonly OR = 195; | ||
static readonly ORDER = 196; | ||
static readonly OUTPUT = 197; | ||
static readonly PASSWORD = 198; | ||
static readonly PASSWORDS = 199; | ||
static readonly PATH = 200; | ||
static readonly PATHS = 201; | ||
static readonly PERIODIC = 202; | ||
static readonly PLAINTEXT = 203; | ||
static readonly PLUS = 204; | ||
static readonly PLUSEQUAL = 205; | ||
static readonly POINT = 206; | ||
static readonly POPULATED = 207; | ||
static readonly POW = 208; | ||
static readonly PRIMARY = 209; | ||
static readonly PRIMARIES = 210; | ||
static readonly PRIVILEGE = 211; | ||
static readonly PRIVILEGES = 212; | ||
static readonly PROCEDURE = 213; | ||
static readonly PROCEDURES = 214; | ||
static readonly PROPERTIES = 215; | ||
static readonly PROPERTY = 216; | ||
static readonly PROVIDER = 217; | ||
static readonly PROVIDERS = 218; | ||
static readonly QUESTION = 219; | ||
static readonly RANGE = 220; | ||
static readonly RBRACKET = 221; | ||
static readonly RCURLY = 222; | ||
static readonly READ = 223; | ||
static readonly REALLOCATE = 224; | ||
static readonly REDUCE = 225; | ||
static readonly RENAME = 226; | ||
static readonly REGEQ = 227; | ||
static readonly REL = 228; | ||
static readonly RELATIONSHIP = 229; | ||
static readonly RELATIONSHIPS = 230; | ||
static readonly REMOVE = 231; | ||
static readonly REPEATABLE = 232; | ||
static readonly REPLACE = 233; | ||
static readonly REPORT = 234; | ||
static readonly REQUIRE = 235; | ||
static readonly REQUIRED = 236; | ||
static readonly RESTRICT = 237; | ||
static readonly RETURN = 238; | ||
static readonly REVOKE = 239; | ||
static readonly ROLE = 240; | ||
static readonly ROLES = 241; | ||
static readonly ROW = 242; | ||
static readonly ROWS = 243; | ||
static readonly RPAREN = 244; | ||
static readonly SCAN = 245; | ||
static readonly SEC = 246; | ||
static readonly SECOND = 247; | ||
static readonly SECONDARY = 248; | ||
static readonly SECONDARIES = 249; | ||
static readonly SECONDS = 250; | ||
static readonly SEEK = 251; | ||
static readonly SEMICOLON = 252; | ||
static readonly SERVER = 253; | ||
static readonly SERVERS = 254; | ||
static readonly SET = 255; | ||
static readonly SETTING = 256; | ||
static readonly SETTINGS = 257; | ||
static readonly SHORTEST_PATH = 258; | ||
static readonly SHORTEST = 259; | ||
static readonly SHOW = 260; | ||
static readonly SIGNED = 261; | ||
static readonly SINGLE = 262; | ||
static readonly SKIPROWS = 263; | ||
static readonly START = 264; | ||
static readonly STARTS = 265; | ||
static readonly STATUS = 266; | ||
static readonly STOP = 267; | ||
static readonly STRING = 268; | ||
static readonly SUPPORTED = 269; | ||
static readonly SUSPENDED = 270; | ||
static readonly TARGET = 271; | ||
static readonly TERMINATE = 272; | ||
static readonly TEXT = 273; | ||
static readonly THEN = 274; | ||
static readonly TIME = 275; | ||
static readonly TIMES = 276; | ||
static readonly TIMESTAMP = 277; | ||
static readonly TIMEZONE = 278; | ||
static readonly TO = 279; | ||
static readonly TOPOLOGY = 280; | ||
static readonly TRAILING = 281; | ||
static readonly TRANSACTION = 282; | ||
static readonly TRANSACTIONS = 283; | ||
static readonly TRAVERSE = 284; | ||
static readonly TRIM = 285; | ||
static readonly TRUE = 286; | ||
static readonly TYPE = 287; | ||
static readonly TYPED = 288; | ||
static readonly TYPES = 289; | ||
static readonly UNION = 290; | ||
static readonly UNIQUE = 291; | ||
static readonly UNIQUENESS = 292; | ||
static readonly UNWIND = 293; | ||
static readonly URL = 294; | ||
static readonly USE = 295; | ||
static readonly USER = 296; | ||
static readonly USERS = 297; | ||
static readonly USING = 298; | ||
static readonly VALUE = 299; | ||
static readonly VARCHAR = 300; | ||
static readonly VECTOR = 301; | ||
static readonly VERBOSE = 302; | ||
static readonly VERTEX = 303; | ||
static readonly WAIT = 304; | ||
static readonly WHEN = 305; | ||
static readonly WHERE = 306; | ||
static readonly WITH = 307; | ||
static readonly WITHOUT = 308; | ||
static readonly WRITE = 309; | ||
static readonly XOR = 310; | ||
static readonly YIELD = 311; | ||
static readonly ZONE = 312; | ||
static readonly ZONED = 313; | ||
static readonly IDENTIFIER = 314; | ||
static readonly ARROW_LINE = 315; | ||
static readonly ARROW_LEFT_HEAD = 316; | ||
static readonly ARROW_RIGHT_HEAD = 317; | ||
static readonly ErrorChar = 318; | ||
static readonly EOF: number; | ||
@@ -315,0 +323,0 @@ static readonly channelNames: string[]; |
@@ -7,3 +7,3 @@ export type { ParserRuleContext } from 'antlr4'; | ||
export { CypherTokenType, lexerSymbols } from './lexerSymbols'; | ||
export { parse, parserWrapper } from './parserWrapper'; | ||
export { parse, parserWrapper, parseStatementsStrs } from './parserWrapper'; | ||
export { signatureHelp, toSignatureInformation } from './signatureHelp'; | ||
@@ -15,5 +15,6 @@ export { applySyntaxColouring, mapCypherToSemanticTokenIndex, syntaxColouringLegend, } from './syntaxColouring/syntaxColouring'; | ||
export { testData } from './tests/testData'; | ||
export type { Neo4jFunction, Neo4jProcedure } from './types'; | ||
export { textMateGrammar } from './textMateGrammar'; | ||
export type { CompletionItem, Neo4jFunction, Neo4jProcedure } from './types'; | ||
export { CypherLexer, CypherParser }; | ||
import CypherLexer from './generated-parser/CypherCmdLexer'; | ||
import CypherParser from './generated-parser/CypherCmdParser'; |
@@ -40,1 +40,2 @@ export declare enum CypherTokenType { | ||
export declare const keywordNames: Set<string>; | ||
export declare const operatorSymbols: Set<string>; |
@@ -9,6 +9,8 @@ import type { ParserRuleContext, Token } from 'antlr4'; | ||
ctx: StatementsOrCommandsContext; | ||
diagnostics: SyntaxDiagnostic[]; | ||
syntaxErrors: SyntaxDiagnostic[]; | ||
stopNode: ParserRuleContext; | ||
collectedLabelOrRelTypes: LabelOrRelType[]; | ||
collectedVariables: string[]; | ||
collectedFunctions: ParsedFunction[]; | ||
collectedProcedures: ParsedProcedure[]; | ||
} | ||
@@ -43,3 +45,15 @@ export interface ParsingResult { | ||
}; | ||
export type ParsedFunction = { | ||
name: string; | ||
rawText: string; | ||
line: number; | ||
column: number; | ||
offsets: { | ||
start: number; | ||
end: number; | ||
}; | ||
}; | ||
export type ParsedProcedure = ParsedFunction; | ||
export declare function createParsingScaffolding(query: string): ParsingScaffolding; | ||
export declare function parseStatementsStrs(query: string): string[]; | ||
export declare function parse(query: string): StatementOrCommandContext[]; | ||
@@ -46,0 +60,0 @@ export declare function createParsingResult(query: string): ParsingResult; |
@@ -1,3 +0,3 @@ | ||
import { CompletionItem } from 'vscode-languageserver-types'; | ||
import { DbSchema } from '../../dbSchema'; | ||
import { CompletionItem } from '../../types'; | ||
export declare function testCompletionsExactly({ query, offset, dbSchema, expected, }: { | ||
@@ -4,0 +4,0 @@ query: string; |
@@ -0,1 +1,2 @@ | ||
import { CompletionItem as VSCodeCompletionItem } from 'vscode-languageserver-types'; | ||
export type ReturnDescription = { | ||
@@ -34,3 +35,7 @@ name: string; | ||
aggregating: boolean; | ||
isDeprecated: boolean; | ||
}; | ||
export type CompletionItem = VSCodeCompletionItem & { | ||
signature?: string; | ||
}; | ||
export {}; |
@@ -21,3 +21,3 @@ { | ||
], | ||
"version": "2.0.0-canary-dbe560f", | ||
"version": "2.0.0-canary-dcbe67d", | ||
"main": "./dist/cjs/index.cjs", | ||
@@ -59,9 +59,8 @@ "module": "./dist/esm/index.mjs", | ||
"clean": "rm -rf {dist,src/generated-parser}", | ||
"test": "jest" | ||
"test": "vitest run" | ||
}, | ||
"devDependencies": { | ||
"@types/benchmark": "^2.1.5", | ||
"@types/jest": "^29.5.5", | ||
"benchmark": "^2.1.4" | ||
} | ||
} |
@@ -1,6 +0,5 @@ | ||
import { CompletionItem } from 'vscode-languageserver-types'; | ||
import { DbSchema } from '../dbSchema'; | ||
import { findCaret } from '../helpers'; | ||
import { parserWrapper } from '../parserWrapper'; | ||
import { CompletionItem } from '../types'; | ||
import { completionCoreCompletion } from './completionCoreCompletions'; | ||
@@ -15,4 +14,3 @@ | ||
const parsingResult = parserWrapper.parse(query); | ||
/* We try to locate the latest statement by finding the latest available `;` | ||
in the query and take from that point to the end of the query | ||
/* We try to locate the statement where the caret is and the token of the caret | ||
@@ -25,4 +23,2 @@ The reason for doing that is we need a way to "resynchronise" when the | ||
If there was no ;, we don't want to reparse, so we return undefined | ||
inside findLatestStatement | ||
*/ | ||
@@ -32,3 +28,4 @@ const caret = findCaret(parsingResult, caretPosition); | ||
const statement = caret.statement; | ||
return completionCoreCompletion(statement, dbSchema, manual); | ||
const caretToken = caret.token; | ||
return completionCoreCompletion(statement, dbSchema, caretToken, manual); | ||
} | ||
@@ -35,0 +32,0 @@ |
@@ -5,4 +5,4 @@ import { Token } from 'antlr4'; | ||
import { | ||
CompletionItem, | ||
CompletionItemKind, | ||
CompletionItemTag, | ||
InsertTextFormat, | ||
@@ -26,2 +26,3 @@ } from 'vscode-languageserver-types'; | ||
import { _internalFeatureFlags } from '../featureFlags'; | ||
import { CompletionItem, Neo4jFunction, Neo4jProcedure } from '../types'; | ||
@@ -46,7 +47,7 @@ const uniq = <T>(arr: T[]) => Array.from(new Set(arr)); | ||
dbSchema: DbSchema, | ||
) => | ||
): CompletionItem[] => | ||
namespacedCompletion( | ||
candidateRule, | ||
tokens, | ||
Object.keys(dbSchema?.functions ?? {}), | ||
dbSchema?.functions ?? {}, | ||
'function', | ||
@@ -59,16 +60,54 @@ ); | ||
dbSchema: DbSchema, | ||
) => | ||
): CompletionItem[] => | ||
namespacedCompletion( | ||
candidateRule, | ||
tokens, | ||
Object.keys(dbSchema?.procedures ?? {}), | ||
dbSchema?.procedures ?? {}, | ||
'procedure', | ||
); | ||
function isDeprecated(arg: Neo4jFunction | Neo4jProcedure | undefined) { | ||
if ('option' in arg) { | ||
return arg.option.deprecated; | ||
} else if ('isDeprecated' in arg) { | ||
return arg.isDeprecated; | ||
} else { | ||
return false; | ||
} | ||
} | ||
function getMethodCompletionItem( | ||
label: string, | ||
fullName: string, | ||
signatures: Record<string, Neo4jFunction | Neo4jProcedure>, | ||
type: 'procedure' | 'function', | ||
kind: CompletionItemKind, | ||
): CompletionItem { | ||
const maybeSignature = signatures[fullName]; | ||
const typeDetail = type === 'procedure' ? '(procedure)' : '(function)'; | ||
const deprecated = isDeprecated(maybeSignature); | ||
const maybeTags: { tags?: CompletionItemTag[] } = deprecated | ||
? { tags: [CompletionItemTag.Deprecated] } | ||
: {}; | ||
const maybeMethodSignature = maybeSignature | ||
? { signature: maybeSignature.signature } | ||
: {}; | ||
return { | ||
...maybeTags, | ||
...maybeMethodSignature, | ||
label, | ||
kind, | ||
detail: typeDetail, | ||
documentation: maybeSignature?.description ?? '', | ||
}; | ||
} | ||
const namespacedCompletion = ( | ||
candidateRule: CandidateRule, | ||
tokens: Token[], | ||
fullNames: string[], | ||
signatures: Record<string, Neo4jFunction> | Record<string, Neo4jProcedure>, | ||
type: 'procedure' | 'function', | ||
) => { | ||
): CompletionItem[] => { | ||
const fullNames = Object.keys(signatures); | ||
const namespacePrefix = calculateNamespacePrefix(candidateRule, tokens); | ||
@@ -83,3 +122,2 @@ if (namespacePrefix === null) { | ||
: CompletionItemKind.Function; | ||
const detail = type === 'procedure' ? '(procedure)' : '(function)'; | ||
@@ -93,4 +131,18 @@ if (namespacePrefix === '') { | ||
return uniq(topLevelPrefixes) | ||
.map((label) => ({ label, kind, detail: `(namespace)` })) | ||
.concat(fullNames.map((label) => ({ label, kind, detail }))); | ||
.map( | ||
(label) => ({ label, kind, detail: `(namespace)` } as CompletionItem), | ||
) | ||
.concat( | ||
fullNames.map((label) => { | ||
const result = getMethodCompletionItem( | ||
label, | ||
label, | ||
signatures, | ||
type, | ||
kind, | ||
); | ||
return result; | ||
}), | ||
); | ||
} else { | ||
@@ -100,3 +152,3 @@ // if we have a namespace prefix, complete on the namespace level: | ||
const funcOptions = new Set<string>(); | ||
const funcOptions = new Set<{ completion: string; fullName: string }>(); | ||
const namespaceOptions = new Set<string>(); | ||
@@ -114,3 +166,3 @@ | ||
if (isFunctionName) { | ||
funcOptions.add(option); | ||
funcOptions.add({ completion: option, fullName: name }); | ||
} else { | ||
@@ -124,9 +176,11 @@ namespaceOptions.add(option); | ||
// test handle namespace with same name as function | ||
const functionNameCompletions = Array.from(funcOptions).map((label) => ({ | ||
label, | ||
kind, | ||
detail, | ||
})); | ||
const functionNameCompletions: CompletionItem[] = Array.from( | ||
funcOptions, | ||
).map(({ completion: label, fullName }) => | ||
getMethodCompletionItem(label, fullName, signatures, type, kind), | ||
); | ||
const namespaceCompletions = Array.from(namespaceOptions).map((label) => ({ | ||
const namespaceCompletions: CompletionItem[] = Array.from( | ||
namespaceOptions, | ||
).map((label) => ({ | ||
label, | ||
@@ -311,2 +365,3 @@ kind, | ||
dbSchema: DbSchema, | ||
caretToken: Token, | ||
manualTrigger = false, | ||
@@ -319,6 +374,7 @@ ): CompletionItem[] { | ||
let caretIndex = caretToken.tokenIndex; | ||
// Move the caret index to the end of the query | ||
let caretIndex = tokens.length > 0 ? tokens.length - 1 : 0; | ||
const eofIndex = tokens.length > 0 ? tokens.length - 1 : 0; | ||
const eof = tokens[caretIndex]; | ||
const eof = tokens[eofIndex]; | ||
@@ -328,3 +384,3 @@ // When we have EOF with a different text in the token, it means the parser has failed to parse it. | ||
// point of completion (e.g. an unclosed string) | ||
if (eof.text !== '<EOF>') { | ||
if (eof.type === CypherLexer.EOF && eof.text !== '<EOF>') { | ||
return []; | ||
@@ -578,20 +634,27 @@ } | ||
// parameters are valid values in all cases of symbolicAliasName | ||
const baseSuggestions = parameterCompletions( | ||
const parameterSuggestions = parameterCompletions( | ||
dbSchema, | ||
ExpectedParameterType.String, | ||
); | ||
const rulesCreatingNewAliasOrDb = [ | ||
CypherParser.RULE_createAlias, | ||
const rulesCreatingNewDb = [ | ||
CypherParser.RULE_createDatabase, | ||
CypherParser.RULE_createCompositeDatabase, | ||
]; | ||
// avoid suggesting existing database names when creating a new alias or database | ||
// avoid suggesting existing database names when creating a new database | ||
if ( | ||
rulesCreatingNewAliasOrDb.some((rule) => | ||
candidateRule.ruleList.includes(rule), | ||
) | ||
rulesCreatingNewDb.some((rule) => candidateRule.ruleList.includes(rule)) | ||
) { | ||
return baseSuggestions; | ||
return parameterSuggestions; | ||
} | ||
// For `CREATE ALIAS aliasName FOR DATABASE databaseName` | ||
// Should not suggest existing aliases for aliasName but should suggest existing databases for databaseName | ||
// so we return base suggestions if we're at the `aliasName` rule | ||
if ( | ||
candidateRule.ruleList.includes(CypherParser.RULE_createAlias) && | ||
candidateRule.ruleList.includes(CypherParser.RULE_aliasName) | ||
) { | ||
return parameterSuggestions; | ||
} | ||
const rulesThatOnlyAcceptAlias = [ | ||
@@ -608,3 +671,3 @@ CypherParser.RULE_dropAlias, | ||
return [ | ||
...baseSuggestions, | ||
...parameterSuggestions, | ||
...(dbSchema?.aliasNames ?? []).map((aliasName) => ({ | ||
@@ -619,3 +682,3 @@ label: aliasName, | ||
return [ | ||
...baseSuggestions, | ||
...parameterSuggestions, | ||
...(dbSchema.databaseNames ?? []) | ||
@@ -636,3 +699,3 @@ .concat(dbSchema.aliasNames ?? []) | ||
// parameters are valid values in all cases of symbolic name | ||
const baseSuggestions = parameterCompletions( | ||
const parameterSuggestions = parameterCompletions( | ||
dbSchema, | ||
@@ -662,3 +725,3 @@ inferExpectedParameterTypeFromContext(candidateRule), | ||
) { | ||
return baseSuggestions; | ||
return parameterSuggestions; | ||
} | ||
@@ -675,3 +738,3 @@ | ||
const result = [ | ||
...baseSuggestions, | ||
...parameterSuggestions, | ||
...(dbSchema?.userNames ?? []).map((userName) => ({ | ||
@@ -694,3 +757,3 @@ label: userName, | ||
return [ | ||
...baseSuggestions, | ||
...parameterSuggestions, | ||
...(dbSchema?.roleNames ?? []).map((roleName) => ({ | ||
@@ -697,0 +760,0 @@ label: roleName, |
@@ -7,3 +7,3 @@ export type { ParserRuleContext } from 'antlr4'; | ||
export { CypherTokenType, lexerSymbols } from './lexerSymbols'; | ||
export { parse, parserWrapper } from './parserWrapper'; | ||
export { parse, parserWrapper, parseStatementsStrs } from './parserWrapper'; | ||
export { signatureHelp, toSignatureInformation } from './signatureHelp'; | ||
@@ -23,5 +23,6 @@ export { | ||
export { testData } from './tests/testData'; | ||
export type { Neo4jFunction, Neo4jProcedure } from './types'; | ||
export { textMateGrammar } from './textMateGrammar'; | ||
export type { CompletionItem, Neo4jFunction, Neo4jProcedure } from './types'; | ||
export { CypherLexer, CypherParser }; | ||
import CypherLexer from './generated-parser/CypherCmdLexer'; | ||
import CypherParser from './generated-parser/CypherCmdParser'; |
@@ -118,2 +118,3 @@ import CypherLexer from './generated-parser/CypherCmdLexer'; | ||
CypherLexer.AT, | ||
CypherLexer.AUTH, | ||
CypherLexer.BINDINGS, | ||
@@ -130,2 +131,3 @@ CypherLexer.BOOL, | ||
CypherLexer.CALL, | ||
CypherLexer.CASCADE, | ||
CypherLexer.CASE, | ||
@@ -204,2 +206,3 @@ CypherLexer.CIDR, | ||
CypherLexer.HOME, | ||
CypherLexer.ID, | ||
CypherLexer.IF, | ||
@@ -250,2 +253,3 @@ CypherLexer.IMMUTABLE, | ||
CypherLexer.OF, | ||
CypherLexer.OFFSET, | ||
CypherLexer.ON, | ||
@@ -275,2 +279,4 @@ CypherLexer.ONLY, | ||
CypherLexer.PROPERTY, | ||
CypherLexer.PROVIDER, | ||
CypherLexer.PROVIDERS, | ||
CypherLexer.RANGE, | ||
@@ -290,2 +296,3 @@ CypherLexer.READ, | ||
CypherLexer.REQUIRED, | ||
CypherLexer.RESTRICT, | ||
CypherLexer.RETURN, | ||
@@ -362,2 +369,3 @@ CypherLexer.REVOKE, | ||
CypherLexer.YIELD, | ||
CypherLexer.ZONE, | ||
CypherLexer.ZONED, | ||
@@ -416,1 +424,4 @@ // Preparser tokens | ||
export const keywordNames = new Set(lexerKeywords.map((i) => tokenNames[i])); | ||
export const operatorSymbols = new Set( | ||
lexerOperators.map((i) => literalNames[i].replaceAll("'", '')), | ||
); |
@@ -11,7 +11,10 @@ import type { ParserRuleContext, Token } from 'antlr4'; | ||
default as CypherParser, | ||
FunctionNameContext, | ||
LabelNameContext, | ||
LabelNameIsContext, | ||
LabelOrRelTypeContext, | ||
ProcedureNameContext, | ||
StatementOrCommandContext, | ||
StatementsOrCommandsContext, | ||
SymbolicNameStringContext, | ||
VariableContext, | ||
@@ -22,2 +25,3 @@ } from './generated-parser/CypherCmdParser'; | ||
findStopNode, | ||
getTokens, | ||
inNodeLabel, | ||
@@ -41,6 +45,8 @@ inRelationshipType, | ||
ctx: StatementsOrCommandsContext; | ||
diagnostics: SyntaxDiagnostic[]; | ||
syntaxErrors: SyntaxDiagnostic[]; | ||
stopNode: ParserRuleContext; | ||
collectedLabelOrRelTypes: LabelOrRelType[]; | ||
collectedVariables: string[]; | ||
collectedFunctions: ParsedFunction[]; | ||
collectedProcedures: ParsedProcedure[]; | ||
} | ||
@@ -98,2 +104,14 @@ | ||
export type ParsedFunction = { | ||
name: string; | ||
rawText: string; | ||
line: number; | ||
column: number; | ||
offsets: { | ||
start: number; | ||
end: number; | ||
}; | ||
}; | ||
export type ParsedProcedure = ParsedFunction; | ||
export function createParsingScaffolding(query: string): ParsingScaffolding { | ||
@@ -123,2 +141,24 @@ const inputStream = CharStreams.fromString(query); | ||
export function parseStatementsStrs(query: string): string[] { | ||
const statements = parse(query); | ||
const result: string[] = []; | ||
for (const statement of statements) { | ||
const tokenStream = statement.parser?.getTokenStream() ?? []; | ||
const tokens = getTokens(tokenStream as CommonTokenStream); | ||
const statementStr = tokens | ||
.filter((token) => token.type !== CypherLexer.EOF) | ||
.map((token) => token.text) | ||
.join(''); | ||
// Do not return empty statements | ||
if (statementStr.trimLeft().length != 0) { | ||
result.push(statementStr); | ||
} | ||
} | ||
return result; | ||
} | ||
/* Parses a query without storing it in the cache */ | ||
export function parse(query: string): StatementOrCommandContext[] { | ||
@@ -142,4 +182,5 @@ const statementScaffolding = | ||
const variableFinder = new VariableCollector(); | ||
const methodsFinder = new MethodsCollector(tokens); | ||
const errorListener = new SyntaxErrorsListener(tokens); | ||
parser._parseListeners = [labelsCollector, variableFinder]; | ||
parser._parseListeners = [labelsCollector, variableFinder, methodsFinder]; | ||
parser.addErrorListener(errorListener); | ||
@@ -152,7 +193,7 @@ const ctx = parser.statementsOrCommands(); | ||
) === undefined; | ||
const diagnostics = isEmptyStatement ? [] : errorListener.errors; | ||
const syntaxErrors = isEmptyStatement ? [] : errorListener.errors; | ||
const collectedCommand = parseToCommand(ctx, isEmptyStatement); | ||
if (!_internalFeatureFlags.consoleCommands) { | ||
diagnostics.push(...errorOnNonCypherCommands(collectedCommand)); | ||
syntaxErrors.push(...errorOnNonCypherCommands(collectedCommand)); | ||
} | ||
@@ -164,3 +205,3 @@ | ||
tokens: tokens, | ||
diagnostics: diagnostics, | ||
syntaxErrors: syntaxErrors, | ||
ctx: ctx, | ||
@@ -170,2 +211,4 @@ stopNode: findStopNode(ctx), | ||
collectedVariables: variableFinder.variables, | ||
collectedFunctions: methodsFinder.functions, | ||
collectedProcedures: methodsFinder.procedures, | ||
}; | ||
@@ -182,3 +225,3 @@ }); | ||
// This listener is collects all labels and relationship types | ||
// This listener collects all labels and relationship types | ||
class LabelAndRelTypesCollector extends ParseTreeListener { | ||
@@ -280,2 +323,87 @@ labelOrRelTypes: LabelOrRelType[] = []; | ||
// This listener collects all functions and procedures | ||
class MethodsCollector extends ParseTreeListener { | ||
public procedures: ParsedProcedure[] = []; | ||
public functions: ParsedFunction[] = []; | ||
private tokens: Token[]; | ||
constructor(tokens: Token[]) { | ||
super(); | ||
this.tokens = tokens; | ||
} | ||
enterEveryRule() { | ||
/* no-op */ | ||
} | ||
visitTerminal() { | ||
/* no-op */ | ||
} | ||
visitErrorNode() { | ||
/* no-op */ | ||
} | ||
exitEveryRule(ctx: unknown) { | ||
if ( | ||
ctx instanceof FunctionNameContext || | ||
ctx instanceof ProcedureNameContext | ||
) { | ||
const methodName = this.getMethodName(ctx); | ||
const startTokenIndex = ctx.start.tokenIndex; | ||
const stopTokenIndex = ctx.stop.tokenIndex; | ||
const rawText = this.tokens | ||
.slice(startTokenIndex, stopTokenIndex + 1) | ||
.map((token) => { | ||
return token.text; | ||
}) | ||
.join(''); | ||
const result = { | ||
name: methodName, | ||
rawText: rawText, | ||
line: ctx.start.line, | ||
column: ctx.start.column, | ||
offsets: { | ||
start: ctx.start.start, | ||
end: ctx.stop.stop + 1, | ||
}, | ||
}; | ||
if (ctx instanceof FunctionNameContext) { | ||
this.functions.push(result); | ||
} else { | ||
this.procedures.push(result); | ||
} | ||
} | ||
} | ||
private getMethodName( | ||
ctx: ProcedureNameContext | FunctionNameContext, | ||
): string { | ||
const namespaces = ctx.namespace().symbolicNameString_list(); | ||
const methodName = ctx.symbolicNameString(); | ||
const normalizedName = [...namespaces, methodName] | ||
.map((symbolicName) => { | ||
return this.getNamespaceString(symbolicName); | ||
}) | ||
.join('.'); | ||
return normalizedName; | ||
} | ||
private getNamespaceString(ctx: SymbolicNameStringContext): string { | ||
const text = ctx.getText(); | ||
const isEscaped = Boolean(ctx.escapedSymbolicNameString()); | ||
const hasDot = text.includes('.'); | ||
if (isEscaped && !hasDot) { | ||
return text.slice(1, -1); | ||
} | ||
return text; | ||
} | ||
} | ||
type CypherCmd = { type: 'cypher'; query: string }; | ||
@@ -282,0 +410,0 @@ type RuleTokens = { |
@@ -37,8 +37,6 @@ /* eslint-disable @typescript-eslint/no-unsafe-call */ | ||
if (dbSchema.functions && dbSchema.procedures) { | ||
updateSignatureResolver({ | ||
procedures: Object.values(dbSchema.procedures), | ||
functions: Object.values(dbSchema.functions), | ||
}); | ||
} | ||
updateSignatureResolver({ | ||
procedures: Object.values(dbSchema.procedures ?? {}), | ||
functions: Object.values(dbSchema.functions ?? {}), | ||
}); | ||
semanticAnalysis([query], (a) => { | ||
@@ -45,0 +43,0 @@ semanticErrorsResult = a; |
@@ -8,5 +8,8 @@ import { DiagnosticSeverity, Position } from 'vscode-languageserver-types'; | ||
LabelType, | ||
ParsedFunction, | ||
ParsedProcedure, | ||
ParsedStatement, | ||
parserWrapper, | ||
} from '../parserWrapper'; | ||
import { Neo4jFunction } from '../types'; | ||
import { | ||
@@ -61,2 +64,78 @@ SemanticAnalysisElement, | ||
function detectNonDeclaredFunction( | ||
parsedFunction: ParsedFunction, | ||
functionsSchema: Record<string, Neo4jFunction>, | ||
): SyntaxDiagnostic | undefined { | ||
const lowercaseFunctionName = parsedFunction.name.toLowerCase(); | ||
const caseInsensitiveFunctionInDatabase = | ||
functionsSchema[lowercaseFunctionName]; | ||
// Built-in functions are case-insensitive in the database | ||
if ( | ||
caseInsensitiveFunctionInDatabase && | ||
caseInsensitiveFunctionInDatabase.isBuiltIn | ||
) { | ||
return undefined; | ||
} | ||
const functionExistsWithExactName = Boolean( | ||
functionsSchema[parsedFunction.name], | ||
); | ||
if (!functionExistsWithExactName) { | ||
return generateFunctionNotFoundError(parsedFunction); | ||
} | ||
} | ||
function generateFunctionNotFoundError( | ||
parsedFunction: ParsedFunction, | ||
): SyntaxDiagnostic { | ||
const rawText = parsedFunction.rawText; | ||
const nameChunks = rawText.split('\n'); | ||
const linesOffset = nameChunks.length - 1; | ||
const lineIndex = parsedFunction.line - 1; | ||
const startColumn = parsedFunction.column; | ||
const endColumn = | ||
linesOffset == 0 | ||
? startColumn + rawText.length | ||
: nameChunks.at(-1)?.length ?? 0; | ||
const error: SyntaxDiagnostic = { | ||
severity: DiagnosticSeverity.Error, | ||
range: { | ||
start: Position.create(lineIndex, startColumn), | ||
end: Position.create(lineIndex + linesOffset, endColumn), | ||
}, | ||
offsets: parsedFunction.offsets, | ||
message: `Function ${parsedFunction.name} is not present in the database. Make sure you didn't misspell it or that it is available when you run this statement in your application`, | ||
}; | ||
return error; | ||
} | ||
function generateProcedureNotFoundError( | ||
parsedProcedure: ParsedProcedure, | ||
): SyntaxDiagnostic { | ||
const rawText = parsedProcedure.rawText; | ||
const nameChunks = rawText.split('\n'); | ||
const linesOffset = nameChunks.length - 1; | ||
const lineIndex = parsedProcedure.line - 1; | ||
const startColumn = parsedProcedure.column; | ||
const endColumn = | ||
linesOffset == 0 | ||
? startColumn + rawText.length | ||
: nameChunks.at(-1)?.length ?? 0; | ||
const error: SyntaxDiagnostic = { | ||
severity: DiagnosticSeverity.Error, | ||
range: { | ||
start: Position.create(lineIndex, startColumn), | ||
end: Position.create(lineIndex + linesOffset, endColumn), | ||
}, | ||
offsets: parsedProcedure.offsets, | ||
message: `Procedure ${parsedProcedure.name} is not present in the database. Make sure you didn't misspell it or that it is available when you run this statement in your application`, | ||
}; | ||
return error; | ||
} | ||
function warnOnUndeclaredLabels( | ||
@@ -179,3 +258,4 @@ parsingResult: ParsedStatement, | ||
const syntaxErrors = validateSyntax(query, dbSchema); | ||
if (syntaxErrors.length > 0) { | ||
// If there are any syntactic errors in the query, do not run the semantic validation | ||
if (syntaxErrors.find((d) => d.severity === DiagnosticSeverity.Error)) { | ||
return syntaxErrors; | ||
@@ -185,3 +265,3 @@ } | ||
const semanticErrors = validateSemantics(query, dbSchema); | ||
return semanticErrors; | ||
return syntaxErrors.concat(semanticErrors); | ||
} | ||
@@ -198,5 +278,5 @@ | ||
const result = statements.statementsParsing.flatMap((statement) => { | ||
const diagnostics = statement.diagnostics; | ||
const syntaxErrors = statement.syntaxErrors; | ||
const labelWarnings = warnOnUndeclaredLabels(statement, dbSchema); | ||
return diagnostics.concat(labelWarnings).sort(sortByPositionAndMessage); | ||
return syntaxErrors.concat(labelWarnings).sort(sortByPositionAndMessage); | ||
}); | ||
@@ -218,5 +298,11 @@ | ||
const semanticErrors = statements.flatMap((current) => { | ||
if (current.diagnostics.length === 0) { | ||
if (current.syntaxErrors.length === 0) { | ||
const cmd = current.command; | ||
if (cmd.type === 'cypher' && cmd.statement.length > 0) { | ||
const functionErrors = errorOnUndeclaredFunctions(current, dbSchema); | ||
const procedureErrors = errorOnUndeclaredProcedures( | ||
current, | ||
dbSchema, | ||
); | ||
const { notifications, errors } = wrappedSemanticAnalysis( | ||
@@ -228,7 +314,9 @@ cmd.statement, | ||
const elements = notifications.concat(errors); | ||
const result = fixSemanticAnalysisPositions({ | ||
const semanticDiagnostics = fixSemanticAnalysisPositions({ | ||
semanticElements: elements, | ||
parseResult: current, | ||
}).sort(sortByPositionAndMessage); | ||
return result; | ||
}); | ||
return semanticDiagnostics | ||
.concat(functionErrors, procedureErrors) | ||
.sort(sortByPositionAndMessage); | ||
} | ||
@@ -244,1 +332,45 @@ } | ||
} | ||
function errorOnUndeclaredFunctions( | ||
parsingResult: ParsedStatement, | ||
dbSchema: DbSchema, | ||
): SyntaxDiagnostic[] { | ||
const warnings: SyntaxDiagnostic[] = []; | ||
if (dbSchema.functions) { | ||
const functionsInQuery = parsingResult.collectedFunctions; | ||
functionsInQuery.forEach((parsedFunction) => { | ||
const warning = detectNonDeclaredFunction( | ||
parsedFunction, | ||
dbSchema.functions, | ||
); | ||
if (warning) warnings.push(warning); | ||
}); | ||
} | ||
return warnings; | ||
} | ||
function errorOnUndeclaredProcedures( | ||
parsingResult: ParsedStatement, | ||
dbSchema: DbSchema, | ||
): SyntaxDiagnostic[] { | ||
const errors: SyntaxDiagnostic[] = []; | ||
if (dbSchema.procedures) { | ||
const proceduresInQuery = parsingResult.collectedProcedures; | ||
proceduresInQuery.forEach((parsedProcedure) => { | ||
const procedureExists = Boolean( | ||
dbSchema.procedures[parsedProcedure.name], | ||
); | ||
if (!procedureExists) { | ||
errors.push(generateProcedureNotFoundError(parsedProcedure)); | ||
} | ||
}); | ||
} | ||
return errors; | ||
} |
@@ -13,3 +13,3 @@ import { | ||
} from 'vscode-languageserver-types'; | ||
import { CypherLexer } from '..'; | ||
import CypherLexer from '../generated-parser/CypherCmdLexer'; | ||
import CypherParser from '../generated-parser/CypherCmdParser'; | ||
@@ -16,0 +16,0 @@ import { isCommentOpener } from '../helpers'; |
@@ -1,4 +0,4 @@ | ||
import { CompletionItem } from 'vscode-languageserver-types'; | ||
import { autocomplete } from '../../autocompletion/autocompletion'; | ||
import { DbSchema } from '../../dbSchema'; | ||
import { CompletionItem } from '../../types'; | ||
@@ -62,3 +62,3 @@ export function testCompletionsExactly({ | ||
const unexpectedCompletions = excluded.map((notExpectedItem) => | ||
const unexpectedCompletions = excluded.filter((notExpectedItem) => | ||
actualCompletionList.find((value) => { | ||
@@ -65,0 +65,0 @@ // if label is left out -> only check kind and vice versa |
@@ -114,2 +114,48 @@ import { CompletionItemKind } from 'vscode-languageserver-types'; | ||
test('Correctly completes database name even in a create alias statement', () => { | ||
testCompletions({ | ||
query: 'CREATE ALIAS fo for DATABASE ', | ||
dbSchema, | ||
expected: [ | ||
{ label: 'db1', kind: CompletionItemKind.Value }, | ||
{ label: 'db2', kind: CompletionItemKind.Value }, | ||
{ label: 'movies', kind: CompletionItemKind.Value }, | ||
{ label: 'myMovies', kind: CompletionItemKind.Value }, | ||
{ label: 'scoped.alias', kind: CompletionItemKind.Value }, | ||
{ label: 'a.b.c.d', kind: CompletionItemKind.Value }, | ||
{ label: '$param1', kind: CompletionItemKind.Variable }, | ||
], | ||
excluded: [ | ||
// validate invalid keyword bug isn't present | ||
{ label: '', kind: CompletionItemKind.Keyword }, | ||
// do not suggest non-string parameters | ||
{ label: '$param2', kind: CompletionItemKind.Variable }, | ||
{ label: '$param3', kind: CompletionItemKind.Variable }, | ||
], | ||
}); | ||
}); | ||
test('Correctly completes database name even in a create alias statement including extra of spaces', () => { | ||
testCompletions({ | ||
query: 'CREATE ALIAS foo FOR DATABASE ', | ||
dbSchema, | ||
expected: [ | ||
{ label: 'db1', kind: CompletionItemKind.Value }, | ||
{ label: 'db2', kind: CompletionItemKind.Value }, | ||
{ label: 'movies', kind: CompletionItemKind.Value }, | ||
{ label: 'myMovies', kind: CompletionItemKind.Value }, | ||
{ label: 'scoped.alias', kind: CompletionItemKind.Value }, | ||
{ label: 'a.b.c.d', kind: CompletionItemKind.Value }, | ||
{ label: '$param1', kind: CompletionItemKind.Variable }, | ||
], | ||
excluded: [ | ||
// validate invalid keyword bug isn't present | ||
{ label: '', kind: CompletionItemKind.Keyword }, | ||
// do not suggest non-string parameters | ||
{ label: '$param2', kind: CompletionItemKind.Variable }, | ||
{ label: '$param3', kind: CompletionItemKind.Variable }, | ||
], | ||
}); | ||
}); | ||
test('Suggests only aliases when dropping alias', () => { | ||
@@ -116,0 +162,0 @@ const query = 'DROP ALIAS '; |
@@ -1,2 +0,5 @@ | ||
import { CompletionItemKind } from 'vscode-languageserver-types'; | ||
import { | ||
CompletionItemKind, | ||
CompletionItemTag, | ||
} from 'vscode-languageserver-types'; | ||
import { DbSchema } from '../../dbSchema'; | ||
@@ -8,2 +11,3 @@ import { testData } from '../testData'; | ||
const dbSchema: DbSchema = testData.mockSchema; | ||
const functions = dbSchema.functions; | ||
@@ -20,2 +24,4 @@ test('Correctly completes unstarted function name in left hand side of WHERE', () => { | ||
detail: '(function)', | ||
signature: functions['acos'].signature, | ||
documentation: functions['acos'].description, | ||
}, | ||
@@ -31,2 +37,4 @@ { | ||
detail: '(function)', | ||
signature: functions['apoc.agg.graph'].signature, | ||
documentation: functions['apoc.agg.graph'].description, | ||
}, | ||
@@ -37,3 +45,13 @@ { | ||
detail: '(function)', | ||
signature: functions['apoc.coll.pairs'].signature, | ||
documentation: functions['apoc.coll.pairs'].description, | ||
}, | ||
{ | ||
label: 'apoc.create.uuid', | ||
kind: CompletionItemKind.Function, | ||
detail: '(function)', | ||
signature: functions['apoc.create.uuid'].signature, | ||
documentation: functions['apoc.create.uuid'].description, | ||
tags: [CompletionItemTag.Deprecated], | ||
}, | ||
], | ||
@@ -53,2 +71,4 @@ }); | ||
detail: '(function)', | ||
signature: functions['acos'].signature, | ||
documentation: functions['acos'].description, | ||
}, | ||
@@ -59,2 +79,4 @@ { | ||
detail: '(function)', | ||
signature: functions['apoc.agg.graph'].signature, | ||
documentation: functions['apoc.agg.graph'].description, | ||
}, | ||
@@ -65,3 +87,13 @@ { | ||
detail: '(function)', | ||
signature: functions['apoc.coll.pairs'].signature, | ||
documentation: functions['apoc.coll.pairs'].description, | ||
}, | ||
{ | ||
label: 'apoc.create.uuid', | ||
kind: CompletionItemKind.Function, | ||
detail: '(function)', | ||
signature: functions['apoc.create.uuid'].signature, | ||
documentation: functions['apoc.create.uuid'].description, | ||
tags: [CompletionItemTag.Deprecated], | ||
}, | ||
], | ||
@@ -88,3 +120,9 @@ }); | ||
], | ||
excluded: [{ label: 'acos', kind: CompletionItemKind.Function }], | ||
excluded: [ | ||
{ label: 'acos', kind: CompletionItemKind.Function }, | ||
{ | ||
label: 'agg.graph', | ||
kind: CompletionItemKind.Function, | ||
}, | ||
], | ||
}); | ||
@@ -103,2 +141,4 @@ }); | ||
detail: '(function)', | ||
signature: functions['apoc.agg.first'].signature, | ||
documentation: functions['apoc.agg.first'].description, | ||
}, | ||
@@ -109,2 +149,4 @@ { | ||
detail: '(function)', | ||
signature: functions['apoc.agg.last'].signature, | ||
documentation: functions['apoc.agg.last'].description, | ||
}, | ||
@@ -115,2 +157,4 @@ { | ||
detail: '(function)', | ||
signature: functions['apoc.agg.slice'].signature, | ||
documentation: functions['apoc.agg.slice'].description, | ||
}, | ||
@@ -148,12 +192,18 @@ ], | ||
detail: '(function)', | ||
signature: functions['apoc.agg.first'].signature, | ||
documentation: functions['apoc.agg.first'].description, | ||
}, | ||
{ | ||
label: 'first', | ||
label: 'last', | ||
kind: CompletionItemKind.Function, | ||
detail: '(function)', | ||
signature: functions['apoc.agg.last'].signature, | ||
documentation: functions['apoc.agg.last'].description, | ||
}, | ||
{ | ||
label: 'first', | ||
label: 'slice', | ||
kind: CompletionItemKind.Function, | ||
detail: '(function)', | ||
signature: functions['apoc.agg.slice'].signature, | ||
documentation: functions['apoc.agg.slice'].description, | ||
}, | ||
@@ -271,2 +321,4 @@ ], | ||
detail: '(function)', | ||
signature: functions['apoc.agg.percentiles'].signature, | ||
documentation: functions['apoc.agg.percentiles'].description, | ||
}, | ||
@@ -277,2 +329,4 @@ { | ||
detail: '(function)', | ||
signature: functions['acos'].signature, | ||
documentation: functions['acos'].description, | ||
}, | ||
@@ -293,2 +347,4 @@ ], | ||
detail: '(function)', | ||
signature: functions['apoc.agg.percentiles'].signature, | ||
documentation: functions['apoc.agg.percentiles'].description, | ||
}, | ||
@@ -304,2 +360,4 @@ { | ||
detail: '(function)', | ||
signature: functions['acos'].signature, | ||
documentation: functions['acos'].description, | ||
}, | ||
@@ -330,2 +388,4 @@ ], | ||
detail: '(function)', | ||
documentation: '', | ||
signature: '', | ||
}, | ||
@@ -335,2 +395,20 @@ ], | ||
}); | ||
test('Correctly completes deprecated functions when namespace started', () => { | ||
const query = 'RETURN apoc.create.'; | ||
testCompletions({ | ||
query, | ||
dbSchema, | ||
expected: [ | ||
{ | ||
label: 'uuid', | ||
kind: CompletionItemKind.Function, | ||
detail: '(function)', | ||
signature: functions['apoc.create.uuid'].signature, | ||
documentation: functions['apoc.create.uuid'].description, | ||
tags: [CompletionItemTag.Deprecated], | ||
}, | ||
], | ||
}); | ||
}); | ||
}); |
@@ -49,11 +49,2 @@ import { CompletionItemKind } from 'vscode-languageserver-types'; | ||
describe('Auto completion of back to back keywords', () => { | ||
test('Correctly completes OPTIONAL MATCH', () => { | ||
const query = 'OP'; | ||
testCompletions({ | ||
query, | ||
expected: [{ label: 'OPTIONAL MATCH', kind: CompletionItemKind.Keyword }], | ||
}); | ||
}); | ||
test('Correctly completes DEFAULT DATABASE and HOME DATABASE', () => { | ||
@@ -60,0 +51,0 @@ const query = 'SHOW '; |
@@ -14,8 +14,11 @@ import { CompletionItemKind } from 'vscode-languageserver-types'; | ||
test('Correctly completes OPTIONAL MATCH', () => { | ||
const query = 'OP'; | ||
test('Correctly completes OPTIONAL MATCH and OPTIONAL CALL', () => { | ||
const query = 'OPTIONAL '; | ||
testCompletions({ | ||
query, | ||
expected: [{ label: 'OPTIONAL MATCH', kind: CompletionItemKind.Keyword }], | ||
expected: [ | ||
{ label: 'MATCH', kind: CompletionItemKind.Keyword }, | ||
{ label: 'CALL', kind: CompletionItemKind.Keyword }, | ||
], | ||
}); | ||
@@ -67,2 +70,34 @@ }); | ||
test('Correctly completes unstarted label when caret is passed', () => { | ||
const query = 'MATCH (n:)'; | ||
testCompletions({ | ||
query, | ||
offset: query.length - 1, | ||
dbSchema: { labels: ['Cat', 'Person', 'Dog'] }, | ||
expected: [{ label: 'Person', kind: CompletionItemKind.TypeParameter }], | ||
}); | ||
}); | ||
test('Correctly completes unstarted label when caret is passed and there is a space', () => { | ||
const query = 'MATCH (n : '; | ||
testCompletions({ | ||
query, | ||
dbSchema: { labels: ['Cat', 'Person', 'Dog'] }, | ||
expected: [{ label: 'Person', kind: CompletionItemKind.TypeParameter }], | ||
}); | ||
}); | ||
test('Correctly completes unstarted label for a first statement when caret is passed', () => { | ||
const query = 'MATCH (n:); MATCH (m:)'; | ||
testCompletions({ | ||
query, | ||
offset: 'MATCH (n:)'.length - 1, | ||
dbSchema: { labels: ['Cat', 'Person', 'Dog'] }, | ||
expected: [{ label: 'Person', kind: CompletionItemKind.TypeParameter }], | ||
}); | ||
}); | ||
test('Correctly completes started barred label inside a node pattern', () => { | ||
@@ -69,0 +104,0 @@ const query = 'MATCH (n:A|B'; |
@@ -1,2 +0,5 @@ | ||
import { CompletionItemKind } from 'vscode-languageserver-types'; | ||
import { | ||
CompletionItemKind, | ||
CompletionItemTag, | ||
} from 'vscode-languageserver-types'; | ||
import { DbSchema } from '../../dbSchema'; | ||
@@ -7,5 +10,14 @@ import { testData } from '../testData'; | ||
describe('Procedures auto-completion', () => { | ||
const procedures = testData.mockSchema.procedures; | ||
const dbSchema: DbSchema = { | ||
procedures: { | ||
'tx.getMetaData': { ...testData.emptyProcedure, name: 'tx.getMetaData' }, | ||
'tx.getMetaData': procedures['tx.getMetaData'], | ||
'db.index.fulltext.awaitEventuallyConsistentIndexRefresh': | ||
procedures['db.index.fulltext.awaitEventuallyConsistentIndexRefresh'], | ||
'db.ping': procedures['db.ping'], | ||
'db.stats.retrieve': procedures['db.stats.retrieve'], | ||
'db.stats.collect': procedures['db.stats.collect'], | ||
'db.stats.clear': procedures['db.stats.clear'], | ||
'cdc.current': procedures['cdc.current'], | ||
'jwt.security.requestAccess': { | ||
@@ -15,16 +27,2 @@ ...testData.emptyProcedure, | ||
}, | ||
'db.index.fulltext.awaitEventuallyConsistentIndexRefresh': { | ||
...testData.emptyProcedure, | ||
name: 'db.index.fulltext.awaitEventuallyConsistentIndexRefresh', | ||
}, | ||
'db.ping': { ...testData.emptyProcedure, name: 'db.ping' }, | ||
'db.stats.retrieve': { | ||
...testData.emptyProcedure, | ||
name: 'db.stats.retrieve', | ||
}, | ||
'db.stats.collect': { | ||
...testData.emptyProcedure, | ||
name: 'db.stats.collect', | ||
}, | ||
'db.stats.clear': { ...testData.emptyProcedure, name: 'db.stats.clear' }, | ||
}, | ||
@@ -71,7 +69,19 @@ }; | ||
detail: '(procedure)', | ||
signature: procedures['tx.getMetaData'].signature, | ||
documentation: procedures['tx.getMetaData'].description, | ||
}, | ||
{ | ||
label: 'cdc.current', | ||
kind: CompletionItemKind.Method, | ||
detail: '(procedure)', | ||
signature: procedures['cdc.current'].signature, | ||
documentation: procedures['cdc.current'].description, | ||
tags: [CompletionItemTag.Deprecated], | ||
}, | ||
{ | ||
label: 'jwt.security.requestAccess', | ||
kind: CompletionItemKind.Method, | ||
detail: '(procedure)', | ||
documentation: '', | ||
signature: '', | ||
}, | ||
@@ -97,2 +107,4 @@ ], | ||
detail: '(procedure)', | ||
signature: procedures['db.ping'].signature, | ||
documentation: procedures['db.ping'].description, | ||
}, | ||
@@ -118,2 +130,4 @@ ], | ||
detail: '(procedure)', | ||
signature: procedures['db.stats.retrieve'].signature, | ||
documentation: procedures['db.stats.retrieve'].description, | ||
}, | ||
@@ -124,2 +138,4 @@ { | ||
detail: '(procedure)', | ||
signature: procedures['db.stats.collect'].signature, | ||
documentation: procedures['db.stats.collect'].description, | ||
}, | ||
@@ -130,2 +146,4 @@ { | ||
detail: '(procedure)', | ||
signature: procedures['db.stats.clear'].signature, | ||
documentation: procedures['db.stats.clear'].description, | ||
}, | ||
@@ -163,2 +181,4 @@ ], | ||
detail: '(procedure)', | ||
signature: procedures['db.stats.retrieve'].signature, | ||
documentation: procedures['db.stats.retrieve'].description, | ||
}, | ||
@@ -169,2 +189,4 @@ { | ||
detail: '(procedure)', | ||
signature: procedures['db.stats.collect'].signature, | ||
documentation: procedures['db.stats.collect'].description, | ||
}, | ||
@@ -175,2 +197,4 @@ { | ||
detail: '(procedure)', | ||
signature: procedures['db.stats.clear'].signature, | ||
documentation: procedures['db.stats.clear'].description, | ||
}, | ||
@@ -247,2 +271,4 @@ ], | ||
detail: '(procedure)', | ||
signature: procedures['db.ping'].signature, | ||
documentation: procedures['db.ping'].description, | ||
}, | ||
@@ -258,2 +284,20 @@ { | ||
}); | ||
test('Correctly completes deprecated procedures when namespace started', () => { | ||
const query = 'CALL cdc.'; | ||
testCompletions({ | ||
query, | ||
dbSchema, | ||
expected: [ | ||
{ | ||
label: 'current', | ||
kind: CompletionItemKind.Method, | ||
detail: '(procedure)', | ||
signature: procedures['cdc.current'].signature, | ||
documentation: procedures['cdc.current'].description, | ||
tags: [CompletionItemTag.Deprecated], | ||
}, | ||
], | ||
}); | ||
}); | ||
}); |
@@ -13,3 +13,3 @@ import { autocomplete } from '../autocompletion/autocompletion'; | ||
expect( | ||
result.statementsParsing.flatMap((statement) => statement.diagnostics), | ||
result.statementsParsing.flatMap((statement) => statement.syntaxErrors), | ||
).toEqual([]); | ||
@@ -34,3 +34,3 @@ expect( | ||
result.statementsParsing | ||
.flatMap((statement) => statement.diagnostics) | ||
.flatMap((statement) => statement.syntaxErrors) | ||
.map((e) => e.message), | ||
@@ -264,3 +264,9 @@ ).toContain(msg); | ||
{ detail: '(namespace)', kind: 3, label: 'duration' }, | ||
{ detail: '(function)', kind: 3, label: 'duration.inSeconds' }, | ||
{ | ||
detail: '(function)', | ||
kind: 3, | ||
label: 'duration.inSeconds', | ||
signature: '', | ||
documentation: '', | ||
}, | ||
{ kind: 14, label: 'TRUE' }, | ||
@@ -267,0 +273,0 @@ { kind: 14, label: 'FALSE' }, |
@@ -116,2 +116,21 @@ import { _internalFeatureFlags } from '../../featureFlags'; | ||
message: | ||
'CALL subquery without a variable scope clause is now deprecated. Use CALL () { ... }', | ||
offsets: { | ||
end: 100, | ||
start: 26, | ||
}, | ||
range: { | ||
end: { | ||
character: 5, | ||
line: 5, | ||
}, | ||
start: { | ||
character: 4, | ||
line: 2, | ||
}, | ||
}, | ||
severity: 2, | ||
}, | ||
{ | ||
message: | ||
'Variable in subquery is shadowing a variable with the same name from the outer scope. If you want to use that variable instead, it must be imported into the subquery using importing WITH clause. (the shadowing variable is: shadowed)', | ||
@@ -138,3 +157,3 @@ offsets: { | ||
test('Accumulates several semantic errors', () => { | ||
const query = `CALL { MATCH (n) RETURN m} IN TRANSACTIONS OF -1 ROWS`; | ||
const query = `CALL () { MATCH (n) RETURN m} IN TRANSACTIONS OF -1 ROWS`; | ||
@@ -146,3 +165,3 @@ expect(getDiagnosticsForQuery({ query })).toEqual([ | ||
offsets: { | ||
end: 53, | ||
end: 56, | ||
start: 0, | ||
@@ -152,3 +171,3 @@ }, | ||
end: { | ||
character: 53, | ||
character: 56, | ||
line: 0, | ||
@@ -166,12 +185,12 @@ }, | ||
offsets: { | ||
end: 25, | ||
start: 24, | ||
end: 28, | ||
start: 27, | ||
}, | ||
range: { | ||
end: { | ||
character: 25, | ||
character: 28, | ||
line: 0, | ||
}, | ||
start: { | ||
character: 24, | ||
character: 27, | ||
line: 0, | ||
@@ -186,12 +205,12 @@ }, | ||
offsets: { | ||
end: 48, | ||
start: 46, | ||
end: 51, | ||
start: 49, | ||
}, | ||
range: { | ||
end: { | ||
character: 48, | ||
character: 51, | ||
line: 0, | ||
}, | ||
start: { | ||
character: 46, | ||
character: 49, | ||
line: 0, | ||
@@ -206,6 +225,6 @@ }, | ||
test('Shows errors for CALL IN TXs used in UNION', () => { | ||
const query = `CALL { CREATE (x) } IN TRANSACTIONS | ||
const query = `CALL () { CREATE (x) } IN TRANSACTIONS | ||
RETURN 1 AS result | ||
UNION | ||
CALL { CREATE (x) } IN TRANSACTIONS | ||
CALL () { CREATE (x) } IN TRANSACTIONS | ||
RETURN 2 AS result`; | ||
@@ -217,3 +236,3 @@ | ||
offsets: { | ||
end: 139, | ||
end: 145, | ||
start: 0, | ||
@@ -236,4 +255,4 @@ }, | ||
offsets: { | ||
end: 139, | ||
start: 79, | ||
end: 145, | ||
start: 82, | ||
}, | ||
@@ -257,3 +276,3 @@ range: { | ||
const query = `WITH 1 AS i | ||
CALL { | ||
CALL () { | ||
WITH 2 AS i | ||
@@ -269,25 +288,6 @@ RETURN * | ||
{ | ||
message: | ||
'Variable in subquery is shadowing a variable with the same name from the outer scope. If you want to use that variable instead, it must be imported into the subquery using importing WITH clause. (the shadowing variable is: i)', | ||
offsets: { | ||
end: 44, | ||
start: 43, | ||
}, | ||
range: { | ||
end: { | ||
character: 19, | ||
line: 2, | ||
}, | ||
start: { | ||
character: 18, | ||
line: 2, | ||
}, | ||
}, | ||
severity: 2, | ||
}, | ||
{ | ||
message: 'Variable `i` already declared in outer scope', | ||
offsets: { | ||
end: 61, | ||
start: 53, | ||
end: 64, | ||
start: 56, | ||
}, | ||
@@ -309,4 +309,4 @@ range: { | ||
offsets: { | ||
end: 61, | ||
start: 60, | ||
end: 64, | ||
start: 63, | ||
}, | ||
@@ -326,19 +326,18 @@ range: { | ||
{ | ||
message: | ||
'Variable in subquery is shadowing a variable with the same name from the outer scope. If you want to use that variable instead, it must be imported into the subquery using importing WITH clause. (the shadowing variable is: i)', | ||
message: 'Variable `i` already declared', | ||
offsets: { | ||
end: 97, | ||
start: 96, | ||
end: 117, | ||
start: 75, | ||
}, | ||
range: { | ||
end: { | ||
character: 19, | ||
line: 5, | ||
character: 16, | ||
line: 6, | ||
}, | ||
start: { | ||
character: 18, | ||
line: 5, | ||
character: 10, | ||
line: 4, | ||
}, | ||
}, | ||
severity: 2, | ||
severity: 1, | ||
}, | ||
@@ -348,4 +347,4 @@ { | ||
offsets: { | ||
end: 114, | ||
start: 106, | ||
end: 117, | ||
start: 109, | ||
}, | ||
@@ -367,4 +366,4 @@ range: { | ||
offsets: { | ||
end: 114, | ||
start: 113, | ||
end: 117, | ||
start: 116, | ||
}, | ||
@@ -387,3 +386,3 @@ range: { | ||
test('Shows errors for subquery with only WITH', () => { | ||
const query = 'WITH 1 AS a CALL { WITH a } RETURN a'; | ||
const query = 'WITH 1 AS a CALL (a) { WITH a } RETURN a'; | ||
@@ -393,14 +392,14 @@ expect(getDiagnosticsForQuery({ query })).toEqual([ | ||
message: | ||
'Query must conclude with a RETURN clause, a FINISH clause, an update clause, a unit subquery call, or a procedure call with no YIELD.', | ||
'Query cannot conclude with WITH (must be a RETURN clause, a FINISH clause, an update clause, a unit subquery call, or a procedure call with no YIELD).', | ||
offsets: { | ||
end: 25, | ||
start: 19, | ||
end: 29, | ||
start: 23, | ||
}, | ||
range: { | ||
end: { | ||
character: 25, | ||
character: 29, | ||
line: 0, | ||
}, | ||
start: { | ||
character: 19, | ||
character: 23, | ||
line: 0, | ||
@@ -417,3 +416,3 @@ }, | ||
WITH 1 AS a | ||
CALL { | ||
CALL () { | ||
USE x | ||
@@ -430,3 +429,3 @@ RETURN 2 AS b | ||
WITH 1 AS a | ||
CALL { | ||
CALL () { | ||
USE other | ||
@@ -442,4 +441,4 @@ RETURN 2 AS b | ||
offsets: { | ||
end: 88, | ||
start: 55, | ||
end: 91, | ||
start: 58, | ||
}, | ||
@@ -665,4 +664,3 @@ range: { | ||
UNWIND [0, 1, 2] AS x | ||
CALL { | ||
WITH x | ||
CALL (x) { | ||
RETURN x * 10 AS y | ||
@@ -678,4 +676,4 @@ } | ||
offsets: { | ||
end: 137, | ||
start: 136, | ||
end: 120, | ||
start: 119, | ||
}, | ||
@@ -685,7 +683,7 @@ range: { | ||
character: 32, | ||
line: 5, | ||
line: 4, | ||
}, | ||
start: { | ||
character: 31, | ||
line: 5, | ||
line: 4, | ||
}, | ||
@@ -771,14 +769,15 @@ }, | ||
{ | ||
message: 'Path selectors such as `ANY 2 PATHS` are not supported yet', | ||
message: | ||
'Multiple path patterns cannot be used in the same clause in combination with a selective path selector.', | ||
offsets: { | ||
end: 32, | ||
start: 21, | ||
end: 82, | ||
start: 16, | ||
}, | ||
range: { | ||
end: { | ||
character: 26, | ||
line: 1, | ||
character: 31, | ||
line: 2, | ||
}, | ||
start: { | ||
character: 15, | ||
character: 10, | ||
line: 1, | ||
@@ -916,4 +915,3 @@ }, | ||
{ | ||
message: | ||
'Type mismatch: p defined with conflicting type List<T> (expected Path)', | ||
message: 'Variable `p` already declared', | ||
offsets: { | ||
@@ -1202,3 +1200,3 @@ end: 35, | ||
message: | ||
'A pattern expression should only be used in order to test the existence of a pattern. It should therefore only be used in contexts that evaluate to a boolean, e.g. inside the function exists() or in a WHERE-clause. No other uses are allowed, instead they should be replaced by a pattern comprehension.', | ||
'A pattern expression should only be used in order to test the existence of a pattern. It can no longer be used inside the function size(), an alternative is to replace size() with COUNT {}.', | ||
offsets: { | ||
@@ -1451,6 +1449,9 @@ end: 29, | ||
test('Does not provide semantic validation for pluggeable functions when schema is not available', () => { | ||
test('Does not error on SHORTEST k', () => { | ||
expect( | ||
getDiagnosticsForQuery({ | ||
query: `RETURN apoc.coll.sum(['a', 'b'])`, | ||
query: `MATCH p = SHORTEST 2 (a)-[]-+(b) | ||
WHERE a.name = "foo" AND b.name = "bar" | ||
RETURN [n in nodes(p) | n.name] AS stops;`, | ||
dbSchema: testData.mockSchema, | ||
}), | ||
@@ -1460,147 +1461,17 @@ ).toEqual([]); | ||
test('Provides semantic validation for built-in functions', () => { | ||
test('Does not error on dynamic labels', () => { | ||
expect( | ||
getDiagnosticsForQuery({ | ||
query: `WITH character_length() AS a | ||
WITH character_length(1) AS b, a | ||
RETURN a,b`, | ||
}), | ||
).toEqual([ | ||
{ | ||
message: "Insufficient parameters for function 'character_length'", | ||
offsets: { | ||
end: 28, | ||
start: 5, | ||
}, | ||
range: { | ||
end: { | ||
character: 28, | ||
line: 0, | ||
}, | ||
start: { | ||
character: 5, | ||
line: 0, | ||
}, | ||
}, | ||
severity: 1, | ||
}, | ||
{ | ||
message: 'Type mismatch: expected String but was Integer', | ||
offsets: { | ||
end: 60, | ||
start: 59, | ||
}, | ||
range: { | ||
end: { | ||
character: 31, | ||
line: 1, | ||
}, | ||
start: { | ||
character: 30, | ||
line: 1, | ||
}, | ||
}, | ||
severity: 1, | ||
}, | ||
]); | ||
}); | ||
test('Provides semantic validation for procedures when a schema is available', () => { | ||
expect( | ||
getDiagnosticsForQuery({ | ||
query: ` | ||
CALL db.awaitIndex('index', 'time') | ||
CALL db.awaitIndex() | ||
`, | ||
query: `MATCH (n) | ||
SET n:$("label")`, | ||
dbSchema: testData.mockSchema, | ||
}), | ||
).toEqual([ | ||
{ | ||
message: 'Type mismatch: expected Integer but was String', | ||
offsets: { | ||
end: 43, | ||
start: 37, | ||
}, | ||
range: { | ||
end: { | ||
character: 42, | ||
line: 1, | ||
}, | ||
start: { | ||
character: 36, | ||
line: 1, | ||
}, | ||
}, | ||
severity: 1, | ||
}, | ||
{ | ||
message: `Procedure call does not provide the required number of arguments: got 0 expected at least 1 (total: 2, 1 of which have default values). | ||
Procedure db.awaitIndex has signature: db.awaitIndex(indexName :: STRING, timeOutSeconds = 300 :: INTEGER) :: | ||
meaning that it expects at least 1 argument of type STRING | ||
`, | ||
offsets: { | ||
end: 73, | ||
start: 53, | ||
}, | ||
range: { | ||
end: { | ||
character: 28, | ||
line: 2, | ||
}, | ||
start: { | ||
character: 8, | ||
line: 2, | ||
}, | ||
}, | ||
severity: 1, | ||
}, | ||
]); | ||
}); | ||
test('Shows default values correctly for external procedures', () => { | ||
expect( | ||
getDiagnosticsForQuery({ | ||
query: 'CALL apoc.load.xml()', | ||
dbSchema: testData.mockSchema, | ||
}), | ||
).toEqual([ | ||
{ | ||
message: `Procedure call does not provide the required number of arguments: got 0 expected at least 1 (total: 4, 3 of which have default values). | ||
Procedure apoc.load.xml has signature: apoc.load.xml(urlOrBinary :: ANY, path = / :: STRING, config = {} :: MAP, simple = false :: BOOLEAN) :: value :: MAP | ||
meaning that it expects at least 1 argument of type ANY | ||
`, | ||
offsets: { | ||
end: 20, | ||
start: 0, | ||
}, | ||
range: { | ||
end: { | ||
character: 20, | ||
line: 0, | ||
}, | ||
start: { | ||
character: 0, | ||
line: 0, | ||
}, | ||
}, | ||
severity: 1, | ||
}, | ||
]); | ||
}); | ||
test('Does not fail if default arguments for procedure not provided', () => { | ||
expect( | ||
getDiagnosticsForQuery({ | ||
query: `CALL apoc.load.xml('url', '/path')`, | ||
dbSchema: testData.mockSchema, | ||
}), | ||
).toEqual([]); | ||
}); | ||
test('Does not fail semantic validation for functions that expect LIST<ANY>', () => { | ||
test('Does not error on dynamic properties', () => { | ||
expect( | ||
getDiagnosticsForQuery({ | ||
query: `RETURN apoc.coll.max(['a'])`, | ||
query: `MATCH (n) | ||
SET n["prop"] = "some value"`, | ||
dbSchema: testData.mockSchema, | ||
@@ -1611,6 +1482,6 @@ }), | ||
test('Provides semantic validation for functions that expect LIST<NUMBER>', () => { | ||
test('Shows deprecation for CALL IN TXs without parentheses', () => { | ||
expect( | ||
getDiagnosticsForQuery({ | ||
query: `RETURN apoc.coll.sum(['a', 'b'])`, | ||
query: `CALL { MATCH (n) RETURN n} IN TRANSACTIONS OF 50 ROWS RETURN 1`, | ||
dbSchema: testData.mockSchema, | ||
@@ -1621,18 +1492,18 @@ }), | ||
message: | ||
'Type mismatch: expected List<Float>, List<Integer> or List<Number> but was List<String>', | ||
'CALL subquery without a variable scope clause is now deprecated. Use CALL () { ... }', | ||
offsets: { | ||
end: 31, | ||
start: 21, | ||
end: 62, | ||
start: 0, | ||
}, | ||
range: { | ||
end: { | ||
character: 31, | ||
character: 62, | ||
line: 0, | ||
}, | ||
start: { | ||
character: 21, | ||
character: 0, | ||
line: 0, | ||
}, | ||
}, | ||
severity: 1, | ||
severity: 2, | ||
}, | ||
@@ -1642,52 +1513,12 @@ ]); | ||
// TODO This doesn't seem to warn on deprecated | ||
// arguments for either functions or procedures, | ||
// needs to be solved in the database first | ||
test('Notifies of deprecated returns in procedures', () => { | ||
test('Shows deprecation for CALL without parentheses', () => { | ||
expect( | ||
getDiagnosticsForQuery({ | ||
query: `CALL apoc.meta.graphSample({})`, | ||
dbSchema: { | ||
functions: {}, | ||
procedures: { | ||
'apoc.meta.graphSample': { | ||
name: 'apoc.meta.graphSample', | ||
description: | ||
'Examines the full graph and returns a meta-graph.\nUnlike `apoc.meta.graph`, this procedure does not filter away non-existing paths.', | ||
mode: 'DEFAULT', | ||
worksOnSystem: false, | ||
argumentDescription: [ | ||
{ | ||
isDeprecated: false, | ||
default: 'DefaultParameterValue{value={}, type=MAP}', | ||
description: 'config = {} :: MAP', | ||
name: 'config', | ||
type: 'MAP', | ||
}, | ||
], | ||
signature: | ||
'apoc.meta.graphSample(config = {} :: MAP) :: (nodes :: LIST<NODE>, relationships :: LIST<RELATIONSHIP>)', | ||
returnDescription: [ | ||
{ | ||
isDeprecated: true, | ||
description: 'nodes :: LIST<NODE>', | ||
name: 'nodes', | ||
type: 'LIST<NODE>', | ||
}, | ||
{ | ||
isDeprecated: false, | ||
description: 'relationships :: LIST<RELATIONSHIP>', | ||
name: 'relationships', | ||
type: 'LIST<RELATIONSHIP>', | ||
}, | ||
], | ||
admin: false, | ||
option: { | ||
deprecated: false, | ||
}, | ||
}, | ||
}, | ||
}, | ||
query: `WITH 1 AS i | ||
CALL { | ||
RETURN 3 AS j | ||
} | ||
RETURN i | ||
`, | ||
dbSchema: testData.mockSchema, | ||
}), | ||
@@ -1697,15 +1528,15 @@ ).toEqual([ | ||
message: | ||
"The query used a deprecated field from a procedure. ('nodes' returned by 'apoc.meta.graphSample' is deprecated.)", | ||
'CALL subquery without a variable scope clause is now deprecated. Use CALL () { ... }', | ||
offsets: { | ||
end: 30, | ||
start: 0, | ||
end: 67, | ||
start: 22, | ||
}, | ||
range: { | ||
end: { | ||
character: 30, | ||
line: 0, | ||
character: 11, | ||
line: 3, | ||
}, | ||
start: { | ||
character: 0, | ||
line: 0, | ||
character: 10, | ||
line: 1, | ||
}, | ||
@@ -1712,0 +1543,0 @@ }, |
@@ -39,3 +39,3 @@ import { getDiagnosticsForQuery } from './helpers'; | ||
message: | ||
'Expected any of ALTER, CALL, CREATE, DEALLOCATE, DELETE, DENY, DETACH, DROP, DRYRUN, ENABLE, EXPLAIN, FINISH, FOREACH, GRANT, INSERT, LOAD, MATCH, MERGE, NODETACH, OPTIONAL, PROFILE, REALLOCATE, REMOVE, RENAME, RETURN, REVOKE, SET, SHOW, START, STOP, TERMINATE, UNWIND, USE, USING or WITH', | ||
'Expected any of ALTER, CALL, CREATE, DEALLOCATE, DELETE, DENY, DETACH, DROP, DRYRUN, ENABLE, EXPLAIN, FINISH, FOREACH, GRANT, INSERT, LIMIT, LOAD, MATCH, MERGE, NODETACH, OFFSET, OPTIONAL, ORDER, PROFILE, REALLOCATE, REMOVE, RENAME, RETURN, REVOKE, SET, SHOW, SKIP, START, STOP, TERMINATE, UNWIND, USE, USING or WITH', | ||
range: { | ||
@@ -120,3 +120,3 @@ end: { | ||
message: | ||
"Expected any of '}', AND, CALL, CREATE, DELETE, DETACH, FINISH, FOREACH, INSERT, LOAD, MATCH, MERGE, NODETACH, OPTIONAL, OR, REMOVE, RETURN, SET, UNION, UNWIND, USE, WITH, XOR or an expression", | ||
"Expected any of '}', AND, CALL, CREATE, DELETE, DETACH, FINISH, FOREACH, INSERT, LIMIT, LOAD, MATCH, MERGE, NODETACH, OFFSET, OPTIONAL, OR, ORDER, REMOVE, RETURN, SET, SKIP, UNION, UNWIND, USE, WITH, XOR or an expression", | ||
range: { | ||
@@ -700,3 +700,3 @@ end: { | ||
{ | ||
message: "Expected '{' or a procedure name", | ||
message: "Expected any of '{', '(' or a procedure name", | ||
offsets: { | ||
@@ -818,3 +818,3 @@ end: 6, | ||
{ | ||
message: 'Expected a string or a parameter', | ||
message: 'Expected any of CHANGE, a string or a parameter', | ||
offsets: { | ||
@@ -983,2 +983,17 @@ end: 39, | ||
}); | ||
test.each([ | ||
`MATCH (n:Test1) RETURN n.profile`, | ||
`CREATE (n:Test1 {explain: 'Explain'});`, | ||
`RETURN { clear: 'Clear', params: 'params', history: 'history'}`, | ||
])( | ||
'Syntax validation should not fail if cmd keywords are used in map properties %s', | ||
(query) => { | ||
expect( | ||
getDiagnosticsForQuery({ | ||
query, | ||
}), | ||
).toEqual([]); | ||
}, | ||
); | ||
}); |
@@ -0,1 +1,3 @@ | ||
import { CompletionItem as VSCodeCompletionItem } from 'vscode-languageserver-types'; | ||
export type ReturnDescription = { | ||
@@ -37,2 +39,7 @@ name: string; | ||
aggregating: boolean; | ||
isDeprecated: boolean; | ||
}; | ||
export type CompletionItem = VSCodeCompletionItem & { | ||
signature?: string; | ||
}; |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
Sorry, the diff of this file is too big to display
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Native code
Supply chain riskContains native code (e.g., compiled binaries or shared libraries). Including native code can obscure malicious behavior.
Found 1 instance in 1 package
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Native code
Supply chain riskContains native code (e.g., compiled binaries or shared libraries). Including native code can obscure malicious behavior.
Found 1 instance in 1 package
159543869
2
130
157371
2