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)) { return Number(templateLiteral); } 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.jq = (expr) => { const output = jsonQuery(expr, { data: response.data }); return output ? output.value : null; }; return res; }; module.exports = { evaluateJsExpression, evaluateJsTemplateLiteral, createResponseParser, internalExpressionCache };