@steemit/steem-js
Version:
JavaScript library for the Steem blockchain
143 lines (134 loc) • 5.06 kB
JavaScript
;
var BigInteger = require('bigi');
var ecurve = require('ecurve');
var secp256k1 = ecurve.getCurveByName('secp256k1');
BigInteger = require('bigi');
var base58 = require('bs58');
var hash = require('./hash');
var config = require('../../../config');
var assert = require('assert');
var G = secp256k1.G;
var n = secp256k1.n;
class PublicKey {
/** @param {ecurve.Point} public key */
constructor(Q) {
this.Q = Q;
}
static fromBinary(bin) {
return PublicKey.fromBuffer(new Buffer.from(bin, 'binary'));
}
static fromBuffer(buffer) {
if (buffer.toString("hex") === "000000000000000000000000000000000000000000000000000000000000000000") return new PublicKey(null);
return new PublicKey(ecurve.Point.decodeFrom(secp256k1, buffer));
}
toBuffer(compressed = this.Q ? this.Q.compressed : null) {
if (this.Q === null) return Buffer.from("000000000000000000000000000000000000000000000000000000000000000000", "hex");
return this.Q.getEncoded(compressed);
}
static fromPoint(point) {
return new PublicKey(point);
}
toUncompressed() {
var buf = this.Q.getEncoded(false);
var point = ecurve.Point.decodeFrom(secp256k1, buf);
return PublicKey.fromPoint(point);
}
/** bts::blockchain::address (unique but not a full public key) */
toBlockchainAddress() {
var pub_buf = this.toBuffer();
var pub_sha = hash.sha512(pub_buf);
return hash.ripemd160(pub_sha);
}
toString(address_prefix = config.get('address_prefix')) {
return this.toPublicKeyString(address_prefix);
}
/**
Full public key
{return} string
*/
toPublicKeyString(address_prefix = config.get('address_prefix')) {
if (this.pubdata) return address_prefix + this.pubdata;
const pub_buf = this.toBuffer();
const checksum = hash.ripemd160(pub_buf);
const addy = Buffer.concat([pub_buf, checksum.slice(0, 4)]);
this.pubdata = base58.encode(addy);
return address_prefix + this.pubdata;
}
/**
@arg {string} public_key - like STMXyz...
@arg {string} address_prefix - like STM
@return PublicKey or `null` (if the public_key string is invalid)
@deprecated fromPublicKeyString (use fromString instead)
*/
static fromString(public_key, address_prefix = config.get('address_prefix')) {
try {
return PublicKey.fromStringOrThrow(public_key, address_prefix);
} catch (e) {
return null;
}
}
/**
@arg {string} public_key - like STMXyz...
@arg {string} address_prefix - like STM
@throws {Error} if public key is invalid
@return PublicKey
*/
static fromStringOrThrow(public_key, address_prefix = config.get('address_prefix')) {
var prefix = public_key.slice(0, address_prefix.length);
assert.equal(address_prefix, prefix, `Expecting key to begin with ${address_prefix}, instead got ${prefix}`);
public_key = public_key.slice(address_prefix.length);
public_key = new Buffer.from(base58.decode(public_key), 'binary');
var checksum = public_key.slice(-4);
public_key = public_key.slice(0, -4);
var new_checksum = hash.ripemd160(public_key);
new_checksum = new_checksum.slice(0, 4);
assert.deepEqual(checksum, new_checksum, 'Checksum did not match');
return PublicKey.fromBuffer(public_key);
}
toAddressString(address_prefix = config.get('address_prefix')) {
var pub_buf = this.toBuffer();
var pub_sha = hash.sha512(pub_buf);
var addy = hash.ripemd160(pub_sha);
var checksum = hash.ripemd160(addy);
addy = Buffer.concat([addy, checksum.slice(0, 4)]);
return address_prefix + base58.encode(addy);
}
toPtsAddy() {
var pub_buf = this.toBuffer();
var pub_sha = hash.sha256(pub_buf);
var addy = hash.ripemd160(pub_sha);
addy = Buffer.concat([new Buffer.from([0x38]), addy]); //version 56(decimal)
var checksum = hash.sha256(addy);
checksum = hash.sha256(checksum);
addy = Buffer.concat([addy, checksum.slice(0, 4)]);
return base58.encode(addy);
}
child(offset) {
assert(Buffer.isBuffer(offset), "Buffer required: offset");
assert.equal(offset.length, 32, "offset length");
offset = Buffer.concat([this.toBuffer(), offset]);
offset = hash.sha256(offset);
let c = BigInteger.fromBuffer(offset);
if (c.compareTo(n) >= 0) throw new Error("Child offset went out of bounds, try again");
let cG = G.multiply(c);
let Qprime = this.Q.add(cG);
if (secp256k1.isInfinity(Qprime)) throw new Error("Child offset derived to an invalid key, try again");
return PublicKey.fromPoint(Qprime);
}
// toByteBuffer() {
// var b = new ByteBuffer(ByteBuffer.DEFAULT_CAPACITY, ByteBuffer.LITTLE_ENDIAN);
// this.appendByteBuffer(b);
// return b.copy(0, b.offset);
// }
static fromHex(hex) {
return PublicKey.fromBuffer(new Buffer.from(hex, 'hex'));
}
toHex() {
return this.toBuffer().toString('hex');
}
static fromStringHex(hex) {
return PublicKey.fromString(new Buffer.from(hex, 'hex'));
}
/* </HEX> */
}
module.exports = PublicKey;