Files
bruno/packages/bruno-js/src/runtime/script-runtime.js
Karan Pradhan 5672745b76 Mark test script errors as failed in runner (#6261)
* Mark test script errors as failed in runner
    and CLI

* Unify handling of post-response and pre-request script errors in both CLI and Electron

* feat: Enhance error handling in script execution by preserving partial results for pre-request and post-response scripts across CLI and Electron. This ensures that tests passing before an error are still reported.

* Preserving stopExecution in test script error handler

---------

Co-authored-by: Pragadesh-45 <temporaryg7904@gmail.com>
2026-02-04 13:12:10 +05:30

250 lines
8.0 KiB
JavaScript

const chai = require('chai');
const Bru = require('../bru');
const BrunoRequest = require('../bruno-request');
const BrunoResponse = require('../bruno-response');
const { cleanJson } = require('../utils');
const { createBruTestResultMethods } = require('../utils/results');
const { runScriptInNodeVm } = require('../sandbox/node-vm');
const { executeQuickJsVmAsync } = require('../sandbox/quickjs');
class ScriptRuntime {
constructor(props) {
this.runtime = props?.runtime || 'quickjs';
}
// This approach is getting out of hand
// Need to refactor this to use a single arg (object) instead of 7
async runRequestScript(
script,
request,
envVariables,
runtimeVariables,
collectionPath,
onConsoleLog,
processEnvVars,
scriptingConfig,
runRequestByItemPathname,
collectionName
) {
const globalEnvironmentVariables = request?.globalEnvironmentVariables || {};
const oauth2CredentialVariables = request?.oauth2CredentialVariables || {};
const collectionVariables = request?.collectionVariables || {};
const folderVariables = request?.folderVariables || {};
const requestVariables = request?.requestVariables || {};
const promptVariables = request?.promptVariables || {};
const assertionResults = request?.assertionResults || [];
const certsAndProxyConfig = request?.certsAndProxyConfig;
const bru = new Bru(this.runtime, envVariables, runtimeVariables, processEnvVars, collectionPath, collectionVariables, folderVariables, requestVariables, globalEnvironmentVariables, oauth2CredentialVariables, collectionName, promptVariables, certsAndProxyConfig);
const req = new BrunoRequest(request);
// extend bru with result getter methods
const { __brunoTestResults, test } = createBruTestResultMethods(bru, assertionResults, chai);
const context = {
bru,
req,
test,
expect: chai.expect,
assert: chai.assert,
__brunoTestResults: __brunoTestResults
};
if (onConsoleLog && typeof onConsoleLog === 'function') {
const customLogger = (type) => {
return (...args) => {
onConsoleLog(type, cleanJson(args));
};
};
context.console = {
log: customLogger('log'),
debug: customLogger('debug'),
info: customLogger('info'),
warn: customLogger('warn'),
error: customLogger('error')
};
}
if (runRequestByItemPathname) {
context.bru.runRequest = runRequestByItemPathname;
}
// Helper to build the result object for pre-request scripts
// Extracted to avoid duplication across runtime branches
const buildRequestScriptResult = () => ({
request,
envVariables: cleanJson(envVariables),
runtimeVariables: cleanJson(runtimeVariables),
persistentEnvVariables: bru.persistentEnvVariables,
globalEnvironmentVariables: cleanJson(globalEnvironmentVariables),
results: cleanJson(__brunoTestResults.getResults()),
nextRequestName: bru.nextRequest,
skipRequest: bru.skipRequest,
stopExecution: bru.stopExecution
});
// Track script errors to attach partial results before re-throwing
// This ensures that any test() calls that passed before the error are preserved
// Similar pattern to test-runtime.js which already handles this correctly
let scriptError = null;
if (this.runtime === 'nodevm') {
try {
await runScriptInNodeVm({
script,
context,
collectionPath,
scriptingConfig
});
} catch (error) {
scriptError = error;
}
// If script errored, attach partial results so callers can display passed tests
// before the error occurred (e.g., 2 tests pass, then script throws)
if (scriptError) {
scriptError.partialResults = buildRequestScriptResult();
throw scriptError;
}
return buildRequestScriptResult();
}
// default runtime is `quickjs`
try {
await executeQuickJsVmAsync({
script: script,
context: context,
collectionPath
});
} catch (error) {
scriptError = error;
}
if (scriptError) {
scriptError.partialResults = buildRequestScriptResult();
throw scriptError;
}
return buildRequestScriptResult();
}
async runResponseScript(
script,
request,
response,
envVariables,
runtimeVariables,
collectionPath,
onConsoleLog,
processEnvVars,
scriptingConfig,
runRequestByItemPathname,
collectionName
) {
const globalEnvironmentVariables = request?.globalEnvironmentVariables || {};
const oauth2CredentialVariables = request?.oauth2CredentialVariables || {};
const collectionVariables = request?.collectionVariables || {};
const folderVariables = request?.folderVariables || {};
const requestVariables = request?.requestVariables || {};
const promptVariables = request?.promptVariables || {};
const assertionResults = request?.assertionResults || {};
const certsAndProxyConfig = request?.certsAndProxyConfig;
const bru = new Bru(this.runtime, envVariables, runtimeVariables, processEnvVars, collectionPath, collectionVariables, folderVariables, requestVariables, globalEnvironmentVariables, oauth2CredentialVariables, collectionName, promptVariables, certsAndProxyConfig);
const req = new BrunoRequest(request);
const res = new BrunoResponse(response);
// extend bru with result getter methods
const { __brunoTestResults, test } = createBruTestResultMethods(bru, assertionResults, chai);
const context = {
bru,
req,
res,
test,
expect: chai.expect,
assert: chai.assert,
__brunoTestResults: __brunoTestResults
};
if (onConsoleLog && typeof onConsoleLog === 'function') {
const customLogger = (type) => {
return (...args) => {
onConsoleLog(type, cleanJson(args));
};
};
context.console = {
log: customLogger('log'),
info: customLogger('info'),
warn: customLogger('warn'),
error: customLogger('error'),
debug: customLogger('debug')
};
}
if (runRequestByItemPathname) {
context.bru.runRequest = runRequestByItemPathname;
}
// Helper to build the result object for post-response scripts
// Extracted to avoid duplication across runtime branches
const buildResponseScriptResult = () => ({
response,
envVariables: cleanJson(envVariables),
persistentEnvVariables: cleanJson(bru.persistentEnvVariables),
runtimeVariables: cleanJson(runtimeVariables),
globalEnvironmentVariables: cleanJson(globalEnvironmentVariables),
results: cleanJson(__brunoTestResults.getResults()),
nextRequestName: bru.nextRequest,
skipRequest: bru.skipRequest,
stopExecution: bru.stopExecution
});
// Track script errors to attach partial results before re-throwing
// This ensures that any test() calls that passed before the error are preserved
// Similar pattern to test-runtime.js which already handles this correctly
let scriptError = null;
if (this.runtime === 'nodevm') {
try {
await runScriptInNodeVm({
script,
context,
collectionPath,
scriptingConfig
});
} catch (error) {
scriptError = error;
}
// If script errored, attach partial results so callers can display passed tests
// before the error occurred (e.g., 2 tests pass, then script throws)
if (scriptError) {
scriptError.partialResults = buildResponseScriptResult();
throw scriptError;
}
return buildResponseScriptResult();
}
// default runtime is `quickjs`
try {
await executeQuickJsVmAsync({
script: script,
context: context,
collectionPath
});
} catch (error) {
scriptError = error;
}
if (scriptError) {
scriptError.partialResults = buildResponseScriptResult();
throw scriptError;
}
return buildResponseScriptResult();
}
}
module.exports = ScriptRuntime;