Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add Ansible Automation for Lava Docker Provider Service #6

Open
wants to merge 1 commit into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
118 changes: 118 additions & 0 deletions impulse-automation/provider/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,118 @@
# RPC Provider Service Deployment

This repository includes Ansible playbooks and supporting files for deploying and managing the RPC Provider Service. The service uses Docker containers managed via Docker Compose to ensure easy and scalable deployments across multiple environments.

## Prerequisites

- **Ansible 2.9+**: Ensure Ansible is installed on your control machine.
- **Docker**: Must be installed on the target hosts.
- **Docker Compose**: Required for managing Dockerized applications.
- **SSH Access**: Root or sudo access on the target hosts.

## Repository Structure

- **`group_vars`** and **`host_vars`**: Contains variables specific to hosts and groups. Customize these to fit the deployment context.
- **`roles`**: Contains the tasks used for setting up the RPC provider.
- **`templates`**: Jinja2 templates for generating Docker Compose and environment configuration files.
- **`inventory`**: Hosts file defining the servers on which the RPC service will be deployed.

## Installation and Setup

### Clone the Repository

Start by cloning this repository to your Ansible control machine:

```bash
git clone <repository-url>
cd <repository-dir>
```

### Configure Inventory

Edit the `inventory/hosts` file to add the IP addresses or hostnames of the machines where the service should be deployed.

Example:

```yaml
all:
children:
lava_provider_eu:
hosts:
192.168.1.100:
ansible_user: root
ansible_ssh_private_key_file: ~/.ssh/id_rsa
```

> You can declare certain parameters in the `ansible.cfg` configuration file or in `group_vars/all.yml`. These settings will be applied to all hosts at each startup

ansible.cfg

```ini
[defaults]
private_key_file = ~/.ssh/id_rsa
```

group_vars/all.yml

```yml
# group_vars/all.yml
---
ansible_user: root
ansible_port: 22
```

### Set Role Variables

Adjust role-specific variables in `group_vars/all.yml` and `host_vars/*.yml` to match your environment:

## Deployment

The deployment process involves setting directly configuring Docker networks, generating necessary configuration claims, and managing Docker containers through Docker Compose.

### Prepare VM on clean system

If you plan to deploy on a clean Debian or Ubuntu system without an installed Docker engine, use the `prepare` tag to install additional software:

```bash
ansible-playbook main.yml --tags prepare
```

![lava_prepare_gifsicle.gif](..%2Fguides%2Flava_prepare_gifsicle.gif)

### Deploy the Service

To deploy the RPC Provider Service:

```bash
ansible-playbook main.yml --tags deploy
```

> Note that by default ```anisble-playbook main.yml``` command deploys and runs the service.

![lava_cache_provider_gifsicle.gif](..%2Fguides%2Flava_cache_provider_gifsicle.gif)

## Managing

Start the Service: Ensure the service is up and running:

```bash
ansible-playbook main.yml --tags start
```

Stop the Service: Safely stop the service when needed:

```bash
ansible-playbook main.yml --tags stop
```

Restart the Service: Restart the service to apply updates or changes:

```bash
ansible-playbook main.yml --tags restart
```

## Configuration Files

- Docker Compose Configuration: Located at `{{ project_path }}/docker-compose.yml`, it defines the service setup, including image, ports, and environment variables.
- Environment Variables: Stored in `{{ project_path }}/provider.env`, this file includes environment-specific variables like log level and cache expiration.
- Chains configuration: Stored in `{{ volume_path }}`. This config includes specified chain settings for the Lava RPC Provider.
30 changes: 30 additions & 0 deletions impulse-automation/provider/ansible.cfg
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
[defaults]
# Defines the location of the inventory file that Ansible will use to find the host information.
inventory = ./inventory/hosts

# Path where Ansible will look for the roles.
roles_path = roles

# The number of parallel processes to use when Ansible executes tasks on multiple hosts.
forks = 20

# Disables SSH host key checking, making Ansible automatically accept unknown host keys.
# This is useful in automated environments to avoid manual intervention.
host_key_checking = False

# Changes the default merging behavior of variables. With 'merge', hashes will be deeply merged.
hash_behaviour = merge

# Enables pipelining, which reduces the number of SSH operations required to execute a module.
# This can result in a significant performance improvement but may not be compatible with all setups.
pipelining = True

# Specifies the SSH private key file to use for SSH authentication.
private_key_file = ~/.ssh/id_rsa

[ssh_connection]
# SSH arguments used when connecting to hosts.
# - ForwardAgent=yes: Allows SSH agent forwarding.
# - ControlMaster=auto: Enables the sharing of multiple sessions over a single network connection.
# - ControlPersist=60s: Makes the master connection stay open in the background for up to 60 seconds after the initial connection, improving subsequent connection times.
ssh_args = -o ForwardAgent=yes -o ControlMaster=auto -o ControlPersist=60s
10 changes: 10 additions & 0 deletions impulse-automation/provider/inventory/group_vars/all.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
# group_vars/all.yml
---
ansible_user: root
ansible_port: 22
project_name: lava
project_type: provider
project_unique_name: "{{ project_name }}-{{ project_type }}"
service_path: /opt/services # The docker-compose.yml and a variables file are located at this path.
project_path: "{{ service_path }}/{{ project_unique_name }}" # Configuration files and a wallet are located at this path.
volume_path: "{{ container.volume_path }}"
140 changes: 140 additions & 0 deletions impulse-automation/provider/inventory/host_vars/lava_provider_eu.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,140 @@
# host_vars/lava_provider_eu.yml
---
ansible_host: xxx.xxx.xxx.xxx # IP address of the host
network: testnet

# Container configuration
container:
image: svetekllc/lava # Docker image name
tag: v2.0.1-provider # Docker image tag to specify version
limits:
cpu: 2 # Maximum number of CPU cores the container can use
memory: 4gb # Maximum amount of memory the container can use
volume_path: /opt/{{ project_unique_name }} # Path where volumes are mounted in the host

# Provider-specific configuration
provider_config:
cache:
enable: false # Whether caching is enabled (true | false)
address: lava-cache # Lava Cache service address (IP address | FQDN)
port: 23100 # Lava Cache service port
chain_id: lava-testnet-2 # Blockchain network identifier
config_path: /root/.lava # Path to provider configuration files
geolocation: 2 # Geolocation ID, used for regional distinctions
log_level: info # Logging level
moniker: Lava # Custom name for the provider instance
rewards_storage_dir: rewards-storage # Directory for storing rewards data
public_rpc_url: https://public-rpc-testnet2.lavanet.xyz:443/rpc/ # Public URL for RPC connections
total_connections: 25 # Max number of simultaneous network connections

# Wallet configuration
wallet:
name: lava # Wallet name
password: "" # Wallet password. Required if the keyring_backend parameter is equal: os, file
keyring_backend: test # Backend system for managing keys and secrets

# Network ports used by the provider
provider_ports:
grpc: 22001 # Port for gRPC service
metrics: 23001 # Port for exposing metrics

networks:
- lava # Network label used for Docker networking

# Blockchain chains configurations
chains:
- name: "Arbitrum mainnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "ARB1", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://arbitrum-mainnet:8545"] }

- name: "Avalanche mainnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "AVAX", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://avalanche-mainnet/ext/bc"] }

- name: "Axelar testnet"
endpoints:
- { api_interface: "tendermintrpc", chain_id: "AXELART", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://axelar-testnet:26657", "ws://axelar-testnet:26657/websocket"] }
- { api_interface: "grpc", chain_id: "AXELART", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["axelar-testnet:9090"] }
- { api_interface: "rest", chain_id: "AXELART", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://axelar-testnet:26317"] }

- name: "Axelar mainnet"
endpoints:
- { api_interface: "tendermintrpc", chain_id: "AXELAR", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://axelar-mainnet:26657", "ws://axelar-mainnet:26657/websocket"] }
- { api_interface: "grpc", chain_id: "AXELAR", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["axelar-mainnet:16090"] }
- { api_interface: "rest", chain_id: "AXELAR", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://axelar-mainnet:26317"] }

- name: "Binance mainnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "BSC", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://binance-mainnet"] }

- name: "Canto mainnet"
endpoints:
- { api_interface: "tendermintrpc", chain_id: "CANTO", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://canto-mainnet:26657", "ws://canto-mainnet:26657/websocket"] }
- { api_interface: "grpc", chain_id: "CANTO", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["canto-mainnet:9090"] }
- { api_interface: "jsonrpc", chain_id: "CANTO", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://canto-mainnet:8545"] }
- { api_interface: "rest", chain_id: "CANTO", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://canto-mainnet:1317"] }

- name: "Celestia mainnet"
endpoints:
- { api_interface: "tendermintrpc", chain_id: "CELESTIA", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://celestia-mainnet:26657", "ws://celestia-mainnet:26657/websocket"] }

Check failure on line 79 in impulse-automation/provider/inventory/host_vars/lava_provider_eu.yml

View workflow job for this annotation

GitHub Actions / Ansible Lint

yaml[line-length]

Line too long (204 > 200 characters)
- { api_interface: "grpc", chain_id: "CELESTIA", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["celestia-mainnet:9090"] }
- { api_interface: "jsonrpc", chain_id: "CELESTIA", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://celestia-mainnet-light:26658"] }
- { api_interface: "rest", chain_id: "CELESTIA", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://celestia-mainnet:1317"] }

- name: "Celo mainnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "CELO", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://celo-mainnet:8545"] }

- name: "CosmosHub mainnet"
endpoints:
- { api_interface: "tendermintrpc", chain_id: "COSMOSHUB", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://cosmos-mainnet:26657", "ws://cosmos-mainnet:26657/websocket"] }

Check failure on line 90 in impulse-automation/provider/inventory/host_vars/lava_provider_eu.yml

View workflow job for this annotation

GitHub Actions / Ansible Lint

yaml[line-length]

Line too long (201 > 200 characters)
- { api_interface: "grpc", chain_id: "COSMOSHUB", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["cosmos-mainnet:9090"] }
- { api_interface: "rest", chain_id: "COSMOSHUB", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://cosmos-mainnet:1317"] }

- name: "Ethereum mainnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "ETH1", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://ethereum-mainnet:8549"] }

- name: "Evmos testnet"
endpoints:
- { api_interface: "tendermintrpc", chain_id: "EVMOST", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://evmos-testnet:26657", "ws://evmos-testnet:26657/websocket"] }
- { api_interface: "grpc", chain_id: "EVMOST", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["evmos-testnet:9090"] }
- { api_interface: "jsonrpc", chain_id: "EVMOST", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://evmos-testnet:8545"] }
- { api_interface: "rest", chain_id: "EVMOST", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://evmos-testnet:1317"] }

- name: "Evmos mainnet"
endpoints:
- { api_interface: "tendermintrpc", chain_id: "EVMOS", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://evmos-mainnet:26657", "ws://evmos-mainnet:26657/websocket"] }
- { api_interface: "grpc", chain_id: "EVMOS", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["evmos-mainnet:9090"] }
- { api_interface: "jsonrpc", chain_id: "EVMOS", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://evmos-mainnet:8545"] }
- { api_interface: "rest", chain_id: "EVMOS", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://evmos-mainnet:1317"] }

- name: "Fantom mainnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "FTM250", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://fantom-mainnet:8545"] }

- name: "Lava testnet"
endpoints:
- { api_interface: "tendermintrpc", chain_id: "LAV1", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://lava-testnet:26657", "ws://lava-testnet:26657/websocket"] }
- { api_interface: "grpc", chain_id: "LAV1", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["lava-testnet:9090"] }
- { api_interface: "rest", chain_id: "LAV1", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://lava-testnet:1317"] }

- name: "Near testnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "NEART", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://near-testnet:3030"] }

- name: "Near mainnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "NEAR", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://near-mainnet:3030"] }

- name: "Polygon mainnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "POLYGON1", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://polygon-mainnet"] }

- name: "Solana mainnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "SOLANA", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://solana-mainnet"] }

- name: "Starknet mainnet"
endpoints:
- { api_interface: "jsonrpc", chain_id: "STRK", network_address: "0.0.0.0:22001", disable_tls: true, node_urls: ["http://starknet-mainnet:9545"] }
3 changes: 3 additions & 0 deletions impulse-automation/provider/inventory/hosts
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
all:
hosts:
lava_provider_eu:
35 changes: 35 additions & 0 deletions impulse-automation/provider/main.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
---
# The main playbook for rpc provider deployment
- name: Deploying and managing the RPC Provider Service
hosts: all
become: true
gather_facts: true
roles:
- role: prepare
tags:
- never
- prepare
- role: deploy
tags:
- deploy
tasks:
- name: Run the service
community.docker.docker_compose_v2:
project_src: "{{ project_path }}"
state: present
tags:
- start
- name: Stop the service
community.docker.docker_compose_v2:
project_src: "{{ project_path }}"
state: stopped
tags:
- never
- stop
- name: Restart the service
community.docker.docker_compose_v2:
project_src: "{{ project_path }}"
state: restarted
tags:
- never
- restart
1 change: 1 addition & 0 deletions impulse-automation/provider/roles/deploy/defaults/main.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
# roles/deploy/defaults/main.yml
16 changes: 16 additions & 0 deletions impulse-automation/provider/roles/deploy/handlers/main.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
# roles/deploy/handlers/main.yml
---
- name: Check and possibly restart docker service
block:
- name: Get container info
community.docker.docker_container_info:
name: "{{ project_unique_name }}"
register: container_info
listen: Check and possibly restart docker service

- name: Restart docker service if container exists
community.docker.docker_compose_v2:
project_src: "{{ project_path }}"
state: restarted
when: container_info.exists
listen: Check and possibly restart docker service
19 changes: 19 additions & 0 deletions impulse-automation/provider/roles/deploy/meta/main.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
# roles/deploy/meta/main.yml
---
dependencies: []
galaxy_info:
author: Michael
description: The role for deploying the lava RPC provider
company: Impulse Expert | https://impulse.expert
license: GPL
min_ansible_version: "2.9"
platforms:
- name: Ubuntu
versions:
- xenial
- bionic
- focal
galaxy_tags:
- lava
- rpc
- docker
Loading
Loading