|
|
|
const assert = require('assert');
|
|
|
|
const util = require('./../util/util.js');
|
|
|
|
const Coverage = require('./../../lib/coverage');
|
|
|
|
const Api = require('./../../lib/api')
|
|
|
|
|
|
|
|
describe('logical OR branches', () => {
|
|
|
|
let coverage;
|
|
|
|
let api;
|
|
|
|
|
|
|
|
before(async () => api = new Api({silent: true}));
|
|
|
|
beforeEach(() => coverage = new Coverage());
|
|
|
|
after(async() => await api.finish());
|
|
|
|
|
|
|
|
// if (x == 1 || x == 2) { } else ...
|
|
|
|
it('should cover an if statement with a simple OR condition (single branch)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/if-or', api, this.provider, );
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 1, 8: 0
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [1, 0], 2: [1, 0]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 1, 2: 0,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// if (x == 1 || x == 2) { } else ...
|
|
|
|
it('should cover an if statement with a simple OR condition (both branches)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/if-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
await contract.instance.a(2, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 2, 8: 0
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [2, 0], 2: [1, 1]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 2, 2: 0,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 2,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// require(x == 1 || x == 2)
|
|
|
|
it('should cover a require statement with a simple OR condition (single branch)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/require-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 1,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [1, 0], 2: [1, 0]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// require(
|
|
|
|
// x == 1 ||
|
|
|
|
// x == 2 ||
|
|
|
|
// x == 3
|
|
|
|
// )
|
|
|
|
it('should cover a require statement with multiline OR condition (two branches)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/require-multiline-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
await contract.instance.a(3, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 2,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [2, 0], 2: [1, 0], 3: [0, 1]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 2,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 2,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// require(x == 1 || x == 2)
|
|
|
|
it('should cover a require statement with a simple OR condition (both branches)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/require-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
await contract.instance.a(2, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 2,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [2, 0], 2: [1, 1]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 2,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 2,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// while( (x == 1 || x == 2) && counter < 2 ){
|
|
|
|
it('should cover a while statement with a simple OR condition (single branch)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/while-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 1, 6: 1, 7: 2
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [3, 0]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 1, 2: 1
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// while( (x == 1 || x == 2) && counter < 2 ){
|
|
|
|
it('should cover a while statement with a simple OR condition (both branches)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/while-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
await contract.instance.a(2, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 2, 6: 2, 7: 4
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [3, 3]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 2, 2: 2
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 2,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// return (x == 1 && true) || (x == 2 && true);
|
|
|
|
it('should cover a return statement with ANDED OR conditions (single branch)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/return-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 1,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [ 1, 0 ]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// return (x == 1 && true) || (x == 2 && true);
|
|
|
|
it('should cover a return statement with ANDED OR conditions (both branches)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/return-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
await contract.instance.a(2, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 2,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [ 1, 1 ]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 2,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 2,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
//if (x == 1 && true || x == 2) {
|
|
|
|
it('should cover an if statement with OR and AND conditions (single branch)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/and-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 1, 8: 0
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [1, 0], 2: [1, 0]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 1, 2: 0,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
//if (x == 1 && true || x == 2) {
|
|
|
|
it('should cover an if statement with OR and AND conditions (both branches)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/and-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
await contract.instance.a(2, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 2, 8: 0
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [2, 0], 2: [1, 1]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 2, 2: 0,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 2,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// if ((x == 1) && (x == 2 || true)) {
|
|
|
|
it('should cover an if statement with bracked ANDED OR conditions (rightmost sub-branch)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/and-or-brackets', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 1, 8: 0
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [1, 0], 2: [0, 1]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 1, 2: 0,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// if ((x == 1) || (x == 2 || true)) {
|
|
|
|
it('should cover an if statement with multiple (bracketed) OR conditions (branch 1)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/multi-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(1, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 1, 8: 0
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [1, 0], 2: [0, 0], 3: [1, 0] // Item 3 is the "outer" (left) OR clause
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 1, 2: 0,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// if ((x == 1) || (x == 2 || true)) {
|
|
|
|
it('should cover an if statement with multiple (bracketed) OR conditions (branch 2)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/multi-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(2, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 1, 8: 0
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [1, 0], 2: [1, 0], 3: [0, 1] // Item 3 is the "outer" (left) OR clause
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 1, 2: 0,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// if ((x == 1) || (x == 2 || true)) {
|
|
|
|
it('should cover an if statement with multiple (bracketed) OR conditions (branch 3)', async function() {
|
|
|
|
const contract = await util.bootstrapCoverage('or/multi-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(3, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 1, 8: 0
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [1, 0], 2: [0, 1], 3: [0, 1]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 1, 2: 0,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 1,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should cover the bzx example', async function(){
|
|
|
|
const contract = await util.bootstrapCoverage('or/bzx-or', api, this.provider);
|
|
|
|
coverage.addContract(contract.instrumented, util.filePath);
|
|
|
|
await contract.instance.a(3, contract.gas);
|
|
|
|
|
|
|
|
const mapping = coverage.generate(contract.data, util.pathPrefix);
|
|
|
|
|
|
|
|
assert.deepEqual(mapping[util.filePath].l, {
|
|
|
|
5: 1, 9: 1
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].b, {
|
|
|
|
1: [1, 0], 2: [0, 1]
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].s, {
|
|
|
|
1: 1, 2: 1,
|
|
|
|
});
|
|
|
|
assert.deepEqual(mapping[util.filePath].f, {
|
|
|
|
1: 1, 2: 1
|
|
|
|
});
|
|
|
|
})
|
|
|
|
});
|