Go to file
Bryan Stitt 3cd463127b another todo 2022-08-06 05:33:32 +00:00
.cargo keep RUSTFLAGS in one place 2022-05-17 16:36:42 +00:00
.vscode minimal app served its purpose 2022-05-20 02:01:02 +00:00
bin less verbose 2022-07-25 19:04:11 +00:00
config make it work 2022-08-06 05:26:43 +00:00
entities back to ids 2022-08-06 00:07:12 +00:00
linkedhashmap store blocks in Arcs 2022-07-22 05:11:26 +00:00
migration auto increment 2022-08-06 04:39:33 +00:00
redis-cell-client make it work 2022-08-06 05:26:43 +00:00
redis-cell-server flatten transaction sending (untested) 2022-05-28 04:26:24 +00:00
web3_proxy make it work 2022-08-06 05:26:43 +00:00
wrk move data files 2022-05-06 01:40:43 +00:00
.dockerignore start adding redis-cell for rate limits 2022-05-21 20:40:22 +00:00
.env actually add the files 2022-07-26 00:38:32 +00:00
.gitignore setup database and stub migrations 2022-07-26 04:55:07 +00:00
Cargo.lock use Uuid and bools in our generated types 2022-08-05 19:47:50 +00:00
Cargo.toml fix directory structure 2022-08-05 19:22:23 +00:00
Dockerfile keep RUSTFLAGS in one place 2022-05-17 16:36:42 +00:00
LICENSE add LICENSE 2022-03-04 19:56:05 -08:00
README.md use Uuid and bools in our generated types 2022-08-05 19:47:50 +00:00
TODO.md another todo 2022-08-06 05:33:32 +00:00
docker-compose.common.yml log less 2022-06-03 22:32:48 +00:00
docker-compose.prod.yml first pass at a schema 2022-07-26 00:38:00 +00:00
docker-compose.yml setup database and stub migrations 2022-07-26 04:55:07 +00:00

web3-proxy

Web3-proxy is a fast caching and load balancing proxy for web3 (Ethereum or similar) JsonRPC servers.

Under construction! This code is under active development. The basics seem to work, but theres lots of tests and features to write still.

Signed transactions (eth_sendRawTransaction) are sent in parallel to the configured private RPCs (eden, ethermine, flashbots, etc.).

All other requests are sent to an RPC server on the latest block (alchemy, moralis, rivet, your own node, or one of many other providers). If multiple servers are in sync, they are prioritized by active_requests/soft_limit. Note that this means that the fastest server is most likely to serve requests and slow servers are unlikely to ever get any requests.

Each server has different limits to configure. The soft_limit is the number of parallel active requests where a server starts to slow down. The hard_limit is where a server starts giving rate limits or other errors.

$ cargo install sea-orm-cli
$ cargo run --release -- --help
   Compiling web3-proxy v0.1.0 (/home/bryan/src/web3-proxy/web3-proxy)
    Finished release [optimized + debuginfo] target(s) in 17.69s
     Running `target/release/web3-proxy --help`
Usage: web3-proxy [--port <port>] [--workers <workers>] [--config <config>]

Web3-proxy is a fast caching and load balancing proxy for web3 (Ethereum or similar) JsonRPC servers.

Options:
  --port            what port the proxy should listen on
  --workers         number of worker threads
  --config          path to a toml of rpc servers
  --help            display usage information

Start the server with the defaults (listen on http://localhost:8544 and use ./config/example.toml which proxies to a bunch of public nodes:

cargo run --release -- --config ./config/example.toml

Common commands

Check that the proxy is working:

curl -X POST -H "Content-Type: application/json" --data '{"jsonrpc":"2.0","method":"web3_clientVersion","id":1}' 127.0.0.1:8544

Check that the websocket is working:

$ websocat ws://127.0.0.1:8544

{"id": 1, "method": "eth_subscribe", "params": ["newHeads"]}

{"id": 2, "method": "eth_subscribe", "params": ["newPendingTransactions"]}

{"id": 3, "method": "eth_subscribe", "params": ["newPendingFullTransactions"]}

{"id": 4, "method": "eth_subscribe", "params": ["newPendingRawTransactions"]}

You can copy config/example.toml to config/production-$CHAINNAME.toml and then run docker-compose up --build -d start proxies for many chains.

Database entities

Types for database entities are generated from a development database.

sea-orm-cli generate entity -u mysql://root:dev_web3_proxy@127.0.0.1:3306/dev_web3_proxy -o entities/src

Then manually fix some columns: Vec<u8> -> sea_orm::prelude::Uuid and i8 -> bool. Related: https://github.com/SeaQL/sea-query/issues/375 https://github.com/SeaQL/sea-orm/issues/924

Flame Graphs

Flame graphs make finding slow code painless:

$ cat /proc/sys/kernel/kptr_restrict
1
$ echo 0 | sudo tee /proc/sys/kernel/kptr_restrict
0
$ CARGO_PROFILE_RELEASE_DEBUG=true cargo flamegraph

GDB

Run the proxy under gdb for advanced debugging:

cargo build --release && RUST_LOG=web3_proxy=debug rust-gdb --args target/debug/web3-proxy --listen-port 7503 --rpc-config-path ./config/production-eth.toml

TODO: also enable debug symbols in the release build by modifying the root Cargo.toml

Load Testing

Test the proxy:

wrk -s ./data/wrk/getBlockNumber.lua -t12 -c400 -d30s --latency http://127.0.0.1:8544
wrk -s ./data/wrk/getLatestBlockByNumber.lua -t12 -c400 -d30s --latency http://127.0.0.1:8544

Test geth (assuming it is on 8545):

wrk -s ./data/wrk/getBlockNumber.lua -t12 -c400 -d30s --latency http://127.0.0.1:8545
wrk -s ./data/wrk/getLatestBlockByNumber.lua -t12 -c400 -d30s --latency http://127.0.0.1:8545

Test erigon (assuming it is on 8945):

wrk -s ./data/wrk/getBlockNumber.lua -t12 -c400 -d30s --latency http://127.0.0.1:8945
wrk -s ./data/wrk/getLatestBlockByNumber.lua -t12 -c400 -d30s --latency http://127.0.0.1:8945

Note: Testing with getLatestBlockByNumber.lua is not great because the latest block changes and so one run is likely to be very different than another.

Run ethspam for a more realistic load test:

docker run --rm --name spam shazow/ethspam --rpc http://$LOCAL_IP:8544 | versus --concurrency=100 --stop-after=10000 http://$LOCAL_IP:8544; docker stop spam