You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
129 lines
3.5 KiB
129 lines
3.5 KiB
var bn = require('./bn');
|
|
|
|
var Signature = function Signature(r, s) {
|
|
if (!(this instanceof Signature))
|
|
return new Signature(r, s);
|
|
this.r = r;
|
|
this.s = s;
|
|
};
|
|
|
|
Signature.prototype.fromCompressed = function(buf) {
|
|
var b1 = buf.slice(0, 1)[0];
|
|
var b2 = buf.slice(1, 33);
|
|
var b3 = buf.slice(33, 65);
|
|
|
|
if (!(b1 === 0 || b1 === 1 || b1 === 2 || b1 === 3))
|
|
throw new Error('signature: i must be 0, 1, 2, or 3');
|
|
if (b2.length !== 32)
|
|
throw new Error('signature: r must be 32 bytes');
|
|
if (b3.length !== 32)
|
|
throw new Error('signature: s must be 32 bytes');
|
|
|
|
this.r = bn.fromBuffer(b2);
|
|
this.s = bn.fromBuffer(b3);
|
|
};
|
|
|
|
Signature.prototype.fromDER = function(buf) {
|
|
var obj = Signature.parseDER(buf);
|
|
this.r = obj.r;
|
|
this.s = obj.s;
|
|
};
|
|
|
|
Signature.prototype.fromString = function(str) {
|
|
var buf = new Buffer(str, 'hex');
|
|
this.fromDER(buf);
|
|
};
|
|
|
|
Signature.parseDER = function(buf) {
|
|
if (!Buffer.isBuffer(buf))
|
|
throw new Error('signature: DER formatted signature should be a buffer');
|
|
|
|
var header = buf[0];
|
|
|
|
if (header !== 0x30)
|
|
throw new Error('signature: Header byte should be 0x30');
|
|
|
|
var length = buf[1];
|
|
if (length !== buf.slice(2).length)
|
|
throw new Error('signature: Length byte should length of what follows');
|
|
|
|
var rheader = buf[2 + 0];
|
|
if (rheader !== 0x02)
|
|
throw new Error('signature: Integer byte for r should be 0x02');
|
|
|
|
var rlength = buf[2 + 1];
|
|
var rbuf = buf.slice(2 + 2, 2 + 2 + rlength);
|
|
var r = bn.fromBuffer(rbuf);
|
|
var rneg = buf[2 + 1 + 1] === 0x00 ? true : false;
|
|
if (rlength !== rbuf.length)
|
|
throw new Error('signature: Length of r incorrect');
|
|
|
|
var sheader = buf[2 + 2 + rlength + 0];
|
|
if (sheader !== 0x02)
|
|
throw new Error('signature: Integer byte for s should be 0x02');
|
|
|
|
var slength = buf[2 + 2 + rlength + 1];
|
|
var sbuf = buf.slice(2 + 2 + rlength + 2, 2 + 2 + rlength + 2 + slength);
|
|
var s = bn.fromBuffer(sbuf);
|
|
var sneg = buf[2 + 2 + rlength + 2 + 2] === 0x00 ? true : false;
|
|
if (slength !== sbuf.length)
|
|
throw new Error('signature: Length of s incorrect');
|
|
|
|
var sumlength = 2 + 2 + rlength + 2 + slength;
|
|
if (length !== sumlength - 2)
|
|
throw new Error('signature: Length of signature incorrect');
|
|
|
|
var obj = {
|
|
header: header,
|
|
length: length,
|
|
rheader: rheader,
|
|
rlength: rlength,
|
|
rneg: rneg,
|
|
rbuf: rbuf,
|
|
r: r,
|
|
sheader: sheader,
|
|
slength: slength,
|
|
sneg: sneg,
|
|
sbuf: sbuf,
|
|
s: s
|
|
};
|
|
|
|
return obj;
|
|
};
|
|
|
|
Signature.prototype.toCompressed = function(i) {
|
|
if (!(i === 0 || i === 1 || i ===2 || i ===3))
|
|
throw new Error('signature: i must be equal to 0, 1, 2, or 3');
|
|
|
|
var b1 = new Buffer([i]);
|
|
var b2 = this.r.toBuffer({size: 32});
|
|
var b3 = this.s.toBuffer({size: 32});
|
|
return Buffer.concat([b1, b2, b3]);
|
|
};
|
|
|
|
Signature.prototype.toDER = function() {
|
|
var rnbuf = this.r.toBuffer();
|
|
var snbuf = this.s.toBuffer();
|
|
|
|
var rneg = rnbuf[0] & 0x80 ? true : false;
|
|
var sneg = snbuf[0] & 0x80 ? true : false;
|
|
|
|
var rbuf = rneg ? Buffer.concat([new Buffer([0x00]), rnbuf]) : rnbuf;
|
|
var sbuf = sneg ? Buffer.concat([new Buffer([0x00]), snbuf]) : snbuf;
|
|
|
|
var length = 2 + rbuf.length + 2 + sbuf.length;
|
|
var rlength = rbuf.length;
|
|
var slength = sbuf.length;
|
|
var rheader = 0x02;
|
|
var sheader = 0x02;
|
|
var header = 0x30;
|
|
|
|
var der = Buffer.concat([new Buffer([header, length, rheader, rlength]), rbuf, new Buffer([sheader, slength]), sbuf]);
|
|
return der;
|
|
};
|
|
|
|
Signature.prototype.toString = function() {
|
|
var buf = this.toDER();
|
|
return buf.toString('hex');
|
|
};
|
|
module.exports = Signature;
|
|
|