Skip to content

Commit

Permalink
Terminal Link Provider
Browse files Browse the repository at this point in the history
This feature allows the user to click on a resource addess inside terraform plan terminal output and have it open the file in the editor.
  • Loading branch information
jpogran committed Jan 26, 2024
1 parent b29c824 commit e835455
Show file tree
Hide file tree
Showing 3 changed files with 153 additions and 0 deletions.
4 changes: 4 additions & 0 deletions src/extension.ts
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ import { TerraformLSCommands } from './commands/terraformls';
import { TerraformCommands } from './commands/terraform';
import * as lsStatus from './status/language';
import { TerraformCloudFeature } from './features/terraformCloud';
import { TerminalLinkProvider } from './providers/terminalLinkProvider';

const id = 'terraform';
const brand = `HashiCorp Terraform`;
Expand Down Expand Up @@ -216,6 +217,9 @@ export async function activate(context: vscode.ExtensionContext): Promise<void>
} catch (error) {
await handleLanguageClientStartError(error, context, reporter);
}

const provider = new TerminalLinkProvider(client);
context.subscriptions.push(vscode.window.registerTerminalLinkProvider(provider));
}

export async function deactivate(): Promise<void> {
Expand Down
54 changes: 54 additions & 0 deletions src/features/terraformPlanLookup.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
/**
* Copyright (c) HashiCorp, Inc.
* SPDX-License-Identifier: MPL-2.0
*/

import * as vscode from 'vscode';
import {
BaseLanguageClient,
ClientCapabilities,
FeatureState,
ServerCapabilities,
StaticFeature,
} from 'vscode-languageclient';
import { ModuleCallsDataProvider } from '../providers/moduleCalls';
import { ExperimentalClientCapabilities } from './types';

const CLIENT_MODULE_CALLS_CMD_ID = 'client.terraformPlanLookup';

export class TerraformPlanLookupFeature implements StaticFeature {
private disposables: vscode.Disposable[] = [];

constructor(private client: BaseLanguageClient, private view: ModuleCallsDataProvider) {}

getState(): FeatureState {
return {
kind: 'static',
};
}

public fillClientCapabilities(capabilities: ClientCapabilities & ExperimentalClientCapabilities): void {
if (!capabilities['experimental']) {
capabilities['experimental'] = {};
}
capabilities['experimental']['terraformPlanLookupCommandId'] = CLIENT_MODULE_CALLS_CMD_ID;
}

public async initialize(capabilities: ServerCapabilities): Promise<void> {
this.disposables.push(vscode.window.registerTreeDataProvider('terraform.modules', this.view));

if (!capabilities.experimental?.terraformPlanLookup) {
console.log('Server does not support client.terraformPlanLookup');
return;
}

const d = this.client.onRequest(CLIENT_MODULE_CALLS_CMD_ID, () => {
this.view?.refresh();
});
this.disposables.push(d);
}

public dispose(): void {
this.disposables.forEach((d: vscode.Disposable) => d.dispose());
}
}
95 changes: 95 additions & 0 deletions src/providers/terminalLinkProvider.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
import * as vscode from 'vscode';
import { ExecuteCommandParams, ExecuteCommandRequest } from 'vscode-languageclient';
import { LanguageClient } from 'vscode-languageclient/node';

export class TerminalLinkProvider implements vscode.TerminalLinkProvider {
constructor(private client: LanguageClient) {}

async handleTerminalLink(link: TerraformTerminalLink): Promise<void> {
const document = await vscode.workspace.openTextDocument(link.uri);
await vscode.window.showTextDocument(document, { selection: link.range });
}

async provideTerminalLinks(
context: vscode.TerminalLinkContext,
token: vscode.CancellationToken,

Check warning on line 15 in src/providers/terminalLinkProvider.ts

View workflow job for this annotation

GitHub Actions / lint

'token' is defined but never used
): Promise<vscode.TerminalLink[]> {
const links: vscode.TerminalLink[] = [];

// # docker_image.nginx will be created
const potential = context.line.trimStart();
if (!potential.startsWith('#')) {
return links;
}
let address = potential.split(' ')[1];
const startIndex = potential.indexOf(address);
const endIndex = startIndex + address.length;

if (address.startsWith('module')) {
address = 'module.' + address.split('.')[1];
}

// TODO: Figure out cwd to send as moduleDir
// this isn't ideal, but it's the only way to get the folder for the terminal
// right now. terminal doesn't reliably expose the cwd in all cases
let folder: vscode.Uri | vscode.WorkspaceFolder | undefined;
if ('cwd' in context.terminal.creationOptions && context.terminal.creationOptions.cwd) {
folder = vscode.workspace.getWorkspaceFolder(
typeof context.terminal.creationOptions.cwd === 'string'
? vscode.Uri.file(context.terminal.creationOptions.cwd)
: context.terminal.creationOptions.cwd,
);
} else {
folder = vscode.workspace.workspaceFolders?.[0].uri;
}
const modDir = vscode.Uri.parse(folder?.toString() ?? '');

const params: ExecuteCommandParams = {
command: 'terraform-ls.terraform.plan.lookup',
arguments: [`uri=${modDir.toString()}`, `line=${address}`],
};

const response = await this.client.sendRequest<ExecuteCommandParams, TerraformPlanLookupResponse, void>(
ExecuteCommandRequest.type,
params,
);
if (!response || !response.fileUri || !response.range) {
return links;
}

const uri = vscode.Uri.parse(response.fileUri);
const range = new vscode.Range(
response.range.startLine,
response.range.startCharacter,
response.range.endLine,
response.range.endCharacter,
);

links.push(new TerraformTerminalLink(startIndex, endIndex, 'Open the file', uri, range));

return links;
}
}

class TerraformTerminalLink extends vscode.TerminalLink {
constructor(
startIndex: number,
endIndex: number,
tooltip: string,
public uri: vscode.Uri,
public range: vscode.Range,
) {
super(startIndex, endIndex, tooltip);
}
}

interface TerraformPlanLookupResponse {
v: number;
range: {
startLine: number;
startCharacter: number;
endLine: number;
endCharacter: number;
};
fileUri: string;
}

0 comments on commit e835455

Please sign in to comment.