# Council daemon v2
## Hardware requirements
- 2-core CPU
- 6GB RAM
- - Council Deamon — 500MB
- - Keys-API-DB — 500MB
- - Keys-API — 5GB
- EL Node (full node)
## Updates
The new version of the Lido Council Daemon has some important changes:
- We now use RabbitMQ instead of Kafka
- To run the Council Daemon you need to run Keys-API
- Using a PostgreSQL database to store the key cache
*Keys-API application for syncing Staking-Modules keys
First, you need to prepare the ENV variables:
### ENV variables
We now use RabbitMQ as the transport:
### RabbitMQ
```env
...
PUBSUB_SERVICE=rabbitmq
RABBITMQ_URL=<rabbitmq url that supports ws>
RABBITMQ_LOGIN=<rabbitmq login>
RABBITMQ_PASSCODE=<rabbitmq password>
...
```
### Wallet private key
```env
...
WALLET_PRIVATE_KEY=<wallet private key>
...
```
The private key can be omitted, in which case a random key will be generated and the daemon will run in test mode. But in production it is required.
The account balance should have some ETH to send transactions. In regular mode, the daemon does not spend any funds. The transaction will be sent only if a potential attack is detected. 1 ETH is enough.
### Keys-api configuration
Now we use the Keys-API to download the keys. Below you can see a sample configuration for the Keys-API:
```env
# Keys API
KEYS_API_PORT=3001
# chain id
# for mainnet 1
# for testnet 5
CHAIN_ID=5
RPC_URL=
# KeysAPI DB config
KEYS_API_DB_NAME=keys_service_db
KEYS_API_DB_PORT=5452
KEYS_API_DB_HOST=localhost
KEYS_API_DB_USER=test
KEYS_API_DB_PASSWORD=test
```
The Keys-API is publicly available and you can read more at this link: https://github.com/lidofinance/lido-keys-api
### Example ENV config file, don't forget to change the data to your own
<details>
<summary>sample.env</summary>
```
# App
PORT=3000
# Log level: debug, info, notice, warning or error
LOG_LEVEL=info
# Log format: simple or json
LOG_FORMAT=simple
# Pubsub (default: rabbitmq)
PUBSUB_SERVICE=rabbitmq
# RabbitMQ
RABBITMQ_URL=rabbitmq_url
RABBITMQ_LOGIN=test
RABBITMQ_PASSCODE=test
# Private key
# Used to sign transactions and stop the protocol.
# Make sure there are enough ETH on the balance to send a transaction to stop the protocol
WALLET_PRIVATE_KEY=0x0000000000000000000000000000000000000000000000000000000000000001
KEYS_API_HOST=http://127.0.0.1
# Keys API
KEYS_API_PORT=3001
# chain id
# for mainnet 1
# for testnet 5
CHAIN_ID=5
RPC_URL=
# KeysAPI DB config
KEYS_API_DB_NAME=keys_service_db
KEYS_API_DB_PORT=5452
KEYS_API_DB_HOST=localhost
KEYS_API_DB_USER=test
KEYS_API_DB_PASSWORD=test
```
</details>
## Running the application
At this point, it is most convenient to run the application with docker-compose. Below is a configuration template for running the entire application:
<details>
<summary>docker-compose.yml</summary>
```yaml=
version: '3.7'
services:
keys_api_service_db:
image: postgres:14-alpine
container_name: keys_api_service_db
restart: unless-stopped
environment:
- POSTGRES_DB=${KEYS_API_DB_NAME}
- POSTGRES_USER=${KEYS_API_DB_USER}
- POSTGRES_PASSWORD=${KEYS_API_DB_PASSWORD}
ports:
- ${KEYS_API_DB_PORT}:5432
volumes:
- ./.volumes/pgdata-${CHAIN_ID}/:/var/lib/postgresql/data
keys_api_service_api:
image: lidofinance/lido-keys-api:dev
container_name: keys_api_service_api
ports:
- '127.0.0.1:${KEYS_API_PORT}:3001'
environment:
- PORT=3001
- LOG_LEVEL=${LOG_LEVEL}
- LOG_FORMAT=${LOG_FORMAT}
- CHAIN_ID=${CHAIN_ID}
- PROVIDERS_URLS=${RPC_URL}
- VALIDATOR_REGISTRY_ENABLE=false
- DB_NAME=${KEYS_API_DB_NAME}
- DB_PORT=5432
- DB_HOST=keys_api_service_db
- DB_USER=${KEYS_API_DB_USER}
- DB_PASSWORD=${KEYS_API_DB_PASSWORD}
depends_on:
- keys_api_service_db
council-daemon:
image: lidofinance/lido-council-daemon:dev
ports:
- "127.0.0.1:${PORT}:3000" # port is used for prometheus metrics
environment:
- PORT=3000
- LOG_LEVEL=${LOG_LEVEL}
- LOG_FORMAT=${LOG_FORMAT}
- RPC_URL=${RPC_URL}
- WALLET_PRIVATE_KEY=${WALLET_PRIVATE_KEY}
- KEYS_API_HOST=${KEYS_API_HOST}
- KEYS_API_PORT=${KEYS_API_PORT}
- PUBSUB_SERVICE=rabbitmq
- RABBITMQ_URL=${RABBITMQ_URL}
- RABBITMQ_LOGIN=${RABBITMQ_LOGIN}
- RABBITMQ_PASSCODE=${RABBITMQ_PASSCODE}
depends_on:
- keys_api_service_api
volumes:
- ./.volumes/cache/:/council/cache/
```
</details>
### Run with docker-compose
After updating the docker-compose file and the .env configuration file, simply enter the command:
```bash
docker-compose up -d
```
Next, we can read the log:
```bash
docker-compose logs -f
```
## Logs
On startup, the daemon checks if the provided wallet address belongs to the list of guardians, as well as account balance. If something goes wrong you will see warnings:
```log
warn: Private key is not provided, a random address will be generated for the test run
warn: Account balance is too low {"balance":"1.0 ETH"}
warn: Your address is not in the Guardian List {"address":"0x0000000000000000000000000000000000000000"}
```
If all goes well, it will be in the logs:
```log
info: Account balance is sufficient {"balance":"1.0 ETH"}
info: Your address is in the Guardian List {"address":"0x0000000000000000000000000000000000000000"}
```
At the first startup the daemon will collect historical data:
```log
info: Historical events are fetched {"endBlock":4487322,"events":3,"startBlock":4467323}
```
If the daemon works correctly, the logs will look like this:
```log
debug: Fresh events are fetched {"startBlock":5679826,"endBlock":5679976,"events":6}
debug: Fresh events are fetched {"startBlock":5679827,"endBlock":5679977,"events":6}
debug: Fresh events are fetched {"startBlock":5679828,"endBlock":5679978,"events":7}
info: No problems found {"type":"deposit","depositRoot":"0xc2c9308fa425a64ef9cac1837412ba462b6429fce2f170184284a260b735638c","keysOpIndex":12,"blockNumber":5679978,"blockHash":"0x87762c941f653f2f70157f86deac78f19e4d1549e231a52d1191289592d1a0ab","guardianAddress":"0x3dc4cF780F2599B528F37dedB34449Fb65Ef7d4A","guardianIndex":0,"signature":{"r":"0x44fec2e6fd34e74b8f001ef0e5bbd2db6d3179925fb82cb43231e19af46f0ddd","s":"0x2ff4326af760e353803458b75279eb8f58e5735b3565ea16bcd0f773bce106a4","_vs":"0xaff4326af760e353803458b75279eb8f58e5735b3565ea16bcd0f773bce106a4","recoveryParam":1,"v":28}}
debug: Fresh events are fetched {"startBlock":5679829,"endBlock":5679979,"events":7}
```
## Development
```bash
# development
$ yarn start
# watch mode
$ yarn start:dev
```
## Prometheus metrics
Prometheus metrics are exposed via HTTP `/metrics` endpoint.
## Cache
Cache warming takes a lot of RPC queries and up to 30m of time (for mainnet). That cache is fully deterministic, fairly easily repopulated and you shouldn't be afraid to lose it.
To clear the cache use:
```bash
yarn cache:clear
```
## Test
```bash
# unit tests
$ yarn test
# e2e tests
$ yarn test:e2e
# test coverage
$ yarn test:cov
```