Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Implement MirrorNodeContractQuery #2723

Merged
merged 29 commits into from
Jan 9, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
360798c
feat: add mirror node contract call and estimate queries
ivaylonikolov7 Dec 16, 2024
e7ed338
chore: add new classes to export.js
ivaylonikolov7 Dec 16, 2024
50d6c41
test: add integration and unit tests
ivaylonikolov7 Dec 16, 2024
fb15585
feat: add example for estimates from mirror node
ivaylonikolov7 Dec 16, 2024
a2a2eef
chore: add new package
ivaylonikolov7 Dec 16, 2024
d3d0864
refactor: change js doc return
ivaylonikolov7 Dec 19, 2024
a08329d
feat: get mirror network from client
ivaylonikolov7 Dec 20, 2024
1561164
fix: urls
0xivanov Dec 20, 2024
a3dafbd
fix: local node condition
ivaylonikolov7 Jan 6, 2025
ff46317
chore: formatting
ivaylonikolov7 Jan 6, 2025
040fc06
chore: formatting
ivaylonikolov7 Jan 6, 2025
0abde7c
fix: make islocalnode optional property
ivaylonikolov7 Jan 6, 2025
c72e210
fix: add missing props
ivaylonikolov7 Jan 6, 2025
3af5c92
fix: add missing props
ivaylonikolov7 Jan 6, 2025
c5802ec
feat: use only sender evm address
ivaylonikolov7 Jan 7, 2025
757351e
feat: remove old tests
ivaylonikolov7 Jan 7, 2025
90332e0
test: add new tests
ivaylonikolov7 Jan 7, 2025
2fefd69
test: remove redundant test
ivaylonikolov7 Jan 8, 2025
bf7f885
docs: add jsdoc comments
ivaylonikolov7 Jan 8, 2025
04dc8c4
Merge branch 'main' into feat/estimate-contract-call-gas
ivaylonikolov7 Jan 8, 2025
6079090
test: temporary skip non-working test
ivaylonikolov7 Jan 8, 2025
9a9716c
fix: revert testnet environment test
ivaylonikolov7 Jan 8, 2025
f9bc882
fix: skip test temporary until we find out why
ivaylonikolov7 Jan 8, 2025
b667457
fix: skip test temporary until we find out why
ivaylonikolov7 Jan 8, 2025
0fd4e44
fix: skip test temporary until we find out why
ivaylonikolov7 Jan 8, 2025
5647fb5
refactor: formatting
ivaylonikolov7 Jan 9, 2025
9e3dd56
chore: fix unskippable test error
ivaylonikolov7 Jan 9, 2025
c5b178f
refactor: remove casting of number
ivaylonikolov7 Jan 9, 2025
b8a942c
test: update
ivaylonikolov7 Jan 9, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
98 changes: 98 additions & 0 deletions examples/mirror-node-contract-queries-example.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,98 @@
import ABI from "@ethersproject/abi";
import {
PrivateKey,
MirrorNodeContractCallQuery,
MirrorNodeContractEstimateQuery,
ContractCallQuery,
Hbar,
ContractCreateTransaction,
Client,
ContractFunctionParameters,
AccountId,
FileCreateTransaction,
Long,
} from "@hashgraph/sdk";
import { setTimeout } from "timers/promises";
import dotenv from "dotenv";

dotenv.config();

const OPERATOR_ID = AccountId.fromString(process.env.OPERATOR_ID);
const OPERATOR_KEY = PrivateKey.fromStringED25519(process.env.OPERATOR_KEY);
const HEDERA_NETWORK = process.env.HEDERA_NETWORK || "testnet";

async function main() {
console.log("Mirror Node contract queries Example Start!");

// Step 0: Create and configure the SDK Client.
const client = Client.forName(HEDERA_NETWORK);
client.setOperator(OPERATOR_ID, OPERATOR_KEY);

const BYTECODE =
"60806040526040518060400160405280600581526020017f68656c6c6f0000000000000000000000000000000000000000000000000000008152505f90816100479190610293565b50348015610053575f80fd5b50610362565b5f81519050919050565b7f4e487b71000000000000000000000000000000000000000000000000000000005f52604160045260245ffd5b7f4e487b71000000000000000000000000000000000000000000000000000000005f52602260045260245ffd5b5f60028204905060018216806100d457607f821691505b6020821081036100e7576100e6610090565b5b50919050565b5f819050815f5260205f209050919050565b5f6020601f8301049050919050565b5f82821b905092915050565b5f600883026101497fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8261010e565b610153868361010e565b95508019841693508086168417925050509392505050565b5f819050919050565b5f819050919050565b5f61019761019261018d8461016b565b610174565b61016b565b9050919050565b5f819050919050565b6101b08361017d565b6101c46101bc8261019e565b84845461011a565b825550505050565b5f90565b6101d86101cc565b6101e38184846101a7565b505050565b5b81811015610206576101fb5f826101d0565b6001810190506101e9565b5050565b601f82111561024b5761021c816100ed565b610225846100ff565b81016020851015610234578190505b610248610240856100ff565b8301826101e8565b50505b505050565b5f82821c905092915050565b5f61026b5f1984600802610250565b1980831691505092915050565b5f610283838361025c565b9150826002028217905092915050565b61029c82610059565b67ffffffffffffffff8111156102b5576102b4610063565b5b6102bf82546100bd565b6102ca82828561020a565b5f60209050601f8311600181146102fb575f84156102e9578287015190505b6102f38582610278565b86555061035a565b601f198416610309866100ed565b5f5b828110156103305784890151825560018201915060208501945060208101905061030b565b8683101561034d5784890151610349601f89168261025c565b8355505b6001600288020188555050505b505050505050565b6102178061036f5f395ff3fe608060405234801561000f575f80fd5b5060043610610029575f3560e01c8063ce6d41de1461002d575b5f80fd5b61003561004b565b6040516100429190610164565b60405180910390f35b60605f8054610059906101b1565b80601f0160208091040260200160405190810160405280929190818152602001828054610085906101b1565b80156100d05780601f106100a7576101008083540402835291602001916100d0565b820191905f5260205f20905b8154815290600101906020018083116100b357829003601f168201915b5050505050905090565b5f81519050919050565b5f82825260208201905092915050565b5f5b838110156101115780820151818401526020810190506100f6565b5f8484015250505050565b5f601f19601f8301169050919050565b5f610136826100da565b61014081856100e4565b93506101508185602086016100f4565b6101598161011c565b840191505092915050565b5f6020820190508181035f83015261017c818461012c565b905092915050565b7f4e487b71000000000000000000000000000000000000000000000000000000005f52602260045260245ffd5b5f60028204905060018216806101c857607f821691505b6020821081036101db576101da610184565b5b5091905056fea26469706673582212202a86c27939bfab6d4a2c61ebbf096d8424e17e22dfdd42320f6e2654863581e964736f6c634300081a0033";

// Step 1: Create a new file with the contract bytecode
const { fileId } = await (
await new FileCreateTransaction().setContents(BYTECODE).execute(client)
).getReceipt(client);

const { contractId } = await (
await new ContractCreateTransaction()
.setBytecodeFileId(fileId)
.setGas(200000)
.execute(client)
).getReceipt(client);

console.log("Created new contract with ID: " + contractId.toString());

// Step 2: Wait for mirror node to import data
await setTimeout(5000);

const gasLimit = Long.fromNumber(30000);
const gasPrice = Long.fromNumber(1234);

// Step 3: Estimate the gas needed
const gas = await new MirrorNodeContractEstimateQuery()
.setContractId(contractId)
.setSender(client.operatorAccountId)
.setGasLimit(gasLimit)
.setGasPrice(gasPrice)
.setFunction("getMessage", new ContractFunctionParameters())
.execute(client);

// Step 4: Do the query against the consensus node using the estimated gas
const callQuery = new ContractCallQuery()
.setContractId(contractId)
.setGas(gas)
.setFunction("getMessage")
.setQueryPayment(new Hbar(1));

const result = await callQuery.execute(client);

// Step 5: Simulate the transaction for free, using the mirror node
const simulationResult = await new MirrorNodeContractCallQuery()
.setContractId(contractId)
.setSender(client.operatorAccountId)
.setGasLimit(Long.fromString("30000"))
.setBlockNumber(Long.fromString("10000"))
.setGasPrice(Long.fromString("1234"))
.setFunction("getMessage", new ContractFunctionParameters())
.execute(client);

// need to do this to remove the readonly property of the array
/**
* @type {string[]}
*/
const decodedSimulationResult = ABI.defaultAbiCoder
.decode(["string"], simulationResult)
.concat();

/**
* @type {string}
*/
const decodedStringMessage = decodedSimulationResult[0];
console.log("Simulation result: " + decodedStringMessage);
console.log("Contract call result: " + result.getString(0));
}

void main();
1 change: 1 addition & 0 deletions examples/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@
"node": ">=14.0.0"
},
"dependencies": {
"@ethersproject/abi": "^5.7.0",
"@hashgraph/sdk": "link:..",
"axios": "^1.6.4",
"dotenv": "^16.3.1"
Expand Down
2 changes: 2 additions & 0 deletions src/exports.js
Original file line number Diff line number Diff line change
Expand Up @@ -105,6 +105,8 @@ export { default as LiveHashAddTransaction } from "./account/LiveHashAddTransact
export { default as LiveHashDeleteTransaction } from "./account/LiveHashDeleteTransaction.js";
export { default as LiveHashQuery } from "./account/LiveHashQuery.js";
export { default as MaxQueryPaymentExceeded } from "./MaxQueryPaymentExceeded.js";
export { default as MirrorNodeContractCallQuery } from "./query/MirrorNodeContractCallQuery.js";
export { default as MirrorNodeContractEstimateQuery } from "./query/MirrorNodeContractEstimateQuery.js";
export { default as NodeAddressBook } from "./address_book/NodeAddressBook.js";
export { default as NetworkVersionInfo } from "./network/NetworkVersionInfo.js";
export { default as NetworkVersionInfoQuery } from "./network/NetworkVersionInfoQuery.js";
Expand Down
43 changes: 43 additions & 0 deletions src/query/MirrorNodeContractCallQuery.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
import MirrorNodeContractQuery from "./MirrorNodeContractQuery.js";

/**
* @typedef {import("../channel/Channel.js").default} Channel
* @typedef {import("../client/Client.js").default<*, *>} Client
*/
export default class MirrorNodeContractCallQuery extends MirrorNodeContractQuery {
/**
* @returns {Object}
*/
get JSONPayload() {
if (this.callData == null) {
throw new Error("Call data is required.");
}

return {
data: Buffer.from(this.callData).toString("hex"),
from: this.senderEvmAddress,
to: this.contractEvmAddress,
estimate: false,
gasPrice: this.gasPrice?.toString(),
gas: this.gasLimit?.toString(),
blockNumber: this.blockNumber?.toString(),
value: this.value?.toString(),
};
}

/**
* @param {Client} client
* @returns {Promise<string>}
*/
async execute(client) {
/**
* @type { { data: { result: string } } }
*/
const mirrorNodeRequest = await this.performMirrorNodeRequest(
client,
this.JSONPayload,
);

return mirrorNodeRequest.data.result;
}
}
43 changes: 43 additions & 0 deletions src/query/MirrorNodeContractEstimateQuery.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
import MirrorNodeContractQuery from "./MirrorNodeContractQuery.js";

/**
* @typedef {import("../channel/Channel.js").default} Channel
* @typedef {import("../client/Client.js").default<*, *>} Client
*/
export default class MirrorNodeContractCallQuery extends MirrorNodeContractQuery {
/**
* @returns {Object}
*/
get JSONPayload() {
if (this.callData == null) {
throw new Error("Call data is required.");
}

return {
data: Buffer.from(this.callData).toString("hex"),
from: this.senderEvmAddress,
to: this.contractEvmAddress,
estimate: true,
gasPrice: this.gasPrice?.toString(),
gas: this.gasLimit?.toString(),
blockNumber: this.blockNumber?.toString(),
value: this.value?.toString(),
};
}

/**
* @param {Client} client
* @returns {Promise<number>}
*/
async execute(client) {
/**
* @type { { data: { result: string } } }
*/
const mirrorNodeRequest = await this.performMirrorNodeRequest(
client,
this.JSONPayload,
);

return Number(mirrorNodeRequest.data.result);
}
}
Loading