Merge pull request #2 from maraoz/standardize
Use standard precondition checks
This commit is contained in:
commit
33476229d9
|
@ -1,6 +1,7 @@
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
var bitcore = require('bitcore');
|
var bitcore = require('bitcore');
|
||||||
|
var _ = require('lodash');
|
||||||
var PrivateKey = bitcore.PrivateKey;
|
var PrivateKey = bitcore.PrivateKey;
|
||||||
var PublicKey = bitcore.PublicKey;
|
var PublicKey = bitcore.PublicKey;
|
||||||
var Address = bitcore.Address;
|
var Address = bitcore.Address;
|
||||||
|
@ -9,9 +10,10 @@ var ECDSA = bitcore.crypto.ECDSA;
|
||||||
var Signature = bitcore.crypto.Signature;
|
var Signature = bitcore.crypto.Signature;
|
||||||
var sha256sha256 = bitcore.crypto.Hash.sha256sha256;
|
var sha256sha256 = bitcore.crypto.Hash.sha256sha256;
|
||||||
var JSUtil = bitcore.util.js;
|
var JSUtil = bitcore.util.js;
|
||||||
|
var $ = bitcore.util.preconditions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Will construct a new message to sign and verify.
|
* constructs a new message to sign and verify.
|
||||||
*
|
*
|
||||||
* @param {String} message
|
* @param {String} message
|
||||||
* @returns {Message}
|
* @returns {Message}
|
||||||
|
@ -20,9 +22,7 @@ var Message = function Message(message) {
|
||||||
if (!(this instanceof Message)) {
|
if (!(this instanceof Message)) {
|
||||||
return new Message(message);
|
return new Message(message);
|
||||||
}
|
}
|
||||||
if (typeof message !== 'string') {
|
$.checkArgument(_.isString(message), 'First argument should be a string');
|
||||||
throw new TypeError('First argument should be a string');
|
|
||||||
}
|
|
||||||
this.message = message;
|
this.message = message;
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
|
@ -40,9 +40,8 @@ Message.prototype.magicHash = function magicHash() {
|
||||||
};
|
};
|
||||||
|
|
||||||
Message.prototype._sign = function _sign(privateKey) {
|
Message.prototype._sign = function _sign(privateKey) {
|
||||||
if (!(privateKey instanceof PrivateKey)) {
|
$.checkArgument(privateKey instanceof PrivateKey,
|
||||||
throw new TypeError('First argument should be an instance of PrivateKey');
|
'First argument should be an instance of PrivateKey');
|
||||||
}
|
|
||||||
var hash = this.magicHash();
|
var hash = this.magicHash();
|
||||||
var ecdsa = new ECDSA();
|
var ecdsa = new ECDSA();
|
||||||
ecdsa.hashbuf = hash;
|
ecdsa.hashbuf = hash;
|
||||||
|
@ -65,12 +64,8 @@ Message.prototype.sign = function sign(privateKey) {
|
||||||
};
|
};
|
||||||
|
|
||||||
Message.prototype._verify = function _verify(publicKey, signature) {
|
Message.prototype._verify = function _verify(publicKey, signature) {
|
||||||
if (!(publicKey instanceof PublicKey)) {
|
$.checkArgument(publicKey instanceof PublicKey, 'First argument should be an instance of PublicKey');
|
||||||
throw new TypeError('First argument should be an instance of PublicKey');
|
$.checkArgument(signature instanceof Signature, 'Second argument should be an instance of Signature');
|
||||||
}
|
|
||||||
if (!(signature instanceof Signature)) {
|
|
||||||
throw new TypeError('Second argument should be an instance of Signature');
|
|
||||||
}
|
|
||||||
var hash = this.magicHash();
|
var hash = this.magicHash();
|
||||||
var verified = ECDSA.verify(hash, signature, publicKey);
|
var verified = ECDSA.verify(hash, signature, publicKey);
|
||||||
if (!verified) {
|
if (!verified) {
|
||||||
|
@ -83,11 +78,17 @@ Message.prototype._verify = function _verify(publicKey, signature) {
|
||||||
* Will return a boolean of the signature is valid for a given bitcoin address.
|
* Will return a boolean of the signature is valid for a given bitcoin address.
|
||||||
* If it isn't the specific reason is accessible via the "error" member.
|
* If it isn't the specific reason is accessible via the "error" member.
|
||||||
*
|
*
|
||||||
* @param {String} bitcoinAddress - A bitcoin address
|
* @param {Address|String} bitcoinAddress - A bitcoin address
|
||||||
* @param {String} signatureString - A base64 encoded compact signature
|
* @param {String} signatureString - A base64 encoded compact signature
|
||||||
* @returns {Boolean}
|
* @returns {Boolean}
|
||||||
*/
|
*/
|
||||||
Message.prototype.verify = function verify(bitcoinAddress, signatureString) {
|
Message.prototype.verify = function verify(bitcoinAddress, signatureString) {
|
||||||
|
$.checkArgument(bitcoinAddress);
|
||||||
|
$.checkArgument(signatureString && _.isString(signatureString));
|
||||||
|
|
||||||
|
if (_.isString(bitcoinAddress)) {
|
||||||
|
bitcoinAddress = Address.fromString(bitcoinAddress);
|
||||||
|
}
|
||||||
var signature = Signature.fromCompact(new Buffer(signatureString, 'base64'));
|
var signature = Signature.fromCompact(new Buffer(signatureString, 'base64'));
|
||||||
|
|
||||||
// recover the public key
|
// recover the public key
|
||||||
|
@ -96,11 +97,10 @@ Message.prototype.verify = function verify(bitcoinAddress, signatureString) {
|
||||||
ecdsa.sig = signature;
|
ecdsa.sig = signature;
|
||||||
var publicKey = ecdsa.toPublicKey();
|
var publicKey = ecdsa.toPublicKey();
|
||||||
|
|
||||||
var expectedAddress = Address.fromString(bitcoinAddress);
|
var signatureAddress = Address.fromPublicKey(publicKey, bitcoinAddress.network);
|
||||||
var signatureAddress = Address.fromPublicKey(publicKey, expectedAddress.network);
|
|
||||||
|
|
||||||
// check that the recovered address and specified address match
|
// check that the recovered address and specified address match
|
||||||
if (expectedAddress.toString() !== signatureAddress.toString()) {
|
if (bitcoinAddress.toString() !== signatureAddress.toString()) {
|
||||||
this.error = 'The signature did not match the message digest';
|
this.error = 'The signature did not match the message digest';
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,8 @@
|
||||||
"url": "https://github.com/bitpay/bitcore-message.git"
|
"url": "https://github.com/bitpay/bitcore-message.git"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"bitcore": "^0.9.4"
|
"bitcore": "^0.9.4",
|
||||||
|
"lodash": "^3.1.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"bitcore-build": "bitpay/bitcore-build",
|
"bitcore-build": "bitpay/bitcore-build",
|
||||||
|
|
|
@ -5,6 +5,7 @@ var expect = chai.expect;
|
||||||
var should = chai.should();
|
var should = chai.should();
|
||||||
|
|
||||||
var bitcore = require('bitcore');
|
var bitcore = require('bitcore');
|
||||||
|
var Address = bitcore.Address;
|
||||||
var Signature = bitcore.crypto.Signature;
|
var Signature = bitcore.crypto.Signature;
|
||||||
var Message = require('../');
|
var Message = require('../');
|
||||||
|
|
||||||
|
@ -24,9 +25,9 @@ describe('Message', function() {
|
||||||
var publicKey = privateKey.toPublicKey();
|
var publicKey = privateKey.toPublicKey();
|
||||||
|
|
||||||
it('will error with incorrect message type', function() {
|
it('will error with incorrect message type', function() {
|
||||||
expect(function(){
|
expect(function() {
|
||||||
return new Message(new Date());
|
return new Message(new Date());
|
||||||
}).to.throw(TypeError);
|
}).to.throw('First argument should be a string');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('will instantiate without "new"', function() {
|
it('will instantiate without "new"', function() {
|
||||||
|
@ -46,10 +47,10 @@ describe('Message', function() {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('sign will error with incorrect private key argument', function() {
|
it('sign will error with incorrect private key argument', function() {
|
||||||
expect(function(){
|
expect(function() {
|
||||||
var message3 = new Message(text);
|
var message3 = new Message(text);
|
||||||
return message3.sign('not a private key');
|
return message3.sign('not a private key');
|
||||||
}).to.throw(TypeError);
|
}).to.throw('First argument should be an instance of PrivateKey');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('can verify a message with signature', function() {
|
it('can verify a message with signature', function() {
|
||||||
|
@ -65,17 +66,17 @@ describe('Message', function() {
|
||||||
});
|
});
|
||||||
|
|
||||||
it('verify will error with incorrect public key argument', function() {
|
it('verify will error with incorrect public key argument', function() {
|
||||||
expect(function(){
|
expect(function() {
|
||||||
var message6 = new Message(text);
|
var message6 = new Message(text);
|
||||||
return message6._verify('not a public key', signature);
|
return message6._verify('not a public key', signature);
|
||||||
}).to.throw(TypeError);
|
}).to.throw('First argument should be an instance of PublicKey');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('verify will error with incorrect signature argument', function() {
|
it('verify will error with incorrect signature argument', function() {
|
||||||
expect(function(){
|
expect(function() {
|
||||||
var message7 = new Message(text);
|
var message7 = new Message(text);
|
||||||
return message7._verify(publicKey, 'not a signature');
|
return message7._verify(publicKey, 'not a signature');
|
||||||
}).to.throw(TypeError);
|
}).to.throw('Second argument should be an instance of Signature');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('verify will correctly identify a bad signature', function() {
|
it('verify will correctly identify a bad signature', function() {
|
||||||
|
@ -146,5 +147,10 @@ describe('Message', function() {
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
it('accepts Address for verification', function() {
|
||||||
|
var verified = Message(text)
|
||||||
|
.verify(new Address(address), signatureString);
|
||||||
|
verified.should.equal(true);
|
||||||
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in New Issue