This repository contains backend code for Cow Protocol Services written in Rust.
The orderbook
crate provides the http api through which users (usually through a frontend web application) interact with the order book.
Users can add signed orders to the order book and query the state of their orders.
They can also use the API to estimate fee amounts and limit prices before placing their order.
Solvers also interact with the order book by querying a list of open orders that they can attempt to settle.
The api is documented with openapi. A simple example script that uses the API to place random orders can be found in this repo
The order book service itself uses PostgreSQL as a backend to persist orders. In addition to connecting the http api to the database it also checks order validity based on the block time, trade events, erc20 funding and approval so that solvers can query only valid orders.
Multiple concurrent orderbook
s can run at the same time, allowing the user-facing API to scale horizontally with increased traffic.
The autopilot
crate is responsible for driving the protocol forward.
Concretely, it is responsible for "cutting" new auctions (i.e. determining auction boundaries and which orders are to be included, as well as various parameters important for settlement objective value computation).
The autopilot
connects to the same PostgreSQL database as the orderbook
and uses it to query orders as well as storing the most recent auction and settlement competition.
There are additional crates that live in the cargo workspace.
alerter
provides a custom alerter binary that looks at the current orderbook and counts metrics for orders that should be solved but aren'tcontracts
provides ethcontract-rs based smart contract bindingsdatabase
provides the shared database and storage layer logic shared between theautopilot
andorderbook
driver
an in-development binary that intends to replace thesolver
; it has a slightly different design that allows co-location with external solverse2e
end-to-end testsethrpc
ethrpc client with a few extensionsmodel
provides the serialization model for orders in the order book apinumber
extensions to number types, such as numerical conversions between 256-bit integers, nonzero types and de/serialization implementationsobserve
initialization and helper functions for logging and metricsshared
provides other shared functionality between the solver and order booktestlib
shared helpers for writing unit and end-to-end tests
To run the services locally you should use the playground
.
You can launch it with the following command:
docker compose -f playground/docker-compose.fork.yml up --build
Optionally you can limit the services run by the playground by specifying the desired service's names (ex. driver autopilot
).
Once stabilized, the playground will watch your local directory for changes and automatically recompile and restart the services as needed.
You can read more about the services available and their respective ports in the playground's README.
Binaries like
autopilot
,orderbook
,driver
andsolvers
have-h
and--help
for, respectively, short and long descriptions over available commands and options. Furthermore, there's OpenAPI pages for theorderbook
,driver
andsolver
APIs, you can find more information about the services and the CoW Protocol at <docs.cow.fi>.
The CI (check .github/workflows/pull-request.yaml
) runs
- doc tests:
just test-doc
- unit tests:
just test-unit
- DB tests:
just test-db
- E2E tests with a local node:
just test-e2e-local
- E2E tests with a forked node:
just test-e2e-forked
- driver tests:
just test-driver
The CI system uses cargo-nextest and therefore all tests are getting verified by it.
cargo-nextest
and cargo test
handle global state slightly differently which can cause some tests to fail with cargo test
.
That's why it's recommended to run tests with the provided just
commands.
In case a test is flaky and only fails sometimes in CI you can use the run-flaky-test
github action to test your fix with the CI to get confidence that the fix that works locally also works in CI.
The tests that require postgres connect to the default database of a locally running postgres instance on the default port. To achieve this, open a new shell and run the command below: Note: The migrations will be applied as well.
docker-compose up
In order to run the e2e forked_network
tests you have to have anvil installed,
if you haven't installed anvil
yet, refer to foundry
's installation guide to get started.
All forked_node
tests will require a FORK_MAINNET_URL
, you can refer to Chainlist to find some publicly available RPCs (terms and conditions may apply).
A subset of the forked_node
tests will require a FORK_GNOSIS_URL
, refer to the list of Gnosis RPC Providers for publicly available nodes.
All binaries are compiled with support for tokio-console by default to allow you to look inside the tokio runtime.
However, this feature is not enabled at runtime by default because it comes with a pretty significant memory overhead. To enable it you just have to set the environment variable TOKIO_CONSOLE=true
and run the binary you want to instrument.
You can install and run tokio-console
with:
cargo install --locked tokio-console
tokio-console
It's possible to change the tracing log filter while the process is running. This can be useful to debug an error that requires more verbose logs but which might no longer appear after restarting the system.
Each process opens a UNIX socket at /tmp/log_filter_override_<program_name>_<pid>.sock
. To change the log filter connect to it with nc -U <path>
and enter a new log filter.
You can also reset the log filter to the filter the program was initially started with by entering reset
.
See here for documentation on the supported log filter format.