Guides
On this page
Wallet Integration #
Learn how to properly integrate Metamask or Keplr with a dApp on ICTech.
:::tip Note: want to learn more about wallet integration beyond what’s covered here? Check out both the MetaMask Wallet documentation and Keplr Wallet documentation. :::
Implementation Checklist #
The integration implementation checklist for dApp developers consists of three categories:
[Frontend features]
[Transactions and wallet interactions]
[Client-side provider]
Frontend #
Make sure to create a wallet-connection button for Metamask and/or Keplr on the frontend of the application. For instance, consider the “Connect to a wallet” button on the interface of Diffusion Finance or the analagous button on the interface of [ICTech wap]
Transactions #
Developers enabling transactions on their dApp have to [determine wallet type]of the user, [create the transaction], [request signatures] from the corresponding wallet, and finally [broadcast the transaction]to the network.
Determining Wallet Type #
Developers should determine whether users are using Keplr or MetaMask. Whether MetaMask or Keplr is installed on the user device can be determined by checking the corresponding window.ethereum
or window.keplr
value.
For MetaMask:
await window.ethereum.enable(chainId);
For Keplr:
await window.keplr.enable(chainId);
If either window.ethereum
or window.keplr
returns undefined
after document.load
, then MetaMask (or, correspondingly, Keplr) is not installed. There are several ways to wait for the load event to check the status: for instance, developers can register functions to window.onload
, or they can track the document’s ready state through the document event listener.
After the user’s wallet type has been determined, developers can proceed with creating, signing, and sending transactions.
Create the Transaction #
:::tip Note: The example below uses the ICTech Testnet chainID
. For more info, check the ICPlaza Chain IDs reference document here. :::
Developers can create MsgSend
transactions using the [ICTechjs] library.
Sign and Broadcast the Transaction #
:::tip Note: The example below uses an ICTech Testnet [RPC node]. :::
After creating the transaction, developers need to send the payload to the appropriate wallet to be signed (msg.signDirect
is the transaction in Keplr format, and msg.eipToSign
is the EIP712
data to sign with MetaMask).
With the signature, we add a Web3Extension to the transaction and broadcast it to the ICTech node.
Connections #
For Ethereum RPC, ICTech gRPC, and/or REST queries, dApp developers should implement providers client-side, and store RPC details in the environment variable as secrets.
DApp Store Contract Registration #
This guide explains how to register your smart contract in the ICTech dApp store, and start earning income every time a user interacts with your smart contract. {synopsis}
The ICTech dApp store is a revenue-per-transaction model, which allows developers to get paid for deploying their decentralized application (dApps) on ICTech. Developers generate revenue every time a user interacts with their dApp in the dApp store, providing them a steady income. Users can discover new applications in the dApp store and pay for the transaction fees that finance the dApp’s revenue. This value-reward exchange of dApp services for transaction fees is implemented by the [x/feesplit module].
Requirements #
Address of a deployed smart contract.
Capability to sign transactions with the address that deployed the contract. If your smart contract was deployed by a contract using a factory pattern, then the signing capaility is required for the address that deployed the factory.
The nonce of the contract deployment transaction. You can query the nonce, e.g. using the
eth_getTransactionByHash
JSON-RPC endpoint.Withdrawer address, in case you wish to receive your earnings at a specified address.
::: warning IMPORTANT: If your contract is part of a development project, please ensure that the deployer of the contract (or the factory that deploys the contract) is an account that is owned by that project. This avoids the situation of a malicious individual/employee deployer (including former contributors) who leaves your project and could later change the withdrawal address unilaterally. :::
Register Contract #
To add your contract in the ICTech dApp Store, you need to register a feesplit
for that contract. The feesplit
includes the details for receiving a cut of the transaction fees, which users pay for interacting with your smart contract. Every time a user submits a transaction to your registered smart contract, a part of the transaction fees (50% by default) is transferred to the withdrawer address specified in the feesplit
. If the withdrawer is not specified, the transaction fees are sent to the contract deployer.
You can register a contract by signing a transaction with the address that originally deployed the contract. You can use the following CLI command, where
$NONCE
is the nonce of transaction that deployed the contract (e.g.0
),$CONTRACT
is the hex address of the deployed contract (e.g0x5f6659B6F712c729c46786bA9562eC50907c67CF
) and(optional)
$WITHDRAWER
is the bech32 address of the address to receive the transaction fees (e.g.ICPlaza1keyy3teyq7t7kuxgeek3v65n0j27k20v2ugysf
):
After your transaction is submitted successfully, you can query your feesplit
with :
Congrats ☄️☄️☄️ Now that you’ve registered a feesplit for your contract, it is part of the ICPlaza dApp store and you will receive a cut of the transaction fees every time a user interacts with your contract. If you wondering how large your cut is, have a look at the [feesplit parameter DeveloperShares
], which is controlled through governance. You can query the parameters using our [OpenAPI documentation].
Deployed Factory Pattern #
You can also register a contract which has been deployed by a smart contract instead of an [EOA]. In this case, you need to provide a sequence of nonces that proves the trace from an original deployer who deployed the factory to the contract that is being registered.
Example DeployerEOA
-> FactoryA
-> FactoryB
-> MyContract
: DeployerEOA
deploys a FactoryA
smart contract with nonce 5
. Then, DeployerEOA
sends a transaction to FactoryA
through which a FactoryB
smart contract is created. If we assume FactoryB
is the second contract created by FactoryA
, then FactoryA
’s nonce is 2
. Then, DeployerEOA
sends a transaction to the FactoryB
contract, through which MyContract
is created. If this is the first contract created by FactoryB - the nonce is 1
. To be able to verify that DeployerEOA
can register MyContract
, we need to provide the following nonces: [5, 2, 1]
.
Update Contract #
Registered contracts can also be updated. To update the withdrawer address of your feesplit
, use the following CLI command:
If the specified withdrawer is the same address as the deployer, then the feesplit is updated with an empty withdrawer address, so that all transaction fees are sent to the deployer address.
Cancel Contract #
Feesplits can also be canceled. In order to stop receiving transaction fees for interaction with your contract, use the following CLI command:
ERC-20 Registration #
Learn how to register interoperable ERC-20s through ICPlaza Governance.
:::tip Note: Not sure what the difference between ICTech Coin and ERC-20 Registration is? You’re in the right place if an ERC-20 contract corresponding to your token already exists, and you want to add functionality to convert the ERC-20 token to a native ICTech Coin denomination. If you instead want to add functionality to convert a native ICTech Coin to an ERC-20 token representation, then check out [ICTech Coin Registration].
Still confused? Learn more about the differences [here]. :::
The ERC-20 Module (also known as x/erc20
) allows users to instantly convert ERC-20 tokens into native ICTech Coins, and vice versa. This allows users to exchange assets interchangeably in two entirely different layers, the EVM and ICTech.
Application-wise, the ERC-20 module allows DeFi protocols to seamlessly integrate with ICPlaza and the ICTech ecosystem. Using the module, developers can build smart contracts on ICTech and use the generated ERC-20 tokens for other applications on the ICPlaza ecosystem, such as:
earning $ICT staking rewards
taking part in governance proposals by voting with $ICT
Registering an interoperable ERC-20 means registering a new mapping between an existing ERC-20 token contract and a ICTech Coin denomination, also known as a Token Pair. Token Pairs enable users to convert ERC-20 tokens into their native ICTech Coin representation (and vice versa), and can only be created via a governance proposal.
More information can be found in [this blog post], which introduced the ERC-20 Module on ICTech.
To register an ERC-20, consider the following stages:
[Drafting the ERC-20 Proposal]
[Submitting the ERC-20 Proposal]
[The On-Chain ERC-20 Proposal]
Drafting the ERC-20 Proposal #
The following topics must be addressed when drafting an ERC-20 Proposal:
Provide the profile of the person(s)/entity making the proposal.
Who are you? What is your involvement in ICTech and/or other blockchain networks? If you are working with a team, who are the team members involved and what is their relevant experience? What is the mission statement of your organization or business? Do you have a website? Showcase some work you’ve done and some proof of who you are.
Promote understanding of the ERC-20 Module.
Make sure to mention the original [blog post] that introduced the ERC-20 Module, along with a brief explanation of what the ERC-20 Module does. It’s also a good idea to link the [ERC-20 Module documentation]!
Describe how ERC-20 Module changes will be made.
Give a breakdown of the proposal’s payload, and explain in layman terms what the proposal will do if it passes. Detail precautions taken during contract and proposal formulation, if applicable (including consultations made prior to proposal creation, how contracts were tested, and any third-party reviews). Finally, mention the risks involved in the proposal, depending on the direction of IBC Coin and ERC-20.
Document adherence to ERC-20 Contract expectations.
Ensure that the ERC-20 contracts are verified (either through the [EVM explorer] or via Sourcify), and that the contracts are deployed open-source. Security-wise, the following are required: - the contracts use the main libraries for ERC-20s (eg. OpenZeppelin, dapp.tools) - the contracts do not extend the
IERC20.sol
interface through a malicious implementation - the transfer logic is not modified (i.e. transfer logic is not directly manipulated) - no maliciousApprove
events can directly manipulate users’ balance through a delayed granted allowance Take note of the above in your proposal description!
Remember to provide links to the relevant [Commonwealth ICTech community] discussions concerning your proposal, as well as the [proposal on testnet].
Submitting the ERC-20 Proposal #
After the drafting process, the ERC-20 Proposal can be submitted.
Formatting the Proposal’s Text #
The ideal format for a proposal is as a Markdown file (ie. .md
) in a Github repo or HackMd. Markdown is a simple and accessible format for writing plain text files that is easy to
learn. See the Github Markdown Guide for details on writing markdown files.
Submit the Proposal to Testnet #
To submit the proposal to testnet through the command line with [ICPlazad
], use the following command with register-erc20
:
However, note that if the CLI is used to create a proposal, and description
is set using a flag, the text will be escaped which may have undesired effects. If the proposal creator is using markdown or line breaks it’s recommended to put the proposal text into a json file and include that file as part of the CLI proposal, as opposed to individual fields in flags. The process of creating a json file containing the proposal can be found here, and the CLI command for submitting the file is below:
You may want to submit your proposal to the testnet chain before the mainnet for a number of reasons, such as wanting to see what the proposal description will look like, to share what the proposal will look like in advance with stakeholders, and to signal that your proposal is about to go live on the mainnet.
Submitting your proposal to the testnet increases the likelihood of engagement and the possibility that you will be alerted to a flaw before deploying your proposal to mainnet.
The On-Chain ERC-20 Proposal #
A majority of the voting community should probably be aware of the proposal and have considered it before the proposal goes live on-chain. If you’re taking a conservative approach, you should have reasonable confidence that your proposal will pass before risking deposit contributions by submitting the proposal. Make revisions to your draft proposal after each stage of engagement.
The Deposit Period #
The deposit period currently lasts 14 days. If you submitted your transaction with the minimum deposit (64 ICTech), your proposal will immediately enter the voting period. If you didn’t submit the minimum deposit amount (currently 64 ICTech), then this may be an opportunity for others to show their support by contributing (and risking) their ICTech as a bond for your proposal. You can request contributions openly and also contact stakeholders directly (particularly stakeholders who are enthusiastic about your proposal). Remember that each contributor is risking their funds, and you can read more about the conditions for burning deposits here.
This is a stage where proposals may begin to get broader attention. Most popular explorers currently display proposals that are in the deposit period, but due to proposal spamming, this may change.
A large cross-section of the blockchain/cryptocurrency community exists on Twitter. Having your proposal in the deposit period is a good time to engage the ICTech community to prepare validators to vote and ICTech-holders that are staking.
The Voting Period #
At this point you’ll want to track which validator has voted and which has not. You’ll want to re-engage directly with top stake-holders, ie. the highest-ranking validator operators, to ensure that:
they are aware of your proposal;
they can ask you any questions about your proposal; and
they are prepared to vote.
Remember that any voter may change their vote at any time before the voting period ends. That historically doesn’t happen often, but there may be an opportunity to convince a voter to change their vote. The biggest risk is that stakeholders won’t vote at all (for a number of reasons). Validator operators tend to need multiple reminders to vote. How you choose to contact validator operators, how often, and what you say is up to you–remember that no validator is obligated to vote, and that operators are likely occupied by competing demands for their attention. Take care not to stress any potential relationship with validator operators.
Query Balances #
Learn how to query balances of IBC ICTech Coins and ERC-20s on ICTech.
This guide will cover the following query methods:
:::warning Note: In this document, the command line is used to interact with endpoints. For dApp developers, using libraries such as [cosmjs] and [ICTechjs] is recommended instead. :::
ICTech & Tendermint RPC #
Upon installation and [configuration] of the ICTech Daemon, developers can query account balances using ICTech with the following CLI command:
where:
$
ICTechADDRESS
is the ICTech address with balances of interest (eg.ICPlaza1...
).(optional)
$COUNTTOTAL
counts the total number of records in all balances to query for.(optional)
$HEIGHT
is the specific height to query state at (can error if node is pruning state).(optional)
$OUTPUT
is the output format (eg.text
).(optional if running local node)
$NODE
is the Tendermint RPC node information is requested from (eg.https://tendermint.bd.
ICTech,xyz:26657
).
Details of non-native currencies (ie. not ICTech) can be queried with the following CLI command:
where $DENOM
is the denomination of the coin (eg. ibc/ED07A3391A1...
).
JSON-RPC #
Developers can query account balances of aICPlaza
using the [eth_getBalance
]JSON-RPC method in conjunction with curl
:
where:
$ETHADDRESS
is the Etherum hex-address the balance is to be queried from. Note that ICPlaza addresses (those beginning withICPlaza1...
) can be converte.d to Ethereum addresses using libraries such as [ICPlazajs].$BLOCK
is the block number or block hash (eg."0x0"
). The reasoning for this parameter is due to EIP-1898.(optional if running local node)
$NODE
is the JSON-RPC node information is requested from (eg.https://eth.bd.ICPlaza.org:8545
).
Developers can also query account balances of x/erc20
-module registered coins using the [eth_call
] JSON-RPC method in conjunction with curl
:
where:
$SENDERCONTRACTADDRESS
is the Ethereum hex-address this smart contract call is sent from.$ERCCONTRACTADDRESS
is the Ethereum hex-address of the ERC-20 contract corresponding to the coin denomination being queried.$DATA
is the hash of thebalanceof
method signature and encoded parameters.balanceOf
is a required method in every ERC-20 contract, and the encoded parameter is the address which is having its balance queried. For additional information, see the Ethereum Contract ABI.$BLOCK
is the block number or block hash (eg."0x0"
). The reasoning for this parameter is due to EIP-1898.(optional if running local node)
$NODE
is the JSON-RPC node information is requested from (eg.https://eth.bd.
ICTech.xyz:8545
).
gRPC #
Developers can use grpcurl
with the AllBalances
endpoint to query account balance by address for all denominations:
where:
$
ICTechADDRESS
is the ICTech address with balances of interest (eg."ICPlaza1..."
).$NODE
is the ICTech gRPC node information is requested from (eg.https://grpc.bd.ICPlaza.org:9090
).(optional)
$OUTPUT
is the output format (eg.plaintext
).
State can also be queried using gRPC within a Go program. The idea is to create a gRPC connection, then use the Protobuf-generated client code to query the gRPC server.
:::tip Note: The following tools will be useful when using gRPC:
[ICTech Swagger API] a comprehensive description of all gRPC endpoints
ICTech SDK Go API &ICTech Go API: packages to implement queries in Go scripts
:::
Trace Transactions #
Learn how to trace transactions and blocks on ICTech. {synopsis}
This guide will cover the following topics:
Basic EVM Tracing with JS #
Tracing a transaction means requesting an ICTech node to re-execute the desired transaction with varying degrees of data collection.
Tracing Prerequisites #
Re-executing a transaction has a few prerequisites to be met. All historical state accessed by the transaction must be available, including:
Balance, nonce, bytecode, and storage of both the recipient as well as all internally invoked contracts
Block metadata referenced during execution of the outer as well as all internally created transactions
Intermediate state generated by all preceding transactions contained in the same block as well as the one being traced
This means there are limits on the transactions that can be traced and imported based on the synchronization and pruning configuration of a node.
Archive nodes: retain all historical data back to genesis, can trace arbitrary transactions at any point in the history of the chain.
Fully synced nodes: transactions within a recent range (depending on how much history is stored) are accessible.
Light synced nodes: these nodes retrieve data on demand, so in theory they can trace transactions for which all required historical state is readily available in the network (however, data availability cannot be reasonably assumed).
Basic Traces #
The simplest type of transaction trace that Geth can generate are raw EVM opcode traces. For every VM instruction the transaction executes, a structured log entry is emitted, contained all contextual metadata deemed useful. This includes:
program counter
opcode name & cost
remaining gas
execution depth
occurred errors
as well as (optionally) the execution stack, execution memory, and contract storage.
The entire output of a raw EVM opcode trace is a JSON object having a few metadata fields: consumed gas, failure status, return value, and a list of opcode entries:
An example log for a single opcode entry has the following format:
Limits of Basic Traces #
Although raw opcode traces generated above are useful, having an individual log entry for every single opcode is too low level for most use cases, and will require developers to create additional tools to post-process the traces. Additionally, a single opcode trace can easily be hundreds of megabytes, making them very resource intensive to extract from the node and process extenally.
To avoid these issues, Geth supports running custom JavaScript traces within the ICTech (or any EVM-compatible) node, which have full access to the EVM stack, memory, and contract storage. This means developers only have to gather data that they actually need, and do any processing at the source.
Filtered EVM Tracing with JS #
Basic traces can include the complete status of the EVM at every point in the transaction’s execution, which is huge space-wise. Usually, developers are only interested in a small subset of this information, which can be obtained by specifying a JavaScript filter.
Running a Simple Trace #
:::warning Note: debug.traceTransaction
must be invoked from within the Geth console, although it can be invoked from outside the node using JSON-RPC (eg. using Curl), as seen in the [following section]. If developers want to use debug.traceTransaction
as it is used here, maintainence of a node is required, so see [this document]. :::
Create a file,
filterTrace_1.js
, with this content:
Run the JavaScript console.
Get a hash of a recent transaction.
Run this command to run the script:
Run the tracer from the script:
The bottom of the output looks similar to:
This output isn’t very readable. Run this command to get a more readable output with each string on its own line:
The JSON.stringify function’s documentation is here. If we just return the output, we get for newlines, which is why we need to use console.log
.
How Does it Work? #
We call the same debug.traceTransaction
function used for [basic traces], but with a new parameter, tracer
. This parameter is a string, which is the JavaScript object we use. In the case of the trace above, it is:
This object has to have three member functions:
step
, called for each opcodefault
, called if there is a problem in the executionresult
, called to produce the results that are returned bydebug.traceTransaction
after the execution is done
It can have additional members. In this case, we use retVal
to store the list of strings that we’ll return in result
.
The step
function here adds to retVal
: the program counter, and the name of the opcode there. Then, in result
, we return this list to be sent to the caller.
Actual Filtering #
For actual filtered tracing, we need an if
statement to only log revelant information. For example, if we are interested in the transaction’s interaction with storage, we might use:
The step
function here looks at the opcode number of the op, and only pushes an entry if the opcode is SLOAD
or SSTORE
(here is a list of all EVM opcodes and their corresponding numbers). We could have used log.op.toString
instead, but it is faster to compare numbers rather than strings.
The output looks similar to this:
Stack Information #
The trace above tells us the program counter and whether the program read from storage or wrote to it. To know more, you can use the log.stack.peek
function to peek into the stack. log.stack.peek(0)
is the stack top, log.stack.peek(1)
is the entry beow it, etc. The values returned by log.stack.peek
are Go big.int
objects. By default they are converted to JavaScript floating point numbers, so you need toString(16) to get them as hexadecimals, which is how we normally represent 256-bit values such as storage cells and their content.
There are several other facets of filtered EVM tracing, including:
determining operation results
dealing with calls between contracts
accessing memory
using the
db
parameter to know the state of the chain at the time of execution
This information is covered in this reference.
JSON-RPC debug_trace*
Endpoints #
debug_trace*
Endpoints #ICTech supports the following debug_trace*
JSON-RPC Methods, which follow Geth’s debug API guidelines.
debug_traceTransaction
#
debug_traceTransaction
#The traceTransaction
debugging method will attempt to run the transaction in the exact same manner as it was executed on the network. It will replay any transaction that may have been executed prior to this one, before it will finally attempt to execute the transaction that corresponds to the given hash.
Parameters:
trace configuration
debug_traceBlockByNumber
#
debug_traceBlockByNumber
#The traceBlockByNumber
endpoint accepts a block number, and will replay the block that is already present in the database.
Parameters:
trace configuration
debug_traceBlockByHash
#
debug_traceBlockByHash
#The traceBlockByNumber
endpoint accepts a block hash, and will replay the block that is already present in the database.
Parameters:
trace configuration
Additional Information #
See the sources below for details not covered above:
[JSON-RPC Methods on ICPlaza]
Last updated