/****************************************************************************** * Copyright 2022 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 { CallHierarchyIncomingCall, CallHierarchyIncomingCallsParams, CallHierarchyItem, CallHierarchyOutgoingCall, CallHierarchyOutgoingCallsParams, CallHierarchyPrepareParams, } from 'vscode-languageserver'; import type { CancellationToken } from '../utils/cancellation.js'; import type { GrammarConfig } from '../languages/grammar-config.js'; import type { NameProvider } from '../references/name-provider.js'; import type { References } from '../references/references.js'; import type { LangiumServices } from './lsp-services.js'; import type { AstNode } from '../syntax-tree.js'; import type { Stream } from '../utils/stream.js'; import type { ReferenceDescription } from '../workspace/ast-descriptions.js'; import type { LangiumDocument, LangiumDocuments } from '../workspace/documents.js'; import type { MaybePromise } from '../utils/promise-utils.js'; import { SymbolKind } from 'vscode-languageserver'; import { findDeclarationNodeAtOffset } from '../utils/cst-utils.js'; import { URI } from '../utils/uri-utils.js'; /** * Language-specific service for handling call hierarchy requests. */ export interface CallHierarchyProvider { prepareCallHierarchy(document: LangiumDocument, params: CallHierarchyPrepareParams, cancelToken?: CancellationToken): MaybePromise; incomingCalls(params: CallHierarchyIncomingCallsParams, cancelToken?: CancellationToken): MaybePromise; outgoingCalls(params: CallHierarchyOutgoingCallsParams, cancelToken?: CancellationToken): MaybePromise; } export abstract class AbstractCallHierarchyProvider implements CallHierarchyProvider { protected readonly grammarConfig: GrammarConfig; protected readonly nameProvider: NameProvider; protected readonly documents: LangiumDocuments; protected readonly references: References; constructor(services: LangiumServices) { this.grammarConfig = services.parser.GrammarConfig; this.nameProvider = services.references.NameProvider; this.documents = services.shared.workspace.LangiumDocuments; this.references = services.references.References; } prepareCallHierarchy(document: LangiumDocument, params: CallHierarchyPrepareParams): MaybePromise { const rootNode = document.parseResult.value; const targetNode = findDeclarationNodeAtOffset( rootNode.$cstNode, document.textDocument.offsetAt(params.position), this.grammarConfig.nameRegexp ); if (!targetNode) { return undefined; } const declarationNode = this.references.findDeclarationNode(targetNode); if (!declarationNode) { return undefined; } return this.getCallHierarchyItems(declarationNode.astNode, document); } protected getCallHierarchyItems(targetNode: AstNode, document: LangiumDocument): CallHierarchyItem[] | undefined { const nameNode = this.nameProvider.getNameNode(targetNode); const name = this.nameProvider.getName(targetNode); if (!nameNode || !targetNode.$cstNode || name === undefined) { return undefined; } return [{ kind: SymbolKind.Method, name, range: targetNode.$cstNode.range, selectionRange: nameNode.range, uri: document.uri.toString(), ...this.getCallHierarchyItem(targetNode) }]; } protected getCallHierarchyItem(_targetNode: AstNode): Partial | undefined { return undefined; } async incomingCalls(params: CallHierarchyIncomingCallsParams): Promise { const document = await this.documents.getOrCreateDocument(URI.parse(params.item.uri)); const rootNode = document.parseResult.value; const targetNode = findDeclarationNodeAtOffset( rootNode.$cstNode, document.textDocument.offsetAt(params.item.range.start), this.grammarConfig.nameRegexp ); if (!targetNode) { return undefined; } const references = this.references.findReferences( targetNode.astNode, { includeDeclaration: false } ); return this.getIncomingCalls(targetNode.astNode, references); } /** * Override this method to collect the incoming calls for your language */ protected abstract getIncomingCalls(node: AstNode, references: Stream): MaybePromise; async outgoingCalls(params: CallHierarchyOutgoingCallsParams): Promise { const document = await this.documents.getOrCreateDocument(URI.parse(params.item.uri)); const rootNode = document.parseResult.value; const targetNode = findDeclarationNodeAtOffset( rootNode.$cstNode, document.textDocument.offsetAt(params.item.range.start), this.grammarConfig.nameRegexp ); if (!targetNode) { return undefined; } return this.getOutgoingCalls(targetNode.astNode); } /** * Override this method to collect the outgoing calls for your language */ protected abstract getOutgoingCalls(node: AstNode): MaybePromise; }