2023-09-21 21:12:14 +02:00
|
|
|
const { describe, it, expect } = require('@jest/globals');
|
|
|
|
const { evaluateJsExpression, internalExpressionCache: cache, createResponseParser } = require('../src/utils');
|
2023-02-11 04:55:05 +01:00
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
describe('utils', () => {
|
|
|
|
describe('expression evaluation', () => {
|
2023-02-11 04:55:05 +01:00
|
|
|
const context = {
|
|
|
|
res: {
|
2023-11-23 23:38:43 +01:00
|
|
|
data: { pets: ['bruno', 'max'] },
|
|
|
|
context: 'testContext',
|
|
|
|
_BrunoNewFunctionInnerContext: 0
|
2023-02-11 04:55:05 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-02-11 05:24:28 +01:00
|
|
|
beforeEach(() => cache.clear());
|
2023-02-11 04:55:05 +01:00
|
|
|
afterEach(() => cache.clear());
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should evaluate expression', () => {
|
2023-02-11 04:55:05 +01:00
|
|
|
let result;
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
result = evaluateJsExpression('res.data.pets', context);
|
|
|
|
expect(result).toEqual(['bruno', 'max']);
|
2023-02-11 04:55:05 +01:00
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
result = evaluateJsExpression('res.data.pets[0].toUpperCase()', context);
|
|
|
|
expect(result).toEqual('BRUNO');
|
2023-02-11 04:55:05 +01:00
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should cache expression', () => {
|
2023-02-11 04:55:05 +01:00
|
|
|
expect(cache.size).toBe(0);
|
2023-09-21 21:12:14 +02:00
|
|
|
evaluateJsExpression('res.data.pets', context);
|
2023-02-11 04:55:05 +01:00
|
|
|
expect(cache.size).toBe(1);
|
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should use cached expression', () => {
|
|
|
|
const expr = 'res.data.pets';
|
2023-02-11 06:34:46 +01:00
|
|
|
|
|
|
|
evaluateJsExpression(expr, context);
|
|
|
|
|
|
|
|
const fn = cache.get(expr);
|
|
|
|
expect(fn).toBeDefined();
|
|
|
|
|
|
|
|
evaluateJsExpression(expr, context);
|
|
|
|
|
|
|
|
// cache should not be overwritten
|
|
|
|
expect(cache.get(expr)).toBe(fn);
|
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should identify top level variables', () => {
|
|
|
|
const expr = 'res.data.pets[0].toUpperCase()';
|
2023-02-11 04:55:05 +01:00
|
|
|
evaluateJsExpression(expr, context);
|
2023-11-23 23:38:43 +01:00
|
|
|
expect(cache.get(expr).toString()).toContain('let { res } = _BrunoNewFunctionInnerContext;');
|
2023-02-11 04:55:05 +01:00
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should not duplicate variables', () => {
|
|
|
|
const expr = 'res.data.pets[0] + res.data.pets[1]';
|
2023-02-11 04:55:05 +01:00
|
|
|
evaluateJsExpression(expr, context);
|
2023-11-23 23:38:43 +01:00
|
|
|
expect(cache.get(expr).toString()).toContain('let { res } = _BrunoNewFunctionInnerContext;');
|
2023-02-11 04:55:05 +01:00
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should exclude js keywords like true false from vars', () => {
|
|
|
|
const expr = 'res.data.pets.length > 0 ? true : false';
|
2023-02-11 04:55:05 +01:00
|
|
|
evaluateJsExpression(expr, context);
|
2023-11-23 23:38:43 +01:00
|
|
|
expect(cache.get(expr).toString()).toContain('let { res } = _BrunoNewFunctionInnerContext;');
|
2023-02-11 04:55:05 +01:00
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should exclude numbers from vars', () => {
|
|
|
|
const expr = 'res.data.pets.length + 10';
|
2023-02-11 04:55:05 +01:00
|
|
|
evaluateJsExpression(expr, context);
|
2023-11-23 23:38:43 +01:00
|
|
|
expect(cache.get(expr).toString()).toContain('let { res } = _BrunoNewFunctionInnerContext;');
|
2023-02-11 04:55:05 +01:00
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should pick variables from complex expressions', () => {
|
|
|
|
const expr = 'res.data.pets.map(pet => pet.length)';
|
2023-02-11 04:55:05 +01:00
|
|
|
const result = evaluateJsExpression(expr, context);
|
|
|
|
expect(result).toEqual([5, 3]);
|
2023-11-23 23:38:43 +01:00
|
|
|
expect(cache.get(expr).toString()).toContain('let { res, pet } = _BrunoNewFunctionInnerContext;');
|
2023-02-11 04:55:05 +01:00
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should be ok picking extra vars from strings', () => {
|
2023-02-11 04:55:05 +01:00
|
|
|
const expr = "'hello' + ' ' + res.data.pets[0]";
|
|
|
|
const result = evaluateJsExpression(expr, context);
|
2023-09-21 21:12:14 +02:00
|
|
|
expect(result).toBe('hello bruno');
|
2023-02-11 04:55:05 +01:00
|
|
|
// extra var hello is harmless
|
2023-11-23 23:38:43 +01:00
|
|
|
expect(cache.get(expr).toString()).toContain('let { hello, res } = _BrunoNewFunctionInnerContext;');
|
2023-02-11 15:57:27 +01:00
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should evaluate expressions referencing globals', () => {
|
|
|
|
const startTime = new Date('2022-02-01').getTime();
|
|
|
|
const currentTime = new Date('2022-02-02').getTime();
|
2023-02-11 15:57:27 +01:00
|
|
|
|
|
|
|
jest.useFakeTimers({ now: currentTime });
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
const expr = 'Math.max(Date.now(), startTime)';
|
2023-02-11 15:57:27 +01:00
|
|
|
const result = evaluateJsExpression(expr, { startTime });
|
|
|
|
|
|
|
|
expect(result).toBe(currentTime);
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
expect(cache.get(expr).toString()).toContain('Math = Math ?? globalThis.Math;');
|
|
|
|
expect(cache.get(expr).toString()).toContain('Date = Date ?? globalThis.Date;');
|
2023-02-11 15:57:27 +01:00
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should use global overridden in context', () => {
|
|
|
|
const startTime = new Date('2022-02-01').getTime();
|
|
|
|
const currentTime = new Date('2022-02-02').getTime();
|
2023-02-11 15:57:27 +01:00
|
|
|
|
|
|
|
jest.useFakeTimers({ now: currentTime });
|
|
|
|
|
|
|
|
const context = {
|
2023-09-21 21:12:14 +02:00
|
|
|
Date: { now: () => new Date('2022-01-31').getTime() },
|
2023-02-11 15:57:27 +01:00
|
|
|
startTime
|
|
|
|
};
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
const expr = 'Math.max(Date.now(), startTime)';
|
2023-02-11 15:57:27 +01:00
|
|
|
const result = evaluateJsExpression(expr, context);
|
|
|
|
|
|
|
|
expect(result).toBe(startTime);
|
2023-02-11 04:55:05 +01:00
|
|
|
});
|
2023-11-23 23:38:43 +01:00
|
|
|
|
|
|
|
it('should allow "context" as a var name', () => {
|
|
|
|
const expr = 'res["context"].toUpperCase()';
|
|
|
|
evaluateJsExpression(expr, context);
|
|
|
|
expect(cache.get(expr).toString()).toContain('let { res, context } = _BrunoNewFunctionInnerContext;');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should throw an error when we use "_BrunoNewFunctionInnerContext" as a var name', () => {
|
|
|
|
const expr = 'res["_BrunoNewFunctionInnerContext"].toUpperCase()';
|
|
|
|
expect(() => evaluateJsExpression(expr, context)).toThrow(SyntaxError);
|
|
|
|
expect(() => evaluateJsExpression(expr, context)).toThrow(
|
|
|
|
"Identifier '_BrunoNewFunctionInnerContext' has already been declared"
|
|
|
|
);
|
|
|
|
});
|
2023-02-11 04:55:05 +01:00
|
|
|
});
|
2023-02-20 06:35:49 +01:00
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
describe('response parser', () => {
|
2023-02-20 06:35:49 +01:00
|
|
|
const res = createResponseParser({
|
|
|
|
status: 200,
|
|
|
|
data: {
|
|
|
|
order: {
|
|
|
|
items: [
|
|
|
|
{ id: 1, amount: 10 },
|
|
|
|
{ id: 2, amount: 20 }
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should default to bruno query', () => {
|
|
|
|
const value = res('..items[?].amount[0]', (i) => i.amount > 10);
|
2023-02-20 06:35:49 +01:00
|
|
|
expect(value).toBe(20);
|
|
|
|
});
|
|
|
|
|
2023-09-21 21:12:14 +02:00
|
|
|
it('should allow json-query', () => {
|
|
|
|
const value = res.jq('order.items[amount > 10].amount');
|
2023-02-20 06:35:49 +01:00
|
|
|
expect(value).toBe(20);
|
|
|
|
});
|
|
|
|
});
|
2023-02-11 05:24:28 +01:00
|
|
|
});
|