UNPKG

langium

Version:

A language engineering tool for the Language Server Protocol

346 lines (317 loc) 16.3 kB
/****************************************************************************** * Copyright 2021 TypeFox GmbH * This program and the accompanying materials are made available under the * terms of the MIT License, which is available in the project root. ******************************************************************************/ import type { LangiumCoreServices } from '../services.js'; import type { AstNode, AstNodeDescription, AstReflection, CstNode, LinkingError, MultiReference, MultiReferenceItem, Reference, ReferenceInfo } from '../syntax-tree.js'; import type { AstNodeLocator } from '../workspace/ast-node-locator.js'; import type { LangiumDocument, LangiumDocuments } from '../workspace/documents.js'; import type { ScopeProvider } from './scope-provider.js'; import { CancellationToken } from '../utils/cancellation.js'; import { isAstNode, isAstNodeDescription, isLinkingError } from '../syntax-tree.js'; import { findRootNode, streamAst, streamReferences } from '../utils/ast-utils.js'; import { interruptAndCheck } from '../utils/promise-utils.js'; import { DocumentState } from '../workspace/documents.js'; /** * Language-specific service for resolving cross-references in the AST. */ export interface Linker { /** * Links all cross-references within the specified document. The default implementation loads only target * elements from documents that are present in the `LangiumDocuments` service. The linked references are * stored in the document's `references` property. * * @param document A LangiumDocument that shall be linked. * @param cancelToken A token for cancelling the operation. * * @throws `OperationCancelled` if a cancellation event is detected */ link(document: LangiumDocument, cancelToken?: CancellationToken): Promise<void>; /** * Unlinks all references within the specified document and removes them from the list of `references`. * * @param document A LangiumDocument that shall be unlinked. */ unlink(document: LangiumDocument): void; /** * Determines a candidate AST node description for linking the given reference. * * @param refInfo Information about the reference. */ getCandidate(refInfo: ReferenceInfo): AstNodeDescription | LinkingError; /** * Determines a candidate AST node description for linking the given reference. * * @param node The AST node containing the reference. * @param refId The reference identifier used to build a scope. * @param reference The actual reference to resolve. */ getCandidates(refInfo: ReferenceInfo): AstNodeDescription[] | LinkingError; /** * Creates a cross reference node being aware of its containing AstNode, the corresponding CstNode, * the cross reference text denoting the target AstNode being already extracted of the document text, * as well as the unique cross reference identifier. * * Default behavior: * - The returned Reference's 'ref' property pointing to the target AstNode is populated lazily on its * first visit. * - If the target AstNode cannot be resolved on the first visit, an error indicator will be installed * and further resolution attempts will *not* be performed. * * @param node The containing AST node * @param property The AST node property being referenced * @param refNode The corresponding CST node * @param refText The cross reference text denoting the target AstNode * @returns the desired Reference node, whose behavior wrt. resolving the cross reference is implementation specific. */ buildReference(node: AstNode, property: string, refNode: CstNode | undefined, refText: string): Reference; buildMultiReference(node: AstNode, property: string, refNode: CstNode | undefined, refText: string): MultiReference; } export const RefResolving = Symbol('RefResolving'); export interface DefaultReference extends Reference { _ref?: AstNode | LinkingError | typeof RefResolving; _nodeDescription?: AstNodeDescription; } export interface DefaultMultiReference extends MultiReference { _items: MultiReferenceItem[] | typeof RefResolving | undefined; _linkingError?: LinkingError; } export class DefaultLinker implements Linker { protected readonly reflection: AstReflection; protected readonly scopeProvider: ScopeProvider; protected readonly astNodeLocator: AstNodeLocator; protected readonly langiumDocuments: () => LangiumDocuments; constructor(services: LangiumCoreServices) { this.reflection = services.shared.AstReflection; this.langiumDocuments = () => services.shared.workspace.LangiumDocuments; this.scopeProvider = services.references.ScopeProvider; this.astNodeLocator = services.workspace.AstNodeLocator; } async link(document: LangiumDocument, cancelToken = CancellationToken.None): Promise<void> { for (const node of streamAst(document.parseResult.value)) { await interruptAndCheck(cancelToken); streamReferences(node).forEach(ref => this.doLink(ref, document)); } } protected doLink(refInfo: ReferenceInfo, document: LangiumDocument): void { const ref = refInfo.reference as DefaultReference | DefaultMultiReference; // The reference may already have been resolved lazily by accessing its `ref` property. if ('_ref' in ref && ref._ref === undefined) { ref._ref = RefResolving; try { const description = this.getCandidate(refInfo); if (isLinkingError(description)) { ref._ref = description; } else { ref._nodeDescription = description; const linkedNode = this.loadAstNode(description); ref._ref = linkedNode ?? this.createLinkingError(refInfo, description); } } catch (err) { console.error(`An error occurred while resolving reference to '${ref.$refText}':`, err); const errorMessage = (err as Error).message ?? String(err); ref._ref = { info: refInfo, message: `An error occurred while resolving reference to '${ref.$refText}': ${errorMessage}` }; } document.references.push(ref); } else if ('_items' in ref && ref._items === undefined) { ref._items = RefResolving; try { const descriptions = this.getCandidates(refInfo); const items: MultiReferenceItem[] = []; if (isLinkingError(descriptions)) { ref._linkingError = descriptions; } else { for (const description of descriptions) { const linkedNode = this.loadAstNode(description); if (linkedNode) { items.push({ ref: linkedNode, $nodeDescription: description }); } } } ref._items = items; } catch (err) { ref._linkingError = { info: refInfo, message: `An error occurred while resolving reference to '${ref.$refText}': ${err}` }; ref._items = []; } document.references.push(ref); } } unlink(document: LangiumDocument): void { for (const ref of document.references) { if ('_ref' in ref) { (ref as DefaultReference)._ref = undefined; delete (ref as DefaultReference)._nodeDescription; } else if ('_items' in ref) { (ref as DefaultMultiReference)._items = undefined; delete (ref as DefaultMultiReference)._linkingError; } } document.references = []; } getCandidate(refInfo: ReferenceInfo): AstNodeDescription | LinkingError { const scope = this.scopeProvider.getScope(refInfo); const description = scope.getElement(refInfo.reference.$refText); return description ?? this.createLinkingError(refInfo); } getCandidates(refInfo: ReferenceInfo): AstNodeDescription[] | LinkingError { const scope = this.scopeProvider.getScope(refInfo); const descriptions = scope.getElements(refInfo.reference.$refText).distinct(desc => `${desc.documentUri}#${desc.path}`).toArray(); return descriptions.length > 0 ? descriptions : this.createLinkingError(refInfo); } buildReference(node: AstNode, property: string, refNode: CstNode | undefined, refText: string): Reference { // See behavior description in doc of Linker, update that on changes in here. // eslint-disable-next-line @typescript-eslint/no-this-alias const linker = this; const reference: DefaultReference = { $refNode: refNode, $refText: refText, _ref: undefined, get ref() { if (isAstNode(this._ref)) { // Most frequent case: the target is already resolved. return this._ref; } else if (isAstNodeDescription(this._nodeDescription)) { // A candidate has been found before, but it is not loaded yet. const linkedNode = linker.loadAstNode(this._nodeDescription); this._ref = linkedNode ?? linker.createLinkingError({ reference, container: node, property }, this._nodeDescription); } else if (this._ref === undefined) { // The reference has not been linked yet, so do that now. this._ref = RefResolving; const document = findRootNode(node).$document; const refData = linker.getLinkedNode({ reference, container: node, property }); if (refData.error && document && document.state < DocumentState.ComputedScopes) { // Document scope is not ready, don't set `this._ref` so linker can retry later. return this._ref = undefined; } this._ref = refData.node ?? refData.error; this._nodeDescription = refData.descr; document?.references.push(this); } else if (this._ref === RefResolving) { linker.throwCyclicReferenceError(node, property, refText); } return isAstNode(this._ref) ? this._ref : undefined; }, get $nodeDescription() { return this._nodeDescription; }, get error() { return isLinkingError(this._ref) ? this._ref : undefined; } }; return reference; } buildMultiReference(node: AstNode, property: string, refNode: CstNode | undefined, refText: string): MultiReference { // See behavior description in doc of Linker, update that on changes in here. // eslint-disable-next-line @typescript-eslint/no-this-alias const linker = this; const reference: DefaultMultiReference = { $refNode: refNode, $refText: refText, _items: undefined, get items() { if (Array.isArray(this._items)) { return this._items; } else if (this._items === undefined) { this._items = RefResolving; const document = findRootNode(node).$document; const descriptions = linker.getCandidates({ reference, container: node, property }); const items: MultiReferenceItem[] = []; if (isLinkingError(descriptions)) { this._linkingError = descriptions; } else { for (const description of descriptions) { const linkedNode = linker.loadAstNode(description); if (linkedNode) { items.push({ ref: linkedNode, $nodeDescription: description }); } } } this._items = items; document?.references.push(this); } else if (this._items === RefResolving) { linker.throwCyclicReferenceError(node, property, refText); } return Array.isArray(this._items) ? this._items : []; }, get error() { if (this._linkingError) { return this._linkingError; } const refs = this.items; if (refs.length > 0) { return undefined; } else { return (this._linkingError = linker.createLinkingError({ reference, container: node, property })); } } }; return reference; } protected throwCyclicReferenceError(node: AstNode, property: string, refText: string): never { throw new Error(`Cyclic reference resolution detected: ${this.astNodeLocator.getAstNodePath(node)}/${property} (symbol '${refText}')`); } protected getLinkedNode(refInfo: ReferenceInfo): { node?: AstNode, descr?: AstNodeDescription, error?: LinkingError } { try { const description = this.getCandidate(refInfo); if (isLinkingError(description)) { return { error: description }; } const linkedNode = this.loadAstNode(description); if (linkedNode) { return { node: linkedNode, descr: description }; } else { return { descr: description, error: this.createLinkingError(refInfo, description) }; } } catch (err) { console.error(`An error occurred while resolving reference to '${refInfo.reference.$refText}':`, err); const errorMessage = (err as Error).message ?? String(err); return { error: { info: refInfo, message: `An error occurred while resolving reference to '${refInfo.reference.$refText}': ${errorMessage}` } }; } } protected loadAstNode(nodeDescription: AstNodeDescription): AstNode | undefined { if (nodeDescription.node) { return nodeDescription.node; } const doc = this.langiumDocuments().getDocument(nodeDescription.documentUri); if (!doc) { return undefined; } return this.astNodeLocator.getAstNode(doc.parseResult.value, nodeDescription.path); } protected createLinkingError(refInfo: ReferenceInfo, targetDescription?: AstNodeDescription): LinkingError { // Check whether the document is sufficiently processed by the DocumentBuilder. If not, this is a hint for a bug // in the language implementation. const document = findRootNode(refInfo.container).$document; if (document && document.state < DocumentState.ComputedScopes) { console.warn(`Attempted reference resolution before document reached ComputedScopes state (${document.uri}).`); } const referenceType = this.reflection.getReferenceType(refInfo); return { info: refInfo, message: `Could not resolve reference to ${referenceType} named '${refInfo.reference.$refText}'.`, targetDescription }; } }