Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

move builtin terminal quick fixes to contribution model #164099

Merged
merged 30 commits into from
Oct 24, 2022
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
30 commits
Select commit Hold shift + click to select a range
2fdae24
add stuff
meganrogge Oct 20, 2022
1899843
tweak
meganrogge Oct 20, 2022
ce05e51
Revert "add stuff"
meganrogge Oct 20, 2022
55bac09
Revert "tweak"
meganrogge Oct 20, 2022
0f5d4df
delete proposed api
meganrogge Oct 20, 2022
b8c2800
add contribution point
meganrogge Oct 20, 2022
6849d34
don't register another terminal contrib point
meganrogge Oct 20, 2022
6f03181
combine contrib points
meganrogge Oct 20, 2022
7f24e91
adopt contributed quick fixes
meganrogge Oct 20, 2022
0d97708
contribute git push quick fix as test
meganrogge Oct 20, 2022
5491e28
delete logs, replace all
meganrogge Oct 20, 2022
074e3c1
Revert "contribute git push quick fix as test"
meganrogge Oct 20, 2022
331c596
return [] if resolved variable is not in expected command/link
meganrogge Oct 20, 2022
0cdc829
add group: prefix to variables
meganrogge Oct 20, 2022
ca231dd
Merge branch 'main' into merogge/provider-quick-fix
meganrogge Oct 20, 2022
701c201
allow multiple groups to be resolved
meganrogge Oct 20, 2022
d34164e
improve docs
meganrogge Oct 20, 2022
45f9614
improve description
meganrogge Oct 20, 2022
19daf71
fix test
meganrogge Oct 20, 2022
33c66e4
Merge branch 'main' into merogge/provider-quick-fix
meganrogge Oct 24, 2022
34a85a4
fix test
meganrogge Oct 24, 2022
0cc0ae8
fix #163965
meganrogge Oct 24, 2022
f58c431
move 3 fixes over to extension contrib model
meganrogge Oct 24, 2022
09264dc
add proposed api placeholder file
meganrogge Oct 24, 2022
2a65a5c
fix test
meganrogge Oct 24, 2022
cbd24be
rename
meganrogge Oct 24, 2022
b091327
reinsert notebook proposal
meganrogge Oct 24, 2022
f6b1ddb
revert change bc args were lost w git similar
meganrogge Oct 24, 2022
db9ff8b
Merge branch 'main' into merogge/provider-quick-fix
meganrogge Oct 24, 2022
476360f
revert accidental deletion
meganrogge Oct 24, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 15 additions & 1 deletion src/vs/platform/terminal/common/terminal.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import { Event } from 'vs/base/common/event';
import { IProcessEnvironment, OperatingSystem } from 'vs/base/common/platform';
import { URI, UriComponents } from 'vs/base/common/uri';
import { createDecorator } from 'vs/platform/instantiation/common/instantiation';
import { IPtyHostProcessReplayEvent, ISerializedCommandDetectionCapability, ITerminalCapabilityStore } from 'vs/platform/terminal/common/capabilities/capabilities';
import { IPtyHostProcessReplayEvent, ISerializedCommandDetectionCapability, ITerminalCapabilityStore, ITerminalOutputMatcher } from 'vs/platform/terminal/common/capabilities/capabilities';
import { IGetTerminalLayoutInfoArgs, IProcessDetails, ISetTerminalLayoutInfoArgs } from 'vs/platform/terminal/common/terminalProcess';
import { ThemeIcon } from 'vs/platform/theme/common/themeService';
import { ISerializableEnvironmentVariableCollections } from 'vs/platform/terminal/common/environmentVariable';
Expand Down Expand Up @@ -797,6 +797,16 @@ export interface ITerminalProfileSource extends IBaseUnresolvedTerminalProfile {

export interface ITerminalContributions {
profiles?: ITerminalProfileContribution[];
quickFixes?: ITerminalQuickFixContribution[];
}

export interface ITerminalQuickFixContribution {
id: string;
commandLineMatcher: string | RegExp;
outputMatcher: ITerminalOutputMatcher;
exitStatus?: boolean;
meganrogge marked this conversation as resolved.
Show resolved Hide resolved
commandToRun?: string;
meganrogge marked this conversation as resolved.
Show resolved Hide resolved
linkToOpen?: string;
meganrogge marked this conversation as resolved.
Show resolved Hide resolved
}

export interface ITerminalProfileContribution {
Expand All @@ -810,6 +820,10 @@ export interface IExtensionTerminalProfile extends ITerminalProfileContribution
extensionIdentifier: string;
}

export interface IExtensionTerminalQuickFix extends ITerminalQuickFixContribution {
extensionIdentifier: string;
}

export type ITerminalProfileObject = ITerminalExecutable | ITerminalProfileSource | IExtensionTerminalProfile | null;
export type ITerminalProfileType = ITerminalProfile | IExtensionTerminalProfile;

Expand Down
1 change: 1 addition & 0 deletions src/vs/workbench/contrib/terminal/browser/terminal.ts
Original file line number Diff line number Diff line change
Expand Up @@ -959,6 +959,7 @@ export interface ITerminalQuickFixOptions {
outputMatcher?: ITerminalOutputMatcher;
getQuickFixes: TerminalQuickFixCallback;
exitStatus?: boolean;
source: string;
}
export type TerminalQuickFixMatchResult = { commandLineMatch: RegExpMatchArray; outputMatch?: RegExpMatchArray | null };
export type TerminalQuickFixAction = IAction | ITerminalQuickFixCommandAction | ITerminalQuickFixOpenerAction;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ export const GitCreatePrOutputRegex = /remote:\s*(https:\/\/github\.com\/.+\/.+\

export function gitSimilarCommand(): ITerminalQuickFixOptions {
return {
source: 'builtin',
id: 'Git Similar',
commandLineMatcher: GitCommandLineRegex,
outputMatcher: {
Expand Down Expand Up @@ -52,6 +53,7 @@ export function gitSimilarCommand(): ITerminalQuickFixOptions {
}
export function gitTwoDashes(): ITerminalQuickFixOptions {
return {
source: 'builtin',
id: 'Git Two Dashes',
commandLineMatcher: GitCommandLineRegex,
outputMatcher: {
Expand All @@ -76,6 +78,7 @@ export function gitTwoDashes(): ITerminalQuickFixOptions {
}
export function freePort(terminalInstance?: Partial<ITerminalInstance>): ITerminalQuickFixOptions {
return {
source: 'builtin',
id: 'Free Port',
commandLineMatcher: AnyCommandLineRegex,
outputMatcher: {
Expand All @@ -102,8 +105,10 @@ export function freePort(terminalInstance?: Partial<ITerminalInstance>): ITermin
}
};
}

export function gitPushSetUpstream(): ITerminalQuickFixOptions {
return {
source: 'builtin',
id: 'Git Push Set Upstream',
commandLineMatcher: GitPushCommandLineRegex,
meganrogge marked this conversation as resolved.
Show resolved Hide resolved
outputMatcher: {
Expand All @@ -129,6 +134,7 @@ export function gitPushSetUpstream(): ITerminalQuickFixOptions {

export function gitCreatePr(): ITerminalQuickFixOptions {
return {
source: 'builtin',
id: 'Git Create Pr',
commandLineMatcher: GitPushCommandLineRegex,
outputMatcher: {
Expand Down
67 changes: 66 additions & 1 deletion src/vs/workbench/contrib/terminal/browser/xterm/quickFixAddon.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ import { IOpenerService } from 'vs/platform/opener/common/opener';
import { IColorTheme, ICssStyleCollector, registerThemingParticipant } from 'vs/platform/theme/common/themeService';
import { PANEL_BACKGROUND } from 'vs/workbench/common/theme';
import { AudioCue, IAudioCueService } from 'vs/workbench/contrib/audioCues/browser/audioCueService';
import { ITerminalQuickFixOptions } from 'vs/workbench/contrib/terminal/browser/terminal';
import { ITerminalQuickFixOpenerAction, ITerminalQuickFixOptions, TerminalQuickFixAction, TerminalQuickFixMatchResult } from 'vs/workbench/contrib/terminal/browser/terminal';
import { DecorationSelector, TerminalDecorationHoverManager, updateLayout } from 'vs/workbench/contrib/terminal/browser/xterm/decorationStyles';
import { TerminalCommandId } from 'vs/workbench/contrib/terminal/common/terminal';
import { TERMINAL_BACKGROUND_COLOR } from 'vs/workbench/contrib/terminal/common/terminalColorRegistry';
Expand All @@ -28,6 +28,9 @@ import { IDecoration, Terminal } from 'xterm';
import type { ITerminalAddon } from 'xterm-headless';
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
import { ILogService } from 'vs/platform/log/common/log';
import { ITerminalContributionService } from 'vs/workbench/contrib/terminal/common/terminalExtensionPoints';
import { IExtensionTerminalQuickFix } from 'vs/platform/terminal/common/terminal';
import { URI } from 'vs/base/common/uri';
const quickFixTelemetryTitle = 'terminal/quick-fix';
type QuickFixResultTelemetryEvent = {
id: string;
Expand Down Expand Up @@ -75,6 +78,7 @@ export class TerminalQuickFixAddon extends Disposable implements ITerminalAddon,
constructor(private readonly _capabilities: ITerminalCapabilityStore,
@IContextMenuService private readonly _contextMenuService: IContextMenuService,
@IConfigurationService private readonly _configurationService: IConfigurationService,
@ITerminalContributionService private readonly _terminalContributionService: ITerminalContributionService,
@IInstantiationService instantiationService: IInstantiationService,
@IAudioCueService private readonly _audioCueService: IAudioCueService,
@IKeybindingService private readonly _keybindingService: IKeybindingService,
Expand All @@ -94,6 +98,9 @@ export class TerminalQuickFixAddon extends Disposable implements ITerminalAddon,
});
}
this._terminalDecorationHoverService = instantiationService.createInstance(TerminalDecorationHoverManager);
for (const quickFix of this._terminalContributionService.quickFixes) {
this.registerCommandFinishedListener(convertExtensionQuickFixOptions(quickFix));
}
}

activate(terminal: Terminal): void {
Expand Down Expand Up @@ -331,3 +338,61 @@ registerThemingParticipant((theme: IColorTheme, collector: ICssStyleCollector) =
collector.addRule(`.${DecorationSelector.CommandDecoration}.${DecorationSelector.QuickFix} { background-color: ${backgroundColor.toString()}; } `);
}
});

export function convertExtensionQuickFixOptions(quickFix: IExtensionTerminalQuickFix): ITerminalQuickFixOptions {
const type = quickFix.commandToRun ? 'command' : quickFix.linkToOpen ? 'opener' : undefined;
const options = {
id: quickFix.id,
commandLineMatcher: quickFix.commandLineMatcher,
outputMatcher: quickFix.outputMatcher,
type,
getQuickFixes: type === 'command' ? (matchResult: TerminalQuickFixMatchResult) => {
const matches = matchResult.outputMatch;
const commandToRun = quickFix.commandToRun;
if (!matches || !commandToRun) {
return;
}
const groups = matches.groups;
if (!groups) {
return;
}
for (const [key, value] of Object.entries(groups)) {
const varToResolve = '${group:' + `${key}` + '}';
meganrogge marked this conversation as resolved.
Show resolved Hide resolved
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

nit: can probably do '${group:' + key + '}' since key is already a string?

const actions: TerminalQuickFixAction[] = [];
if (!commandToRun.includes(varToResolve)) {
return [];
}
const fixedCommand = commandToRun.replaceAll(varToResolve, value);
actions.push({
type: 'command',
command: fixedCommand,
addNewLine: true
});
return actions;
}
return;
} : (matchResult: TerminalQuickFixMatchResult) => {
const matches = matchResult.outputMatch;
const linkToOpen = quickFix.linkToOpen;
if (!matches || !linkToOpen) {
return;
}
const groups = matches.groups;
if (!groups) {
return;
}
for (const [key, value] of Object.entries(groups)) {
const varToResolve = '${group:' + `${key}` + '}';
if (!linkToOpen?.includes(varToResolve)) {
return [];
}
const link = linkToOpen.replaceAll(varToResolve, value);
return { uri: URI.parse(link) } as ITerminalQuickFixOpenerAction;
}
return;
},
exitStatus: quickFix.exitStatus,
source: quickFix.extensionIdentifier
};
return options;
}
48 changes: 48 additions & 0 deletions src/vs/workbench/contrib/terminal/common/terminal.ts
Original file line number Diff line number Diff line change
Expand Up @@ -717,6 +717,54 @@ export const terminalContributionsDescriptor: IExtensionPointDescriptor = {
description: nls.localize('vscode.extension.contributes.terminal', 'Contributes terminal functionality.'),
type: 'object',
properties: {
quickFixes: {
type: 'array',
description: nls.localize('vscode.extension.contributes.terminal.quickFixes', "Defines quick fixes for terminals with shell integration enabled."),
items: {
type: 'object',
required: ['id', 'commandLineMatcher', 'outputMatcher'],
defaultSnippets: [{
body: {
id: '$1',
commandLineMatcher: '$2',
outputMatcher: '$3'
}
}],
properties: {
id: {
description: nls.localize('vscode.extension.contributes.terminal.quickFixes.id', "The ID of the quick fix."),
type: 'string',
},
commandLineMatcher: {
description: nls.localize('vscode.extension.contributes.terminal.quickFixes.commandLineMatcher', "The command line to match."),
type: 'string',
},
outputMatcher: {
description: nls.localize('vscode.extension.contributes.terminal.quickFixes.outputMatcher', "The output to match."),
type: 'object',
required: ['lineMatcher', 'anchor', 'offset', 'length'],
properties: {
lineMatcher: {
description: 'The command line to match',
type: 'string'
},
anchor: {
description: 'Which side of the output to anchor the offset and length against',
enum: ['top', 'bottom']
},
offset: {
description: 'How far from either the top or the bottom of the butter to start matching against.',
type: 'number'
},
length: {
description: 'The number of rows to match against, this should be as small as possible for performance reasons',
type: 'number'
}
}
},
},
}
},
profiles: {
type: 'array',
description: nls.localize('vscode.extension.contributes.terminal.profiles', "Defines additional terminal profiles that the user can create."),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import * as extensionsRegistry from 'vs/workbench/services/extensions/common/ext
import { terminalContributionsDescriptor } from 'vs/workbench/contrib/terminal/common/terminal';
import { flatten } from 'vs/base/common/arrays';
import { createDecorator } from 'vs/platform/instantiation/common/instantiation';
import { IExtensionTerminalProfile, ITerminalContributions, ITerminalProfileContribution } from 'vs/platform/terminal/common/terminal';
import { IExtensionTerminalProfile, IExtensionTerminalQuickFix, ITerminalContributions, ITerminalProfileContribution } from 'vs/platform/terminal/common/terminal';
import { URI } from 'vs/base/common/uri';

// terminal extension point
Expand All @@ -17,6 +17,7 @@ export interface ITerminalContributionService {
readonly _serviceBrand: undefined;

readonly terminalProfiles: ReadonlyArray<IExtensionTerminalProfile>;
readonly quickFixes: Array<IExtensionTerminalQuickFix>;
}

export const ITerminalContributionService = createDecorator<ITerminalContributionService>('terminalContributionsService');
Expand All @@ -27,13 +28,17 @@ export class TerminalContributionService implements ITerminalContributionService
private _terminalProfiles: ReadonlyArray<IExtensionTerminalProfile> = [];
get terminalProfiles() { return this._terminalProfiles; }

private _quickFixes: Array<IExtensionTerminalQuickFix> = [];
get quickFixes() { return this._quickFixes; }

constructor() {
terminalsExtPoint.setHandler(contributions => {
this._terminalProfiles = flatten(contributions.map(c => {
return c.value?.profiles?.filter(p => hasValidTerminalIcon(p)).map(e => {
return { ...e, extensionIdentifier: c.description.identifier.value };
}) || [];
}));
this._quickFixes = flatten(contributions.map(c => c.value.quickFixes ? c.value.quickFixes.map(fix => { return { ...fix, extensionIdentifier: c.description.identifier.value }; }) : []));
meganrogge marked this conversation as resolved.
Show resolved Hide resolved
});
}
}
Expand Down