forked from noir-lang/vscode-noir
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Enhance dynamic resolution of debugger configuration (#1)
* Gracefully handle debugger loading/init errors * Refactor: extract debugger functionality to own file * Properly handle launch.json if exists * Add a bit more context on preflight check
- Loading branch information
Showing
4 changed files
with
186 additions
and
80 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,160 @@ | ||
import { | ||
debug, | ||
window, | ||
workspace, | ||
DebugAdapterDescriptorFactory, | ||
DebugSession, | ||
DebugAdapterExecutable, | ||
DebugAdapterDescriptor, | ||
ExtensionContext, | ||
OutputChannel, | ||
DebugConfigurationProvider, | ||
CancellationToken, | ||
DebugConfiguration, | ||
ProviderResult, | ||
} from 'vscode'; | ||
|
||
import { spawn } from 'child_process'; | ||
import findNargo from './find-nargo'; | ||
import findNearestPackageFrom from './find-nearest-package'; | ||
|
||
let outputChannel: OutputChannel; | ||
|
||
export function activateDebugger(context: ExtensionContext) { | ||
outputChannel = window.createOutputChannel('NoirDebugger'); | ||
|
||
context.subscriptions.push( | ||
debug.registerDebugAdapterDescriptorFactory('noir', new NoirDebugAdapterDescriptorFactory()), | ||
debug.registerDebugConfigurationProvider('noir', new NoirDebugConfigurationProvider()), | ||
debug.onDidTerminateDebugSession(() => { | ||
outputChannel.appendLine(`Debug session ended.`); | ||
}), | ||
); | ||
} | ||
|
||
export class NoirDebugAdapterDescriptorFactory implements DebugAdapterDescriptorFactory { | ||
async createDebugAdapterDescriptor( | ||
_session: DebugSession, | ||
_executable: DebugAdapterExecutable, | ||
): ProviderResult<DebugAdapterDescriptor> { | ||
const config = workspace.getConfiguration('noir'); | ||
|
||
const configuredNargoPath = config.get<string | undefined>('nargoPath'); | ||
const nargoPath = configuredNargoPath || findNargo(); | ||
|
||
return new DebugAdapterExecutable(nargoPath, ['dap']); | ||
} | ||
} | ||
|
||
class NoirDebugConfigurationProvider implements DebugConfigurationProvider { | ||
async resolveDebugConfiguration( | ||
_folder: WorkspaceFolder | undefined, | ||
config: DebugConfiguration, | ||
_token?: CancellationToken, | ||
): ProviderResult<DebugConfiguration> { | ||
if (window.activeTextEditor?.document.languageId != 'noir') | ||
return window.showInformationMessage(`Select a Noir file to debug`); | ||
|
||
const workspaceConfig = workspace.getConfiguration('noir'); | ||
const nargoPath = workspaceConfig.get<string | undefined>('nargoPath') || findNargo(); | ||
|
||
const currentFilePath = window.activeTextEditor.document.uri.fsPath; | ||
const projectFolder = | ||
config.projectFolder && config.projectFolder !== `` | ||
? config.projectFolder | ||
: findNearestPackageFrom(currentFilePath); | ||
|
||
const resolvedConfig = { | ||
type: config.type || 'noir', | ||
name: config.name || 'Noir binary package', | ||
request: 'launch', | ||
program: currentFilePath, | ||
projectFolder, | ||
package: config.package || ``, | ||
proverName: config.proverName || `Prover`, | ||
generateAcir: config.generateAcir || false, | ||
skipInstrumentation: config.skipInstrumentation || false, | ||
}; | ||
|
||
outputChannel.clear(); | ||
|
||
outputChannel.appendLine(`Using nargo at ${nargoPath}`); | ||
outputChannel.appendLine(`Compiling Noir project...`); | ||
outputChannel.appendLine(``); | ||
|
||
// Run Nargo's DAP in "pre-flight mode", which test runs | ||
// the DAP initialization code without actually starting the DAP server. | ||
// This lets us gracefully handle errors that happen *before* | ||
// the DAP loop is established, which otherwise are considered | ||
// "out of band". | ||
// This was necessary due to the VS Code project being reluctant to let extension authors capture | ||
// stderr output generated by a DAP server wrapped in DebugAdapterExecutable. | ||
// | ||
// More details here: https://github.com/microsoft/vscode/issues/108138 | ||
const preflightArgs = [ | ||
'dap', | ||
'--preflight-check', | ||
'--preflight-project-folder', | ||
resolvedConfig.projectFolder, | ||
'--preflight-prover-name', | ||
resolvedConfig.proverName, | ||
]; | ||
|
||
if (resolvedConfig.package !== ``) { | ||
preflightArgs.push(`--preflight-package`); | ||
preflightArgs.push(config.package); | ||
} | ||
|
||
if (resolvedConfig.generateAcir) { | ||
preflightArgs.push(`--preflight-generate-acir`); | ||
} | ||
|
||
if (resolvedConfig.skipInstrumentation) { | ||
preflightArgs.push(`--preflight-skip-instrumentation`); | ||
} | ||
|
||
const preflightCheck = spawn(nargoPath, preflightArgs); | ||
|
||
// Create a promise to block until the preflight check child process | ||
// ends. | ||
let ready: (r: boolean) => void; | ||
const preflightCheckMonitor = new Promise((resolve) => (ready = resolve)); | ||
|
||
preflightCheck.stderr.on('data', (ev_buffer) => preflightCheckPrinter(ev_buffer, outputChannel)); | ||
preflightCheck.stdout.on('data', (ev_buffer) => preflightCheckPrinter(ev_buffer, outputChannel)); | ||
preflightCheck.on('data', (ev_buffer) => preflightCheckPrinter(ev_buffer, outputChannel)); | ||
preflightCheck.on('exit', async (code) => { | ||
if (code !== 0) { | ||
outputChannel.appendLine(`Exited with code ${code}`); | ||
} | ||
ready(code == 0); | ||
}); | ||
|
||
if (!(await preflightCheckMonitor)) { | ||
outputChannel.show(); | ||
throw new Error(`Error launching debugger. Please inspect the Output pane for more details.`); | ||
} else { | ||
outputChannel.appendLine(`Starting debugger session...`); | ||
} | ||
|
||
return resolvedConfig; | ||
} | ||
} | ||
|
||
/** | ||
* Takes stderr or stdout output from the Nargo's DAP | ||
* preflight check and formats it in an Output pane friendly way, | ||
* by removing all special characters. | ||
* | ||
* Note: VS Code's output panes only support plain text. | ||
* | ||
*/ | ||
function preflightCheckPrinter(buffer: Buffer, output: OutputChannel) { | ||
const formattedOutput = buffer | ||
.toString() | ||
// eslint-disable-next-line no-control-regex | ||
.replace(/\x1b(?:[@-Z\\-_]|\[[0-?]*[ -/]*[@-~])/g, '') | ||
.replace(/[^ -~\n\t]/g, ''); | ||
|
||
output.appendLine(formattedOutput); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters