fcoin/test/bloom-test.js
2017-08-11 18:25:54 -07:00

184 lines
5.3 KiB
JavaScript

/* eslint-env mocha */
/* eslint prefer-arrow-callback: "off" */
'use strict';
const assert = require('./util/assert');
const Bloom = require('../lib/utils/bloom');
const RollingFilter = require('../lib/utils/rollingfilter');
const murmur3 = require('../lib/utils/murmur3');
function testMurmur(str, seed, expect, enc) {
if (!enc)
enc = 'ascii';
const data = Buffer.from(str, enc);
const hash = murmur3(data, seed);
assert.strictEqual(hash, expect);
}
describe('Bloom', function() {
this.timeout(20000);
it('should do proper murmur3', () => {
testMurmur('', 0, 0);
testMurmur('', 0xfba4c795, 0x6a396f08);
testMurmur('00', 0xfba4c795, 0x2a101837);
testMurmur('hello world', 0, 0x5e928f0f);
testMurmur('', 0x00000000, 0x00000000, 'hex');
testMurmur('', 0xfba4c795, 0x6a396f08, 'hex');
testMurmur('', 0xffffffff, 0x81f16f39, 'hex');
testMurmur('00', 0x00000000, 0x514e28b7, 'hex');
testMurmur('00', 0xfba4c795, 0xea3f0b17, 'hex');
testMurmur('ff', 0x00000000, 0xfd6cf10d, 'hex');
testMurmur('0011', 0x00000000, 0x16c6b7ab, 'hex');
testMurmur('001122', 0x00000000, 0x8eb51c3d, 'hex');
testMurmur('00112233', 0x00000000, 0xb4471bf8, 'hex');
testMurmur('0011223344', 0x00000000, 0xe2301fa8, 'hex');
testMurmur('001122334455', 0x00000000, 0xfc2e4a15, 'hex');
testMurmur('00112233445566', 0x00000000, 0xb074502c, 'hex');
testMurmur('0011223344556677', 0x00000000, 0x8034d2a0, 'hex');
testMurmur('001122334455667788', 0x00000000, 0xb4698def, 'hex');
});
it('should test and add stuff', () => {
const filter = new Bloom(512, 10, 156);
filter.add('hello', 'ascii');
assert(filter.test('hello', 'ascii'));
assert(!filter.test('hello!', 'ascii'));
assert(!filter.test('ping', 'ascii'));
filter.add('hello!', 'ascii');
assert(filter.test('hello!', 'ascii'));
assert(!filter.test('ping', 'ascii'));
filter.add('ping', 'ascii');
assert(filter.test('ping', 'ascii'));
});
it('should serialize to the correct format', () => {
const filter = new Bloom(952, 6, 3624314491, Bloom.flags.NONE);
const item1 = '8e7445bbb8abd4b3174d80fa4c409fea6b94d96b';
const item2 = '047b00000078da0dca3b0ec2300c00d0ab4466ed10'
+ 'e763272c6c9ca052972c69e3884a9022084215e2eef'
+ '0e6f781656b5d5a87231cd4349e534b6dea55ad4ff55e';
const expected = Buffer.from(''
+ '000000000000000000000000000000000000000000000000088004000000000000000'
+ '000000000200000000000000000000000000000000800000000000000000002000000'
+ '000000000000002000000000000000000000000000000000000000000040000200000'
+ '0000000001000000800000080000000',
'hex');
filter.add(item1, 'hex');
filter.add(item2, 'hex');
assert.bufferEqual(filter.filter, expected);
});
it('should handle 1m ops with regular filter', () => {
const filter = Bloom.fromRate(210000, 0.00001, -1);
filter.tweak = 0xdeadbeef;
// ~1m operations
for (let i = 0; i < 1000; i++) {
const str = 'foobar' + i;
let j = i;
filter.add(str, 'ascii');
do {
const str = 'foobar' + j;
assert(filter.test(str, 'ascii'));
assert(!filter.test(str + '-', 'ascii'));
} while (j--);
}
});
it('should handle 1m ops with rolling filter', () => {
const filter = new RollingFilter(210000, 0.00001);
filter.tweak = 0xdeadbeef;
// ~1m operations
for (let i = 0; i < 1000; i++) {
const str = 'foobar' + i;
let j = i;
filter.add(str, 'ascii');
do {
const str = 'foobar' + j;
assert(filter.test(str, 'ascii'));
assert(!filter.test(str + '-', 'ascii'));
} while (j--);
}
});
it('should handle rolling generations', () => {
const filter = new RollingFilter(50, 0.00001);
filter.tweak = 0xdeadbeee;
for (let i = 0; i < 25; i++) {
const str = 'foobar' + i;
let j = i;
filter.add(str, 'ascii');
do {
const str = 'foobar' + j;
assert(filter.test(str, 'ascii'));
assert(!filter.test(str + '-', 'ascii'));
} while (j--);
}
for (let i = 25; i < 50; i++) {
const str = 'foobar' + i;
let j = i;
filter.add(str, 'ascii');
do {
const str = 'foobar' + j;
assert(filter.test(str, 'ascii'));
assert(!filter.test(str + '-', 'ascii'));
} while (j--);
}
for (let i = 50; i < 75; i++) {
const str = 'foobar' + i;
let j = i;
filter.add(str, 'ascii');
do {
const str = 'foobar' + j;
assert(filter.test(str, 'ascii'));
assert(!filter.test(str + '-', 'ascii'));
} while (j--);
}
for (let i = 75; i < 100; i++) {
const str = 'foobar' + i;
let j = i;
filter.add(str, 'ascii');
do {
const str = 'foobar' + j;
assert(filter.test(str, 'ascii'));
assert(!filter.test(str + '-', 'ascii'));
} while (j-- > 25);
assert(!filter.test('foobar 24', 'ascii'));
}
for (let i = 100; i < 125; i++) {
const str = 'foobar' + i;
let j = i;
filter.add(str, 'ascii');
do {
const str = 'foobar' + j;
assert(filter.test(str, 'ascii'));
assert(!filter.test(str + '-', 'ascii'));
} while (j-- > 50);
}
assert(!filter.test('foobar 49', 'ascii'));
});
});