|
|
|
var Point = require('./point');
|
|
|
|
var bn = require('./bn');
|
|
|
|
|
|
|
|
var Pubkey = function Pubkey(point) {
|
|
|
|
if (!(this instanceof Pubkey))
|
|
|
|
return new Pubkey(point);
|
|
|
|
if (point && !point.getX() && !point.getY())
|
|
|
|
throw new Error('Invalid point');
|
|
|
|
this.point = point;
|
|
|
|
};
|
|
|
|
|
|
|
|
Pubkey.prototype.fromDER = function(buf) {
|
|
|
|
if (buf[0] == 0x04) {
|
|
|
|
var xbuf = buf.slice(1, 33);
|
|
|
|
var ybuf = buf.slice(33, 65);
|
|
|
|
if (xbuf.length !== 32 || ybuf.length !== 32 || buf.length !== 65)
|
|
|
|
throw new Error('Length of x and y must be 32 bytes');
|
|
|
|
var x = bn(xbuf);
|
|
|
|
var y = bn(ybuf);
|
|
|
|
this.point = Point(x, y);
|
|
|
|
} else if (buf[0] == 0x03) {
|
|
|
|
var xbuf = buf.slice(1);
|
|
|
|
var x = bn(xbuf);
|
|
|
|
this.fromX(true, x);
|
|
|
|
} else if (buf[0] == 0x02) {
|
|
|
|
var xbuf = buf.slice(1);
|
|
|
|
var x = bn(xbuf);
|
|
|
|
this.fromX(false, x);
|
|
|
|
} else {
|
|
|
|
throw new Error('Invalid DER format pubkey');
|
|
|
|
}
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
Pubkey.prototype.fromString = function(str) {
|
|
|
|
this.fromDER(new Buffer(str, 'hex'));
|
|
|
|
};
|
|
|
|
|
|
|
|
Pubkey.prototype.fromX = function(odd, x) {
|
|
|
|
if (typeof odd !== 'boolean')
|
|
|
|
throw new Error('Must specify whether y is odd or not (true or false)');
|
|
|
|
this.point = Point.fromX(odd, x);
|
|
|
|
};
|
|
|
|
|
|
|
|
Pubkey.prototype.toBuffer = function() {
|
|
|
|
return this.toDER(true);
|
|
|
|
};
|
|
|
|
|
|
|
|
Pubkey.prototype.toDER = function(compressed) {
|
|
|
|
if (typeof compressed !== 'boolean')
|
|
|
|
throw new Error('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});
|
|
|
|
|
|
|
|
if (!compressed) {
|
|
|
|
var prefix = new Buffer([0x04]);
|
|
|
|
return Buffer.concat([prefix, xbuf, ybuf]);
|
|
|
|
} else {
|
|
|
|
var odd = ybuf[ybuf.length - 1] % 2;
|
|
|
|
if (odd)
|
|
|
|
var prefix = new Buffer([0x03]);
|
|
|
|
else
|
|
|
|
var prefix = new Buffer([0x02]);
|
|
|
|
return Buffer.concat([prefix, xbuf]);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
Pubkey.prototype.toString = function() {
|
|
|
|
return this.toDER(true).toString('hex');
|
|
|
|
};
|
|
|
|
|
|
|
|
//https://www.iacr.org/archive/pkc2003/25670211/25670211.pdf
|
|
|
|
Pubkey.prototype.validate = function() {
|
|
|
|
if (this.point.isInfinity())
|
|
|
|
throw new Error('point: Point cannot be equal to Infinity');
|
|
|
|
if (this.point.eq(Point(bn(0), bn(0))))
|
|
|
|
throw new Error('point: Point cannot be equal to 0, 0');
|
|
|
|
this.point.validate();
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = Pubkey;
|