From b54f48876833b6cda160295f9de43ad2f0d48be3 Mon Sep 17 00:00:00 2001 From: Paula Camargo Date: Thu, 18 Apr 2024 14:06:28 -0700 Subject: [PATCH 1/7] Add debug Visualizer --- package.json | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/package.json b/package.json index 3fc1c3e..d0c4c48 100644 --- a/package.json +++ b/package.json @@ -6,7 +6,8 @@ "publisher": "ms-python", "enabledApiProposals": [ "portsAttributes", - "contribIssueReporter" + "contribIssueReporter", + "debugVisualization" ], "license": "MIT", "homepage": "https://github.com/Microsoft/vscode-python-debugger", @@ -508,7 +509,13 @@ }, "when": "!virtualWorkspace && shellExecutionSupported" } - ] + ], + "debugVisualizers": [ + { + "id": "inlineHexDecoder", + "when": "debugConfigurationType == 'debugpy' && (variableType == 'float' || variableType == 'int')" + } + ] }, "extensionDependencies": [ "ms-python.python" From fc05af16122423cc1dd119519752f8550c3c48e0 Mon Sep 17 00:00:00 2001 From: Paula Camargo Date: Thu, 18 Apr 2024 14:06:51 -0700 Subject: [PATCH 2/7] Add proposal --- vscode.proposed.debugVisualization.d.ts | 170 ++++++++++++++++++++++++ 1 file changed, 170 insertions(+) create mode 100644 vscode.proposed.debugVisualization.d.ts diff --git a/vscode.proposed.debugVisualization.d.ts b/vscode.proposed.debugVisualization.d.ts new file mode 100644 index 0000000..cb12eaa --- /dev/null +++ b/vscode.proposed.debugVisualization.d.ts @@ -0,0 +1,170 @@ +/*--------------------------------------------------------------------------------------------- + * Copyright (c) Microsoft Corporation. All rights reserved. + * Licensed under the MIT License. See License.txt in the project root for license information. + *--------------------------------------------------------------------------------------------*/ + + +declare module 'vscode' { + export namespace debug { + /** + * Registers a custom data visualization for variables when debugging. + * + * @param id The corresponding ID in the package.json `debugVisualizers` contribution point. + * @param provider The {@link DebugVisualizationProvider} to register + */ + export function registerDebugVisualizationProvider( + id: string, + provider: DebugVisualizationProvider + ): Disposable; + + /** + * Registers a tree that can be referenced by {@link DebugVisualization.visualization}. + * @param id + * @param provider + */ + export function registerDebugVisualizationTreeProvider( + id: string, + provider: DebugVisualizationTree + ): Disposable; + } + + /** + * An item from the {@link DebugVisualizationTree} + */ + export interface DebugTreeItem { + /** + * A human-readable string describing this item. + */ + label: string; + + /** + * A human-readable string which is rendered less prominent. + */ + description?: string; + + /** + * {@link TreeItemCollapsibleState} of the tree item. + */ + collapsibleState?: TreeItemCollapsibleState; + + /** + * Context value of the tree item. This can be used to contribute item specific actions in the tree. + * For example, a tree item is given a context value as `folder`. When contributing actions to `view/item/context` + * using `menus` extension point, you can specify context value for key `viewItem` in `when` expression like `viewItem == folder`. + * ```json + * "contributes": { + * "menus": { + * "view/item/context": [ + * { + * "command": "extension.deleteFolder", + * "when": "viewItem == folder" + * } + * ] + * } + * } + * ``` + * This will show action `extension.deleteFolder` only for items with `contextValue` is `folder`. + */ + contextValue?: string; + + /** + * Whether this item can be edited by the user. + */ + canEdit?: boolean; + } + + /** + * Provides a tree that can be referenced in debug visualizations. + */ + export interface DebugVisualizationTree { + /** + * Gets the tree item for an element or the base context item. + */ + getTreeItem(context: DebugVisualizationContext): ProviderResult; + /** + * Gets children for the tree item or the best context item. + */ + getChildren(element: T): ProviderResult; + /** + * Handles the user editing an item. + */ + editItem?(item: T, value: string): ProviderResult; + } + + export class DebugVisualization { + /** + * The name of the visualization to show to the user. + */ + name: string; + + /** + * An icon for the view when it's show in inline actions. + */ + iconPath?: Uri | { light: Uri; dark: Uri } | ThemeIcon; + + /** + * Visualization to use for the variable. This may be either: + * - A command to run when the visualization is selected for a variable. + * - A reference to a previously-registered {@link DebugVisualizationTree} + */ + visualization?: Command | { treeId: string }; + + /** + * Creates a new debug visualization object. + * @param name Name of the visualization to show to the user. + */ + constructor(name: string); + } + + export interface DebugVisualizationProvider { + /** + * Called for each variable when the debug session stops. It should return + * any visualizations the extension wishes to show to the user. + * + * Note that this is only called when its `when` clause defined under the + * `debugVisualizers` contribution point in the `package.json` evaluates + * to true. + */ + provideDebugVisualization(context: DebugVisualizationContext, token: CancellationToken): ProviderResult; + + /** + * Invoked for a variable when a user picks the visualizer. + * + * It may return a {@link TreeView} that's shown in the Debug Console or + * inline in a hover. A visualizer may choose to return `undefined` from + * this function and instead trigger other actions in the UI, such as opening + * a custom {@link WebviewView}. + */ + resolveDebugVisualization?(visualization: T, token: CancellationToken): ProviderResult; + } + + export interface DebugVisualizationContext { + /** + * The Debug Adapter Protocol Variable to be visualized. + * @see https://microsoft.github.io/debug-adapter-protocol/specification#Types_Variable + */ + variable: any; + /** + * The Debug Adapter Protocol variable reference the type (such as a scope + * or another variable) that contained this one. Empty for variables + * that came from user evaluations in the Debug Console. + * @see https://microsoft.github.io/debug-adapter-protocol/specification#Types_Variable + */ + containerId?: number; + /** + * The ID of the Debug Adapter Protocol StackFrame in which the variable was found, + * for variables that came from scopes in a stack frame. + * @see https://microsoft.github.io/debug-adapter-protocol/specification#Types_StackFrame + */ + frameId?: number; + /** + * The ID of the Debug Adapter Protocol Thread in which the variable was found. + * @see https://microsoft.github.io/debug-adapter-protocol/specification#Types_StackFrame + */ + threadId: number; + /** + * The debug session the variable belongs to. + */ + session: DebugSession; + } +} From 4dadd7e9559b0fa049ce92d7210d1aee849e6138 Mon Sep 17 00:00:00 2001 From: Paula Camargo Date: Thu, 18 Apr 2024 16:23:12 -0700 Subject: [PATCH 3/7] add hex view --- src/extension/extensionInit.ts | 64 ++++++++++++++++++++++++++++++++-- 1 file changed, 62 insertions(+), 2 deletions(-) diff --git a/src/extension/extensionInit.ts b/src/extension/extensionInit.ts index 904621c..8fb4f49 100644 --- a/src/extension/extensionInit.ts +++ b/src/extension/extensionInit.ts @@ -3,8 +3,25 @@ 'use strict'; -import { debug, DebugConfigurationProviderTriggerKind, languages, Uri, window, workspace } from 'vscode'; -import { executeCommand, getConfiguration, registerCommand, startDebugging } from './common/vscodeapi'; +import { + debug, + DebugConfigurationProviderTriggerKind, + DebugTreeItem, + DebugVisualization, + DebugVisualizationContext, + languages, + ThemeIcon, + Uri, + window, + workspace, +} from 'vscode'; +import { + executeCommand, + getConfiguration, + registerCommand, + showInformationMessage, + startDebugging, +} from './common/vscodeapi'; import { DebuggerTypeName } from './constants'; import { DynamicPythonDebugConfigurationService } from './debugger/configuration/dynamicdebugConfigurationService'; import { IExtensionContext } from './common/types'; @@ -177,5 +194,48 @@ export async function registerDebugger(context: IExtensionContext): Promise('inlineHexDecoder', { + getTreeItem(context) { + const decoded = `0x${Number(context.variable.value).toString(16)}`; + return { + label: context.variable.name.toString(), + description: decoded.toString(), + buffer: decoded, + canEdit: true, + context, + }; + }, + getChildren(_element) { + return undefined; + }, + editItem(item, value) { + item.buffer = `0x${Number(value).toString(16)}`; + item.description = item.buffer.toString(); + + item.context.session.customRequest('setExpression', { + expression: item.context.variable.evaluateName, + frameId: item.context.frameId, + value: JSON.stringify(item.buffer.toString()), + }); + + return item; + }, + }), + ); + + context.subscriptions.push( + debug.registerDebugVisualizationProvider('inlineHexDecoder', { + provideDebugVisualization(_context, _token) { + const v = new DebugVisualization('Show as Hex'); + v.iconPath = new ThemeIcon('rocket'); + v.visualization = { treeId: 'inlineHexDecoder' }; + return [v]; + }, + }), + ); + return buildApi(); } From 672016ade4a73aa96a4d103c5369e4f9eb0ca128 Mon Sep 17 00:00:00 2001 From: Paula Camargo Date: Thu, 18 Apr 2024 16:33:58 -0700 Subject: [PATCH 4/7] fux lint --- src/extension/extensionInit.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/src/extension/extensionInit.ts b/src/extension/extensionInit.ts index 8fb4f49..abacab9 100644 --- a/src/extension/extensionInit.ts +++ b/src/extension/extensionInit.ts @@ -19,7 +19,6 @@ import { executeCommand, getConfiguration, registerCommand, - showInformationMessage, startDebugging, } from './common/vscodeapi'; import { DebuggerTypeName } from './constants'; From bc4ebc51042a9fd1d74984b5c9cfa0a8f6738f43 Mon Sep 17 00:00:00 2001 From: Paula Camargo Date: Fri, 19 Apr 2024 09:57:07 -0700 Subject: [PATCH 5/7] fix lint --- src/extension/extensionInit.ts | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/src/extension/extensionInit.ts b/src/extension/extensionInit.ts index abacab9..7e571e5 100644 --- a/src/extension/extensionInit.ts +++ b/src/extension/extensionInit.ts @@ -15,12 +15,7 @@ import { window, workspace, } from 'vscode'; -import { - executeCommand, - getConfiguration, - registerCommand, - startDebugging, -} from './common/vscodeapi'; +import { executeCommand, getConfiguration, registerCommand, startDebugging } from './common/vscodeapi'; import { DebuggerTypeName } from './constants'; import { DynamicPythonDebugConfigurationService } from './debugger/configuration/dynamicdebugConfigurationService'; import { IExtensionContext } from './common/types'; From 311a514b341137ffc455378be0a4f2e09f13fcb8 Mon Sep 17 00:00:00 2001 From: Paula Camargo Date: Fri, 19 Apr 2024 13:56:37 -0700 Subject: [PATCH 6/7] Add localization, move tree creation to own file --- src/extension/common/utils/localize.ts | 4 +++ .../debugger/visualizers/inlineHexDecoder.ts | 34 ++++++++++++++++++ src/extension/extensionInit.ts | 35 +++---------------- 3 files changed, 43 insertions(+), 30 deletions(-) create mode 100644 src/extension/debugger/visualizers/inlineHexDecoder.ts diff --git a/src/extension/common/utils/localize.ts b/src/extension/common/utils/localize.ts index c81182c..8636c96 100644 --- a/src/extension/common/utils/localize.ts +++ b/src/extension/common/utils/localize.ts @@ -177,3 +177,7 @@ export namespace pickArgsInput { export const title = l10n.t('Command Line Arguments'); export const prompt = l10n.t('Enter the command line arguments you want to pass to the program'); } + +export namespace DebugVisualizers { + export const hexDecoder = l10n.t('Show as Hex'); +} diff --git a/src/extension/debugger/visualizers/inlineHexDecoder.ts b/src/extension/debugger/visualizers/inlineHexDecoder.ts new file mode 100644 index 0000000..9edd711 --- /dev/null +++ b/src/extension/debugger/visualizers/inlineHexDecoder.ts @@ -0,0 +1,34 @@ +// Copyright (c) Microsoft Corporation. All rights reserved. +// Licensed under the MIT License. + +import { DebugVisualizationContext } from 'vscode'; + +export function registerHexDebugVisualizationTreeProvider() { + return { + getTreeItem(context: DebugVisualizationContext) { + const decoded = `0x${Number(context.variable.value).toString(16)}`; + return { + label: context.variable.name.toString(), + description: decoded.toString(), + buffer: decoded, + canEdit: true, + context, + }; + }, + getChildren(_element) { + return undefined; + }, + editItem(item, value) { + item.buffer = `0x${Number(value).toString(16)}`; + item.description = item.buffer.toString(); + + item.context.session.customRequest('setExpression', { + expression: item.context.variable.evaluateName, + frameId: item.context.frameId, + value: JSON.stringify(item.buffer.toString()), + }); + + return item; + }, + }; +} diff --git a/src/extension/extensionInit.ts b/src/extension/extensionInit.ts index 7e571e5..deb662b 100644 --- a/src/extension/extensionInit.ts +++ b/src/extension/extensionInit.ts @@ -41,13 +41,14 @@ import { DebugSessionTelemetry } from './common/application/debugSessionTelemetr import { JsonLanguages, LaunchJsonCompletionProvider } from './debugger/configuration/launch.json/completionProvider'; import { LaunchJsonUpdaterServiceHelper } from './debugger/configuration/launch.json/updaterServiceHelper'; import { ignoreErrors } from './common/promiseUtils'; -import { pickArgsInput } from './common/utils/localize'; +import { DebugVisualizers, pickArgsInput } from './common/utils/localize'; import { DebugPortAttributesProvider } from './debugger/debugPort/portAttributesProvider'; import { getConfigurationsByUri } from './debugger/configuration/launch.json/launchJsonReader'; import { DebugpySocketsHandler } from './debugger/hooks/debugpySocketsHandler'; import { openReportIssue } from './common/application/commands/reportIssueCommand'; import { buildApi } from './api'; import { IExtensionApi } from './apiTypes'; +import { registerHexDebugVisualizationTreeProvider } from './debugger/visualizers/inlineHexDecoder'; export async function registerDebugger(context: IExtensionContext): Promise { const childProcessAttachService = new ChildProcessAttachService(); @@ -191,40 +192,14 @@ export async function registerDebugger(context: IExtensionContext): Promise('inlineHexDecoder', { - getTreeItem(context) { - const decoded = `0x${Number(context.variable.value).toString(16)}`; - return { - label: context.variable.name.toString(), - description: decoded.toString(), - buffer: decoded, - canEdit: true, - context, - }; - }, - getChildren(_element) { - return undefined; - }, - editItem(item, value) { - item.buffer = `0x${Number(value).toString(16)}`; - item.description = item.buffer.toString(); - - item.context.session.customRequest('setExpression', { - expression: item.context.variable.evaluateName, - frameId: item.context.frameId, - value: JSON.stringify(item.buffer.toString()), - }); - - return item; - }, - }), + >('inlineHexDecoder', registerHexDebugVisualizationTreeProvider()), ); context.subscriptions.push( debug.registerDebugVisualizationProvider('inlineHexDecoder', { provideDebugVisualization(_context, _token) { - const v = new DebugVisualization('Show as Hex'); - v.iconPath = new ThemeIcon('rocket'); + const v = new DebugVisualization(DebugVisualizers.hexDecoder); + v.iconPath = new ThemeIcon('eye'); v.visualization = { treeId: 'inlineHexDecoder' }; return [v]; }, From 07e83ef2f546123f23877499feb12c380f8135de Mon Sep 17 00:00:00 2001 From: Paula Camargo Date: Fri, 19 Apr 2024 14:02:06 -0700 Subject: [PATCH 7/7] fix compile errors --- src/extension/debugger/visualizers/inlineHexDecoder.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/extension/debugger/visualizers/inlineHexDecoder.ts b/src/extension/debugger/visualizers/inlineHexDecoder.ts index 9edd711..630fbb5 100644 --- a/src/extension/debugger/visualizers/inlineHexDecoder.ts +++ b/src/extension/debugger/visualizers/inlineHexDecoder.ts @@ -15,10 +15,10 @@ export function registerHexDebugVisualizationTreeProvider() { context, }; }, - getChildren(_element) { + getChildren(_element: any) { return undefined; }, - editItem(item, value) { + editItem(item: any, value: string) { item.buffer = `0x${Number(value).toString(16)}`; item.description = item.buffer.toString();