Janus is a web3 proxy adapter that can be used as a web3 provider to interact with Qtum. It supports HTTP(s) and websockets and the current version enables self hosting of keys.
Builds successful with Node v18.19.0 after deleting yarn.lock and node_modules NB!Make sure to align the networks in the yml config file for docker.
nano /janus/docker/quick_start/docker-compose.mainnet.yml
Set the exact same network interface under all mentions of networks:
- Sample docker-compose.mainnet.yml
version: "3.3"
networks:
docker_common_network:
external: true
services:
janus_mainnet:
env_file:
- ../.env
image: ipetrov22/janus:latest
container_name: janus_mainnet
build:
context: ../../
cache_from:
- golang:1.18-alpine
ports:
- "23890:23890"
environment:
- QTUM_RPC=${RPC_URL}
- COMPOSE_PROJECT_NAME=mainnet
volumes:
- ../../https:/https
- ../../logs:/logs
command: --bind 0.0.0.0 --port 23890 --ignoreTransactions --dev --https-key /https/key.pem --https-cert /https/cert.pem
networks:
- docker_common_network
After the network is configured, run the Adapter and check the logs of the docker to ensure it is connecting with the RPC node and pulling data.
git clone https://github.com/Hydra-Chain/janus.git
cd janus
Create .env
file in janus/docker
folder:
RPC_URL=""
Run in janus
folder:
make quick-start-mainnet
Runs on ports 23890
janus
running on port 23890
- Quick start
- Public instances
- Requirements
- Installation
- How to use Janus as a Web3 provider
- How to add Janus to Metamask
- Truffle support
- Ethers support
- Supported ETH methods
- Websocket ETH methods
- Janus methods
- Development methods
- Health checks
- Deploying and Interacting with a contract using RPC calls
- Differences between EVM chains
Mainnet: https://janus.qiswap.com/api/
Testnet: https://testnet-janus.qiswap.com/api/
Regtest: run it locally with make quick-start-regtest
If you need to use eth_sendTransaction, you are going to have to run your own instance pointing to your own QTUM instance
See (Beta) QTUM ethers-js library to generate transactions in the browser so you can use public instances
See Differences between EVM chains below
- Golang
- Docker
- linux commands:
make
,curl
$ sudo apt install make git golang docker-compose
# Configure GOPATH if not configured
$ export GOPATH=`go env GOPATH`
$ mkdir -p $GOPATH/src/github.com/qtumproject && \
cd $GOPATH/src/github.com/qtumproject && \
git clone https://github.com/qtumproject/janus
$ cd $GOPATH/src/github.com/qtumproject/janus
# Generate self-signed SSL cert (optional)
# If you do this step, Janus will respond in SSL
# otherwise, Janus will respond unencrypted
$ make docker-configure-https
# Pick a network to quick-start with
$ make quick-start-regtest
$ make quick-start-testnet
$ make quick-start-mainnet
This will build the docker image for the local version of Janus as well as spin up two containers:
-
One named
janus
running on port 23889 -
Another one named
qtum
running on port 3889
make quick-start
will also fund the tests accounts with QTUM in order for you to start testing and developing locally. Additionally, if you need or want to make changes and or additions to Janus, but don't want to go through the hassle of rebuilding the container, you can run the following command at the project root level:
$ make run-janus
# For https
$ make docker-configure-https && make run-janus-https
Which will run the most current local version of Janus on port 23888, but without rebuilding the image or the local docker container.
Note that Janus will use the hex address for the test base58 Qtum addresses that belong the the local qtum node, for example:
- qUbxboqjBRp96j3La8D1RYkyqx5uQbJPoW (hex 0x7926223070547d2d15b2ef5e7383e541c338ffe9 )
- qLn9vqbr2Gx3TsVR9QyTVB5mrMoh4x43Uf (hex 0x2352be3db3177f0a07efbe6da5857615b8c9901d )
SSL keys and certificates go inside the https folder (mounted at /https
in the container) and use --https-key
and --https-cert
parameters. If the specified files do not exist, it will fall back to http.
To generate self-signed certificates with docker for local development the following script will generate SSL certificates and drop them into the https folder
$ make docker-configure-https
Once Janus is successfully running, all one has to do is point your desired framework to Janus in order to use it as your web3 provider. Lets say you want to use truffle for example, in this case all you have to do is go to your truffle-config.js file and add janus as a network:
module.exports = {
networks: {
janus: {
host: "127.0.0.1",
port: 23889,
network_id: "*",
gasPrice: "0x5d21dba000"
},
...
},
...
}
Getting Janus to work with Metamask requires two things
- Configuring Metamask to point to Janus
- Locally signing transactions with a Metamask fork
Hosting your own Janus and blockchain instance works similarly to geth and is supported
Client side transaction signing is supported with hdwallet-provider underneath it uses qtum-ethers-wrapper to construct raw transactions
See truffle unbox qtumproject/react-box for an example truffle-config file
Ethers is supported, use qtum-ethers-wrapper
- web3_clientVersion
- web3_sha3
- net_version
- net_listening
- net_peerCount
- eth_protocolVersion
- eth_chainId
- eth_mining
- eth_hashrate
- eth_gasPrice
- eth_accounts
- eth_blockNumber
- eth_getBalance
- eth_getStorageAt
- eth_getTransactionCount
- eth_getCode
- eth_sign
- eth_signTransaction
- eth_sendTransaction
- eth_sendRawTransaction
- eth_call
- eth_estimateGas
- eth_getBlockByHash
- eth_getBlockByNumber
- eth_getTransactionByHash
- eth_getTransactionByBlockHashAndIndex
- eth_getTransactionByBlockNumberAndIndex
- eth_getTransactionReceipt
- eth_getUncleByBlockHashAndIndex
- eth_getCompilers
- eth_newFilter
- eth_newBlockFilter
- eth_uninstallFilter
- eth_getFilterChanges
- eth_getFilterLogs
- eth_getLogs
- (All the above methods)
- eth_subscribe (only 'logs' for now)
- eth_unsubscribe
Use these to speed up development, but don't rely on them in your dapp
- dev_gethexaddress Convert Qtum base58 address to hex
- dev_fromhexaddress Convert from hex to Qtum base58 address for the connected network (strip 0x prefix from address when calling this)
- dev_generatetoaddress Mines blocks in regtest (accepts hex/base58 addresses - keep in mind that to use these coins, you must mine 2000 blocks)
There are two health check endpoints, GET /live
and GET /ready
they return 200 or 503 depending on health (if they can connect to qtumd)
Assume that you have a contract like this:
pragma solidity ^0.4.18;
contract SimpleStore {
constructor(uint _value) public {
value = _value;
}
function set(uint newValue) public {
value = newValue;
}
function get() public constant returns (uint) {
return value;
}
uint value;
}
so that the bytecode is
solc --optimize --bin contracts/SimpleStore.sol
======= contracts/SimpleStore.sol:SimpleStore =======
Binary:
608060405234801561001057600080fd5b506040516020806100f2833981016040525160005560bf806100336000396000f30060806040526004361060485763ffffffff7c010000000000000000000000000000000000000000000000000000000060003504166360fe47b18114604d5780636d4ce63c146064575b600080fd5b348015605857600080fd5b5060626004356088565b005b348015606f57600080fd5b506076608d565b60408051918252519081900360200190f35b600055565b600054905600a165627a7a7230582049a087087e1fc6da0b68ca259d45a2e369efcbb50e93f9b7fa3e198de6402b810029
constructor parameters is 0000000000000000000000000000000000000000000000000000000000000001
$ curl --header 'Content-Type: application/json' --data \
'{"id":"10","jsonrpc":"2.0","method":"eth_sendTransaction","params":[{"from":"0x7926223070547d2d15b2ef5e7383e541c338ffe9","gas":"0x6691b7","gasPrice":"0x5d21dba000","data":"0x608060405234801561001057600080fd5b506040516020806100f2833981016040525160005560bf806100336000396000f30060806040526004361060485763ffffffff7c010000000000000000000000000000000000000000000000000000000060003504166360fe47b18114604d5780636d4ce63c146064575b600080fd5b348015605857600080fd5b5060626004356088565b005b348015606f57600080fd5b506076608d565b60408051918252519081900360200190f35b600055565b600054905600a165627a7a7230582049a087087e1fc6da0b68ca259d45a2e369efcbb50e93f9b7fa3e198de6402b8100290000000000000000000000000000000000000000000000000000000000000001"}]}' \
'http://localhost:23889'
{
"jsonrpc": "2.0",
"result": "0xa85cacc6143004139fc68808744ea6125ae984454e0ffa6072ac2f2debb0c2e6",
"id": "10"
}
$ curl --header 'Content-Type: application/json' --data \
'{"id":"10","jsonrpc":"2.0","method":"eth_getTransactionByHash","params":["0xa85cacc6143004139fc68808744ea6125ae984454e0ffa6072ac2f2debb0c2e6"]}' \
'localhost:23889'
{
"jsonrpc":"2.0",
"result": {
"blockHash":"0x1e64595e724ea5161c0597d327072074940f519a6fb285ae60e73a4c996b47a4",
"blockNumber":"0xc9b5",
"transactionIndex":"0x5",
"hash":"0xa85cacc6143004139fc68808744ea6125ae984454e0ffa6072ac2f2debb0c2e6",
"nonce":"0x0",
"value":"0x0",
"input":"0x00",
"from":"0x7926223070547d2d15b2ef5e7383e541c338ffe9",
"to":"",
"gas":"0x363639316237",
"gasPrice":"0x5d21dba000"
},
"id":"10"
}
$ curl --header 'Content-Type: application/json' --data \
'{"id":"10","jsonrpc":"2.0","method":"eth_getTransactionReceipt","params":["0x6da39dc909debf70a536bbc108e2218fd7bce23305ddc00284075df5dfccc21b"]}' \
'localhost:23889'
{
"jsonrpc": "2.0",
"result": {
"transactionHash": "0xa85cacc6143004139fc68808744ea6125ae984454e0ffa6072ac2f2debb0c2e6",
"transactionIndex": "0x5",
"blockHash": "0x1e64595e724ea5161c0597d327072074940f519a6fb285ae60e73a4c996b47a4",
"from":"0x7926223070547d2d15b2ef5e7383e541c338ffe9"
"blockNumber": "0xc9b5",
"cumulativeGasUsed": "0x8c235",
"gasUsed": "0x1c071",
"contractAddress": "0x1286595f8683ae074bc026cf0e587177b36842e2",
"logs": [],
"logsBloom": "0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000",
"status": "0x1"
},
"id": "10"
}
the ABI code of set method with param '["2"]' is 60fe47b10000000000000000000000000000000000000000000000000000000000000002
$ curl --header 'Content-Type: application/json' --data \
'{"id":"10","jsonrpc":"2.0","method":"eth_sendTransaction","params":[{"from":"0x7926223070547d2d15b2ef5e7383e541c338ffe9","gas":"0x6691b7","gasPrice":"0x5d21dba000","to":"0x1286595f8683ae074bc026cf0e587177b36842e2","data":"60fe47b10000000000000000000000000000000000000000000000000000000000000002"}]}' \
'localhost:23889'
{
"jsonrpc": "2.0",
"result": "0x51a286c3bc68335274b9fd255e3988918a999608e305475105385f7ccf838339",
"id": "10"
}
get method's ABI code is 6d4ce63c
$ curl --header 'Content-Type: application/json' --data \
'{"id":"10","jsonrpc":"2.0","method":"eth_call","params":[{"from":"0x7926223070547d2d15b2ef5e7383e541c338ffe9","gas":"0x6691b7","gasPrice":"0x5d21dba000","to":"0x1286595f8683ae074bc026cf0e587177b36842e2","data":"6d4ce63c"},"latest"]}' \
'localhost:23889'
{
"jsonrpc": "2.0",
"result": "0x0000000000000000000000000000000000000000000000000000000000000002",
"id": "10"
}
- Transparently translate eth_sendRawTransaction from an EVM transaction to a QTUM transaction if the same key is hosted
- Transparently serve blocks by their Ethereum block hash
- Send all QTUM support via eth_sendTransaction
- For eth_subscribe only the 'logs' type is supported at the moment