1
0
mirror of synced 2025-01-11 00:05:48 +00:00

Docker compose for small libp2p node network (#364)

* Update rust build image version

* Docker compose file for testnet

* Wrap tcp into dns transport in order to resolve hostnames (#346)

* Docker compose for small libp2p node network

* Install etcdctl to node containers

* Register libp2p nodes on etcd

* Register IP address in KV store

* Interconnect libp2p nodes

* Use delimiter in cli and env variables

* Use docker compose initial env config

* Leave main Dockerfile as is, use new ones in testnet dir

* Remove etcd installation script

* run_mixnet.sh placeholder

* Use .env file for docker compose config

* Ignore local .env file

* Wrap sh envvars used in strings

* Remove mixnode placeholders

* Use default values for envconfig

* Update labels in Dockerfiles

* Sanitize scripts via shellcheck

* Export env for nomos node

* Updated to latest libp2p config

* Pass config to bootstrap node
This commit is contained in:
gusto 2023-09-15 17:27:38 +03:00 committed by GitHub
parent 90529dada3
commit 97c653efe3
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
11 changed files with 220 additions and 8 deletions

10
.env.example Normal file
View File

@ -0,0 +1,10 @@
# Environment variables for compose.yml file config.
DOCKER_COMPOSE_LIBP2P_REPLICAS=1
DOCKER_COMPOSE_LIBP2P_NODE_KEY_MASK=2000000000000000000000000000000000000000000000000000000000000000
DOCKER_COMPOSE_MIXNET_REPLICAS=1
DOCKER_COMPOSE_MIXNET_NODE_KEY_MASK=3000000000000000000000000000000000000000000000000000000000000000
DOCKER_COMPOSE_ETCDCTL_ENDPOINTS=etcd:2379
DOCKER_COMPOSE_ETCDCTL_API=3
DOCKER_COMPOSE_BOOSTRAP_NET_NODE_KEY=1000000000000000000000000000000000000000000000000000000000000000
DOCKER_COMPOSE_OVERLAY_NODES=$DOCKER_COMPOSE_BOOSTRAP_NET_NODE_KEY
DOCKER_COMPOSE_NET_INITIAL_PEERS=/dns/bootstrap/tcp/3000

1
.gitignore vendored
View File

@ -14,3 +14,4 @@ config.yml
store.* store.*
sim_config.json sim_config.json
*.txt *.txt
.env

View File

@ -1,6 +1,6 @@
# BUILD IMAGE --------------------------------------------------------- # BUILD IMAGE ---------------------------------------------------------
FROM rust:1.70.0-slim-bullseye AS builder FROM rust:1.72.0-slim-bullseye AS builder
# Using backports for go 1.19 # Using backports for go 1.19
RUN echo 'deb http://deb.debian.org/debian bullseye-backports main' \ RUN echo 'deb http://deb.debian.org/debian bullseye-backports main' \
@ -16,15 +16,15 @@ RUN apt-get update && apt-get install -yq \
WORKDIR /nomos WORKDIR /nomos
COPY . . COPY . .
RUN cargo build --release -p nomos-node RUN cargo build --release -p nomos-node --no-default-features --features libp2p
# NODE IMAGE ---------------------------------------------------------- # NODE IMAGE ----------------------------------------------------------
FROM bitnami/minideb:latest FROM bitnami/minideb:latest
LABEL maintainer="augustinas@status.im" LABEL maintainer="augustinas@status.im" \
LABEL source="https://github.com/logos-co/nomos-research" source="https://github.com/logos-co/nomos-node" \
LABEL description="Nomos node image" description="Nomos node image"
# nomos default ports # nomos default ports
EXPOSE 3000 8080 9000 60000 EXPOSE 3000 8080 9000 60000

43
compose.yml Normal file
View File

@ -0,0 +1,43 @@
services:
bootstrap:
build:
context: .
dockerfile: testnet/Dockerfile
ports:
- "3000:3000/tcp"
- "8080:8080/tcp"
volumes:
- ./testnet:/etc/nomos
environment:
- NET_NODE_KEY=${DOCKER_COMPOSE_BOOSTRAP_NET_NODE_KEY:-1000000000000000000000000000000000000000000000000000000000000000}
command: /etc/nomos/config.yaml
libp2p-node:
build:
context: .
dockerfile: testnet/Dockerfile
volumes:
- ./testnet:/etc/nomos
deploy:
replicas: ${DOCKER_COMPOSE_LIBP2P_REPLICAS:-1}
depends_on:
- bootstrap
- etcd
environment:
- DOCKER_REPLICAS=${DOCKER_COMPOSE_LIBP2P_REPLICAS:-1}
- ETCDCTL_ENDPOINTS=${DOCKER_COMPOSE_ETCDCTL_ENDPOINTS:-etcd:2379}
- ETCDCTL_API=${DOCKER_COMPOSE_ETCDCTL_API:-3}
- NODE_MASK=${DOCKER_COMPOSE_LIBP2P_NODE_KEY_MASK:-2000000000000000000000000000000000000000000000000000000000000000}
- OVERLAY_NODES=${DOCKER_COMPOSE_OVERLAY_NODES:-1000000000000000000000000000000000000000000000000000000000000000}
- NET_INITIAL_PEERS=${DOCKER_COMPOSE_NET_INITIAL_PEERS:-/dns/bootstrap/tcp/3000}
entrypoint: /etc/nomos/configure_node.sh
etcd:
image: quay.io/coreos/etcd:v3.4.15
ports:
- "2379:2379/tcp"
command:
- /usr/local/bin/etcd
- --advertise-client-urls=http://etcd:2379
- --listen-client-urls=http://0.0.0.0:2379

View File

@ -70,7 +70,7 @@ pub struct NetworkArgs {
#[clap(long = "net-node-key", env = "NET_NODE_KEY")] #[clap(long = "net-node-key", env = "NET_NODE_KEY")]
node_key: Option<String>, node_key: Option<String>,
#[clap(long = "net-initial-peers", env = "NET_INITIAL_PEERS")] #[clap(long = "net-initial-peers", env = "NET_INITIAL_PEERS", num_args = 1.., value_delimiter = ',')]
pub initial_peers: Option<Vec<Multiaddr>>, pub initial_peers: Option<Vec<Multiaddr>>,
} }
@ -105,7 +105,7 @@ pub struct OverlayArgs {
#[clap(long = "overlay-type", env = "OVERLAY_TYPE")] #[clap(long = "overlay-type", env = "OVERLAY_TYPE")]
pub overlay_type: Option<OverlayType>, pub overlay_type: Option<OverlayType>,
#[clap(long = "overlay-nodes", env = "OVERLAY_NODES")] #[clap(long = "overlay-nodes", env = "OVERLAY_NODES", num_args = 1.., value_delimiter = ',')]
pub overlay_nodes: Option<Vec<String>>, pub overlay_nodes: Option<Vec<String>>,
#[clap(long = "overlay-leader", env = "OVERLAY_LEADER")] #[clap(long = "overlay-leader", env = "OVERLAY_LEADER")]

View File

@ -8,6 +8,7 @@ multiaddr = "0.18"
tokio = { version = "1", features = ["sync", "macros"] } tokio = { version = "1", features = ["sync", "macros"] }
futures = "0.3" futures = "0.3"
libp2p = { version = "0.52.1", features = [ libp2p = { version = "0.52.1", features = [
"dns",
"yamux", "yamux",
"plaintext", "plaintext",
"macros", "macros",

View File

@ -10,8 +10,10 @@ use blake2::digest::{consts::U32, Digest};
use blake2::Blake2b; use blake2::Blake2b;
use libp2p::gossipsub::{Message, MessageId, TopicHash}; use libp2p::gossipsub::{Message, MessageId, TopicHash};
use libp2p::tcp::tokio::Tcp;
pub use libp2p::{ pub use libp2p::{
core::upgrade, core::upgrade,
dns,
gossipsub::{self, PublishError, SubscriptionError}, gossipsub::{self, PublishError, SubscriptionError},
identity::{self, secp256k1}, identity::{self, secp256k1},
plaintext::PlainText2Config, plaintext::PlainText2Config,
@ -76,7 +78,7 @@ impl Swarm {
log::info!("libp2p peer_id:{}", local_peer_id); log::info!("libp2p peer_id:{}", local_peer_id);
// TODO: consider using noise authentication // TODO: consider using noise authentication
let tcp_transport = tcp::tokio::Transport::new(tcp::Config::default().nodelay(true)) let tcp_transport = tcp::Transport::<Tcp>::new(tcp::Config::default().nodelay(true))
.upgrade(upgrade::Version::V1Lazy) .upgrade(upgrade::Version::V1Lazy)
.authenticate(PlainText2Config { .authenticate(PlainText2Config {
local_public_key: id_keys.public(), local_public_key: id_keys.public(),
@ -85,6 +87,9 @@ impl Swarm {
.timeout(TRANSPORT_TIMEOUT) .timeout(TRANSPORT_TIMEOUT)
.boxed(); .boxed();
// Wrapping TCP transport into DNS transport to resolve hostnames.
let tcp_transport = dns::TokioDnsConfig::system(tcp_transport)?.boxed();
// TODO: consider using Signed or Anonymous. // TODO: consider using Signed or Anonymous.
// For Anonymous, a custom `message_id` function need to be set // For Anonymous, a custom `message_id` function need to be set
// to prevent all messages from a peer being filtered as duplicates. // to prevent all messages from a peer being filtered as duplicates.

38
testnet/Dockerfile Normal file
View File

@ -0,0 +1,38 @@
# BUILD IMAGE ---------------------------------------------------------
FROM rust:1.72.0-slim-bullseye AS builder
# Using backports for go 1.19
RUN echo 'deb http://deb.debian.org/debian bullseye-backports main' \
>> /etc/apt/sources.list
# Dependecies for publishing documentation and building waku-bindings.
RUN apt-get update && apt-get install -yq \
git clang etcd-client \
golang-src/bullseye-backports \
golang-doc/bullseye-backports \
golang/bullseye-backports
WORKDIR /nomos
COPY . .
RUN cargo build --release -p nomos-node --no-default-features --features libp2p
RUN cargo build --release -p mixnode
# NODE IMAGE ----------------------------------------------------------
FROM bitnami/minideb:latest
LABEL maintainer="augustinas@status.im" \
source="https://github.com/logos-co/nomos-node" \
description="Nomos testnet image"
# nomos default ports
EXPOSE 3000 8080 9000 60000
COPY --from=builder /nomos/target/release/nomos-node /usr/bin/nomos-node
COPY --from=builder /nomos/target/release/mixnode /usr/bin/mixnode
COPY --from=builder /usr/bin/etcdctl /usr/bin/etcdctl
COPY nodes/nomos-node/config.yaml /etc/nomos/config.yaml
ENTRYPOINT ["/usr/bin/nomos-node"]

37
testnet/config.yaml Normal file
View File

@ -0,0 +1,37 @@
log:
backend: "Stdout"
format: "Json"
level: "debug"
consensus:
private_key: [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
fountain_settings: null
overlay_settings:
nodes: [[0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0], [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1]]
leader_super_majority_threshold: 1
leader:
cur: 0
network:
backend:
host: 0.0.0.0
port: 3000
log_level: "fatal"
node_key: "0000000000000000000000000000000000000000000000000000000000000001"
discV5BootstrapNodes: []
initial_peers: []
relayTopics: []
mixnet_client:
mode: Sender
topology:
layers:
- nodes:
- address: 127.0.0.1:7777
public_key: "0000000000000000000000000000000000000000000000000000000000000000"
connection_pool_size: 255
mixnet_delay:
start: "0ms"
end: "0ms"
http:
backend:
address: 0.0.0.0:8080
cors_origins: []

31
testnet/configure_node.sh Executable file
View File

@ -0,0 +1,31 @@
#!/bin/sh
set -e
# Set env variables for nomos-node.
NET_NODE_KEY=$(./etc/nomos/register_node.sh)
CONSENSUS_PRIV_KEY=$NET_NODE_KEY
node_ids=$(etcdctl get "/node/" --prefix --keys-only)
for node_id in $node_ids; do
node_key=$(etcdctl get "/config${node_id}/key" --print-value-only)
node_ip=$(etcdctl get "/config${node_id}/ip" --print-value-only)
node_multiaddr="/ip4/${node_ip}/tcp/3000"
if [ -z "$OVERLAY_NODES" ]; then
OVERLAY_NODES=$node_key
NET_INITIAL_PEERS=$node_multiaddr
else
OVERLAY_NODES="${OVERLAY_NODES},${node_key}"
NET_INITIAL_PEERS="${NET_INITIAL_PEERS},${node_multiaddr}"
fi
done
export CONSENSUS_PRIV_KEY \
OVERLAY_NODES \
NET_NODE_KEY \
NET_INITIAL_PEERS
echo "I am a container ${HOSTNAME} node ${NET_NODE_KEY}"
exec /usr/bin/nomos-node /etc/nomos/config.yaml

46
testnet/register_node.sh Executable file
View File

@ -0,0 +1,46 @@
#!/bin/sh
# NODE_MASK is set via compose.yml file.
node_key_from_id() {
echo "${NODE_MASK}" | sed "s/.\{${#NODE_ID}\}$/${NODE_ID}/"
}
END=$DOCKER_REPLICAS
NODE_ID=1
NODE_IP=$(hostname -i)
NODE_KEY=$(node_key_from_id)
register_node() {
## Conditional transaction to set node config key if it doesn't exist.
## Newlines in EOF block are important, more info here:
## https://github.com/etcd-io/etcd/tree/main/etcdctl#examples-3
etcdctl txn <<EOF
mod("/node/${NODE_ID}") = "0"
put /node/${NODE_ID} "${NODE_ID}"
put /config/node/${NODE_ID}/key "${NODE_KEY}"
put /config/node/${NODE_ID}/ip "${NODE_IP}"
EOF
}
while [ "${NODE_ID}" -le "${END}" ]; do
result=$(register_node)
# Check if the key was registered or already exists
if [ "${result}" != "FAILURE" ]; then
break
else
NODE_ID=$((NODE_ID + 1))
NODE_KEY=$(node_key_from_id)
fi
done
if [ "${NODE_ID}" -gt "${END}" ]; then
echo "Reached the limit without registering a ${NODE_ID}."
return 1
fi
echo "${NODE_KEY}"