UNPKG

@steemit/steem-js

Version:

JavaScript library for the Steem blockchain

143 lines (134 loc) 5.06 kB
"use strict"; 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;