@neo4j-cypher/react-codemirror 2.0.0-next.3 → 2.0.0-next.30

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (163) hide show
  1. package/CHANGELOG.md +257 -0
  2. package/README.md +3 -2
  3. package/dist/{types → src}/CypherEditor.d.ts +81 -3
  4. package/dist/src/CypherEditor.js +336 -0
  5. package/dist/src/CypherEditor.js.map +1 -0
  6. package/dist/src/CypherEditor.test.js +154 -0
  7. package/dist/src/CypherEditor.test.js.map +1 -0
  8. package/dist/src/constants.d.ts +1 -0
  9. package/dist/src/constants.js +2 -0
  10. package/dist/src/constants.js.map +1 -0
  11. package/dist/src/e2e_tests/autoCompletion.spec.js +332 -0
  12. package/dist/src/e2e_tests/autoCompletion.spec.js.map +1 -0
  13. package/dist/src/e2e_tests/configuration.spec.js +83 -0
  14. package/dist/src/e2e_tests/configuration.spec.js.map +1 -0
  15. package/dist/src/e2e_tests/debounce.spec.js +66 -0
  16. package/dist/src/e2e_tests/debounce.spec.js.map +1 -0
  17. package/dist/{types/e2e_tests/e2e-utils.d.ts → src/e2e_tests/e2eUtils.d.ts} +2 -0
  18. package/dist/src/e2e_tests/e2eUtils.js +79 -0
  19. package/dist/src/e2e_tests/e2eUtils.js.map +1 -0
  20. package/dist/src/e2e_tests/extraKeybindings.spec.js +43 -0
  21. package/dist/src/e2e_tests/extraKeybindings.spec.js.map +1 -0
  22. package/dist/src/e2e_tests/historyNavigation.spec.js +227 -0
  23. package/dist/src/e2e_tests/historyNavigation.spec.js.map +1 -0
  24. package/dist/src/e2e_tests/performanceTest.spec.d.ts +6 -0
  25. package/dist/src/e2e_tests/performanceTest.spec.js +97 -0
  26. package/dist/src/e2e_tests/performanceTest.spec.js.map +1 -0
  27. package/dist/src/e2e_tests/sanityChecks.spec.js +53 -0
  28. package/dist/src/e2e_tests/sanityChecks.spec.js.map +1 -0
  29. package/dist/src/e2e_tests/signatureHelp.spec.js +228 -0
  30. package/dist/src/e2e_tests/signatureHelp.spec.js.map +1 -0
  31. package/dist/src/e2e_tests/snippets.spec.js +62 -0
  32. package/dist/src/e2e_tests/snippets.spec.js.map +1 -0
  33. package/dist/src/e2e_tests/syntaxHighlighting.spec.d.ts +1 -0
  34. package/dist/src/e2e_tests/syntaxHighlighting.spec.js +90 -0
  35. package/dist/src/e2e_tests/syntaxHighlighting.spec.js.map +1 -0
  36. package/dist/src/e2e_tests/syntaxValidation.spec.d.ts +1 -0
  37. package/dist/src/e2e_tests/syntaxValidation.spec.js +126 -0
  38. package/dist/src/e2e_tests/syntaxValidation.spec.js.map +1 -0
  39. package/dist/src/historyNavigation.js +163 -0
  40. package/dist/src/historyNavigation.js.map +1 -0
  41. package/dist/{types → src}/icons.d.ts +1 -1
  42. package/dist/src/icons.js +62 -0
  43. package/dist/src/icons.js.map +1 -0
  44. package/dist/src/index.d.ts +4 -0
  45. package/dist/src/index.js +5 -0
  46. package/dist/src/index.js.map +1 -0
  47. package/dist/src/lang-cypher/autocomplete.d.ts +6 -0
  48. package/dist/src/lang-cypher/autocomplete.js +113 -0
  49. package/dist/src/lang-cypher/autocomplete.js.map +1 -0
  50. package/dist/{types → src}/lang-cypher/constants.d.ts +11 -0
  51. package/dist/src/lang-cypher/constants.js +69 -0
  52. package/dist/src/lang-cypher/constants.js.map +1 -0
  53. package/dist/src/lang-cypher/contants.test.d.ts +1 -0
  54. package/dist/src/lang-cypher/contants.test.js +103 -0
  55. package/dist/src/lang-cypher/contants.test.js.map +1 -0
  56. package/dist/src/lang-cypher/createCypherTheme.js +183 -0
  57. package/dist/src/lang-cypher/createCypherTheme.js.map +1 -0
  58. package/dist/src/lang-cypher/langCypher.d.ts +13 -0
  59. package/dist/src/lang-cypher/langCypher.js +23 -0
  60. package/dist/src/lang-cypher/langCypher.js.map +1 -0
  61. package/dist/src/lang-cypher/lintWorker.mjs +2022 -0
  62. package/dist/src/lang-cypher/parser-adapter.d.ts +19 -0
  63. package/dist/src/lang-cypher/parser-adapter.js +113 -0
  64. package/dist/src/lang-cypher/parser-adapter.js.map +1 -0
  65. package/dist/src/lang-cypher/signatureHelp.d.ts +4 -0
  66. package/dist/src/lang-cypher/signatureHelp.js +109 -0
  67. package/dist/src/lang-cypher/signatureHelp.js.map +1 -0
  68. package/dist/{types/lang-cypher/syntax-validation.d.ts → src/lang-cypher/syntaxValidation.d.ts} +2 -1
  69. package/dist/src/lang-cypher/syntaxValidation.js +57 -0
  70. package/dist/src/lang-cypher/syntaxValidation.js.map +1 -0
  71. package/dist/src/lang-cypher/themeIcons.js +22 -0
  72. package/dist/src/lang-cypher/themeIcons.js.map +1 -0
  73. package/dist/src/lang-cypher/utils.d.ts +2 -0
  74. package/dist/src/lang-cypher/utils.js +10 -0
  75. package/dist/src/lang-cypher/utils.js.map +1 -0
  76. package/dist/src/ndlTokensCopy.d.ts +570 -0
  77. package/dist/src/ndlTokensCopy.js +571 -0
  78. package/dist/src/ndlTokensCopy.js.map +1 -0
  79. package/dist/src/ndlTokensCopy.test.d.ts +1 -0
  80. package/dist/src/ndlTokensCopy.test.js +12 -0
  81. package/dist/src/ndlTokensCopy.test.js.map +1 -0
  82. package/dist/src/neo4jSetup.d.ts +6 -0
  83. package/dist/src/neo4jSetup.js +120 -0
  84. package/dist/src/neo4jSetup.js.map +1 -0
  85. package/dist/src/richClipboardCopier.d.ts +4 -0
  86. package/dist/src/richClipboardCopier.js +78 -0
  87. package/dist/src/richClipboardCopier.js.map +1 -0
  88. package/dist/src/richClipboardCopier.test.d.ts +1 -0
  89. package/dist/src/richClipboardCopier.test.js +53 -0
  90. package/dist/src/richClipboardCopier.test.js.map +1 -0
  91. package/dist/{types → src}/themes.d.ts +1 -1
  92. package/dist/src/themes.js +93 -0
  93. package/dist/src/themes.js.map +1 -0
  94. package/dist/tsconfig.tsbuildinfo +1 -0
  95. package/package.json +43 -41
  96. package/src/CypherEditor.test.tsx +204 -0
  97. package/src/CypherEditor.tsx +316 -42
  98. package/src/constants.ts +1 -0
  99. package/src/e2e_tests/autoCompletion.spec.tsx +571 -0
  100. package/src/e2e_tests/configuration.spec.tsx +111 -0
  101. package/src/e2e_tests/debounce.spec.tsx +106 -0
  102. package/src/e2e_tests/{e2e-utils.ts → e2eUtils.ts} +41 -3
  103. package/src/e2e_tests/{extra-keybindings.spec.tsx → extraKeybindings.spec.tsx} +1 -3
  104. package/src/e2e_tests/{history-navigation.spec.tsx → historyNavigation.spec.tsx} +137 -18
  105. package/src/e2e_tests/performanceTest.spec.tsx +163 -0
  106. package/src/e2e_tests/{sanity-checks.spec.tsx → sanityChecks.spec.tsx} +7 -22
  107. package/src/e2e_tests/signatureHelp.spec.tsx +444 -0
  108. package/src/e2e_tests/snippets.spec.tsx +92 -0
  109. package/src/e2e_tests/{syntax-highlighting.spec.tsx → syntaxHighlighting.spec.tsx} +26 -24
  110. package/src/e2e_tests/syntaxValidation.spec.tsx +259 -0
  111. package/src/{history-navigation.ts → historyNavigation.ts} +1 -1
  112. package/src/icons.ts +3 -0
  113. package/src/index.ts +2 -2
  114. package/src/lang-cypher/autocomplete.ts +99 -18
  115. package/src/lang-cypher/constants.ts +27 -0
  116. package/src/lang-cypher/contants.test.ts +6 -2
  117. package/src/lang-cypher/{create-cypher-theme.ts → createCypherTheme.ts} +45 -2
  118. package/src/lang-cypher/langCypher.ts +42 -0
  119. package/src/lang-cypher/lintWorker.mjs +2022 -0
  120. package/src/lang-cypher/parser-adapter.ts +145 -0
  121. package/src/lang-cypher/signatureHelp.ts +151 -0
  122. package/src/lang-cypher/syntaxValidation.ts +72 -0
  123. package/src/lang-cypher/utils.ts +9 -0
  124. package/src/{ndl-tokens-copy.test.ts → ndlTokensCopy.test.ts} +2 -1
  125. package/src/ndlTokensCopy.ts +570 -0
  126. package/src/{neo4j-setup.tsx → neo4jSetup.tsx} +78 -17
  127. package/src/richClipboardCopier.test.ts +65 -0
  128. package/src/richClipboardCopier.ts +99 -0
  129. package/src/themes.ts +45 -70
  130. package/src/viteEnv.d.ts +1 -0
  131. package/dist/cjs/index.cjs +0 -1440
  132. package/dist/cjs/index.cjs.map +0 -7
  133. package/dist/esm/index.mjs +0 -1463
  134. package/dist/esm/index.mjs.map +0 -7
  135. package/dist/types/e2e_tests/mock-data.d.ts +0 -3779
  136. package/dist/types/index.d.ts +0 -4
  137. package/dist/types/lang-cypher/ParserAdapter.d.ts +0 -14
  138. package/dist/types/lang-cypher/autocomplete.d.ts +0 -3
  139. package/dist/types/lang-cypher/lang-cypher.d.ts +0 -7
  140. package/dist/types/ndl-tokens-copy.d.ts +0 -379
  141. package/dist/types/neo4j-setup.d.ts +0 -2
  142. package/dist/types/tsconfig.tsbuildinfo +0 -1
  143. package/src/e2e_tests/auto-completion.spec.tsx +0 -232
  144. package/src/e2e_tests/mock-data.ts +0 -4310
  145. package/src/e2e_tests/performance-test.spec.tsx +0 -71
  146. package/src/e2e_tests/syntax-validation.spec.tsx +0 -156
  147. package/src/lang-cypher/ParserAdapter.ts +0 -92
  148. package/src/lang-cypher/lang-cypher.ts +0 -32
  149. package/src/lang-cypher/syntax-validation.ts +0 -24
  150. package/src/ndl-tokens-copy.ts +0 -379
  151. /package/dist/{types/e2e_tests/auto-completion.spec.d.ts → src/CypherEditor.test.d.ts} +0 -0
  152. /package/dist/{types/e2e_tests/extra-keybindings.spec.d.ts → src/e2e_tests/autoCompletion.spec.d.ts} +0 -0
  153. /package/dist/{types/e2e_tests/history-navigation.spec.d.ts → src/e2e_tests/configuration.spec.d.ts} +0 -0
  154. /package/dist/{types/e2e_tests/performance-test.spec.d.ts → src/e2e_tests/debounce.spec.d.ts} +0 -0
  155. /package/dist/{types/e2e_tests/sanity-checks.spec.d.ts → src/e2e_tests/extraKeybindings.spec.d.ts} +0 -0
  156. /package/dist/{types/e2e_tests/syntax-highlighting.spec.d.ts → src/e2e_tests/historyNavigation.spec.d.ts} +0 -0
  157. /package/dist/{types/e2e_tests/syntax-validation.spec.d.ts → src/e2e_tests/sanityChecks.spec.d.ts} +0 -0
  158. /package/dist/{types/lang-cypher/contants.test.d.ts → src/e2e_tests/signatureHelp.spec.d.ts} +0 -0
  159. /package/dist/{types/ndl-tokens-copy.test.d.ts → src/e2e_tests/snippets.spec.d.ts} +0 -0
  160. /package/dist/{types/history-navigation.d.ts → src/historyNavigation.d.ts} +0 -0
  161. /package/dist/{types/lang-cypher/create-cypher-theme.d.ts → src/lang-cypher/createCypherTheme.d.ts} +0 -0
  162. /package/dist/{types/lang-cypher/theme-icons.d.ts → src/lang-cypher/themeIcons.d.ts} +0 -0
  163. /package/src/lang-cypher/{theme-icons.ts → themeIcons.ts} +0 -0
@@ -0,0 +1,145 @@
1
+ import { Facet } from '@codemirror/state';
2
+ import { Input, NodeType, Parser, PartialParse, Tree } from '@lezer/common';
3
+ import {
4
+ applySyntaxColouring,
5
+ ParsedCypherToken,
6
+ } from '@neo4j-cypher/language-support';
7
+
8
+ import Prism from 'prismjs';
9
+ import {
10
+ CodemirrorParseTokenType,
11
+ cypherTokenTypeToNode,
12
+ parserAdapterNodeSet,
13
+ } from './constants';
14
+ // This import will load the cypher support in prisma
15
+ import 'prismjs/components/prism-cypher';
16
+ import { CypherConfig } from './langCypher';
17
+
18
+ const DEFAULT_NODE_GROUP_SIZE = 4;
19
+ Prism.manual = true;
20
+
21
+ export class ParserAdapter extends Parser {
22
+ cypherTokenTypeToNode: Record<CodemirrorParseTokenType, NodeType>;
23
+
24
+ constructor(facet: Facet<unknown>, private config: CypherConfig) {
25
+ super();
26
+ this.cypherTokenTypeToNode = cypherTokenTypeToNode(facet);
27
+ }
28
+
29
+ createParse(input: Input): PartialParse {
30
+ return this.startParse(input);
31
+ }
32
+
33
+ /* There are more arguments, but since we don't do any incremental parsing, they are not useful */
34
+ startParse(input: string | Input): PartialParse {
35
+ const document =
36
+ typeof input === 'string' ? input : input.read(0, input.length);
37
+
38
+ const tree = this.buildTree(document);
39
+
40
+ return {
41
+ stoppedAt: input.length,
42
+ parsedPos: input.length,
43
+ stopAt: () => {
44
+ return undefined;
45
+ },
46
+ advance: () => tree,
47
+ };
48
+ }
49
+
50
+ private buildTree(document: string) {
51
+ const parse = this.config.useLightVersion
52
+ ? this.prismParse(document)
53
+ : this.antlrParse(document);
54
+
55
+ if (parse.tokens.length === 0) {
56
+ return this.createEmptyTree(document);
57
+ }
58
+
59
+ const buffer =
60
+ parse.type === 'prism'
61
+ ? this.createBufferForPrismTokens(parse.tokens)
62
+ : this.createBufferForAntlrTokens(parse.tokens);
63
+
64
+ this.addTopNodeToBuffer(buffer, document);
65
+
66
+ return Tree.build({
67
+ buffer: buffer.flat(),
68
+ nodeSet: parserAdapterNodeSet(this.cypherTokenTypeToNode),
69
+ topID: this.cypherTokenTypeToNode.topNode.id,
70
+ });
71
+ }
72
+
73
+ private antlrParse(document: string) {
74
+ const startTime = performance.now();
75
+ const tokens = applySyntaxColouring(document);
76
+ const timeTaken = performance.now() - startTime;
77
+ if (timeTaken > 300) {
78
+ this.config.setUseLightVersion?.(true);
79
+ }
80
+ return { type: 'antlr' as const, tokens };
81
+ }
82
+
83
+ private prismParse(document: string) {
84
+ if (document.length === 0) {
85
+ this.config.setUseLightVersion?.(false);
86
+ }
87
+ const tokens = Prism.tokenize(document, Prism.languages.cypher);
88
+ return {
89
+ type: 'prism' as const,
90
+ tokens,
91
+ };
92
+ }
93
+ private createBufferForAntlrTokens(tokens: ParsedCypherToken[]) {
94
+ return tokens.map((token) => {
95
+ const nodeTypeId = this.cypherTokenTypeToNode[token.tokenType].id;
96
+ const startOffset = token.position.startOffset;
97
+ const endOffset = token.position.startOffset + token.length;
98
+
99
+ return [nodeTypeId, startOffset, endOffset, DEFAULT_NODE_GROUP_SIZE];
100
+ });
101
+ }
102
+
103
+ private createBufferForPrismTokens(tokens: (string | Prism.Token)[]) {
104
+ let totalOffset = 0;
105
+ return tokens.map((token) => {
106
+ const tokenType = (
107
+ typeof token === 'string' ? 'variable' : token.type
108
+ ) as CodemirrorParseTokenType;
109
+
110
+ const nodeTypeId = this.cypherTokenTypeToNode[tokenType].id;
111
+ const startOffset = totalOffset;
112
+ const endOffset = startOffset + token.length;
113
+ totalOffset = endOffset;
114
+
115
+ return [nodeTypeId, startOffset, endOffset, DEFAULT_NODE_GROUP_SIZE];
116
+ });
117
+ }
118
+
119
+ private createEmptyTree(document: string) {
120
+ return Tree.build({
121
+ buffer: [
122
+ this.cypherTokenTypeToNode.topNode.id,
123
+ 0,
124
+ document.length,
125
+ DEFAULT_NODE_GROUP_SIZE,
126
+ ],
127
+ nodeSet: parserAdapterNodeSet(this.cypherTokenTypeToNode),
128
+ topID: this.cypherTokenTypeToNode.topNode.id,
129
+ });
130
+ }
131
+
132
+ private addTopNodeToBuffer(buffer: number[][], document: string) {
133
+ const id = this.cypherTokenTypeToNode.topNode.id;
134
+ const startOffset = 0;
135
+ const endOffset = document.length;
136
+ const totalBufferLength = buffer.length * DEFAULT_NODE_GROUP_SIZE;
137
+
138
+ buffer.push([
139
+ id,
140
+ startOffset,
141
+ endOffset,
142
+ totalBufferLength + DEFAULT_NODE_GROUP_SIZE,
143
+ ]);
144
+ }
145
+ }
@@ -0,0 +1,151 @@
1
+ import { EditorState, StateField } from '@codemirror/state';
2
+ import { showTooltip, Tooltip } from '@codemirror/view';
3
+ import { signatureHelp } from '@neo4j-cypher/language-support';
4
+ import {
5
+ MarkupContent,
6
+ SignatureInformation,
7
+ } from 'vscode-languageserver-types';
8
+ import { CypherConfig } from './langCypher';
9
+ import { getDocString } from './utils';
10
+
11
+ function getTriggerCharacter(query: string, caretPosition: number) {
12
+ let i = caretPosition - 1;
13
+ let triggerCharacter = query.at(i);
14
+
15
+ // Discard all space characters. Note that a space can be more than just a ' '
16
+ while (/\s/.test(triggerCharacter) && i > 0) {
17
+ i -= 1;
18
+ triggerCharacter = query.at(i);
19
+ }
20
+
21
+ return triggerCharacter;
22
+ }
23
+
24
+ const createSignatureHelpElement =
25
+ ({
26
+ signature,
27
+ activeParameter,
28
+ }: {
29
+ signature: SignatureInformation;
30
+ activeParameter: number;
31
+ }) =>
32
+ () => {
33
+ const parameters = signature.parameters;
34
+ const doc = getDocString(signature.documentation);
35
+ const dom = document.createElement('div');
36
+ dom.className = 'cm-signature-help-panel';
37
+
38
+ const contents = document.createElement('div');
39
+ contents.className = 'cm-signature-help-panel-contents';
40
+ dom.appendChild(contents);
41
+
42
+ const signatureLabel = document.createElement('div');
43
+ signatureLabel.className = 'cm-signature-help-panel-name';
44
+ const methodName = signature.label.slice(0, signature.label.indexOf('('));
45
+ const returnType = signature.label.slice(signature.label.indexOf(')') + 1);
46
+ signatureLabel.appendChild(document.createTextNode(`${methodName}(`));
47
+ let currentParamDescription: string | undefined = undefined;
48
+
49
+ parameters.forEach((param, index) => {
50
+ if (typeof param.label === 'string') {
51
+ const span = document.createElement('span');
52
+ span.appendChild(document.createTextNode(param.label));
53
+ if (index !== parameters.length - 1) {
54
+ span.appendChild(document.createTextNode(', '));
55
+ }
56
+
57
+ if (index === activeParameter) {
58
+ span.className = 'cm-signature-help-panel-current-argument';
59
+ const paramDoc = param.documentation;
60
+ currentParamDescription = MarkupContent.is(paramDoc)
61
+ ? paramDoc.value
62
+ : paramDoc;
63
+ }
64
+ signatureLabel.appendChild(span);
65
+ }
66
+ });
67
+
68
+ signatureLabel.appendChild(document.createTextNode(')'));
69
+ signatureLabel.appendChild(document.createTextNode(returnType));
70
+
71
+ contents.appendChild(signatureLabel);
72
+
73
+ const separator = document.createElement('div');
74
+ separator.className = 'cm-signature-help-panel-separator';
75
+
76
+ contents.appendChild(separator);
77
+
78
+ if (currentParamDescription !== undefined) {
79
+ const argDescription = document.createElement('div');
80
+ argDescription.className = 'cm-signature-help-panel-arg-description';
81
+ argDescription.appendChild(
82
+ document.createTextNode(currentParamDescription),
83
+ );
84
+ contents.appendChild(argDescription);
85
+ }
86
+ const methodDescription = document.createElement('div');
87
+ methodDescription.className = 'cm-signature-help-panel-description';
88
+ methodDescription.appendChild(document.createTextNode(doc));
89
+ contents.appendChild(methodDescription);
90
+
91
+ return { dom };
92
+ };
93
+
94
+ function getSignatureHelpTooltip(
95
+ state: EditorState,
96
+ config: CypherConfig,
97
+ ): Tooltip[] {
98
+ let result: Tooltip[] = [];
99
+ const schema = config.schema;
100
+ const ranges = state.selection.ranges;
101
+ const range = ranges.at(0);
102
+
103
+ if (schema && ranges.length === 1 && range.from === range.to) {
104
+ const caretPosition = range.from;
105
+ const query = state.doc.toString();
106
+
107
+ const triggerCharacter = getTriggerCharacter(query, caretPosition);
108
+
109
+ if (triggerCharacter === '(' || triggerCharacter === ',') {
110
+ const signatureHelpInfo = signatureHelp(query, schema, caretPosition);
111
+ const activeSignature = signatureHelpInfo.activeSignature;
112
+ const signatures = signatureHelpInfo.signatures;
113
+ const activeParameter = signatureHelpInfo.activeParameter;
114
+
115
+ if (
116
+ activeSignature !== undefined &&
117
+ activeSignature >= 0 &&
118
+ activeSignature < signatures.length &&
119
+ signatures[activeSignature].documentation !== undefined
120
+ ) {
121
+ const signature = signatures[activeSignature];
122
+ const showSignatureTooltipBelow =
123
+ config.showSignatureTooltipBelow ?? true;
124
+
125
+ result = [
126
+ {
127
+ pos: caretPosition,
128
+ above: !showSignatureTooltipBelow,
129
+ arrow: true,
130
+ create: createSignatureHelpElement({ signature, activeParameter }),
131
+ },
132
+ ];
133
+ }
134
+ }
135
+ }
136
+
137
+ return result;
138
+ }
139
+
140
+ export function signatureHelpTooltip(config: CypherConfig) {
141
+ return StateField.define<readonly Tooltip[]>({
142
+ create: (state) => getSignatureHelpTooltip(state, config),
143
+
144
+ update(tooltips, tr) {
145
+ if (!tr.docChanged && !tr.selection) return tooltips;
146
+ return getSignatureHelpTooltip(tr.state, config);
147
+ },
148
+
149
+ provide: (f) => showTooltip.computeN([f], (state) => state.field(f)),
150
+ });
151
+ }
@@ -0,0 +1,72 @@
1
+ import { Diagnostic, linter } from '@codemirror/lint';
2
+ import { Extension } from '@codemirror/state';
3
+ import { DiagnosticSeverity, DiagnosticTag } from 'vscode-languageserver-types';
4
+ import workerpool from 'workerpool';
5
+ import type { CypherConfig } from './langCypher';
6
+ import type { LintWorker } from '@neo4j-cypher/lint-worker';
7
+ import { parserWrapper } from '@neo4j-cypher/language-support';
8
+
9
+ const WorkerURL = new URL('./lintWorker.mjs', import.meta.url).pathname;
10
+
11
+ const pool = workerpool.pool(WorkerURL, {
12
+ minWorkers: 1,
13
+ workerOpts: { type: 'module' },
14
+ workerTerminateTimeout: 2000,
15
+ });
16
+
17
+ export const cypherLinter: (config: CypherConfig) => Extension = (config) =>
18
+ linter(async (view) => {
19
+ if (!config.lint) {
20
+ return [];
21
+ }
22
+ const query = view.state.doc.toString();
23
+ if (query.length === 0) {
24
+ return [];
25
+ }
26
+
27
+ try {
28
+ if (pool.stats().busyWorkers > 0) {
29
+ await pool.terminate(true);
30
+ }
31
+
32
+ const proxyWorker = (await pool.proxy()) as unknown as LintWorker;
33
+ const result = await proxyWorker.lintCypherQuery(
34
+ query,
35
+ config.schema ?? {},
36
+ config.featureFlags ?? {},
37
+ );
38
+
39
+ if (result.symbolTables) {
40
+ parserWrapper.setSymbolsInfo({
41
+ query,
42
+ symbolTables: result.symbolTables,
43
+ });
44
+ }
45
+
46
+ const a: Diagnostic[] = result.diagnostics.map((diagnostic) => {
47
+ return {
48
+ from: diagnostic.offsets.start,
49
+ to: diagnostic.offsets.end,
50
+ severity:
51
+ diagnostic.severity === DiagnosticSeverity.Error
52
+ ? 'error'
53
+ : 'warning',
54
+ message: diagnostic.message,
55
+ ...(diagnostic.tags !== undefined &&
56
+ diagnostic.tags.includes(DiagnosticTag.Deprecated)
57
+ ? { markClass: 'cm-deprecated-element' }
58
+ : {}),
59
+ };
60
+ });
61
+ return a;
62
+ } catch (err) {
63
+ if (!String(err).includes('Worker terminated')) {
64
+ console.error(String(err) + ' ' + query);
65
+ }
66
+ }
67
+ return [];
68
+ });
69
+
70
+ export const cleanupWorkers = () => {
71
+ void pool.terminate();
72
+ };
@@ -0,0 +1,9 @@
1
+ import { MarkupContent } from 'vscode-languageserver-types';
2
+
3
+ export function getDocString(result: string | MarkupContent): string {
4
+ if (MarkupContent.is(result)) {
5
+ result.value;
6
+ } else {
7
+ return result;
8
+ }
9
+ }
@@ -1,5 +1,6 @@
1
1
  import { tokens } from '@neo4j-ndl/base';
2
- import { tokens as tokensCopy } from './ndl-tokens-copy';
2
+ import { expect, test } from 'vitest';
3
+ import { tokens as tokensCopy } from './ndlTokensCopy';
3
4
 
4
5
  /*
5
6
  * Needle has some odd package configuration that made playwright stop working