|
|
|
'use strict';
|
|
|
|
|
|
|
|
var Point = require('./crypto/point');
|
|
|
|
var BN = require('./crypto/bn');
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Bitcore PublicKey
|
|
|
|
*
|
|
|
|
* Instantiate a PublicKey from a 'PrivateKey', 'Point', 'string', 'Buffer'.
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
*
|
|
|
|
* var publicKey = new PublicKey(privkey, true);
|
|
|
|
*
|
|
|
|
* @param {String} data - The encoded data in various formats
|
|
|
|
* @param {String} [compressed] - If the public key is compressed
|
|
|
|
* @returns {PublicKey} A new valid instance of an PublicKey
|
|
|
|
*/
|
|
|
|
var PublicKey = function PublicKey(data, compressed) {
|
|
|
|
|
|
|
|
if (!(this instanceof PublicKey)) {
|
|
|
|
return new PublicKey(data, compressed);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!data) {
|
|
|
|
throw new TypeError('First argument is required, please include public key data.');
|
|
|
|
}
|
|
|
|
|
|
|
|
var info = {
|
|
|
|
compressed: typeof(compressed) !== 'undefined' ? compressed : true
|
|
|
|
};
|
|
|
|
|
|
|
|
// detect type of data
|
|
|
|
if (data instanceof Point) {
|
|
|
|
info.point = data;
|
|
|
|
} else if (typeof(data) === 'string'){
|
|
|
|
info = PublicKey._transformDER(new Buffer(data, 'hex' ));
|
|
|
|
} else if (data instanceof Buffer || data instanceof Uint8Array){
|
|
|
|
info = PublicKey._transformDER(data);
|
|
|
|
} else if (data.constructor && (data.constructor.name &&
|
|
|
|
data.constructor.name === 'PrivateKey')) {
|
|
|
|
info = PublicKey._transformPrivateKey(data);
|
|
|
|
} else {
|
|
|
|
throw new TypeError('First argument is an unrecognized data format.');
|
|
|
|
}
|
|
|
|
|
|
|
|
// validation
|
|
|
|
if (info.point.isInfinity()){
|
|
|
|
throw new Error('Point cannot be equal to Infinity');
|
|
|
|
}
|
|
|
|
if (info.point.eq(Point(BN(0), BN(0)))){
|
|
|
|
throw new Error('Point cannot be equal to 0, 0');
|
|
|
|
}
|
|
|
|
|
|
|
|
//https://www.iacr.org/archive/pkc2003/25670211/25670211.pdf
|
|
|
|
info.point.validate();
|
|
|
|
|
|
|
|
this.point = info.point;
|
|
|
|
this.compressed = info.compressed;
|
|
|
|
|
|
|
|
return this;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Internal function to transform a private key into a public key point
|
|
|
|
*
|
|
|
|
* @param {PrivateKey} privkey - An instance of PrivateKey
|
|
|
|
* @returns {Object} An object with keys: point and compressed
|
|
|
|
*/
|
|
|
|
PublicKey._transformPrivateKey = function(privkey) {
|
|
|
|
var info = {};
|
|
|
|
if (!privkey.constructor ||
|
|
|
|
(privkey.constructor.name && privkey.constructor.name !== 'PrivateKey')) {
|
|
|
|
throw new TypeError('Must be an instance of PrivateKey');
|
|
|
|
}
|
|
|
|
info.point = Point.getG().mul(privkey.bn);
|
|
|
|
info.compressed = privkey.compressed;
|
|
|
|
return info;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Internal function to transform DER into a public key point
|
|
|
|
*
|
|
|
|
* @param {Buffer} buf - An hex encoded buffer
|
|
|
|
* @returns {Object} An object with keys: point and compressed
|
|
|
|
*/
|
|
|
|
PublicKey._transformDER = function(buf){
|
|
|
|
var info = {};
|
|
|
|
if (!(buf instanceof Buffer) && !(buf instanceof Uint8Array)){
|
|
|
|
throw new TypeError('Must be a hex buffer of DER encoded public key');
|
|
|
|
}
|
|
|
|
|
|
|
|
var x;
|
|
|
|
var y;
|
|
|
|
var xbuf;
|
|
|
|
var ybuf;
|
|
|
|
|
|
|
|
if (buf[0] === 0x04) {
|
|
|
|
xbuf = buf.slice(1, 33);
|
|
|
|
ybuf = buf.slice(33, 65);
|
|
|
|
if (xbuf.length !== 32 || ybuf.length !== 32 || buf.length !== 65) {
|
|
|
|
throw new TypeError('Length of x and y must be 32 bytes');
|
|
|
|
}
|
|
|
|
x = BN(xbuf);
|
|
|
|
y = BN(ybuf);
|
|
|
|
info.point = Point(x, y);
|
|
|
|
info.compressed = false;
|
|
|
|
} else if (buf[0] === 0x03) {
|
|
|
|
xbuf = buf.slice(1);
|
|
|
|
x = BN(xbuf);
|
|
|
|
info = PublicKey._transformX(true, x);
|
|
|
|
info.compressed = true;
|
|
|
|
} else if (buf[0] == 0x02) {
|
|
|
|
xbuf = buf.slice(1);
|
|
|
|
x = BN(xbuf);
|
|
|
|
info = PublicKey._transformX(false, x);
|
|
|
|
info.compressed = true;
|
|
|
|
} else {
|
|
|
|
throw new TypeError('Invalid DER format public key');
|
|
|
|
}
|
|
|
|
return info;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Internal function to transform X into a public key point
|
|
|
|
*
|
|
|
|
* @param {Boolean} odd - If the point is above or below the x axis
|
|
|
|
* @param {Point} x - The x point
|
|
|
|
* @returns {Object} An object with keys: point and compressed
|
|
|
|
*/
|
|
|
|
PublicKey._transformX = function(odd, x){
|
|
|
|
var info = {};
|
|
|
|
if (typeof odd !== 'boolean') {
|
|
|
|
throw new TypeError('Must specify whether y is odd or not (true or false)');
|
|
|
|
}
|
|
|
|
info.point = Point.fromX(odd, x);
|
|
|
|
return info;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Instantiate a PublicKey from JSON
|
|
|
|
*
|
|
|
|
* @param {String} json - A JSON string of DER encoded public key
|
|
|
|
* @returns {PublicKey} A new valid instance of PublicKey
|
|
|
|
*/
|
|
|
|
PublicKey.fromJSON = function(json) {
|
|
|
|
var buf = new Buffer(json, 'hex');
|
|
|
|
var info = PublicKey._transformDER(buf);
|
|
|
|
return new PublicKey(info.point, info.compressed);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Instantiate a PublicKey from a PrivateKey
|
|
|
|
*
|
|
|
|
* @param {PrivateKey} privkey - An instance of PrivateKey
|
|
|
|
* @returns {PublicKey} A new valid instance of PublicKey
|
|
|
|
*/
|
|
|
|
PublicKey.fromPrivateKey = function(privkey) {
|
|
|
|
var info = PublicKey._transformPrivateKey(privkey);
|
|
|
|
return new PublicKey(info.point, info.compressed);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Instantiate a PublicKey from a Buffer
|
|
|
|
*
|
|
|
|
* @param {Buffer} buf - A DER hex buffer
|
|
|
|
* @returns {PublicKey} A new valid instance of PublicKey
|
|
|
|
*/
|
|
|
|
PublicKey.fromBuffer = function(buf) {
|
|
|
|
var info = PublicKey._transformDER(buf);
|
|
|
|
return new PublicKey(info.point, info.compressed);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Instantiate a PublicKey from a Point
|
|
|
|
*
|
|
|
|
* @param {Point} point - A Point instance
|
|
|
|
* @returns {PublicKey} A new valid instance of PublicKey
|
|
|
|
*/
|
|
|
|
PublicKey.fromPoint = function(point){
|
|
|
|
if (!(point instanceof Point)) {
|
|
|
|
throw new TypeError('First argument must be an instance of Point.');
|
|
|
|
}
|
|
|
|
return new PublicKey(point);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Instantiate a PublicKey from a DER Buffer
|
|
|
|
*
|
|
|
|
* @param {Buffer} buf - A DER Buffer
|
|
|
|
* @returns {PublicKey} A new valid instance of PublicKey
|
|
|
|
*/
|
|
|
|
PublicKey.fromDER = function(buf) {
|
|
|
|
var info = PublicKey._transformDER(buf);
|
|
|
|
return new PublicKey(info.point, info.compressed);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Instantiate a PublicKey from a DER hex encoded string
|
|
|
|
*
|
|
|
|
* @param {String} str - A DER hex string
|
|
|
|
* @param {String} [encoding] - The type of string encoding
|
|
|
|
* @returns {PublicKey} A new valid instance of PublicKey
|
|
|
|
*/
|
|
|
|
PublicKey.fromString = function(str, encoding) {
|
|
|
|
var buf = new Buffer(str, encoding || 'hex');
|
|
|
|
var info = PublicKey._transformDER(buf);
|
|
|
|
return new PublicKey(info.point, info.compressed);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Instantiate a PublicKey from an X Point
|
|
|
|
*
|
|
|
|
* @param {Boolean} odd - If the point is above or below the x axis
|
|
|
|
* @param {Point} x - The x point
|
|
|
|
* @returns {PublicKey} A new valid instance of PublicKey
|
|
|
|
*/
|
|
|
|
PublicKey.fromX = function(odd, x) {
|
|
|
|
var info = PublicKey._transformX(odd, x);
|
|
|
|
return new PublicKey(info.point, info.compressed);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Check if there would be any errors when initializing a PublicKey
|
|
|
|
*
|
|
|
|
* @param {String} data - The encoded data in various formats
|
|
|
|
* @param {String} [compressed] - If the public key is compressed
|
|
|
|
* @returns {null|Error} An error if exists
|
|
|
|
*/
|
|
|
|
PublicKey.getValidationError = function(data, compressed) {
|
|
|
|
var error;
|
|
|
|
try {
|
|
|
|
new PublicKey(data, compressed);
|
|
|
|
} catch (e) {
|
|
|
|
error = e;
|
|
|
|
}
|
|
|
|
return error;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Check if the parameters are valid
|
|
|
|
*
|
|
|
|
* @param {String} data - The encoded data in various formats
|
|
|
|
* @param {String} [compressed] - If the public key is compressed
|
|
|
|
* @returns {Boolean} If the public key would be valid
|
|
|
|
*/
|
|
|
|
PublicKey.isValid = function(data, compressed) {
|
|
|
|
return !PublicKey.getValidationError(data, compressed);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Will output the PublicKey to JSON
|
|
|
|
*
|
|
|
|
* @returns {String} A hex encoded string
|
|
|
|
*/
|
|
|
|
PublicKey.prototype.toJSON = function() {
|
|
|
|
return this.toBuffer().toString('hex');
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Will output the PublicKey to a Buffer
|
|
|
|
*
|
|
|
|
* @returns {Buffer} A DER hex encoded buffer
|
|
|
|
*/
|
|
|
|
PublicKey.prototype.toBuffer = function() {
|
|
|
|
var compressed = typeof this.compressed === 'undefined' ? true : this.compressed;
|
|
|
|
return this.toDER(compressed);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Will output the PublicKey to a DER Buffer
|
|
|
|
*
|
|
|
|
* @returns {Buffer} A DER hex encoded buffer
|
|
|
|
*/
|
|
|
|
PublicKey.prototype.toDER = function(compressed) {
|
|
|
|
compressed = typeof(compressed) !== 'undefined' ? compressed : this.compressed;
|
|
|
|
if (typeof compressed !== 'boolean') {
|
|
|
|
throw new TypeError('Must specify whether the public key is compressed or not (true or false)');
|
|
|
|
}
|
|
|
|
|
|
|
|
var x = this.point.getX();
|
|
|
|
var y = this.point.getY();
|
|
|
|
|
|
|
|
var xbuf = x.toBuffer({size: 32});
|
|
|
|
var ybuf = y.toBuffer({size: 32});
|
|
|
|
|
|
|
|
var prefix;
|
|
|
|
if (!compressed) {
|
|
|
|
prefix = new Buffer([0x04]);
|
|
|
|
return Buffer.concat([prefix, xbuf, ybuf]);
|
|
|
|
} else {
|
|
|
|
var odd = ybuf[ybuf.length - 1] % 2;
|
|
|
|
if (odd) {
|
|
|
|
prefix = new Buffer([0x03]);
|
|
|
|
} else {
|
|
|
|
prefix = new Buffer([0x02]);
|
|
|
|
}
|
|
|
|
return Buffer.concat([prefix, xbuf]);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Will output the PublicKey to a DER encoded hex string
|
|
|
|
*
|
|
|
|
* @returns {String} A DER hex encoded string
|
|
|
|
*/
|
|
|
|
PublicKey.prototype.toString = function() {
|
|
|
|
var compressed = typeof this.compressed === 'undefined' ? true : this.compressed;
|
|
|
|
return this.toDER(compressed).toString('hex');
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Will return a string formatted for the console
|
|
|
|
*
|
|
|
|
* @returns {String} Public key
|
|
|
|
*/
|
|
|
|
PublicKey.prototype.inspect = function() {
|
|
|
|
return '<PublicKey: ' + this.toString() + ', compressed: '+this.compressed+'>';
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = PublicKey;
|