mirror of
https://github.com/usebruno/bruno.git
synced 2026-06-23 12:45:38 +00:00
* feat: add JSON Schema validation support with custom chai assertion - Introduced a new custom assertion for JSON Schema validation in chai, allowing users to validate response bodies against defined schemas. - Updated the postman translation logic to translate `pm.response.to.have.jsonSchema` to the new assertion format. - Enhanced tests to cover various scenarios for JSON Schema validation, ensuring accurate translations and functionality. - Updated package dependencies to include the latest versions of relevant libraries. * refactor: enhance JSON Schema validation assertion and add comprehensive test cases * chore: add @rollup/plugin-json dependency and enhance JSON Schema validation tests - Added @rollup/plugin-json as a development dependency in package.json and package-lock.json. - Introduced new test cases for JSON Schema validation, covering various scenarios including valid schema matching, type mismatches, and required field checks. - Updated existing assertions to utilize the new validation capabilities. * refactor: streamline JSON Schema validation with default Ajv instance - Updated the custom chai assertion for JSON Schema validation to utilize a default Ajv instance, improving consistency and reducing redundancy in the code. - Enhanced the error messages in the assertion to include the actual data being validated, providing clearer feedback during validation failures. * refactor: improve error messaging in JSON Schema validation assertion - Enhanced the custom chai assertion for JSON Schema validation to provide clearer error messages by including a stringified version of the data being validated, improving feedback during validation failures. * refactor: simplify Ajv instance creation in JSON Schema validation - Removed the default Ajv instance and streamlined the creation of Ajv instances in the custom chai assertion for JSON Schema validation, ensuring consistent handling of ajvOptions across the codebase. * feat: add support for negated JSON Schema assertions in Postman translations - Introduced translations for `pm.response.to.not.have.jsonSchema`, `pm.response.not.to.have.jsonSchema`, and `pm.response.to.have.not.jsonSchema` to the new assertion format using `expect`. - Enhanced the translation logic to handle these new patterns and added corresponding test cases to ensure accurate functionality. - Updated existing tests to cover various scenarios for negated assertions, improving overall test coverage for JSON Schema validation. * fix: improve error handling in JSON Schema validation assertions - Added error handling for JSON schema compilation in the custom chai assertion, ensuring that any compilation errors are caught and reported with a clear message. - Updated tests to verify that malformed schemas correctly trigger assertion errors, enhancing the robustness of JSON Schema validation.
432 lines
14 KiB
JavaScript
432 lines
14 KiB
JavaScript
const { describe, it, expect } = require('@jest/globals');
|
|
const TestRuntime = require('../src/runtime/test-runtime');
|
|
const ScriptRuntime = require('../src/runtime/script-runtime');
|
|
const AssertRuntime = require('../src/runtime/assert-runtime');
|
|
const Bru = require('../src/bru');
|
|
const VarsRuntime = require('../src/runtime/vars-runtime');
|
|
|
|
describe('runtime', () => {
|
|
describe('test-runtime', () => {
|
|
const baseRequest = {
|
|
method: 'GET',
|
|
url: 'http://localhost:3000/',
|
|
headers: {},
|
|
data: undefined
|
|
};
|
|
const baseResponse = {
|
|
status: 200,
|
|
statusText: 'OK',
|
|
data: [
|
|
{
|
|
id: 1
|
|
},
|
|
{
|
|
id: 2
|
|
},
|
|
{
|
|
id: 3
|
|
}
|
|
]
|
|
};
|
|
|
|
it('should wait async tests', async () => {
|
|
const testFile = `
|
|
await test('async test', ()=> {
|
|
return new Promise((resolve)=> {
|
|
setTimeout(()=> {resolve()},200)
|
|
})
|
|
})
|
|
`;
|
|
|
|
const runtime = new TestRuntime({ runtime: 'nodevm' });
|
|
const result = await runtime.runTests(
|
|
testFile,
|
|
{ ...baseRequest },
|
|
{ ...baseResponse },
|
|
{},
|
|
{},
|
|
'.',
|
|
null,
|
|
process.env
|
|
);
|
|
expect(result.results.map((el) => ({ description: el.description, status: el.status }))).toEqual([
|
|
{ description: 'async test', status: 'pass' }
|
|
]);
|
|
});
|
|
|
|
it('should have ajv and ajv-formats dependencies available', async () => {
|
|
const testFile = `
|
|
const Ajv = require('ajv');
|
|
const addFormats = require("ajv-formats");
|
|
const ajv = new Ajv();
|
|
addFormats(ajv);
|
|
|
|
const schema = {
|
|
type: 'string',
|
|
format: 'date-time'
|
|
};
|
|
|
|
const validate = ajv.compile(schema)
|
|
|
|
test('format valid', () => {
|
|
const valid = validate(new Date().toISOString())
|
|
expect(valid).to.be.true;
|
|
})
|
|
`;
|
|
|
|
const runtime = new TestRuntime({ runtime: 'nodevm' });
|
|
const result = await runtime.runTests(
|
|
testFile,
|
|
{ ...baseRequest },
|
|
{ ...baseResponse },
|
|
{},
|
|
{},
|
|
'.',
|
|
null,
|
|
process.env
|
|
);
|
|
expect(result.results.map((el) => ({ description: el.description, status: el.status }))).toEqual([
|
|
{ description: 'format valid', status: 'pass' }
|
|
]);
|
|
});
|
|
});
|
|
|
|
describe('script-runtime', () => {
|
|
describe('run-request-script', () => {
|
|
const baseRequest = {
|
|
method: 'GET',
|
|
url: 'http://localhost:3000/',
|
|
headers: {},
|
|
data: undefined
|
|
};
|
|
|
|
it('should have ajv and ajv-formats dependencies available', async () => {
|
|
const script = `
|
|
const Ajv = require('ajv');
|
|
const addFormats = require("ajv-formats");
|
|
const ajv = new Ajv();
|
|
addFormats(ajv);
|
|
|
|
const schema = {
|
|
type: 'string',
|
|
format: 'date-time'
|
|
};
|
|
|
|
const validate = ajv.compile(schema)
|
|
|
|
bru.setVar('validation', validate(new Date().toISOString()))
|
|
`;
|
|
|
|
const runtime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
const result = await runtime.runRequestScript(script, { ...baseRequest }, {}, {}, '.', null, process.env);
|
|
expect(result.runtimeVariables.validation).toBeTruthy();
|
|
});
|
|
});
|
|
|
|
describe('run-response-script', () => {
|
|
const baseRequest = {
|
|
method: 'GET',
|
|
url: 'http://localhost:3000/',
|
|
headers: {},
|
|
data: undefined
|
|
};
|
|
const baseResponse = {
|
|
status: 200,
|
|
statusText: 'OK',
|
|
data: [
|
|
{
|
|
id: 1
|
|
},
|
|
{
|
|
id: 2
|
|
},
|
|
{
|
|
id: 3
|
|
}
|
|
]
|
|
};
|
|
|
|
it('should have ajv and ajv-formats dependencies available', async () => {
|
|
const script = `
|
|
const Ajv = require('ajv');
|
|
const addFormats = require("ajv-formats");
|
|
const ajv = new Ajv();
|
|
addFormats(ajv);
|
|
|
|
const schema = {
|
|
type: 'string',
|
|
format: 'date-time'
|
|
};
|
|
|
|
const validate = ajv.compile(schema)
|
|
|
|
bru.setVar('validation', validate(new Date().toISOString()))
|
|
`;
|
|
|
|
const runtime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
const result = await runtime.runResponseScript(
|
|
script,
|
|
{ ...baseRequest },
|
|
{ ...baseResponse },
|
|
{},
|
|
{},
|
|
'.',
|
|
null,
|
|
process.env
|
|
);
|
|
expect(result.runtimeVariables.validation).toBeTruthy();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('persistent environment variables validation', () => {
|
|
it('should throw error when trying to persist non-string values', async () => {
|
|
const script = `bru.setEnvVar('number', 42, {persist: true});`;
|
|
const runtime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
|
|
await expect(runtime.runRequestScript(script, {}, {}, {}, '.', null, process.env))
|
|
.rejects.toThrow('Persistent environment variables must be strings. Received number for key "number".');
|
|
});
|
|
|
|
it('should throw error when trying to persist boolean values', async () => {
|
|
const script = `bru.setEnvVar('isActive', true, {persist: true});`;
|
|
const runtime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
|
|
await expect(runtime.runRequestScript(script, {}, {}, {}, '.', null, process.env))
|
|
.rejects.toThrow('Persistent environment variables must be strings. Received boolean for key "isActive".');
|
|
});
|
|
|
|
it('should throw error when trying to persist object values', async () => {
|
|
const script = `bru.setEnvVar('config', {port: 3000}, {persist: true});`;
|
|
const runtime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
|
|
await expect(runtime.runRequestScript(script, {}, {}, {}, '.', null, process.env))
|
|
.rejects.toThrow('Persistent environment variables must be strings. Received object for key "config".');
|
|
});
|
|
|
|
it('should throw error when trying to persist array values', async () => {
|
|
const script = `bru.setEnvVar('items', ['item1', 'item2'], {persist: true});`;
|
|
const runtime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
|
|
await expect(runtime.runRequestScript(script, {}, {}, {}, '.', null, process.env))
|
|
.rejects.toThrow('Persistent environment variables must be strings. Received object for key "items".');
|
|
});
|
|
|
|
it('should allow string values when persist is true', async () => {
|
|
const script = `bru.setEnvVar('api_key', 'abc123', {persist: true});`;
|
|
const runtime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
|
|
const result = await runtime.runRequestScript(script, {}, {}, {}, '.', null, process.env);
|
|
|
|
expect(result.envVariables.api_key).toBe('abc123');
|
|
});
|
|
|
|
it('should allow non-string values when persist is false', async () => {
|
|
const script = `
|
|
bru.setEnvVar('number', 42, {persist: false});
|
|
bru.setEnvVar('boolean', true, {persist: false});
|
|
bru.setEnvVar('object', {key: 'value'}, {persist: false});
|
|
bru.setEnvVar('array', [1, 2, 3], {persist: false});
|
|
`;
|
|
const runtime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
|
|
const result = await runtime.runRequestScript(script, {}, {}, {}, '.', null, process.env);
|
|
|
|
expect(result.envVariables.number).toBe(42);
|
|
expect(result.envVariables.boolean).toBe(true);
|
|
expect(result.envVariables.object).toEqual({ key: 'value' });
|
|
expect(result.envVariables.array).toEqual([1, 2, 3]);
|
|
});
|
|
|
|
it('should allow non-string values when persist is not specified', async () => {
|
|
const script = `bru.setEnvVar('number', 42);`;
|
|
const runtime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
|
|
const result = await runtime.runRequestScript(script, {}, {}, {}, '.', null, process.env);
|
|
|
|
expect(result.envVariables.number).toBe(42);
|
|
});
|
|
});
|
|
|
|
describe('bru.setVar random variable', () => {
|
|
it('should be able to set random variables as values', async () => {
|
|
const script = `bru.setVar('title', '{{$randomFirstName}}')`;
|
|
|
|
const runtime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
|
|
const result = await runtime.runRequestScript(script, {}, {}, {}, '.', null, process.env);
|
|
|
|
expect(result.runtimeVariables.title).toBe('{{$randomFirstName}}');
|
|
});
|
|
});
|
|
|
|
describe('assert-runtime', () => {
|
|
const baseRequest = {
|
|
method: 'GET',
|
|
url: 'http://localhost:3000/',
|
|
headers: {},
|
|
data: undefined
|
|
};
|
|
|
|
const makeResponse = (data) => ({
|
|
status: 200,
|
|
statusText: 'OK',
|
|
data,
|
|
headers: {}
|
|
});
|
|
|
|
const runAssertions = (assertions, response, runtime = 'nodevm') => {
|
|
const assertRuntime = new AssertRuntime({ runtime });
|
|
return assertRuntime.runAssertions(assertions, { ...baseRequest }, response, {}, {}, process.env);
|
|
};
|
|
|
|
describe('isJson', () => {
|
|
it('should pass for a plain object', () => {
|
|
const results = runAssertions(
|
|
[{ name: 'res.body', value: 'isJson', enabled: true }],
|
|
makeResponse({ id: 1, name: 'test' })
|
|
);
|
|
expect(results[0].status).toBe('pass');
|
|
});
|
|
|
|
it('should pass for a nested object', () => {
|
|
const results = runAssertions(
|
|
[{ name: 'res.body', value: 'isJson', enabled: true }],
|
|
makeResponse({ user: { id: 1, tags: ['a', 'b'] } })
|
|
);
|
|
expect(results[0].status).toBe('pass');
|
|
});
|
|
|
|
it('should pass for objects from a different realm (e.g. after res.setBody in node-vm)', async () => {
|
|
const response = makeResponse({ id: 1, name: 'original' });
|
|
|
|
// res.setBody() inside node-vm creates a cross-realm object whose
|
|
// constructor is the VM's Object, not the host's Object
|
|
const scriptRuntime = new ScriptRuntime({ runtime: 'nodevm' });
|
|
await scriptRuntime.runResponseScript(
|
|
`res.setBody({ id: 2, name: 'updated' });`,
|
|
{ ...baseRequest },
|
|
response,
|
|
{}, {}, '.', null, process.env
|
|
);
|
|
|
|
const results = runAssertions(
|
|
[{ name: 'res.body', value: 'isJson', enabled: true }],
|
|
response
|
|
);
|
|
expect(results[0].status).toBe('pass');
|
|
});
|
|
|
|
it('should pass for an array', () => {
|
|
const results = runAssertions(
|
|
[{ name: 'res.body', value: 'isJson', enabled: true }],
|
|
makeResponse([1, 2, 3])
|
|
);
|
|
expect(results[0].status).toBe('pass');
|
|
});
|
|
|
|
it('should pass for an array of strings', () => {
|
|
const results = runAssertions(
|
|
[{ name: 'res.body', value: 'isJson', enabled: true }],
|
|
makeResponse(['A55001213ZX0A'])
|
|
);
|
|
expect(results[0].status).toBe('pass');
|
|
});
|
|
|
|
it('should pass for an empty array', () => {
|
|
const results = runAssertions(
|
|
[{ name: 'res.body', value: 'isJson', enabled: true }],
|
|
makeResponse([])
|
|
);
|
|
expect(results[0].status).toBe('pass');
|
|
});
|
|
|
|
it('should pass for an array of objects', () => {
|
|
const results = runAssertions(
|
|
[{ name: 'res.body', value: 'isJson', enabled: true }],
|
|
makeResponse([{ id: 1 }, { id: 2 }])
|
|
);
|
|
expect(results[0].status).toBe('pass');
|
|
});
|
|
|
|
it('should fail for a string', () => {
|
|
const results = runAssertions(
|
|
[{ name: 'res.body', value: 'isJson', enabled: true }],
|
|
makeResponse('hello')
|
|
);
|
|
expect(results[0].status).toBe('fail');
|
|
});
|
|
|
|
it('should fail for null', () => {
|
|
const results = runAssertions(
|
|
[{ name: 'res.body', value: 'isJson', enabled: true }],
|
|
makeResponse(null)
|
|
);
|
|
expect(results[0].status).toBe('fail');
|
|
});
|
|
});
|
|
|
|
describe('jsonSchema', () => {
|
|
const chai = require('chai');
|
|
|
|
it('should pass when body matches a valid schema', () => {
|
|
const body = { name: 'John', age: 30 };
|
|
const schema = {
|
|
type: 'object',
|
|
properties: {
|
|
name: { type: 'string' },
|
|
age: { type: 'number' }
|
|
},
|
|
required: ['name', 'age']
|
|
};
|
|
chai.expect(body).to.have.jsonSchema(schema);
|
|
});
|
|
|
|
it('should fail when body has a type mismatch', () => {
|
|
const body = { name: 'John', age: 'thirty' };
|
|
const schema = {
|
|
type: 'object',
|
|
properties: {
|
|
name: { type: 'string' },
|
|
age: { type: 'number' }
|
|
},
|
|
required: ['name', 'age']
|
|
};
|
|
expect(() => chai.expect(body).to.have.jsonSchema(schema)).toThrow(/validation errors/);
|
|
});
|
|
|
|
it('should fail when a required field is missing', () => {
|
|
const body = { name: 'John' };
|
|
const schema = {
|
|
type: 'object',
|
|
properties: {
|
|
name: { type: 'string' },
|
|
age: { type: 'number' }
|
|
},
|
|
required: ['name', 'age']
|
|
};
|
|
expect(() => chai.expect(body).to.have.jsonSchema(schema)).toThrow(/validation errors/);
|
|
});
|
|
|
|
it('should pass with custom ajvOptions', () => {
|
|
const body = { name: 'John', age: 30 };
|
|
const schema = {
|
|
type: 'object',
|
|
properties: {
|
|
name: { type: 'string' },
|
|
age: { type: 'number' }
|
|
},
|
|
required: ['name', 'age']
|
|
};
|
|
chai.expect(body).to.have.jsonSchema(schema, { allErrors: false });
|
|
});
|
|
|
|
it('should support negation with .not', () => {
|
|
const body = { name: 'John' };
|
|
const schema = { type: 'array' };
|
|
chai.expect(body).to.not.have.jsonSchema(schema);
|
|
});
|
|
});
|
|
});
|
|
});
|