diff --git a/server/src/server.common.ts b/server/src/server.common.ts index ac38152..f440f97 100644 --- a/server/src/server.common.ts +++ b/server/src/server.common.ts @@ -21,7 +21,19 @@ import { TextDocument } from "vscode-languageserver-textdocument"; import { errors, transformer } from "@openfga/syntax-transformer"; import { defaultDocumentationMap } from "./documentation"; import { getDuplicationFix, getMissingDefinitionFix, getReservedTypeNameFix } from "./code-action"; -import { LineCounter, YAMLSeq, parseDocument } from "yaml"; +import { + LineCounter, + YAMLSeq, + parseDocument, + Range as TokenRange, + isScalar, + visitAsync, + Scalar, + Pair, + Document, + visit, +} from "yaml"; +import { stringify } from "json-to-pretty-yaml"; import { YAMLSourceMap, YamlStoreValidateResults, @@ -31,6 +43,7 @@ import { validateYamlStore, getFieldPosition, getRangeFromToken, + DocumentLoc, } from "./yaml-utils"; import { getRangeOfWord } from "./helpers"; import { getDiagnosticsForDsl as validateDSL } from "./dsl-utils"; @@ -100,16 +113,109 @@ export function startServer(connection: _Connection) { connection.languages.diagnostics.refresh(); }); - async function validateYamlSyntaxAndModel(textDocument: TextDocument): Promise { - const diagnostics: Diagnostic[] = []; - const modelDiagnostics: Diagnostic[] = []; - + async function parseYamlStore( + textDocument: TextDocument, + ): Promise<{ yamlDoc: Document; lineCounter: LineCounter; parsedDiagnostics: Diagnostic[] }> { const lineCounter = new LineCounter(); const yamlDoc = parseDocument(textDocument.getText(), { lineCounter, keepSourceTokens: true, }); + const parsedDiagnostics: Diagnostic[] = []; + + // Basic syntax errors + for (const err of yamlDoc.errors) { + parsedDiagnostics.push({ message: err.message, range: rangeFromLinePos(err.linePos) }); + } + + const importedDocs = new Map(); + + await visitAsync(yamlDoc, { + async Pair(_, pair) { + if (!isScalar(pair.key) || !isScalar(pair.value) || pair.key.value !== "tuple_file" || !pair.value.source) { + return; + } + + const originalRange = pair.key.range; + try { + const result = await getFileContents(URI.parse(textDocument.uri), pair.value.source); + if (pair.value.source.match(/.yaml$/)) { + const file = parseDocument(result.contents); + + const diagnosticFromInclusion: Diagnostic[] = []; + + diagnosticFromInclusion.push( + ...file.errors.map((err) => { + return { + source: "ParseError", + message: "error with external file: " + err.message, + range: getRangeFromToken(originalRange, textDocument), + }; + }), + ); + + if (diagnosticFromInclusion.length) { + parsedDiagnostics.push(...diagnosticFromInclusion); + return undefined; + } + + if (originalRange) { + importedDocs.set(pair.value.source, { range: originalRange, doc: file }); + } + return visit.SKIP; + } + } catch (err) { + parsedDiagnostics.push({ + range: getRangeFromToken(originalRange, textDocument), + message: "error with external file: " + (err as Error).message, + source: "ParseError", + }); + } + }, + }); + + // Override all tuples with new location + for (const p of importedDocs.entries()) { + visit(p[1].doc.contents, { + Scalar(key, node) { + node.range = p[1].range; + }, + }); + } + + // Prepare final virtual doc + visit(yamlDoc, { + Pair(_, pair) { + if (!isScalar(pair.key) || !isScalar(pair.value) || pair.key.value !== "tuple_file" || !pair.value.source) { + return; + } + + const value = importedDocs.get(pair.value.source); + + if (value) { + // Import tuples, and point range at where file field used to exist + const scalar = new Scalar("tuples"); + scalar.source = "tuples"; + scalar.range = value?.range; + + return new Pair(scalar, value?.doc.contents); + } + }, + }); + return { yamlDoc, lineCounter, parsedDiagnostics }; + } + + async function validateYamlSyntaxAndModel(textDocument: TextDocument): Promise { + const diagnostics: Diagnostic[] = []; + const modelDiagnostics: Diagnostic[] = []; + + const { yamlDoc, lineCounter, parsedDiagnostics } = await parseYamlStore(textDocument); + + if (parsedDiagnostics.length) { + return { diagnostics: parsedDiagnostics }; + } + const map = new YAMLSourceMap(); map.doMap(yamlDoc.contents); @@ -119,25 +225,6 @@ export function startServer(connection: _Connection) { return { diagnostics }; } - // Basic syntax errors - for (const err of yamlDoc.errors) { - diagnostics.push({ message: err.message, range: rangeFromLinePos(err.linePos) }); - } - - const keys = [...map.nodes.keys()].filter((key) => key.includes("tuple_file")); - for (const fileField of keys) { - const fileName = yamlDoc.getIn(fileField.split(".")) as string; - try { - await getFileContents(URI.parse(textDocument.uri), fileName); - } catch (err) { - diagnostics.push({ - range: getRangeFromToken(map.nodes.get(fileField), textDocument), - message: "error with external file: " + (err as Error).message, - source: "ParseError", - }); - } - } - let model, modelUri = undefined; @@ -147,7 +234,7 @@ export function startServer(connection: _Connection) { diagnostics.push(...parseYamlModel(yamlDoc, lineCounter)); diagnostics.push(...validateYamlStore(yamlDoc.get("model") as string, yamlDoc, textDocument, map)); } else if (yamlDoc.has("model_file")) { - const position = getFieldPosition(yamlDoc, lineCounter, "model_file"); + const position = getFieldPosition(yamlDoc, lineCounter, "model_file")[0]; const modelFile = yamlDoc.get("model_file") as string; try { diff --git a/server/src/yaml-utils.ts b/server/src/yaml-utils.ts index 0c167ce..f4657bd 100644 --- a/server/src/yaml-utils.ts +++ b/server/src/yaml-utils.ts @@ -1,8 +1,21 @@ import { Range, Position, Diagnostic, DiagnosticSeverity } from "vscode-languageserver"; -import { Document, LineCounter, Node, Range as TokenRange, isMap, isPair, isScalar, isSeq } from "yaml"; +import { + Document, + LineCounter, + Node, + Pair, + Range as TokenRange, + isDocument, + isMap, + isNode, + isPair, + isScalar, + isSeq, + parseDocument, + visit, +} from "yaml"; import { LinePos } from "yaml/dist/errors"; -import { BlockMap, SourceToken } from "yaml/dist/parse/cst"; import { getDiagnosticsForDsl } from "./dsl-utils"; import { ErrorObject, ValidateFunction } from "ajv"; import { transformer } from "@openfga/syntax-transformer"; @@ -10,6 +23,11 @@ import { YamlStoreValidator } from "./openfga-yaml-schema"; import { TextDocument } from "vscode-languageserver-textdocument"; import { URI } from "vscode-uri"; +export type DocumentLoc = { + range: TokenRange; + doc: Document; +}; + export type YamlStoreValidateResults = { diagnostics: Diagnostic[]; modelUri?: URI; @@ -32,22 +50,31 @@ export function rangeFromLinePos(linePos: [LinePos] | [LinePos, LinePos] | undef return { start, end }; } -// Only gets the line of 1st depth. This should be deprecated and replaced. +export function parseDocumentWithFixedRange(contents: string, range: TokenRange): Document { + const doc = parseDocument(contents); + visit(doc, (key, node, path) => { + if (isPair(node) && isScalar(node.key)) { + node.key.range = range; + + return new Pair(node); + } + }); + return doc; +} + export function getFieldPosition( yamlDoc: Document, lineCounter: LineCounter, field: string, -): { line: number; col: number } { - let position: { line: number; col: number } = { line: 0, col: 0 }; - - // Get the model token and find its position - (yamlDoc.contents?.srcToken as BlockMap).items.forEach((i) => { - if (i.key?.offset !== undefined && (i.key as SourceToken).source === field) { - position = lineCounter.linePos(i.key?.offset); +): { line: number; col: number }[] { + const positions: { line: number; col: number }[] = []; + visit(yamlDoc, (key, node, path) => { + if (isPair(node) && isScalar(node.key) && node.key.value === field && node.key.srcToken?.offset) { + positions.push(lineCounter.linePos(node.key.srcToken?.offset)); } }); - return position; + return positions; } export function validateYamlStore( @@ -115,7 +142,7 @@ export function validateYamlStore( } export function parseYamlModel(yamlDoc: Document, lineCounter: LineCounter): Diagnostic[] { - const position = getFieldPosition(yamlDoc, lineCounter, "model"); + const position = getFieldPosition(yamlDoc, lineCounter, "model")[0]; // Shift generated diagnostics by line of model, and indent of 2 let dslDiagnostics = getDiagnosticsForDsl(yamlDoc.get("model") as string); @@ -172,7 +199,6 @@ export class YAMLSourceMap { if (isScalar(node) && node.source && node.range) { this.nodes.set(localPath.join("."), node.range); - return; } } }