2014-03-31 11:47:47 +08:00
|
|
|
var assert = require('assert')
|
2014-07-28 13:46:34 +10:00
|
|
|
var crypto = require('crypto')
|
2014-10-16 01:25:39 +11:00
|
|
|
var ecurve = require('ecurve')
|
2014-06-03 21:43:10 +10:00
|
|
|
var networks = require('../src/networks')
|
2014-06-23 18:22:01 +10:00
|
|
|
var sinon = require('sinon')
|
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
var BigInteger = require('bigi')
|
2014-05-13 17:55:53 +10:00
|
|
|
var ECKey = require('../src/eckey')
|
2014-05-16 13:46:06 +10:00
|
|
|
|
2014-05-18 19:47:39 +10:00
|
|
|
var fixtures = require('./fixtures/eckey.json')
|
2014-01-11 13:57:43 +07:00
|
|
|
|
|
|
|
describe('ECKey', function() {
|
2014-03-31 11:47:47 +08:00
|
|
|
describe('constructor', function() {
|
2014-05-16 13:46:06 +10:00
|
|
|
it('defaults to compressed', function() {
|
|
|
|
var privKey = new ECKey(BigInteger.ONE)
|
|
|
|
|
|
|
|
assert.equal(privKey.pub.compressed, true)
|
2014-03-22 18:19:56 +11:00
|
|
|
})
|
2014-02-24 18:31:18 +02:00
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
it('supports the uncompressed flag', function() {
|
|
|
|
var privKey = new ECKey(BigInteger.ONE, false)
|
2014-03-31 11:47:47 +08:00
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
assert.equal(privKey.pub.compressed, false)
|
2014-03-25 03:48:50 +11:00
|
|
|
})
|
2014-03-24 15:11:34 -07:00
|
|
|
|
2014-05-29 16:00:27 +10:00
|
|
|
fixtures.valid.forEach(function(f) {
|
2014-06-07 18:24:16 +10:00
|
|
|
it('calculates the matching pubKey for ' + f.d, function() {
|
|
|
|
var d = new BigInteger(f.d)
|
|
|
|
var privKey = new ECKey(d)
|
2014-03-31 11:47:47 +08:00
|
|
|
|
2014-06-21 17:05:12 +10:00
|
|
|
assert.equal(privKey.pub.Q.toString(), f.Q)
|
2014-05-16 13:46:06 +10:00
|
|
|
})
|
2014-03-31 11:47:47 +08:00
|
|
|
})
|
2014-03-25 03:31:55 +11:00
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
fixtures.invalid.constructor.forEach(function(f) {
|
2014-06-07 18:24:16 +10:00
|
|
|
it('throws on ' + f.d, function() {
|
|
|
|
var d = new BigInteger(f.d)
|
2014-05-31 11:28:00 +10:00
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
assert.throws(function() {
|
2014-06-07 18:24:16 +10:00
|
|
|
new ECKey(d)
|
2014-05-29 16:00:27 +10:00
|
|
|
}, new RegExp(f.exception))
|
2014-05-16 13:46:06 +10:00
|
|
|
})
|
2014-03-31 11:47:47 +08:00
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2014-10-16 01:25:39 +11:00
|
|
|
it('uses the secp256k1 curve by default', function() {
|
|
|
|
var secp256k1 = ecurve.getCurveByName('secp256k1')
|
|
|
|
|
|
|
|
for (var property in secp256k1) {
|
|
|
|
// FIXME: circular structures in ecurve
|
|
|
|
if (property === 'G') continue
|
|
|
|
if (property === 'infinity') continue
|
|
|
|
|
|
|
|
var actual = ECKey.curve[property]
|
|
|
|
var expected = secp256k1[property]
|
|
|
|
|
|
|
|
assert.deepEqual(actual, expected)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
describe('fromWIF', function() {
|
2014-05-29 16:00:27 +10:00
|
|
|
fixtures.valid.forEach(function(f) {
|
|
|
|
f.WIFs.forEach(function(wif) {
|
|
|
|
it('imports ' + wif.string + ' correctly', function() {
|
2014-05-16 13:46:06 +10:00
|
|
|
var privKey = ECKey.fromWIF(wif.string)
|
2014-03-31 11:47:47 +08:00
|
|
|
|
2014-06-07 18:24:16 +10:00
|
|
|
assert.equal(privKey.d.toString(), f.d)
|
2014-05-16 13:46:06 +10:00
|
|
|
assert.equal(privKey.pub.compressed, wif.compressed)
|
|
|
|
})
|
|
|
|
})
|
2014-03-31 11:47:47 +08:00
|
|
|
})
|
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
fixtures.invalid.WIF.forEach(function(f) {
|
2014-05-29 16:00:27 +10:00
|
|
|
it('throws on ' + f.string, function() {
|
2014-05-16 13:46:06 +10:00
|
|
|
assert.throws(function() {
|
|
|
|
ECKey.fromWIF(f.string)
|
2014-05-29 16:00:27 +10:00
|
|
|
}, new RegExp(f.exception))
|
2014-05-16 13:46:06 +10:00
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
2014-03-31 11:47:47 +08:00
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
describe('toWIF', function() {
|
2014-05-29 16:00:27 +10:00
|
|
|
fixtures.valid.forEach(function(f) {
|
|
|
|
f.WIFs.forEach(function(wif) {
|
|
|
|
it('exports ' + wif.string + ' correctly', function() {
|
2014-05-16 13:46:06 +10:00
|
|
|
var privKey = ECKey.fromWIF(wif.string)
|
2014-06-03 21:43:10 +10:00
|
|
|
var network = networks[wif.network]
|
|
|
|
var result = privKey.toWIF(network)
|
2014-05-16 13:46:06 +10:00
|
|
|
|
|
|
|
assert.equal(result, wif.string)
|
|
|
|
})
|
|
|
|
})
|
2014-03-31 11:47:47 +08:00
|
|
|
})
|
2014-05-16 13:46:06 +10:00
|
|
|
})
|
|
|
|
|
2014-06-23 18:22:01 +10:00
|
|
|
describe('makeRandom', function() {
|
|
|
|
var exWIF = 'KwMWvwRJeFqxYyhZgNwYuYjbQENDAPAudQx5VEmKJrUZcq6aL2pv'
|
|
|
|
var exPrivKey = ECKey.fromWIF(exWIF)
|
|
|
|
var exBuffer = exPrivKey.d.toBuffer(32)
|
|
|
|
|
2014-07-28 13:46:34 +10:00
|
|
|
describe('uses default crypto RNG', function() {
|
2014-06-23 18:22:01 +10:00
|
|
|
beforeEach(function() {
|
2014-07-28 13:46:34 +10:00
|
|
|
sinon.stub(crypto, 'randomBytes').returns(exBuffer)
|
2014-06-23 18:22:01 +10:00
|
|
|
})
|
|
|
|
|
|
|
|
afterEach(function() {
|
2014-07-28 13:46:34 +10:00
|
|
|
crypto.randomBytes.restore()
|
2014-06-23 18:22:01 +10:00
|
|
|
})
|
|
|
|
|
|
|
|
it('generates a ECKey', function() {
|
|
|
|
var privKey = ECKey.makeRandom()
|
|
|
|
|
|
|
|
assert.equal(privKey.toWIF(), exWIF)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('supports compression', function() {
|
|
|
|
assert.equal(ECKey.makeRandom(true).pub.compressed, true)
|
|
|
|
assert.equal(ECKey.makeRandom(false).pub.compressed, false)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
it('allows a custom RNG to be used', function() {
|
|
|
|
function rng(size) {
|
|
|
|
return exBuffer.slice(0, size)
|
|
|
|
}
|
|
|
|
|
|
|
|
var privKey = ECKey.makeRandom(undefined, rng)
|
|
|
|
assert.equal(privKey.toWIF(), exWIF)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
describe('signing', function() {
|
2014-07-29 01:01:11 +10:00
|
|
|
var hash = crypto.randomBytes(32)
|
2014-05-16 13:46:06 +10:00
|
|
|
var priv = ECKey.makeRandom()
|
|
|
|
var signature = priv.sign(hash)
|
2014-03-31 11:47:47 +08:00
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
it('should verify against the public key', function() {
|
|
|
|
assert(priv.pub.verify(hash, signature))
|
|
|
|
})
|
2014-03-31 11:47:47 +08:00
|
|
|
|
2014-05-16 13:46:06 +10:00
|
|
|
it('should not verify against the wrong public key', function() {
|
|
|
|
var priv2 = ECKey.makeRandom()
|
2014-03-31 11:47:47 +08:00
|
|
|
|
2014-04-24 07:45:28 +10:00
|
|
|
assert(!priv2.pub.verify(hash, signature))
|
2014-03-31 11:47:47 +08:00
|
|
|
})
|
|
|
|
})
|
2014-01-11 13:57:43 +07:00
|
|
|
})
|