Skip to content

Commit 86314ba

Browse files
authored
Configless debug (#557)
* support no-config debugging in debugger extensions
1 parent f423022 commit 86314ba

File tree

9 files changed

+370
-37
lines changed

9 files changed

+370
-37
lines changed

.vscode/launch.json

+32-37
Original file line numberDiff line numberDiff line change
@@ -3,40 +3,35 @@
33
// Hover to view descriptions of existing attributes.
44
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
55
{
6-
"version": "0.2.0",
7-
"configurations": [
8-
{
9-
"name": "Run Extension",
10-
"type": "extensionHost",
11-
"request": "launch",
12-
"args": [
13-
"--extensionDevelopmentPath=${workspaceFolder}"
14-
],
15-
"outFiles": [
16-
"${workspaceFolder}/dist/**/*.js",
17-
"!${workspaceFolder}/**/node_modules**/*"
18-
],
19-
"preLaunchTask": "npm: watch",
20-
"presentation": {
21-
"hidden": false,
22-
"group": "",
23-
"order": 2
24-
}
25-
},
26-
{
27-
"name": "Unit Tests",
28-
"type": "extensionHost",
29-
"request": "launch",
30-
"runtimeExecutable": "${execPath}",
31-
"args": [
32-
"./out/test/**/*.unit.test.js",
33-
"--extensionDevelopmentPath=${workspaceFolder}",
34-
"--extensionTestsPath=${workspaceFolder}/out/test/unittest/index"
35-
],
36-
"outFiles": [
37-
"${workspaceFolder}/out/**/*.js",
38-
],
39-
"preLaunchTask": "tasks: watch-tests"
40-
},
41-
]
42-
}
6+
"version": "0.2.0",
7+
"configurations": [
8+
{
9+
"name": "Run Extension",
10+
"type": "extensionHost",
11+
"request": "launch",
12+
"args": ["--extensionDevelopmentPath=${workspaceFolder}"],
13+
"outFiles": ["${workspaceFolder}/dist/**/*.js", "!${workspaceFolder}/**/node_modules**/*"],
14+
"preLaunchTask": "npm: watch",
15+
"presentation": {
16+
"hidden": false,
17+
"group": "",
18+
"order": 2
19+
}
20+
},
21+
{
22+
"name": "Unit Tests",
23+
"type": "extensionHost",
24+
"request": "launch",
25+
"runtimeExecutable": "${execPath}",
26+
"args": [
27+
"./out/test/**/*.unit.test.js",
28+
"--extensionDevelopmentPath=${workspaceFolder}",
29+
"--extensionTestsPath=${workspaceFolder}/out/test/unittest/index",
30+
//"--grep", "<suite name>",
31+
"--timeout=300000"
32+
],
33+
"outFiles": ["${workspaceFolder}/out/**/*.js"],
34+
"preLaunchTask": "tasks: watch-tests"
35+
}
36+
]
37+
}
+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
#! /bin/bash
2+
# Bash script
3+
python $BUNDLED_DEBUGPY_PATH --listen 0 --wait-for-client $@
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
@echo off
2+
:: Bat script
3+
python %BUNDLED_DEBUGPY_PATH% --listen 0 --wait-for-client %*
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
# Fish script
2+
python $BUNDLED_DEBUGPY_PATH --listen 0 --wait-for-client $argv
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
# PowerShell script
2+
python $env:BUNDLED_DEBUGPY_PATH --listen 0 --wait-for-client $args

src/extension/extensionInit.ts

+5
Original file line numberDiff line numberDiff line change
@@ -53,6 +53,7 @@ import { IExtensionApi } from './apiTypes';
5353
import { registerHexDebugVisualizationTreeProvider } from './debugger/visualizers/inlineHexDecoder';
5454
import { PythonInlineValueProvider } from './debugger/inlineValue/pythonInlineValueProvider';
5555
import { traceLog } from './common/log/logging';
56+
import { registerNoConfigDebug } from './noConfigDebugInit';
5657

5758
export async function registerDebugger(context: IExtensionContext): Promise<IExtensionApi> {
5859
const childProcessAttachService = new ChildProcessAttachService();
@@ -247,5 +248,9 @@ export async function registerDebugger(context: IExtensionContext): Promise<IExt
247248
window.activeTextEditor?.document.languageId === 'python',
248249
);
249250

251+
context.subscriptions.push(
252+
await registerNoConfigDebug(context.environmentVariableCollection, context.extensionPath),
253+
);
254+
250255
return buildApi();
251256
}

src/extension/noConfigDebugInit.ts

+116
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,116 @@
1+
// Copyright (c) Microsoft Corporation. All rights reserved.
2+
// Licensed under the MIT License.
3+
4+
import * as fs from 'fs';
5+
import * as path from 'path';
6+
import * as crypto from 'crypto';
7+
import * as os from 'os';
8+
import { DebugSessionOptions, Disposable, GlobalEnvironmentVariableCollection, RelativePattern } from 'vscode';
9+
import { createFileSystemWatcher, debugStartDebugging } from './utils';
10+
import { traceError, traceLog, traceVerbose } from './common/log/logging';
11+
12+
/**
13+
* Registers the configuration-less debugging setup for the extension.
14+
*
15+
* This function sets up environment variables and a file system watcher to
16+
* facilitate debugging without requiring a pre-configured launch.json file.
17+
*
18+
* @param envVarCollection - The collection of environment variables to be modified.
19+
* @param extPath - The path to the extension directory.
20+
*
21+
* Environment Variables:
22+
* - `DEBUGPY_ADAPTER_ENDPOINTS`: Path to the file containing the debugger adapter endpoint.
23+
* - `BUNDLED_DEBUGPY_PATH`: Path to the bundled debugpy library.
24+
* - `PATH`: Appends the path to the noConfigScripts directory.
25+
*/
26+
export async function registerNoConfigDebug(
27+
envVarCollection: GlobalEnvironmentVariableCollection,
28+
extPath: string,
29+
): Promise<Disposable> {
30+
const collection = envVarCollection;
31+
32+
// create a temp directory for the noConfigDebugAdapterEndpoints
33+
// file path format: tempDir/noConfigDebugAdapterEndpoints-<randomString>/debuggerAdapterEndpoint.txt
34+
const randomSuffix = crypto.randomBytes(10).toString('hex');
35+
const tempDirName = `noConfigDebugAdapterEndpoints-${randomSuffix}`;
36+
let tempDirPath = path.join(os.tmpdir(), tempDirName);
37+
try {
38+
traceLog('Attempting to use temp directory for noConfigDebugAdapterEndpoints, dir name:', tempDirName);
39+
await fs.promises.mkdir(tempDirPath, { recursive: true });
40+
} catch (error) {
41+
// Handle the error when accessing the temp directory
42+
traceError('Error accessing temp directory:', error, ' Attempt to use extension root dir instead');
43+
// Make new temp directory in extension root dird
44+
tempDirPath = path.join(extPath, '.temp');
45+
await fs.promises.mkdir(tempDirPath, { recursive: true });
46+
}
47+
const tempFilePath = path.join(tempDirPath, 'debuggerAdapterEndpoint.txt');
48+
49+
// Add env vars for DEBUGPY_ADAPTER_ENDPOINTS, BUNDLED_DEBUGPY_PATH, and PATH
50+
collection.replace('DEBUGPY_ADAPTER_ENDPOINTS', tempFilePath);
51+
52+
const noConfigScriptsDir = path.join(extPath, 'bundled', 'scripts', 'noConfigScripts');
53+
const pathSeparator = process.platform === 'win32' ? ';' : ':';
54+
collection.append('PATH', `${pathSeparator}${noConfigScriptsDir}`);
55+
56+
const bundledDebugPath = path.join(extPath, 'bundled', 'libs', 'debugpy');
57+
collection.replace('BUNDLED_DEBUGPY_PATH', bundledDebugPath);
58+
59+
// create file system watcher for the debuggerAdapterEndpointFolder for when the communication port is written
60+
const fileSystemWatcher = createFileSystemWatcher(new RelativePattern(tempDirPath, '**/*'));
61+
const fileCreationEvent = fileSystemWatcher.onDidCreate(async (uri) => {
62+
const filePath = uri.fsPath;
63+
fs.readFile(filePath, (err, data) => {
64+
const dataParse = data.toString();
65+
if (err) {
66+
traceError(`Error reading debuggerAdapterEndpoint.txt file: ${err}`);
67+
return;
68+
}
69+
try {
70+
// parse the client port
71+
const jsonData = JSON.parse(dataParse);
72+
const clientPort = jsonData.client?.port;
73+
traceVerbose(`Parsed client port: ${clientPort}`);
74+
75+
const options: DebugSessionOptions = {
76+
noDebug: false,
77+
};
78+
79+
// start debug session with the client port
80+
debugStartDebugging(
81+
undefined,
82+
{
83+
type: 'python',
84+
request: 'attach',
85+
name: 'Attach to Python',
86+
connect: {
87+
port: clientPort,
88+
host: 'localhost',
89+
},
90+
},
91+
options,
92+
).then(
93+
(started) => {
94+
if (started) {
95+
traceVerbose('Successfully started debug session');
96+
} else {
97+
traceError('Error starting debug session, session not started.');
98+
}
99+
},
100+
(error) => {
101+
traceError(`Error starting debug session: ${error}`);
102+
},
103+
);
104+
} catch (parseErr) {
105+
traceError(`Error parsing JSON: ${parseErr}`);
106+
}
107+
});
108+
JSON.parse;
109+
});
110+
return Promise.resolve(
111+
new Disposable(() => {
112+
fileSystemWatcher.dispose();
113+
fileCreationEvent.dispose();
114+
}),
115+
);
116+
}

src/extension/utils.ts

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
import {
2+
workspace,
3+
debug,
4+
WorkspaceFolder,
5+
DebugConfiguration,
6+
DebugSession,
7+
DebugSessionOptions,
8+
FileSystemWatcher,
9+
} from 'vscode';
10+
11+
export function createFileSystemWatcher(args: any): FileSystemWatcher {
12+
return workspace.createFileSystemWatcher(args);
13+
}
14+
15+
export async function debugStartDebugging(
16+
folder: WorkspaceFolder | undefined,
17+
nameOrConfiguration: string | DebugConfiguration,
18+
parentSessionOrOptions?: DebugSession | DebugSessionOptions,
19+
): Promise<boolean> {
20+
return debug.startDebugging(folder, nameOrConfiguration, parentSessionOrOptions);
21+
}

0 commit comments

Comments
 (0)