|
|
|
var Stealthkey = require('./stealthkey');
|
|
|
|
var Base58check = require('../base58check');
|
|
|
|
var Pubkey = require('../pubkey');
|
|
|
|
var KDF = require('../kdf');
|
|
|
|
var BufferWriter = require('../bufferwriter');
|
|
|
|
var BufferReader = require('../bufferreader');
|
|
|
|
|
|
|
|
var StealthAddress = function StealthAddress(addrstr) {
|
|
|
|
if (!(this instanceof StealthAddress))
|
|
|
|
return new StealthAddress(addrstr);
|
|
|
|
|
|
|
|
if (typeof addrstr === 'string') {
|
|
|
|
this.fromBitcoreString(addrstr)
|
|
|
|
}
|
|
|
|
else if (Buffer.isBuffer(addrstr)) {
|
|
|
|
var buf = addrstr;
|
|
|
|
this.fromBitcoreBuffer(buf);
|
|
|
|
}
|
|
|
|
else if (addrstr) {
|
|
|
|
var obj = addrstr;
|
|
|
|
this.set(obj);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.mainver = 42;
|
|
|
|
StealthAddress.testver = 43;
|
|
|
|
|
|
|
|
StealthAddress.prototype.set = function(obj) {
|
|
|
|
this.payloadPubkey = obj.payloadPubkey || this.payloadPubkey;
|
|
|
|
this.scanPubkey = obj.scanPubkey || this.scanPubkey;
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.fromJSON = function(json) {
|
|
|
|
this.fromBitcoreString(json);
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.toJSON = function() {
|
|
|
|
return this.toBitcoreString();
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.fromStealthkey = function(stealthkey) {
|
|
|
|
this.set({
|
|
|
|
payloadPubkey: stealthkey.payloadKeypair.pubkey,
|
|
|
|
scanPubkey: stealthkey.scanKeypair.pubkey
|
|
|
|
});
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.fromBitcoreBuffer = function(buf) {
|
|
|
|
if (!Buffer.isBuffer(buf) || buf.length !== 66)
|
|
|
|
throw new Error('stealthkey: A stealth address must have length 66');
|
|
|
|
|
|
|
|
var pPubBuf = buf.slice(0, 33);
|
|
|
|
var sPubBuf = buf.slice(33, 66);
|
|
|
|
|
|
|
|
this.payloadPubkey = Pubkey().fromDER(pPubBuf);
|
|
|
|
this.scanPubkey = Pubkey().fromDER(sPubBuf);
|
|
|
|
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.fromBuffer = function(buf) {
|
|
|
|
var parsed = StealthAddress.parseDWBuffer(buf);
|
|
|
|
if ((parsed.version !== StealthAddress.mainver) && (parsed.version !== StealthAddress.testver))
|
|
|
|
throw new Error('Invalid version');
|
|
|
|
if (parsed.options !== 0)
|
|
|
|
throw new Error('Invalid options');
|
|
|
|
if (!parsed.scanPubkey)
|
|
|
|
throw new Error('Invalid scanPubkey');
|
|
|
|
if (parsed.payloadPubkeys.length !== 1)
|
|
|
|
throw new Error('Must have exactly one payloadPubkey');
|
|
|
|
if (parsed.nSigs !== 1)
|
|
|
|
throw new Error('Must require exactly one signature');
|
|
|
|
if (parsed.prefix.toString() !== "")
|
|
|
|
throw new Error('Only blank prefixes supported');
|
|
|
|
this.scanPubkey = parsed.scanPubkey;
|
|
|
|
this.payloadPubkey = parsed.payloadPubkeys[0];
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.fromString = function(str) {
|
|
|
|
return this.fromBuffer(Base58check(str).toBuffer());
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.fromBitcoreString = function(str) {
|
|
|
|
var buf = Base58check.decode(str);
|
|
|
|
this.fromBitcoreBuffer(buf);
|
|
|
|
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.getSharedKeypair = function(senderKeypair) {
|
|
|
|
var sharedSecretPoint = this.scanPubkey.point.mul(senderKeypair.privkey.bn);
|
|
|
|
var sharedSecretPubkey = Pubkey(sharedSecretPoint);
|
|
|
|
var buf = sharedSecretPubkey.toDER(true);
|
|
|
|
var sharedKeypair = KDF.sha256hmac2keypair(buf);
|
|
|
|
|
|
|
|
return sharedKeypair;
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.getReceivePubkey = function(senderKeypair) {
|
|
|
|
var sharedKeypair = this.getSharedKeypair(senderKeypair);
|
|
|
|
var pubkey = Pubkey(this.payloadPubkey.point.add(sharedKeypair.pubkey.point));
|
|
|
|
|
|
|
|
return pubkey;
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.toBitcoreBuffer = function() {
|
|
|
|
var pBuf = this.payloadPubkey.toDER(true);
|
|
|
|
var sBuf = this.scanPubkey.toDER(true);
|
|
|
|
|
|
|
|
return Buffer.concat([pBuf, sBuf]);
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.toBuffer = function(networkstr) {
|
|
|
|
if (networkstr === 'testnet')
|
|
|
|
var version = StealthAddress.testver;
|
|
|
|
else
|
|
|
|
var version = StealthAddress.mainver;
|
|
|
|
var bw = new BufferWriter();
|
|
|
|
bw.writeUInt8(version);
|
|
|
|
bw.writeUInt8(0); //options
|
|
|
|
bw.write(this.scanPubkey.toDER(true));
|
|
|
|
bw.writeUInt8(1); //number of payload keys - we only support 1 (not multisig)
|
|
|
|
bw.write(this.payloadPubkey.toDER(true));
|
|
|
|
bw.writeUInt8(1); //number of signatures - we only support 1 (not multisig)
|
|
|
|
bw.writeUInt8(0); //prefix length - we do not support prefix yet
|
|
|
|
var buf = bw.concat();
|
|
|
|
return buf;
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.toString = function(networkstr) {
|
|
|
|
return Base58check(this.toBuffer(networkstr)).toString();
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.prototype.toBitcoreString = function() {
|
|
|
|
var buf = this.toBitcoreBuffer();
|
|
|
|
var b58 = Base58check.encode(buf);
|
|
|
|
|
|
|
|
return b58;
|
|
|
|
};
|
|
|
|
|
|
|
|
StealthAddress.parseDWBuffer = function(buf) {
|
|
|
|
var br = new BufferReader(buf);
|
|
|
|
var parsed = {};
|
|
|
|
parsed.version = br.readUInt8();
|
|
|
|
parsed.options = br.readUInt8();
|
|
|
|
parsed.scanPubkey = Pubkey().fromBuffer(br.read(33));
|
|
|
|
parsed.nPayloadPubkeys = br.readUInt8();
|
|
|
|
parsed.payloadPubkeys = [];
|
|
|
|
for (var i = 0; i < parsed.nPayloadPubkeys; i++)
|
|
|
|
parsed.payloadPubkeys.push(Pubkey().fromBuffer(br.read(33)));
|
|
|
|
parsed.nSigs = br.readUInt8();
|
|
|
|
parsed.nPrefix = br.readUInt8();
|
|
|
|
parsed.prefix = br.read(parsed.nPrefix / 8);
|
|
|
|
return parsed;
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = StealthAddress;
|