Skip to content

Commit

Permalink
Fix rpc
Browse files Browse the repository at this point in the history
- Remove all .keys().all() that is extremely slow.
- Fixed undefined address account storage related rpc methods.
- Fixed db not closing after get_storageKeys.
  • Loading branch information
nguyenphuminh authored Jul 11, 2024
1 parent 81cca05 commit 9510cd6
Show file tree
Hide file tree
Showing 3 changed files with 71 additions and 74 deletions.
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "jechain",
"version": "0.30.2",
"version": "0.30.3",
"description": "Node for JeChain - an experimental smart contract blockchain network",
"main": "./index.js",
"scripts": {
Expand Down
2 changes: 1 addition & 1 deletion src/node/server.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
const crypto = require("crypto"), SHA256 = message => crypto.createHash("sha256").update(message).digest("hex");
const WS = require("ws");
const EC = require("elliptic").ec, ec = new EC("secp256k1");
const { Level } = require('level');
const { Level } = require("level");
const { fork } = require("child_process");

const Block = require("../core/block");
Expand Down
141 changes: 69 additions & 72 deletions src/rpc/rpc.js
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
"use strict";

const { Level } = require("level");
const crypto = require("crypto"), SHA256 = message => crypto.createHash("sha256").update(message).digest("hex");
const fs = require("fs");
const Transaction = require("../core/transaction");
const Block = require("../core/block");
const { deserializeState, serializeState } = require("../utils/utils");
const { deserializeState } = require("../utils/utils");

const fastify = require("fastify")();

Expand Down Expand Up @@ -88,15 +90,13 @@ function rpc(PORT, client, transactionHandler, keyPair, stateDB, blockDB, bhashD
if (typeof req.body.params !== "object" || typeof req.body.params.hash !== "string") {
throwError("Invalid request.");
} else {
const hashes = (await bhashDB.keys().all());

if (!hashes.find(hash => hash === req.body.params.hash)) {
throwError("Invalid block hash.", 400);
} else {
try {
const blockNumber = parseInt((await bhashDB.get(req.body.params.hash)).toString("hex"), 16).toString();
const block = [...await blockDB.get(blockNumber)];

respond({ block: Block.deserialize(block) });
} catch (e) {
throwError("Invalid block hash.", 400);
}
}

Expand All @@ -106,14 +106,12 @@ function rpc(PORT, client, transactionHandler, keyPair, stateDB, blockDB, bhashD
if (typeof req.body.params !== "object" || typeof req.body.params.blockNumber !== "number") {
throwError("Invalid request.");
} else {
const currentBlockNumber = Math.max(...(await blockDB.keys().all()).map(key => parseInt(key)));

if (req.body.params.blockNumber <= 0 || req.body.params.blockNumber > currentBlockNumber) {
throwError("Invalid block number.", 400);
} else {
try {
const block = [...await blockDB.get( req.body.params.blockNumber.toString() )];

respond({ block: Block.deserialize(block) });
} catch (e) {
throwError("Invalid block number.", 400);
}
}

Expand All @@ -123,15 +121,13 @@ function rpc(PORT, client, transactionHandler, keyPair, stateDB, blockDB, bhashD
if (typeof req.body.params !== "object" || typeof req.body.params.hash !== "string") {
throwError("Invalid request.", 400);
} else {
const hashes = (await bhashDB.keys().all());

if (!hashes.find(hash => hash === req.body.params.hash)) {
throwError("Invalid block hash.", 400);
} else {
try {
const blockNumber = parseInt((await bhashDB.get(req.body.params.hash)).toString("hex"), 16).toString();
const block = Block.deserialize([...await blockDB.get(blockNumber)]);

respond({ count: block.transactions.length });
} catch (e) {
throwError("Invalid block hash.", 400);
}
}

Expand All @@ -141,74 +137,72 @@ function rpc(PORT, client, transactionHandler, keyPair, stateDB, blockDB, bhashD
if (typeof req.body.params !== "object" || typeof req.body.params.blockNumber !== "number") {
throwError("Invalid request.", 400);
} else {
const currentBlockNumber = Math.max(...(await blockDB.keys().all()).map(key => parseInt(key)));

if (req.body.params.blockNumber <= 0 || req.body.params.blockNumber > currentBlockNumber) {
throwError("Invalid block number.", 400);
} else {
try {
const block = Block.deserialize([...await blockDB.get( req.body.params.blockNumber.toString() )]);

respond({ count: block.transactions.length });
} catch (e) {
throwError("Invalid block number.", 400);
}
}

break;

case "get_balance":
if (
typeof req.body.params !== "object" ||
typeof req.body.params.address !== "string" ||
!(await stateDB.keys().all()).includes(req.body.params.address)
) {
if (typeof req.body.params !== "object" || typeof req.body.params.address !== "string") {
throwError("Invalid request.", 400);
} else {
const dataFromTarget = deserializeState(await stateDB.get(req.body.params.address)); // Fetch target's state object
const targetBalance = dataFromTarget.balance; // Get target's balance
try {
const dataFromTarget = deserializeState(await stateDB.get(req.body.params.address)); // Fetch target's state object
const targetBalance = dataFromTarget.balance; // Get target's balance

respond({ balance: targetBalance });
respond({ balance: targetBalance });
} catch (e) {
throwError("Can not find account.", 400);
}
}

break;

case "get_code":
if (
typeof req.body.params !== "object" ||
typeof req.body.params.codeHash !== "string" ||
!(await codeDB.keys().all()).includes(req.body.params.codeHash)
) {
if (typeof req.body.params !== "object" || typeof req.body.params.codeHash !== "string") {
throwError("Invalid request.", 400);
} else {
respond({ code: await codeDB.get(req.body.params.codeHash) });
try {
respond({ code: await codeDB.get(req.body.params.codeHash) });
} catch (e) {
throwError("Can not find code.", 400);
}
}

break;

case "get_codeHash":
if (
typeof req.body.params !== "object" ||
typeof req.body.params.address !== "string" ||
!(await stateDB.keys().all()).includes(req.body.params.address)
) {
if (typeof req.body.params !== "object" || typeof req.body.params.address !== "string") {
throwError("Invalid request.", 400);
} else {
const dataFromTarget = deserializeState(await stateDB.get(req.body.params.address)); // Fetch target's state object
try {
const dataFromTarget = deserializeState(await stateDB.get(req.body.params.address)); // Fetch target's state object

respond({ codeHash: dataFromTarget.codeHash });
respond({ codeHash: dataFromTarget.codeHash });
} catch (e) {
throwError("Can not find account.", 400);
}
}

break;

case "get_nonce":
if (
typeof req.body.params !== "object" ||
typeof req.body.params.address !== "string" ||
!(await stateDB.keys().all()).includes(req.body.params.address)
) {
if (typeof req.body.params !== "object" || typeof req.body.params.address !== "string") {
throwError("Invalid request.", 400);
} else {
const dataFromTarget = deserializeState(await stateDB.get(req.body.params.address)); // Fetch target's state object
try {
const dataFromTarget = deserializeState(await stateDB.get(req.body.params.address)); // Fetch target's state object

respond({ nonce: dataFromTarget.nonce });
respond({ nonce: dataFromTarget.nonce });
} catch (e) {
throwError("Can not find account.", 400);
}
}

break;
Expand All @@ -218,41 +212,48 @@ function rpc(PORT, client, transactionHandler, keyPair, stateDB, blockDB, bhashD
typeof req.body.params !== "object" ||
typeof req.body.params.address !== "string" ||
typeof req.body.params.key !== "string" ||
!(await stateDB.keys().all()).includes(req.body.params.address)
!fs.existsSync("./log/accountStore/" + req.body.params.address)

Check failure

Code scanning / CodeQL

Uncontrolled data used in path expression High

This path depends on a
user-provided value
.
) {
throwError("Invalid request.", 400);
} else {
const storageDB = new Level("./log/accountStore/" + contractInfo.address);
try {
const storageDB = new Level("./log/accountStore/" + req.body.params.address);

respond({ storage: await storageDB.get(req.body.params.key) });
respond({ storage: await storageDB.get(req.body.params.key) });

storageDB.close();
storageDB.close();
} catch (e) {
throwError("Can not find storage slot.", 400);
}
}

break;

case "get_storageKeys":
if (
typeof req.body.params.address !== "string" ||
!(await stateDB.keys().all()).includes(req.body.params.address)
!fs.existsSync("./log/accountStore/" + req.body.params.address)

Check failure

Code scanning / CodeQL

Uncontrolled data used in path expression High

This path depends on a
user-provided value
.
) {
throwError("Invalid request.", 400);
} else {
const storageDB = new Level("./log/accountStore/" + contractInfo.address);
const storageDB = new Level("./log/accountStore/" + req.body.params.address);

respond({ storage: await storageDB.keys().all() });

storageDB.close();
}

break;

case "get_storageRoot":
if (
typeof req.body.params.address !== "string" ||
!(await stateDB.keys().all()).includes(req.body.params.address)
) {
if (typeof req.body.params.address !== "string") {
throwError("Invalid request.", 400);
} else {
respond({ storageRoot: (deserializeState(await stateDB.get(contractInfo.address))).storageRoot });
try {
respond({ storageRoot: (deserializeState(await stateDB.get(req.body.params.address))).storageRoot });
} catch (e) {
throwError("Can not find account.", 400);
}
}

break;
Expand All @@ -265,18 +266,16 @@ function rpc(PORT, client, transactionHandler, keyPair, stateDB, blockDB, bhashD
) {
throwError("Invalid request.", 400);
} else {
const currentBlockNumber = Math.max(...(await blockDB.keys().all()).map(key => parseInt(key)));

if (req.body.params.blockNumber <= 0 || req.body.params.blockNumber > currentBlockNumber) {
throwError("Invalid block number.", 400);
} else {
try {
const block = Block.deserialize([...await blockDB.get( req.body.params.blockNumber.toString() )]);

if (req.body.params.index < 0 || req.body.params.index >= block.transactions.length) {
throwError("Invalid transaction index.", 400);
} else {
respond({ transaction: block.transactions[req.body.params.index] });
}
} catch (e) {
throwError("Invalid block number.", 400);
}
}

Expand All @@ -290,11 +289,7 @@ function rpc(PORT, client, transactionHandler, keyPair, stateDB, blockDB, bhashD
) {
throwError("Invalid request.", 400);
} else {
const hashes = (await bhashDB.keys().all());

if (!hashes.find(hash => hash === req.body.params.hash)) {
throwError("Invalid block hash.", 400);
} else {
try {
const blockNumber = parseInt((await bhashDB.get(req.body.params.hash)).toString("hex"), 16).toString();
const block = Block.deserialize([...await blockDB.get( blockNumber )]);

Expand All @@ -303,6 +298,8 @@ function rpc(PORT, client, transactionHandler, keyPair, stateDB, blockDB, bhashD
} else {
respond({ transaction: block.transactions[req.body.params.index] });
}
} catch (e) {
throwError("Invalid block hash.", 400);
}
}

Expand Down

0 comments on commit 9510cd6

Please sign in to comment.