To generate a private key we are going to use the randomBytes
method of the crypto
package in nodejs
to have a reasonable source of randomness and then use the secp256k1
package to verify the validity of the private key.
Open the main.js
file and add the following.
const { randomBytes } = require("crypto");
const secp256k1 = require("secp256k1");
function makePrivateKey() {
let pk;
do {
pk = randomBytes(32);
} while (!secp256k1.privateKeyVerify(pk));
return pk;
}
const privateKey = makePrivateKey();
console.log(`Private Key: ${privateKey.toString("hex")}`);
To generate the public key we will use the publicKeyCreate
method of the secp256k1
package, this will return a buffer
object that we are later converting into a hexadecimal string.
const secp256k1 = require("secp256k1");
// sample private key
const pk = "1111111111111111111111111111111111111111111111111111111111111111";
function makePublicKey(pk) {
// slice(1) removes the 'type byte' which is hardcoded as '04'
const publicKey = secp256k1.publicKeyCreate(pk, false).slice(1);
return publicKey;
}
function getPublicKey(pubK) {
let pubKHex = null;
try {
pubKHex = Buffer.from(pubK).toString("hex");
} catch (err) {
console.log("Unexpected error parsing public key into hex");
console.log(err);
}
return pubKHex;
}
function privateKeyAsBuffer(pk) {
let pkBuffer = null;
try {
pkBuffer = Buffer.from(pk, "hex");
} catch (err) {
console.log("Unexpected error parsing private key into buffer");
}
return pkBuffer;
}
const pkAsBuffer = privateKeyAsBuffer(pk);
const pubKBuffer = makePublicKey(pkAsBuffer);
const pubKHex = getPublicKey(pubKBuffer);
console.log(`Public Key: ${pubKHex}`);
To generate the wallet address we convert the hexadecimal string of the public key into a Buffer object and then apply the SHA3-256 hashing algorithm to it, we then take the last 40 characters of the resulting hexadecimal string and add “hx” to the beginning.
const sha3_256 = require("js-sha3").sha3_256;
const pubK =
"4f355bdcb7cc0af728ef3cceb9615d90684bb5b2ca5f859ab0f0b704075871aa385b6b1b8ead809ca67454d9683fcf2ba03456d6fe2c4abe2b07f0fbdbb2f1c1";
function getAddress(pubK) {
let address = null;
try {
address = "hx" + sha3_256(pubK).slice(-40);
} catch (err) {
console.log("Unexpected error parsing address");
console.log(err);
}
return address;
}
function hexKeyToBuffer(key) {
let keyAsBuffer = null;
try {
keyAsBuffer = Buffer.from(key, "hex");
} catch (err) {
console.log("Unexpected error parsing key into buffer");
console.log(err);
}
return keyAsBuffer;
}
const pubKAsBuffer = hexKeyToBuffer(pubK);
const address = getAddress(pubKAsBuffer);
console.log(`Address: ${address}`);
To generate a signature is necessary to execute the following steps:
- Serialized transaction data
- Create a transaction signature
- Hash the serialized transaction data
- Create a recoverable ECDSA signature
- Encode the generated recoverable ECDSA signature as a Base64-encoded string
- Add the signature to the RPC JSON request as a param.
Transaction data is serialized by concatenating key, value pairs in params
with .
as a delimiter. Adding the method name, icx_sendTransaction
, to the serialized string as a prefix completes the serialization process.
For a more detailed explanation of how to serialize transaction data you can visit the following tutorial.
An example of a serialized transaction would be the following:
“icx_sendTransaction.” +
“from.hx396031be52ec56955bd7bf15eacdfa1a1c1fe19e.” +
“nid.0x2.” +
“nonce.0x1.” +
“stepLimit.0x1e8480.” +
“timestamp.0x5f569aaf7c100.” +
“to.hx81765cee88107ca5d3de480d25778ec71d8ae65f.” +
“value.0xde0b6b3a7640000.” +
“version.0x3”
To generate the signature we hash the serialized data with the sha3-256
algorithm, convert that data into a Buffer and apply the ecdaSign
method of the secp256k1
library which will return a value of type Buffer. This data is then converted into a base64
string to have our signature.
const secp256k1 = require("secp256k1");
const sha3_256 = require("js-sha3").sha3_256;
function sign(serializedData, pk) {
const serializedHash = sha3_256(serializedData);
const msgAsBuffer = Buffer.from(serializedHash, "hex");
const signing = secp256k1.ecdsaSign(msgAsBuffer, pk);
const recovery = new Uint8Array(1);
recovery[0] = signing.recid;
return concatTypedArrays(signing.signature, recovery);
}
function concatTypedArrays(a, b) {
const c = new a.constructor(a.length + b.length);
c.set(a, 0);
c.set(b, a.length);
return c;
}
function hexKeyToBuffer(key) {
let keyAsBuffer = null;
try {
keyAsBuffer = Buffer.from(key, "hex");
} catch (err) {
console.log("Unexpected error parsing key into buffer");
console.log(err);
}
return keyAsBuffer;
}
const pk = "1111111111111111111111111111111111111111111111111111111111111111";
const serialized = "icx_sendTransaction.from.hx396031be52ec56955bd7bf15eacdfa1a1c1fe19e.nid.0x2.nonce.0x1.stepLimit.0x1e8480.timestamp.0x5f569aaf7c100.to.hx81765cee88107ca5d3de480d25778ec71d8ae65f.value.0xde0b6b3a7640000.version.0x3";
const PK1AsBuffer = hexKeyToBuffer(pk);
const signaturePk1 = sign(serialized, PK1AsBuffer);
const signatureAsBase64 = Buffer.from(signaturePk1).toString("base64");
console.log(signatureAsBase64);