bruno/packages/bruno-js/src/utils.js

152 lines
4.2 KiB
JavaScript

const jsonQuery = require('json-query');
const { get } = require('@usebruno/query');
const JS_KEYWORDS = `
break case catch class const continue debugger default delete do
else export extends false finally for function if import in instanceof
new null return super switch this throw true try typeof var void while with
undefined let static yield arguments of
`
.split(/\s+/)
.filter((word) => word.length > 0);
/**
* Creates a function from a JavaScript expression
*
* When the function is called, the variables used in this expression are picked up from the context
*
* ```js
* res.data.pets.map(pet => pet.name.toUpperCase())
*
* function(context) {
* const { res, pet } = context;
* return res.data.pets.map(pet => pet.name.toUpperCase())
* }
* ```
*/
const compileJsExpression = (expr) => {
// get all dotted identifiers (foo, bar.baz, .baz)
const matches = expr.match(/([\w\.$]+)/g) ?? [];
// get valid js identifiers (foo, bar)
const vars = new Set(
matches
.filter((match) => /^[a-zA-Z$_]/.test(match)) // starts with valid js identifier (foo.bar)
.map((match) => match.split('.')[0]) // top level identifier (foo)
.filter((name) => !JS_KEYWORDS.includes(name)) // exclude js keywords
);
// globals such as Math
const globals = [...vars].filter((name) => name in globalThis);
const code = {
vars: [...vars].join(', '),
// pick global from context or globalThis
globals: globals.map((name) => ` ${name} = ${name} ?? globalThis.${name};`).join('')
};
const body = `let { ${code.vars} } = context; ${code.globals}; return ${expr}`;
return new Function('context', body);
};
const internalExpressionCache = new Map();
const evaluateJsExpression = (expression, context) => {
let fn = internalExpressionCache.get(expression);
if (fn == null) {
internalExpressionCache.set(expression, (fn = compileJsExpression(expression)));
}
return fn(context);
};
const evaluateJsTemplateLiteral = (templateLiteral, context) => {
if (!templateLiteral || !templateLiteral.length || typeof templateLiteral !== 'string') {
return templateLiteral;
}
templateLiteral = templateLiteral.trim();
if (templateLiteral === 'true') {
return true;
}
if (templateLiteral === 'false') {
return false;
}
if (templateLiteral === 'null') {
return null;
}
if (templateLiteral === 'undefined') {
return undefined;
}
if (templateLiteral.startsWith('"') && templateLiteral.endsWith('"')) {
return templateLiteral.slice(1, -1);
}
if (templateLiteral.startsWith("'") && templateLiteral.endsWith("'")) {
return templateLiteral.slice(1, -1);
}
if (!isNaN(templateLiteral)) {
const number = Number(templateLiteral);
// Check if the number is too high. Too high number might get altered, see #1000
if (number > Number.MAX_SAFE_INTEGER) {
return templateLiteral;
}
return number;
}
templateLiteral = '`' + templateLiteral + '`';
return evaluateJsExpression(templateLiteral, context);
};
const createResponseParser = (response = {}) => {
const res = (expr, ...fns) => {
return get(response.data, expr, ...fns);
};
res.status = response.status;
res.statusText = response.statusText;
res.headers = response.headers;
res.body = response.data;
res.responseTime = response.responseTime;
res.jq = (expr) => {
const output = jsonQuery(expr, { data: response.data });
return output ? output.value : null;
};
return res;
};
/**
* Objects that are created inside vm2 execution context result in an serialization error when sent to the renderer process
* Error sending from webFrameMain: Error: Failed to serialize arguments
* at s.send (node:electron/js2c/browser_init:169:631)
* at g.send (node:electron/js2c/browser_init:165:2156)
* How to reproduce
* Remove the cleanJson fix and execute the below post response script
* bru.setVar("a", {b:3});
* Todo: Find a better fix
*/
const cleanJson = (data) => {
try {
return JSON.parse(JSON.stringify(data));
} catch (e) {
return data;
}
};
module.exports = {
evaluateJsExpression,
evaluateJsTemplateLiteral,
createResponseParser,
internalExpressionCache,
cleanJson
};