-
Notifications
You must be signed in to change notification settings - Fork 10
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
3 changed files
with
200 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,168 @@ | ||
/* eslint new-cap: 0 */ | ||
const {Command, flags} = require('@oclif/command') | ||
const {red, white} = require('kleur') | ||
const ora = require('ora') | ||
const validateQrlAddress = require('@theqrl/validate-qrl-address') | ||
const grpc = require('grpc') | ||
const {createClient} = require('grpc-kit') | ||
const tmp = require('tmp') | ||
const fs = require('fs') | ||
const util = require('util') | ||
const CryptoJS = require('crypto-js') | ||
const {QRLPROTO_SHA256} = require('../get-qrl-proto-shasum') | ||
const protoLoader = require('@grpc/proto-loader') | ||
const PROTO_PATH = `${__dirname}/../../src/qrlbase.proto` | ||
|
||
const readFile = util.promisify(fs.readFile) | ||
const writeFile = util.promisify(fs.writeFile) | ||
|
||
const clientGetNodeInfo = client => { | ||
return new Promise((resolve, reject) => { | ||
client.getNodeInfo({}, (error, response) => { | ||
if (error) { | ||
reject(error) | ||
} | ||
resolve(response) | ||
}) | ||
}) | ||
} | ||
|
||
let qrlClient = null | ||
|
||
async function checkProtoHash(file) { | ||
return readFile(file).then(async contents => { | ||
const protoFileWordArray = CryptoJS.lib.WordArray.create(contents) | ||
const calculatedProtoHash = CryptoJS.SHA256(protoFileWordArray).toString(CryptoJS.enc.Hex) | ||
let verified = false | ||
QRLPROTO_SHA256.forEach(value => { | ||
if (value.protoSha256 === calculatedProtoHash) { | ||
verified = true | ||
} | ||
}) | ||
return verified | ||
}).catch(error => { | ||
throw new Error(error) | ||
}) | ||
} | ||
|
||
async function loadGrpcBaseProto(grpcEndpoint) { | ||
return protoLoader.load(PROTO_PATH, {}).then(async packageDefinition => { | ||
const packageObject = grpc.loadPackageDefinition(packageDefinition) | ||
const client = await new packageObject.qrl.Base(grpcEndpoint, grpc.credentials.createInsecure()) | ||
const res = await clientGetNodeInfo(client) | ||
const qrlProtoFilePath = tmp.fileSync({mode: '0644', prefix: 'qrl-', postfix: '.proto'}).name | ||
await writeFile(qrlProtoFilePath, res.grpcProto).then(fsErr => { | ||
if (fsErr) { | ||
throw new Error('tmp filesystem error') | ||
} | ||
}) | ||
return qrlProtoFilePath | ||
}) | ||
} | ||
|
||
async function loadGrpcProto(protofile, endpoint) { | ||
const options = { | ||
keepCase: true, | ||
longs: String, | ||
enums: String, | ||
defaults: true, | ||
oneofs: true, | ||
} | ||
const packageDefinition = await protoLoader.load(protofile, options) | ||
const grpcObject = await grpc.loadPackageDefinition(packageDefinition) | ||
const grpcObjectString = JSON.stringify(util.inspect(grpcObject.qrl, {showHidden: true, depth: 4})) | ||
const protoObjectWordArray = CryptoJS.lib.WordArray.create(grpcObjectString) | ||
const calculatedObjectHash = CryptoJS.SHA256(protoObjectWordArray).toString(CryptoJS.enc.Hex) | ||
let verified = false | ||
QRLPROTO_SHA256.forEach(value => { | ||
if (value.objectSha256 === calculatedObjectHash) { | ||
verified = true | ||
} | ||
}) | ||
// If the grpc object shasum matches, establish the grpc connection. | ||
if (verified) { | ||
qrlClient = createClient({ | ||
protoPath: protofile, | ||
packageName: 'qrl', | ||
serviceName: 'PublicAPI', | ||
options: { | ||
keepCase: true, | ||
longs: String, | ||
enums: String, | ||
defaults: true, | ||
oneofs: true, | ||
}, | ||
}, endpoint) | ||
} else { | ||
throw new Error('Unable to verify proto file') | ||
} | ||
} | ||
|
||
class OTSKey extends Command { | ||
async run() { | ||
const {args, flags} = this.parse(OTSKey) | ||
const address = args.address | ||
if (!validateQrlAddress.hexString(address).result) { | ||
this.log(`${red('⨉')} Unable to get a OTS: invalid QRL address`) | ||
this.exit(1) | ||
} | ||
let grpcEndpoint = 'testnet-4.automated.theqrl.org:19009' | ||
let network = 'Testnet' | ||
if (flags.grpc) { | ||
grpcEndpoint = flags.grpc | ||
network = `Custom GRPC endpoint: [${flags.grpc}]` | ||
} | ||
if (flags.testnet) { | ||
grpcEndpoint = 'testnet-4.automated.theqrl.org:19009' | ||
network = 'Testnet' | ||
} | ||
if (flags.mainnet) { | ||
grpcEndpoint = 'mainnet-4.automated.theqrl.org:19009' | ||
network = 'Mainnet' | ||
} | ||
this.log(white().bgBlue(network)) | ||
const spinner = ora({text: 'Fetching OTS from API...'}).start() | ||
const proto = await loadGrpcBaseProto(grpcEndpoint) | ||
checkProtoHash(proto).then(async protoHash => { | ||
if (!protoHash) { | ||
this.log(`${red('⨉')} Unable to validate .proto file from node`) | ||
this.exit(1) | ||
} | ||
// next load GRPC object and check hash of that too | ||
await loadGrpcProto(proto, grpcEndpoint) | ||
const request = { | ||
address: Buffer.from(address.substring(1), 'hex'), | ||
} | ||
await qrlClient.GetOTS(request, async (error, response) => { | ||
if (error) { | ||
this.log(`${red('⨉')} Unable to read next unused OTS key`) | ||
this.exit(1) | ||
} | ||
// console.log('response:', response) | ||
spinner.succeed(`Next unused OTS key: ${response.next_unused_ots_index}`) | ||
this.exit(0) | ||
}) | ||
}) | ||
} | ||
} | ||
|
||
OTSKey.description = `Get a address's OTS state from the network | ||
... | ||
TODO | ||
` | ||
|
||
OTSKey.args = [ | ||
{ | ||
name: 'address', | ||
description: 'address to return OTS state for', | ||
required: true, | ||
}, | ||
] | ||
|
||
OTSKey.flags = { | ||
testnet: flags.boolean({char: 't', default: false, description: 'queries testnet for the OTS state'}), | ||
mainnet: flags.boolean({char: 'm', default: false, description: 'queries mainnet for the OTS state'}), | ||
grpc: flags.string({char: 'g', required: false, description: 'advanced: grcp endpoint (for devnet/custom QRL network deployments)'}), | ||
} | ||
|
||
module.exports = {OTSKey} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
/* qrl.proto sha256 sum for each release of QRL Node */ | ||
const QRLPROTO_SHA256 = [ | ||
{ | ||
version: '1.1.10 python', | ||
protoSha256: '00032d07d4b4637103db15b3d68ae019c14988e870475832af6eb5bd390e04f5', | ||
objectSha256: 'a93598b15aea7d4d40656e3a824891a6c881cbcb3678c7bec0110f9614b8e271', | ||
}, | ||
{ | ||
version: '1.1.11 python', | ||
protoSha256: '00032d07d4b4637103db15b3d68ae019c14988e870475832af6eb5bd390e04f5', | ||
objectSha256: 'a93598b15aea7d4d40656e3a824891a6c881cbcb3678c7bec0110f9614b8e271', | ||
}, | ||
] | ||
|
||
module.exports = {QRLPROTO_SHA256} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
syntax = "proto3"; | ||
|
||
package qrl; | ||
|
||
service Base { | ||
rpc GetNodeInfo (GetNodeInfoReq) returns (GetNodeInfoResp) {} | ||
} | ||
|
||
message GetNodeInfoReq | ||
{ | ||
} | ||
|
||
message GetNodeInfoResp | ||
{ | ||
string version = 1; | ||
string grpcProto = 2; | ||
} |