diff --git a/app/base.py b/app/base.py
index 526c7c6c..a7721141 100644
--- a/app/base.py
+++ b/app/base.py
@@ -15,7 +15,7 @@
import os
from abc import ABC, abstractmethod
-from .deploy_system import get_stack_status
+from .deploy import get_stack_status
from decouple import config
diff --git a/app/data/compose/docker-compose-fixturenet-laconicd.yml b/app/data/compose/docker-compose-fixturenet-laconicd.yml
index 21c9fb77..5037687c 100644
--- a/app/data/compose/docker-compose-fixturenet-laconicd.yml
+++ b/app/data/compose/docker-compose-fixturenet-laconicd.yml
@@ -25,3 +25,6 @@ services:
image: cerc/laconic-registry-cli:local
volumes:
- ../config/fixturenet-laconicd/registry-cli-config-template.yml:/registry-cli-config-template.yml
+
+volumes:
+ laconicd-data:
diff --git a/app/data/compose/docker-compose-lasso.yml b/app/data/compose/docker-compose-lasso.yml
new file mode 100644
index 00000000..6e54b5ee
--- /dev/null
+++ b/app/data/compose/docker-compose-lasso.yml
@@ -0,0 +1,8 @@
+version: "3.2"
+
+services:
+ lasso:
+ image: cerc/lasso:local
+ restart: always
+ ports:
+ - "0.0.0.0:3000:3000"
diff --git a/app/data/compose/docker-compose-mainnet-laconicd.yml b/app/data/compose/docker-compose-mainnet-laconicd.yml
new file mode 100644
index 00000000..4de68d7b
--- /dev/null
+++ b/app/data/compose/docker-compose-mainnet-laconicd.yml
@@ -0,0 +1,30 @@
+services:
+ laconicd:
+ restart: no
+ image: cerc/laconicd:local
+ command: ["sh", "/docker-entrypoint-scripts.d/create-fixturenet.sh"]
+ volumes:
+ # The cosmos-sdk node's database directory:
+ - laconicd-data:/root/.laconicd/data
+ # TODO: look at folding these scripts into the container
+ - ../config/mainnet-laconicd/create-fixturenet.sh:/docker-entrypoint-scripts.d/create-fixturenet.sh
+ - ../config/mainnet-laconicd/export-mykey.sh:/docker-entrypoint-scripts.d/export-mykey.sh
+ - ../config/mainnet-laconicd/export-myaddress.sh:/docker-entrypoint-scripts.d/export-myaddress.sh
+ # TODO: determine which of the ports below is really needed
+ ports:
+ - "6060"
+ - "26657"
+ - "26656"
+ - "9473:9473"
+ - "8545"
+ - "8546"
+ - "9090"
+ - "9091"
+ - "1317"
+ cli:
+ image: cerc/laconic-registry-cli:local
+ volumes:
+ - ../config/mainnet-laconicd/registry-cli-config-template.yml:/registry-cli-config-template.yml
+
+volumes:
+ laconicd-data:
diff --git a/app/data/compose/docker-compose-mobymask-app.yml b/app/data/compose/docker-compose-mobymask-app.yml
index 7d41264a..1b4b4f6d 100644
--- a/app/data/compose/docker-compose-mobymask-app.yml
+++ b/app/data/compose/docker-compose-mobymask-app.yml
@@ -23,7 +23,7 @@ services:
- peers_ids:/peers
- mobymask_deployment:/server
ports:
- - "0.0.0.0:3002:80"
+ - "127.0.0.1:3002:80"
healthcheck:
test: ["CMD", "nc", "-vz", "localhost", "80"]
interval: 20s
@@ -55,7 +55,7 @@ services:
- peers_ids:/peers
- mobymask_deployment:/server
ports:
- - "0.0.0.0:3004:80"
+ - "127.0.0.1:3004:80"
healthcheck:
test: ["CMD", "nc", "-vz", "localhost", "80"]
interval: 20s
diff --git a/app/data/compose/docker-compose-peer-test-app.yml b/app/data/compose/docker-compose-peer-test-app.yml
index 649ab8f8..87a36228 100644
--- a/app/data/compose/docker-compose-peer-test-app.yml
+++ b/app/data/compose/docker-compose-peer-test-app.yml
@@ -18,7 +18,7 @@ services:
- ../config/watcher-mobymask-v2/test-app-start.sh:/scripts/test-app-start.sh
- peers_ids:/peers
ports:
- - "0.0.0.0:3003:80"
+ - "127.0.0.1:3003:80"
healthcheck:
test: ["CMD", "nc", "-v", "localhost", "80"]
interval: 20s
diff --git a/app/data/compose/docker-compose-test.yml b/app/data/compose/docker-compose-test.yml
index d20c3cfc..19660f89 100644
--- a/app/data/compose/docker-compose-test.yml
+++ b/app/data/compose/docker-compose-test.yml
@@ -5,7 +5,7 @@ services:
environment:
CERC_SCRIPT_DEBUG: ${CERC_SCRIPT_DEBUG}
volumes:
- - test-data:/var
+ - test-data:/data
ports:
- "80"
diff --git a/app/data/compose/docker-compose-watcher-mobymask-v2.yml b/app/data/compose/docker-compose-watcher-mobymask-v2.yml
index 0c743670..db3e3a20 100644
--- a/app/data/compose/docker-compose-watcher-mobymask-v2.yml
+++ b/app/data/compose/docker-compose-watcher-mobymask-v2.yml
@@ -14,7 +14,7 @@ services:
- ../config/postgresql/multiple-postgressql-databases.sh:/docker-entrypoint-initdb.d/multiple-postgressql-databases.sh
- mobymask_watcher_db_data:/var/lib/postgresql/data
ports:
- - "0.0.0.0:15432:5432"
+ - "127.0.0.1:15432:5432"
healthcheck:
test: ["CMD", "nc", "-v", "localhost", "5432"]
interval: 20s
@@ -95,9 +95,9 @@ services:
- mobymask_deployment:/server
# Expose GQL, metrics and relay node ports
ports:
- - "0.0.0.0:3001:3001"
- - "0.0.0.0:9001:9001"
- - "0.0.0.0:9090:9090"
+ - "127.0.0.1:3001:3001"
+ - "127.0.0.1:9001:9001"
+ - "127.0.0.1:9090:9090"
healthcheck:
test: ["CMD", "busybox", "nc", "localhost", "9090"]
interval: 20s
diff --git a/app/data/config/mainnet-laconicd/create-fixturenet.sh b/app/data/config/mainnet-laconicd/create-fixturenet.sh
new file mode 100644
index 00000000..9c30bff8
--- /dev/null
+++ b/app/data/config/mainnet-laconicd/create-fixturenet.sh
@@ -0,0 +1,118 @@
+#!/bin/bash
+
+# TODO: this file is now an unmodified copy of cerc-io/laconicd/init.sh
+# so we should have a mechanism to bundle it inside the container rather than link from here
+# at deploy time.
+
+KEY="mykey"
+CHAINID="laconic_9000-1"
+MONIKER="localtestnet"
+KEYRING="test"
+KEYALGO="eth_secp256k1"
+LOGLEVEL="info"
+# trace evm
+TRACE="--trace"
+# TRACE=""
+
+# validate dependencies are installed
+command -v jq > /dev/null 2>&1 || { echo >&2 "jq not installed. More info: https://stedolan.github.io/jq/download/"; exit 1; }
+
+# remove existing daemon and client
+rm -rf ~/.laconic*
+
+make install
+
+laconicd config keyring-backend $KEYRING
+laconicd config chain-id $CHAINID
+
+# if $KEY exists it should be deleted
+laconicd keys add $KEY --keyring-backend $KEYRING --algo $KEYALGO
+
+# Set moniker and chain-id for Ethermint (Moniker can be anything, chain-id must be an integer)
+laconicd init $MONIKER --chain-id $CHAINID
+
+# Change parameter token denominations to aphoton
+cat $HOME/.laconicd/config/genesis.json | jq '.app_state["staking"]["params"]["bond_denom"]="aphoton"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+cat $HOME/.laconicd/config/genesis.json | jq '.app_state["crisis"]["constant_fee"]["denom"]="aphoton"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+cat $HOME/.laconicd/config/genesis.json | jq '.app_state["gov"]["deposit_params"]["min_deposit"][0]["denom"]="aphoton"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+cat $HOME/.laconicd/config/genesis.json | jq '.app_state["mint"]["params"]["mint_denom"]="aphoton"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+# Custom modules
+cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["record_rent"]["denom"]="aphoton"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_rent"]["denom"]="aphoton"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_auction_commit_fee"]["denom"]="aphoton"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_auction_reveal_fee"]["denom"]="aphoton"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_auction_minimum_bid"]["denom"]="aphoton"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+
+if [[ "$TEST_REGISTRY_EXPIRY" == "true" ]]; then
+ echo "Setting timers for expiry tests."
+
+ cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["record_rent_duration"]="60s"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+ cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_grace_period"]="60s"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+ cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_rent_duration"]="60s"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+fi
+
+if [[ "$TEST_AUCTION_ENABLED" == "true" ]]; then
+ echo "Enabling auction and setting timers."
+
+ cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_auction_enabled"]=true' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+ cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_rent_duration"]="60s"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+ cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_grace_period"]="300s"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+ cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_auction_commits_duration"]="60s"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+ cat $HOME/.laconicd/config/genesis.json | jq '.app_state["registry"]["params"]["authority_auction_reveals_duration"]="60s"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+fi
+
+# increase block time (?)
+cat $HOME/.laconicd/config/genesis.json | jq '.consensus_params["block"]["time_iota_ms"]="1000"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+
+# Set gas limit in genesis
+cat $HOME/.laconicd/config/genesis.json | jq '.consensus_params["block"]["max_gas"]="10000000"' > $HOME/.laconicd/config/tmp_genesis.json && mv $HOME/.laconicd/config/tmp_genesis.json $HOME/.laconicd/config/genesis.json
+
+# disable produce empty block
+if [[ "$OSTYPE" == "darwin"* ]]; then
+ sed -i '' 's/create_empty_blocks = true/create_empty_blocks = false/g' $HOME/.laconicd/config/config.toml
+ else
+ sed -i 's/create_empty_blocks = true/create_empty_blocks = false/g' $HOME/.laconicd/config/config.toml
+fi
+
+if [[ $1 == "pending" ]]; then
+ if [[ "$OSTYPE" == "darwin"* ]]; then
+ sed -i '' 's/create_empty_blocks_interval = "0s"/create_empty_blocks_interval = "30s"/g' $HOME/.laconicd/config/config.toml
+ sed -i '' 's/timeout_propose = "3s"/timeout_propose = "30s"/g' $HOME/.laconicd/config/config.toml
+ sed -i '' 's/timeout_propose_delta = "500ms"/timeout_propose_delta = "5s"/g' $HOME/.laconicd/config/config.toml
+ sed -i '' 's/timeout_prevote = "1s"/timeout_prevote = "10s"/g' $HOME/.laconicd/config/config.toml
+ sed -i '' 's/timeout_prevote_delta = "500ms"/timeout_prevote_delta = "5s"/g' $HOME/.laconicd/config/config.toml
+ sed -i '' 's/timeout_precommit = "1s"/timeout_precommit = "10s"/g' $HOME/.laconicd/config/config.toml
+ sed -i '' 's/timeout_precommit_delta = "500ms"/timeout_precommit_delta = "5s"/g' $HOME/.laconicd/config/config.toml
+ sed -i '' 's/timeout_commit = "5s"/timeout_commit = "150s"/g' $HOME/.laconicd/config/config.toml
+ sed -i '' 's/timeout_broadcast_tx_commit = "10s"/timeout_broadcast_tx_commit = "150s"/g' $HOME/.laconicd/config/config.toml
+ else
+ sed -i 's/create_empty_blocks_interval = "0s"/create_empty_blocks_interval = "30s"/g' $HOME/.laconicd/config/config.toml
+ sed -i 's/timeout_propose = "3s"/timeout_propose = "30s"/g' $HOME/.laconicd/config/config.toml
+ sed -i 's/timeout_propose_delta = "500ms"/timeout_propose_delta = "5s"/g' $HOME/.laconicd/config/config.toml
+ sed -i 's/timeout_prevote = "1s"/timeout_prevote = "10s"/g' $HOME/.laconicd/config/config.toml
+ sed -i 's/timeout_prevote_delta = "500ms"/timeout_prevote_delta = "5s"/g' $HOME/.laconicd/config/config.toml
+ sed -i 's/timeout_precommit = "1s"/timeout_precommit = "10s"/g' $HOME/.laconicd/config/config.toml
+ sed -i 's/timeout_precommit_delta = "500ms"/timeout_precommit_delta = "5s"/g' $HOME/.laconicd/config/config.toml
+ sed -i 's/timeout_commit = "5s"/timeout_commit = "150s"/g' $HOME/.laconicd/config/config.toml
+ sed -i 's/timeout_broadcast_tx_commit = "10s"/timeout_broadcast_tx_commit = "150s"/g' $HOME/.laconicd/config/config.toml
+ fi
+fi
+
+# Allocate genesis accounts (cosmos formatted addresses)
+laconicd add-genesis-account $KEY 100000000000000000000000000aphoton --keyring-backend $KEYRING
+
+# Sign genesis transaction
+laconicd gentx $KEY 1000000000000000000000aphoton --keyring-backend $KEYRING --chain-id $CHAINID
+
+# Collect genesis tx
+laconicd collect-gentxs
+
+# Run this to ensure everything worked and that the genesis file is setup correctly
+laconicd validate-genesis
+
+if [[ $1 == "pending" ]]; then
+ echo "pending mode is on, please wait for the first block committed."
+fi
+
+# Start the node (remove the --pruning=nothing flag if historical queries are not needed)
+laconicd start --pruning=nothing --evm.tracer=json $TRACE --log_level $LOGLEVEL --minimum-gas-prices=0.0001aphoton --json-rpc.api eth,txpool,personal,net,debug,web3,miner --api.enable --gql-server --gql-playground
diff --git a/app/data/config/mainnet-laconicd/export-myaddress.sh b/app/data/config/mainnet-laconicd/export-myaddress.sh
new file mode 100644
index 00000000..e454c0b0
--- /dev/null
+++ b/app/data/config/mainnet-laconicd/export-myaddress.sh
@@ -0,0 +1,2 @@
+#!/bin/sh
+laconicd keys show mykey | grep address | cut -d ' ' -f 3
diff --git a/app/data/config/mainnet-laconicd/export-mykey.sh b/app/data/config/mainnet-laconicd/export-mykey.sh
new file mode 100644
index 00000000..1a5be86e
--- /dev/null
+++ b/app/data/config/mainnet-laconicd/export-mykey.sh
@@ -0,0 +1,2 @@
+#!/bin/sh
+echo y | laconicd keys export mykey --unarmored-hex --unsafe
diff --git a/app/data/config/mainnet-laconicd/registry-cli-config-template.yml b/app/data/config/mainnet-laconicd/registry-cli-config-template.yml
new file mode 100644
index 00000000..16432c18
--- /dev/null
+++ b/app/data/config/mainnet-laconicd/registry-cli-config-template.yml
@@ -0,0 +1,9 @@
+services:
+ cns:
+ restEndpoint: 'http://laconicd:1317'
+ gqlEndpoint: 'http://laconicd:9473/api'
+ userKey: REPLACE_WITH_MYKEY
+ bondId:
+ chainId: laconic_9000-1
+ gas: 250000
+ fees: 200000aphoton
diff --git a/app/data/config/watcher-mobymask-v2/mobymask-params.env b/app/data/config/watcher-mobymask-v2/mobymask-params.env
index 39d55016..5e4d9fb6 100644
--- a/app/data/config/watcher-mobymask-v2/mobymask-params.env
+++ b/app/data/config/watcher-mobymask-v2/mobymask-params.env
@@ -10,7 +10,7 @@ DEFAULT_CERC_RELAY_PEERS=[]
DEFAULT_CERC_RELAY_ANNOUNCE_DOMAIN=
# Base URI for mobymask-app (used for generating invite)
-DEFAULT_CERC_MOBYMASK_APP_BASE_URI="http://127.0.0.1:3002/#"
+DEFAULT_CERC_MOBYMASK_APP_BASE_URI="http://127.0.0.1:3004/#"
# Set to false for disabling watcher peer to send txs to L2
DEFAULT_CERC_ENABLE_PEER_L2_TXS=true
diff --git a/app/data/container-build/cerc-lasso/build.sh b/app/data/container-build/cerc-lasso/build.sh
new file mode 100755
index 00000000..a27f2fcd
--- /dev/null
+++ b/app/data/container-build/cerc-lasso/build.sh
@@ -0,0 +1,4 @@
+#!/usr/bin/env bash
+# Build the lasso image
+source ${CERC_CONTAINER_BASE_DIR}/build-base.sh
+docker build -t cerc/lasso:local -f ${CERC_REPO_BASE_DIR}/lasso/Dockerfile ${build_command_args} ${CERC_REPO_BASE_DIR}/lasso
diff --git a/app/data/container-build/cerc-test-container/run.sh b/app/data/container-build/cerc-test-container/run.sh
index 5aa33467..205c231a 100755
--- a/app/data/container-build/cerc-test-container/run.sh
+++ b/app/data/container-build/cerc-test-container/run.sh
@@ -4,7 +4,7 @@ if [ -n "$CERC_SCRIPT_DEBUG" ]; then
set -x
fi
# Test if the container's filesystem is old (run previously) or new
-EXISTSFILENAME=/var/exists
+EXISTSFILENAME=/data/exists
echo "Test container starting"
if [[ -f "$EXISTSFILENAME" ]];
then
diff --git a/app/data/container-image-list.txt b/app/data/container-image-list.txt
index 412d1fea..43781f8a 100644
--- a/app/data/container-image-list.txt
+++ b/app/data/container-image-list.txt
@@ -42,4 +42,5 @@ cerc/ipld-eth-state-snapshot
cerc/watcher-gelato
cerc/lotus
cerc/go-opera
+cerc/lasso
cerc/reth
diff --git a/app/data/pod-list.txt b/app/data/pod-list.txt
index 077709eb..97a7f369 100644
--- a/app/data/pod-list.txt
+++ b/app/data/pod-list.txt
@@ -5,7 +5,6 @@ go-ethereum-foundry
ipld-eth-beacon-db
ipld-eth-beacon-indexer
ipld-eth-server
-lighthouse
laconicd
fixturenet-laconicd
fixturenet-eth
@@ -29,4 +28,5 @@ watcher-azimuth
watcher-gelato
fixturenet-lotus
mainnet-go-opera
+lasso
reth
diff --git a/app/data/repository-list.txt b/app/data/repository-list.txt
index bae0808d..a9a85342 100644
--- a/app/data/repository-list.txt
+++ b/app/data/repository-list.txt
@@ -35,4 +35,5 @@ github.com/cerc-io/gelato-watcher-ts
github.com/filecoin-project/lotus
git.vdb.to/cerc-io/test-project
github.com/Fantom-foundation/go-opera
+github.com/cerc-io/lasso
github.com/paradigmxyz/reth
diff --git a/app/data/stacks/fixturenet-optimism/README.md b/app/data/stacks/fixturenet-optimism/README.md
index 0fd4f94f..c083ee68 100644
--- a/app/data/stacks/fixturenet-optimism/README.md
+++ b/app/data/stacks/fixturenet-optimism/README.md
@@ -42,12 +42,11 @@ Deploy the stack:
laconic-so --stack fixturenet-optimism deploy up
```
-If you get the error `service "fixturenet-optimism-contracts" didn't complete successfully: exit 1` with ~25 lines of Traceback, wait 15-20 mins then re-run the command.
-
The `fixturenet-optimism-contracts` service takes a while to complete running as it:
1. waits for the 'Merge' to happen on L1
2. waits for a finalized block to exist on L1 (so that it can be taken as a starting block for roll ups)
3. deploys the L1 contracts
+It may restart a few times after running into errors.
To list and monitor the running containers:
@@ -115,6 +114,5 @@ docker volume rm $(docker volume ls -q --filter "name=.*l1_deployment|.*l2_accou
## Known Issues
-* `fixturenet-eth` currently starts fresh on a restart
* Resource requirements (memory + time) for building the `cerc/foundry` image are on the higher side
* `cerc/optimism-contracts` image is currently based on `cerc/foundry` (Optimism requires foundry installation)
diff --git a/app/data/stacks/lasso/README.md b/app/data/stacks/lasso/README.md
new file mode 100644
index 00000000..226e4e39
--- /dev/null
+++ b/app/data/stacks/lasso/README.md
@@ -0,0 +1,7 @@
+# lasso
+
+```
+laconic-so --stack lasso setup-repositories
+laconic-so --stack lasso build-containers
+laconic-so --stack lasso deploy up
+```
diff --git a/app/data/stacks/lasso/stack.yml b/app/data/stacks/lasso/stack.yml
new file mode 100644
index 00000000..e756202c
--- /dev/null
+++ b/app/data/stacks/lasso/stack.yml
@@ -0,0 +1,8 @@
+version: "0.1"
+name: lasso
+repos:
+ - github.com/cerc-io/lasso
+containers:
+ - cerc/lasso
+pods:
+ - lasso
diff --git a/app/data/stacks/mainnet-laconic/README.md b/app/data/stacks/mainnet-laconic/README.md
new file mode 100644
index 00000000..67984e5b
--- /dev/null
+++ b/app/data/stacks/mainnet-laconic/README.md
@@ -0,0 +1,2 @@
+# Laconic Mainnet Deployment (experimental)
+
diff --git a/app/data/stacks/mainnet-laconic/deploy/commands.py b/app/data/stacks/mainnet-laconic/deploy/commands.py
new file mode 100644
index 00000000..0ac4845f
--- /dev/null
+++ b/app/data/stacks/mainnet-laconic/deploy/commands.py
@@ -0,0 +1,57 @@
+# Copyright © 2022, 2023 Cerc
+
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU Affero General Public License for more details.
+
+# You should have received a copy of the GNU Affero General Public License
+# along with this program. If not, see .
+
+import click
+import os
+from shutil import copyfile
+import sys
+from .util import get_stack_config_filename, get_parsed_deployment_spec
+
+default_spec_file_content = """stack: mainnet-laconic
+data_dir: /my/path
+node_name: my-node-name
+"""
+
+
+def make_default_deployment_dir():
+ return "deployment-001"
+
+@click.command()
+@click.option("--output", required=True, help="Write yaml spec file here")
+@click.pass_context
+def init(ctx, output):
+ with open(output, "w") as output_file:
+ output_file.write(default_spec_file_content)
+
+
+@click.command()
+@click.option("--spec-file", required=True, help="Spec file to use to create this deployment")
+@click.option("--deployment-dir", help="Create deployment files in this directory")
+@click.pass_context
+def create(ctx, spec_file, deployment_dir):
+ # This function fails with a useful error message if the file doens't exist
+ parsed_spec = get_parsed_deployment_spec(spec_file)
+ if ctx.debug:
+ print(f"parsed spec: {parsed_spec}")
+ if deployment_dir is None:
+ deployment_dir = make_default_deployment_dir()
+ if os.path.exists(deployment_dir):
+ print(f"Error: {deployment_dir} already exists")
+ sys.exit(1)
+ os.mkdir(deployment_dir)
+ # Copy spec file and the stack file into the deployment dir
+ copyfile(spec_file, os.path.join(deployment_dir, os.path.basename(spec_file)))
+ stack_file = get_stack_config_filename(parsed_spec.stack)
+ copyfile(stack_file, os.path.join(deployment_dir, os.path.basename(stack_file)))
diff --git a/app/data/stacks/mainnet-laconic/stack.yml b/app/data/stacks/mainnet-laconic/stack.yml
new file mode 100644
index 00000000..b5e1f16c
--- /dev/null
+++ b/app/data/stacks/mainnet-laconic/stack.yml
@@ -0,0 +1,31 @@
+version: "1.0"
+name: mainnet-laconic
+description: "Mainnet laconic node"
+repos:
+ - cerc-io/laconicd
+ - lirewine/debug
+ - lirewine/crypto
+ - lirewine/gem
+ - lirewine/sdk
+ - cerc-io/laconic-sdk
+ - cerc-io/laconic-registry-cli
+ - cerc-io/laconic-console
+npms:
+ - laconic-sdk
+ - laconic-registry-cli
+ - debug
+ - crypto
+ - sdk
+ - gem
+ - laconic-console
+containers:
+ - cerc/laconicd
+ - cerc/laconic-registry-cli
+ - cerc/laconic-console-host
+pods:
+ - mainnet-laconicd
+ - fixturenet-laconic-console
+config:
+ cli:
+ key: laconicd.mykey
+ address: laconicd.myaddress
diff --git a/app/data/stacks/mainnet-laconic/test/run-mainnet-laconic-test.sh b/app/data/stacks/mainnet-laconic/test/run-mainnet-laconic-test.sh
new file mode 100755
index 00000000..3e25f5dc
--- /dev/null
+++ b/app/data/stacks/mainnet-laconic/test/run-mainnet-laconic-test.sh
@@ -0,0 +1,48 @@
+#!/usr/bin/env bash
+set -e
+if [ -n "$CERC_SCRIPT_DEBUG" ]; then
+ set -x
+fi
+# Dump environment variables for debugging
+echo "Environment variables:"
+env
+# Test laconic stack
+echo "Running laconic stack test"
+# Bit of a hack, test the most recent package
+TEST_TARGET_SO=$( ls -t1 ./package/laconic-so* | head -1 )
+# Set a non-default repo dir
+export CERC_REPO_BASE_DIR=~/stack-orchestrator-test/repo-base-dir
+echo "Testing this package: $TEST_TARGET_SO"
+echo "Test version command"
+reported_version_string=$( $TEST_TARGET_SO version )
+echo "Version reported is: ${reported_version_string}"
+echo "Cloning repositories into: $CERC_REPO_BASE_DIR"
+rm -rf $CERC_REPO_BASE_DIR
+mkdir -p $CERC_REPO_BASE_DIR
+# Test bringing the test container up and down
+# with and without volume removal
+$TEST_TARGET_SO --stack test setup-repositories
+$TEST_TARGET_SO --stack test build-containers
+$TEST_TARGET_SO --stack test deploy up
+$TEST_TARGET_SO --stack test deploy down
+# The next time we bring the container up the volume will be old (from the previous run above)
+$TEST_TARGET_SO --stack test deploy up
+log_output_1=$( $TEST_TARGET_SO --stack test deploy logs )
+if [[ "$log_output_1" == *"Filesystem is old"* ]]; then
+ echo "Retain volumes test: passed"
+else
+ echo "Retain volumes test: FAILED"
+ exit 1
+fi
+$TEST_TARGET_SO --stack test deploy down --delete-volumes
+# Now when we bring the container up the volume will be new again
+$TEST_TARGET_SO --stack test deploy up
+log_output_2=$( $TEST_TARGET_SO --stack test deploy logs )
+if [[ "$log_output_2" == *"Filesystem is fresh"* ]]; then
+ echo "Delete volumes test: passed"
+else
+ echo "Delete volumes test: FAILED"
+ exit 1
+fi
+$TEST_TARGET_SO --stack test deploy down --delete-volumes
+echo "Test passed"
diff --git a/app/data/stacks/mobymask-v2/README.md b/app/data/stacks/mobymask-v2/README.md
index b4563c42..dfbabd09 100644
--- a/app/data/stacks/mobymask-v2/README.md
+++ b/app/data/stacks/mobymask-v2/README.md
@@ -31,17 +31,21 @@ Deploy the stack:
* Deploy the containers:
```bash
- laconic-so --stack mobymask-v2 deploy-system up
+ laconic-so --stack mobymask-v2 deploy --cluster mobymask_v2 up
```
-* List and check the health status of all the containers using `docker ps` and wait for them to be `healthy`
+ NOTE: The `fixturenet-optimism-contracts` service takes a while to run to completion and it may restart a few times after running into errors.
- NOTE: The `mobymask-app` container might not start; if the app is not running at http://localhost:3002, restart the container using it's id:
+* To list down and monitor the running containers:
```bash
- docker ps -a | grep "mobymask-app"
+ laconic-so --stack mobymask-v2 deploy --cluster mobymask_v2 ps
- docker restart
+ # With status
+ docker ps -a
+
+ # Check logs for a container
+ docker logs -f
```
## Tests
@@ -68,11 +72,12 @@ docker ps | grep -E 'mobymask-app|peer-test-app'
### mobymask-app
-The mobymask-app should be running at http://localhost:3002
+* The mobymask-app should be running at http://localhost:3002
+* The lxdao-mobymask-app should be running at http://localhost:3004
### peer-test-app
-The peer-test-app should be running at http://localhost:3003
+* The peer-test-app should be running at http://localhost:3003
## Details
@@ -91,15 +96,15 @@ Follow the [demo](./demo.md) to try out the MobyMask app with L2 chain
Stop all the services running in background run:
```bash
-laconic-so --stack mobymask-v2 deploy-system down 30
+laconic-so --stack mobymask-v2 deploy --cluster mobymask_v2 down 30
```
Clear volumes created by this stack:
```bash
# List all relevant volumes
-docker volume ls -q --filter "name=.*mobymask_watcher_db_data|.*peers_ids|.*mobymask_deployment|.*l1_deployment|.*l2_accounts|.*l2_config|.*l2_geth_data"
+docker volume ls -q --filter "name=mobymask_v2"
# Remove all the listed volumes
-docker volume rm $(docker volume ls -q --filter "name=.*mobymask_watcher_db_data|.*peers_ids|.*mobymask_deployment|.*l1_deployment|.*l2_accounts|.*l2_config|.*l2_geth_data")
+docker volume rm $(docker volume ls -q --filter "name=mobymask_v2")
```
diff --git a/app/data/stacks/mobymask-v2/demo.md b/app/data/stacks/mobymask-v2/demo.md
index 1e1d6f01..c84df507 100644
--- a/app/data/stacks/mobymask-v2/demo.md
+++ b/app/data/stacks/mobymask-v2/demo.md
@@ -9,23 +9,23 @@
The invite link is seen at the end of the logs. Example log:
```bash
- laconic-bfb01caf98b1b8f7c8db4d33f11b905a-mobymask-1 | http://127.0.0.1:3002/#/members?invitation=%7B%22v%22%3A1%2C%22signedDelegations%22%3A%5B%7B%22signature%22%3A%220x7559bd412f02677d60820e38243acf61547f79339395a34f7d4e1630e645aeb30535fc219f79b6fbd3af0ce3bd05132ad46d2b274a9fbc4c36bc71edd09850891b%22%2C%22delegation%22%3A%7B%22delegate%22%3A%220xc0838c92B2b71756E0eAD5B3C1e1F186baeEAAac%22%2C%22authority%22%3A%220x0000000000000000000000000000000000000000000000000000000000000000%22%2C%22caveats%22%3A%5B%7B%22enforcer%22%3A%220x558024C7d593B840E1BfD83E9B287a5CDad4db15%22%2C%22terms%22%3A%220x0000000000000000000000000000000000000000000000000000000000000000%22%7D%5D%7D%7D%5D%2C%22key%22%3A%220x98da9805821f1802196443e578fd32af567bababa0a249c07c82df01ecaa7d8d%22%7D
+ http://127.0.0.1:3004/#/members?invitation=%7B%22v%22%3A1%2C%22signedDelegations%22%3A%5B%7B%22signature%22%3A%220x7559bd412f02677d60820e38243acf61547f79339395a34f7d4e1630e645aeb30535fc219f79b6fbd3af0ce3bd05132ad46d2b274a9fbc4c36bc71edd09850891b%22%2C%22delegation%22%3A%7B%22delegate%22%3A%220xc0838c92B2b71756E0eAD5B3C1e1F186baeEAAac%22%2C%22authority%22%3A%220x0000000000000000000000000000000000000000000000000000000000000000%22%2C%22caveats%22%3A%5B%7B%22enforcer%22%3A%220x558024C7d593B840E1BfD83E9B287a5CDad4db15%22%2C%22terms%22%3A%220x0000000000000000000000000000000000000000000000000000000000000000%22%7D%5D%7D%7D%5D%2C%22key%22%3A%220x98da9805821f1802196443e578fd32af567bababa0a249c07c82df01ecaa7d8d%22%7D
```
* Open the invite link in a browser to use the mobymask-app.
- NOTE: Before opening the invite link, clear the browser cache (local storage) for http://127.0.0.1:3002 to remove old invitations
+ NOTE: Before opening the invite link, clear the browser cache (local storage) for http://127.0.0.1:3004 to remove old invitations
* In the debug panel, check if it is connected to the p2p network (it should be connected to at least one other peer for pubsub to work).
-* Create an invite link in the app by clicking on `Create new invite link` button.
+* Create an invite link in the app by clicking on `Create new invite link` button in the `My invitees` section.
* Switch to the `MESSAGES` tab in debug panel for viewing incoming messages later.
* Open the invite link in a new browser with different profile (to simulate remote browser)
- * Check that it is connected to any other peer in the network.
+ * Check that it is connected to a peer in the network.
-* In `Report a phishing attempt` section, report multiple phishers using the `Submit` button. Click on the `Submit batch to p2p network` button. This broadcasts signed invocations to the connected peers.
+* In the `Pending reports` section, enter multiple phisher records and click on the `Submit batch to p2p network` button. This broadcasts signed invocations to the connected peers.
* In the `MESSAGES` tab of other browsers, a message can be seen with the signed invocations.
@@ -66,7 +66,7 @@
* Get the deployed contract address:
```bash
- docker exec -it $(docker ps -aq --filter name="mobymask-app") cat /config/config.yml
+ docker exec -it $(docker ps -aq --filter name="lxdao-mobymask-app") cat /config/config.yml
```
The value of `address` field is the deployed contract address
@@ -91,15 +91,14 @@
* Watcher internally is using L2 chain `eth_getStorageAt` method.
* Check the phisher name in mobymask app in `Check Phisher Status` section.
- * Watcher GQL API is used for checking phisher.
+ * Watcher GQL API is used for checking phisher.
-* Manage the invitations by clicking on the `Outstanding Invitations in p2p network`.
-
-* Revoke the created invitation by clicking on `Revoke (p2p network)`
+* Manage invitations in the `Outstanding invitations (p2p network)` tab in `My Invitations` section.
+ * Revoke the created invitation by clicking on the `Revoke` button.
* Revocation messages can be seen in the debug panel `MESSAGES` tab of other browsers.
-* Check the watcher peer logs. It should receive a message and log the transaction receipt for a `revoke` message.
+* Also, check the watcher peer logs. It should receive a message and log the transaction receipt for a `revoke` message.
* Try reporting a phisher from the revoked invitee's browser.
diff --git a/app/data/stacks/mobymask-v2/mobymask-only.md b/app/data/stacks/mobymask-v2/mobymask-only.md
index 1e525f60..eb8f3153 100644
--- a/app/data/stacks/mobymask-v2/mobymask-only.md
+++ b/app/data/stacks/mobymask-v2/mobymask-only.md
@@ -49,7 +49,7 @@ Create and update an env file to be used in the next step ([defaults](../../conf
# Base URI for mobymask-app
# (used for generating a root invite link after deploying the contract)
- CERC_MOBYMASK_APP_BASE_URI="http://127.0.0.1:3002/#"
+ CERC_MOBYMASK_APP_BASE_URI="http://127.0.0.1:3004/#"
# (Optional) Domain to be used in the relay node's announce address
CERC_RELAY_ANNOUNCE_DOMAIN=
@@ -72,16 +72,16 @@ Create and update an env file to be used in the next step ([defaults](../../conf
### Deploy the stack
```bash
-laconic-so --stack mobymask-v2 deploy --include watcher-mobymask-v2 --env-file up
+laconic-so --stack mobymask-v2 deploy --cluster mobymask_v2 --include watcher-mobymask-v2 --env-file up
```
To list down and monitor the running containers:
```bash
-laconic-so --stack mobymask-v2 deploy --include watcher-mobymask-v2 ps
+laconic-so --stack mobymask-v2 deploy --cluster mobymask_v2 --include watcher-mobymask-v2 ps
# With status
-docker ps
+docker ps -a
# Check logs for a container
docker logs -f
@@ -108,15 +108,15 @@ For deploying the web-app(s) separately after deploying the watcher, follow [web
Stop all services running in the background:
```bash
-laconic-so --stack mobymask-v2 deploy --include watcher-mobymask-v2 down
+laconic-so --stack mobymask-v2 deploy --cluster mobymask_v2 --include watcher-mobymask-v2 down
```
Clear volumes created by this stack:
```bash
# List all relevant volumes
-docker volume ls -q --filter "name=.*mobymask_watcher_db_data|.*peers_ids|.*mobymask_deployment"
+docker volume ls -q --filter "name=mobymask_v2"
# Remove all the listed volumes
-docker volume rm $(docker volume ls -q --filter "name=.*mobymask_watcher_db_data|.*peers_ids|.*mobymask_deployment")
+docker volume rm $(docker volume ls -q --filter "name=mobymask_v2")
```
diff --git a/app/data/stacks/mobymask-v2/watcher-p2p-network/watcher.md b/app/data/stacks/mobymask-v2/watcher-p2p-network/watcher.md
index 682ca0e7..84898df6 100644
--- a/app/data/stacks/mobymask-v2/watcher-p2p-network/watcher.md
+++ b/app/data/stacks/mobymask-v2/watcher-p2p-network/watcher.md
@@ -110,8 +110,8 @@ To list down and monitor the running containers:
# Expected output:
# Running containers:
- # id: 25cc3a1cbda27fcd9c2ad4c772bd753ccef1e178f901a70e6ff4191d4a8684e9, name: mobymask_v2-mobymask-watcher-db-1, ports: 0.0.0.0:15432->5432/tcp
- # id: c9806f78680d68292ffe942222af2003aa3ed5d5c69d7121b573f5028444391d, name: mobymask_v2-mobymask-watcher-server-1, ports: 0.0.0.0:3001->3001/tcp, 0.0.0.0:9001->9001/tcp, 0.0.0.0:9090->9090/tcp
+ # id: 25cc3a1cbda27fcd9c2ad4c772bd753ccef1e178f901a70e6ff4191d4a8684e9, name: mobymask_v2-mobymask-watcher-db-1, ports: 127.0.0.1:15432->5432/tcp
+ # id: c9806f78680d68292ffe942222af2003aa3ed5d5c69d7121b573f5028444391d, name: mobymask_v2-mobymask-watcher-server-1, ports: 127.0.0.1:3001->3001/tcp, 127.0.0.1:9001->9001/tcp, 127.0.0.1:9090->9090/tcp
# id: 6b30a1d313a88fb86f8a3b37a1b1a3bc053f238664e4b2d196c3ec74e04faf13, name: mobymask_v2-peer-tests-1, ports:
@@ -122,8 +122,8 @@ To list down and monitor the running containers:
# CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
# 6b30a1d313a8 cerc/watcher-ts:local "docker-entrypoint.s…" 5 minutes ago Up 4 minutes mobymask_v2-peer-tests-1
- # c9806f78680d cerc/watcher-mobymask-v2:local "sh start-server.sh" 5 minutes ago Up 5 minutes (healthy) 0.0.0.0:3001->3001/tcp, 0.0.0.0:9001->9001/tcp, 0.0.0.0:9090->9090/tcp mobymask_v2-mobymask-watcher-server-1
- # 25cc3a1cbda2 postgres:14-alpine "docker-entrypoint.s…" 5 minutes ago Up 5 minutes (healthy) 0.0.0.0:15432->5432/tcp mobymask_v2-mobymask-watcher-db-1
+ # c9806f78680d cerc/watcher-mobymask-v2:local "sh start-server.sh" 5 minutes ago Up 5 minutes (healthy) 127.0.0.1:3001->3001/tcp, 127.0.0.1:9001->9001/tcp, 127.0.0.1:9090->9090/tcp mobymask_v2-mobymask-watcher-server-1
+ # 25cc3a1cbda2 postgres:14-alpine "docker-entrypoint.s…" 5 minutes ago Up 5 minutes (healthy) 127.0.0.1:15432->5432/tcp mobymask_v2-mobymask-watcher-db-1
# Check logs for a container
diff --git a/app/data/stacks/mobymask-v2/watcher-p2p-network/web-app.md b/app/data/stacks/mobymask-v2/watcher-p2p-network/web-app.md
index cf1821b5..f97d44bb 100644
--- a/app/data/stacks/mobymask-v2/watcher-p2p-network/web-app.md
+++ b/app/data/stacks/mobymask-v2/watcher-p2p-network/web-app.md
@@ -78,7 +78,7 @@ To monitor the running container:
# Expected output:
# CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
- # f1369dbae1c9 cerc/mobymask-ui:local "docker-entrypoint.s…" 2 minutes ago Up 2 minutes (healthy) 0.0.0.0:3004->80/tcp mm_v2-lxdao-mobymask-app-1
+ # f1369dbae1c9 cerc/mobymask-ui:local "docker-entrypoint.s…" 2 minutes ago Up 2 minutes (healthy) 127.0.0.1:3004->80/tcp mm_v2-lxdao-mobymask-app-1
# Check logs for a container
docker logs -f mm_v2-lxdao-mobymask-app-1
diff --git a/app/data/stacks/mobymask-v2/web-apps.md b/app/data/stacks/mobymask-v2/web-apps.md
index d1570c93..ade5953b 100644
--- a/app/data/stacks/mobymask-v2/web-apps.md
+++ b/app/data/stacks/mobymask-v2/web-apps.md
@@ -47,14 +47,14 @@ Create and update an env file to be used in the next step ([defaults](../../conf
For running mobymask-app
```bash
-laconic-so --stack mobymask-v2 deploy --include mobymask-app --env-file up
+laconic-so --stack mobymask-v2 deploy --cluster mobymask_v2 --include mobymask-app --env-file up
# Runs mobymask-app on host port 3002 and lxdao-mobymask-app on host port 3004
```
For running peer-test-app
```bash
-laconic-so --stack mobymask-v2 deploy --include peer-test-app --env-file up
+laconic-so --stack mobymask-v2 deploy --cluster mobymask_v2 --include peer-test-app --env-file up
# Runs on host port 3003
```
@@ -62,9 +62,10 @@ laconic-so --stack mobymask-v2 deploy --include peer-test-app --env-file
@@ -80,20 +81,20 @@ Stop all services running in the background:
For mobymask-app
```bash
-laconic-so --stack mobymask-v2 deploy --include mobymask-app down
+laconic-so --stack mobymask-v2 deploy --cluster mobymask_v2 --include mobymask-app down
```
For peer-test-app
```bash
-laconic-so --stack mobymask-v2 deploy --include peer-test-app down
+laconic-so --stack mobymask-v2 deploy --cluster mobymask_v2 --include peer-test-app down
```
Clear volumes created by this stack:
```bash
# List all relevant volumes
-docker volume ls -q --filter "name=.*mobymask_deployment|.*peers_ids"
+docker volume ls -q --filter "name=mobymask_v2"
# Remove all the listed volumes
-docker volume rm $(docker volume ls -q --filter "name=.*mobymask_deployment|.*peers_ids")
+docker volume rm $(docker volume ls -q --filter "name=mobymask_v2")
```
diff --git a/app/deploy_system.py b/app/deploy.py
similarity index 86%
rename from app/deploy_system.py
rename to app/deploy.py
index 2782ce5d..3f769f3e 100644
--- a/app/deploy_system.py
+++ b/app/deploy.py
@@ -21,12 +21,15 @@ import os
import sys
from dataclasses import dataclass
from decouple import config
+from importlib import resources
import subprocess
from python_on_whales import DockerClient, DockerException
import click
-import importlib.resources
from pathlib import Path
-from .util import include_exclude_check, get_parsed_stack_config
+from .util import include_exclude_check, get_parsed_stack_config, global_options2
+from .deployment_create import create as deployment_create
+from .deployment_create import init as deployment_init
+
class DeployCommandContext(object):
def __init__(self, cluster_context, docker):
@@ -43,44 +46,40 @@ class DeployCommandContext(object):
def command(ctx, include, exclude, env_file, cluster):
'''deploy a stack'''
- cluster_context = _make_cluster_context(ctx.obj, include, exclude, cluster, env_file)
-
- # See: https://gabrieldemarmiesse.github.io/python-on-whales/sub-commands/compose/
- docker = DockerClient(compose_files=cluster_context.compose_files, compose_project_name=cluster_context.cluster,
- compose_env_file=cluster_context.env_file)
-
- ctx.obj = DeployCommandContext(cluster_context, docker)
+ if ctx.parent.obj.debug:
+ print(f"ctx.parent.obj: {ctx.parent.obj}")
+ ctx.obj = create_deploy_context(global_options2(ctx), global_options2(ctx).stack, include, exclude, cluster, env_file)
# Subcommand is executed now, by the magic of click
-@command.command()
-@click.argument('extra_args', nargs=-1) # help: command: up
-@click.pass_context
-def up(ctx, extra_args):
+def create_deploy_context(global_context, stack, include, exclude, cluster, env_file):
+ cluster_context = _make_cluster_context(global_context, stack, include, exclude, cluster, env_file)
+ # See: https://gabrieldemarmiesse.github.io/python-on-whales/sub-commands/compose/
+ docker = DockerClient(compose_files=cluster_context.compose_files, compose_project_name=cluster_context.cluster,
+ compose_env_file=cluster_context.env_file)
+ return DeployCommandContext(cluster_context, docker)
+
+
+def up_operation(ctx, services_list):
global_context = ctx.parent.parent.obj
- extra_args_list = list(extra_args) or None
+ deploy_context = ctx.obj
if not global_context.dry_run:
- cluster_context = ctx.obj.cluster_context
+ cluster_context = deploy_context.cluster_context
container_exec_env = _make_runtime_env(global_context)
for attr, value in container_exec_env.items():
os.environ[attr] = value
if global_context.verbose:
- print(f"Running compose up with container_exec_env: {container_exec_env}, extra_args: {extra_args_list}")
+ print(f"Running compose up with container_exec_env: {container_exec_env}, extra_args: {services_list}")
for pre_start_command in cluster_context.pre_start_commands:
_run_command(global_context, cluster_context.cluster, pre_start_command)
- ctx.obj.docker.compose.up(detach=True, services=extra_args_list)
+ deploy_context.docker.compose.up(detach=True, services=services_list)
for post_start_command in cluster_context.post_start_commands:
_run_command(global_context, cluster_context.cluster, post_start_command)
- _orchestrate_cluster_config(global_context, cluster_context.config, ctx.obj.docker, container_exec_env)
+ _orchestrate_cluster_config(global_context, cluster_context.config, deploy_context.docker, container_exec_env)
-@command.command()
-@click.option("--delete-volumes/--preserve-volumes", default=False, help="delete data volumes")
-@click.argument('extra_args', nargs=-1) # help: command: down
-@click.pass_context
-def down(ctx, delete_volumes, extra_args):
+def down_operation(ctx, delete_volumes, extra_args_list):
global_context = ctx.parent.parent.obj
- extra_args_list = list(extra_args) or None
if not global_context.dry_run:
if global_context.verbose:
print("Running compose down")
@@ -91,9 +90,7 @@ def down(ctx, delete_volumes, extra_args):
ctx.obj.docker.compose.down(timeout=timeout_arg, volumes=delete_volumes)
-@command.command()
-@click.pass_context
-def ps(ctx):
+def ps_operation(ctx):
global_context = ctx.parent.parent.obj
if not global_context.dry_run:
if global_context.verbose:
@@ -118,10 +115,7 @@ def ps(ctx):
print("No containers running")
-@command.command()
-@click.argument('extra_args', nargs=-1) # help: command: port
-@click.pass_context
-def port(ctx, extra_args):
+def port_operation(ctx, extra_args):
global_context = ctx.parent.parent.obj
extra_args_list = list(extra_args) or None
if not global_context.dry_run:
@@ -136,10 +130,7 @@ def port(ctx, extra_args):
print(f"{mapped_port_data[0]}:{mapped_port_data[1]}")
-@command.command()
-@click.argument('extra_args', nargs=-1) # help: command: exec
-@click.pass_context
-def exec(ctx, extra_args):
+def exec_operation(ctx, extra_args):
global_context = ctx.parent.parent.obj
extra_args_list = list(extra_args) or None
if not global_context.dry_run:
@@ -157,10 +148,7 @@ def exec(ctx, extra_args):
print(f"container command returned error exit status")
-@command.command()
-@click.argument('extra_args', nargs=-1) # help: command: logs
-@click.pass_context
-def logs(ctx, extra_args):
+def logs_operation(ctx, extra_args):
global_context = ctx.parent.parent.obj
extra_args_list = list(extra_args) or None
if not global_context.dry_run:
@@ -170,12 +158,56 @@ def logs(ctx, extra_args):
print(logs_output)
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: up
+@click.pass_context
+def up(ctx, extra_args):
+ extra_args_list = list(extra_args) or None
+ up_operation(ctx, extra_args_list)
+
+
+@command.command()
+@click.option("--delete-volumes/--preserve-volumes", default=False, help="delete data volumes")
+@click.argument('extra_args', nargs=-1) # help: command: down
+@click.pass_context
+def down(ctx, delete_volumes, extra_args):
+ extra_args_list = list(extra_args) or None
+ down_operation(ctx, delete_volumes, extra_args_list)
+
+
+@command.command()
+@click.pass_context
+def ps(ctx):
+ ps_operation(ctx)
+
+
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: port
+@click.pass_context
+def port(ctx, extra_args):
+ port_operation(ctx, extra_args)
+
+
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: exec
+@click.pass_context
+def exec(ctx, extra_args):
+ exec_operation(ctx, extra_args)
+
+
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: logs
+@click.pass_context
+def logs(ctx, extra_args):
+ logs_operation(ctx, extra_args)
+
+
def get_stack_status(ctx, stack):
ctx_copy = copy.copy(ctx)
ctx_copy.stack = stack
- cluster_context = _make_cluster_context(ctx_copy, None, None, None, None)
+ cluster_context = _make_cluster_context(ctx_copy, stack, None, None, None, None)
docker = DockerClient(compose_files=cluster_context.compose_files, compose_project_name=cluster_context.cluster)
# TODO: refactor to avoid duplicating this code above
if ctx.verbose:
@@ -200,7 +232,8 @@ def _make_runtime_env(ctx):
return container_exec_env
-def _make_cluster_context(ctx, include, exclude, cluster, env_file):
+# stack has to be either PathLike pointing to a stack yml file, or a string with the name of a known stack
+def _make_cluster_context(ctx, stack, include, exclude, cluster, env_file):
if ctx.local_stack:
dev_root_path = os.getcwd()[0:os.getcwd().rindex("stack-orchestrator")]
@@ -208,14 +241,20 @@ def _make_cluster_context(ctx, include, exclude, cluster, env_file):
else:
dev_root_path = os.path.expanduser(config("CERC_REPO_BASE_DIR", default="~/cerc"))
- # See: https://stackoverflow.com/questions/25389095/python-get-path-of-root-project-structure
- compose_dir = Path(__file__).absolute().parent.joinpath("data", "compose")
+ # TODO: huge hack, fix this
+ # If the caller passed a path for the stack file, then we know that we can get the compose files
+ # from the same directory
+ if isinstance(stack, os.PathLike):
+ compose_dir = stack.parent.joinpath("compose")
+ else:
+ # See: https://stackoverflow.com/questions/25389095/python-get-path-of-root-project-structure
+ compose_dir = Path(__file__).absolute().parent.joinpath("data", "compose")
if cluster is None:
# Create default unique, stable cluster name from confile file path and stack name if provided
# TODO: change this to the config file path
path = os.path.realpath(sys.argv[0])
- unique_cluster_descriptor = f"{path},{ctx.stack},{include},{exclude}"
+ unique_cluster_descriptor = f"{path},{stack},{include},{exclude}"
if ctx.debug:
print(f"pre-hash descriptor: {unique_cluster_descriptor}")
hash = hashlib.md5(unique_cluster_descriptor.encode()).hexdigest()
@@ -225,12 +264,12 @@ def _make_cluster_context(ctx, include, exclude, cluster, env_file):
# See: https://stackoverflow.com/a/20885799/1701505
from . import data
- with importlib.resources.open_text(data, "pod-list.txt") as pod_list_file:
+ with resources.open_text(data, "pod-list.txt") as pod_list_file:
all_pods = pod_list_file.read().splitlines()
pods_in_scope = []
- if ctx.stack:
- stack_config = get_parsed_stack_config(ctx.stack)
+ if stack:
+ stack_config = get_parsed_stack_config(stack)
# TODO: syntax check the input here
pods_in_scope = stack_config['pods']
cluster_config = stack_config['config'] if 'config' in stack_config else None
@@ -342,6 +381,7 @@ def _orchestrate_cluster_config(ctx, cluster_config, docker, container_exec_env)
f" = {pd.source_container}.{pd.source_variable}")
# TODO: add a timeout
waiting_for_data = True
+ destination_output = "*** no output received yet ***"
while waiting_for_data:
# TODO: fix the script paths so they're consistent between containers
source_value = None
@@ -376,3 +416,7 @@ def _orchestrate_cluster_config(ctx, cluster_config, docker, container_exec_env)
waiting_for_data = False
if ctx.debug:
print(f"destination output: {destination_output}")
+
+
+command.add_command(deployment_init)
+command.add_command(deployment_create)
diff --git a/app/deployment.py b/app/deployment.py
new file mode 100644
index 00000000..fa2236aa
--- /dev/null
+++ b/app/deployment.py
@@ -0,0 +1,140 @@
+# Copyright © 2022, 2023 Cerc
+
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU Affero General Public License for more details.
+
+# You should have received a copy of the GNU Affero General Public License
+# along with this program. If not, see .
+
+import click
+from dataclasses import dataclass
+from pathlib import Path
+import sys
+from .deploy import up_operation, down_operation, ps_operation, port_operation, exec_operation, logs_operation, create_deploy_context
+from .util import global_options
+
+
+@dataclass
+class DeploymentContext:
+ dir: Path
+
+
+@click.group()
+@click.option("--dir", required=True, help="path to deployment directory")
+@click.pass_context
+def command(ctx, dir):
+ # Check that --stack wasn't supplied
+ if ctx.parent.obj.stack:
+ print("Error: --stack can't be supplied with the deployment command")
+ sys.exit(1)
+ # Check dir is valid
+ dir_path = Path(dir)
+ if not dir_path.exists():
+ print(f"Error: deployment directory {dir} does not exist")
+ sys.exit(1)
+ if not dir_path.is_dir():
+ print(f"Error: supplied deployment directory path {dir} exists but is a file not a directory")
+ sys.exit(1)
+ # Store the deployment context for subcommands
+ ctx.obj = DeploymentContext(dir_path)
+
+
+def make_deploy_context(ctx):
+ # Get the stack config file name
+ stack_file_path = ctx.obj.dir.joinpath("stack.yml")
+ # TODO: add cluster name and env file here
+ return create_deploy_context(ctx.parent.parent.obj, stack_file_path, None, None, None, None)
+
+
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: up
+@click.pass_context
+def up(ctx, extra_args):
+ ctx.obj = make_deploy_context(ctx)
+ services_list = list(extra_args) or None
+ up_operation(ctx, services_list)
+
+
+# start is the preferred alias for up
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: up
+@click.pass_context
+def start(ctx, extra_args):
+ ctx.obj = make_deploy_context(ctx)
+ services_list = list(extra_args) or None
+ up_operation(ctx, services_list)
+
+
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: down
+@click.pass_context
+def down(ctx, extra_args):
+ # Get the stack config file name
+ # TODO: add cluster name and env file here
+ ctx.obj = make_deploy_context(ctx)
+ down_operation(ctx, extra_args, None)
+
+
+# stop is the preferred alias for down
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: down
+@click.pass_context
+def stop(ctx, extra_args):
+ # TODO: add cluster name and env file here
+ ctx.obj = make_deploy_context(ctx)
+ down_operation(ctx, extra_args, None)
+
+
+@command.command()
+@click.pass_context
+def ps(ctx):
+ ctx.obj = make_deploy_context(ctx)
+ ps_operation(ctx)
+
+
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: port
+@click.pass_context
+def port(ctx, extra_args):
+ port_operation(ctx, extra_args)
+
+
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: exec
+@click.pass_context
+def exec(ctx, extra_args):
+ ctx.obj = make_deploy_context(ctx)
+ exec_operation(ctx, extra_args)
+
+
+@command.command()
+@click.argument('extra_args', nargs=-1) # help: command: logs
+@click.pass_context
+def logs(ctx, extra_args):
+ ctx.obj = make_deploy_context(ctx)
+ logs_operation(ctx, extra_args)
+
+
+@command.command()
+@click.pass_context
+def status(ctx):
+ print(f"Context: {ctx.parent.obj}")
+
+
+
+#from importlib import resources, util
+# TODO: figure out how to do this dynamically
+#stack = "mainnet-laconic"
+#module_name = "commands"
+#spec = util.spec_from_file_location(module_name, "./app/data/stacks/" + stack + "/deploy/commands.py")
+#imported_stack = util.module_from_spec(spec)
+#spec.loader.exec_module(imported_stack)
+#command.add_command(imported_stack.init)
+#command.add_command(imported_stack.create)
diff --git a/app/deployment_create.py b/app/deployment_create.py
new file mode 100644
index 00000000..5a9ddbcd
--- /dev/null
+++ b/app/deployment_create.py
@@ -0,0 +1,155 @@
+# Copyright © 2022, 2023 Cerc
+
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as published by
+# the Free Software Foundation, either version 3 of the License, or
+# (at your option) any later version.
+
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU Affero General Public License for more details.
+
+# You should have received a copy of the GNU Affero General Public License
+# along with this program. If not, see .
+
+import click
+import os
+from pathlib import Path
+from shutil import copyfile, copytree
+import sys
+import ruamel.yaml
+from .util import get_stack_file_path, get_parsed_deployment_spec, get_parsed_stack_config, global_options
+
+
+def _get_yaml():
+ # See: https://stackoverflow.com/a/45701840/1701505
+ yaml = ruamel.yaml.YAML()
+ yaml.preserve_quotes = True
+ yaml.indent(sequence=3, offset=1)
+ return yaml
+
+
+def _make_default_deployment_dir():
+ return "deployment-001"
+
+
+def _get_compose_file_dir():
+ # TODO: refactor to use common code with deploy command
+ # See: https://stackoverflow.com/questions/25389095/python-get-path-of-root-project-structure
+ data_dir = Path(__file__).absolute().parent.joinpath("data")
+ source_compose_dir = data_dir.joinpath("compose")
+ return source_compose_dir
+
+
+def _get_named_volumes(stack):
+ # Parse the compose files looking for named volumes
+ named_volumes = []
+ parsed_stack = get_parsed_stack_config(stack)
+ pods = parsed_stack["pods"]
+ yaml = _get_yaml()
+ for pod in pods:
+ pod_file_path = os.path.join(_get_compose_file_dir(), f"docker-compose-{pod}.yml")
+ parsed_pod_file = yaml.load(open(pod_file_path, "r"))
+ if "volumes" in parsed_pod_file:
+ volumes = parsed_pod_file["volumes"]
+ for volume in volumes.keys():
+ # Volume definition looks like:
+ # 'laconicd-data': None
+ named_volumes.append(volume)
+ return named_volumes
+
+
+# If we're mounting a volume from a relatie path, then we
+# assume the directory doesn't exist yet and create it
+# so the deployment will start
+# Also warn if the path is absolute and doesn't exist
+def _create_bind_dir_if_relative(volume, path_string, compose_dir):
+ path = Path(path_string)
+ if not path.is_absolute():
+ absolute_path = Path(compose_dir).parent.joinpath(path)
+ absolute_path.mkdir(parents=True, exist_ok=True)
+ else:
+ if not path.exists():
+ print(f"WARNING: mount path for volume {volume} does not exist: {path_string}")
+
+
+# See: https://stackoverflow.com/questions/45699189/editing-docker-compose-yml-with-pyyaml
+def _fixup_pod_file(pod, spec, compose_dir):
+ # Fix up volumes
+ if "volumes" in spec:
+ spec_volumes = spec["volumes"]
+ if "volumes" in pod:
+ pod_volumes = pod["volumes"]
+ for volume in pod_volumes.keys():
+ if volume in spec_volumes:
+ volume_spec = spec_volumes[volume]
+ volume_spec_fixedup = volume_spec if Path(volume_spec).is_absolute() else f".{volume_spec}"
+ _create_bind_dir_if_relative(volume, volume_spec, compose_dir)
+ new_volume_spec = {"driver": "local",
+ "driver_opts": {
+ "type": "none",
+ "device": volume_spec_fixedup,
+ "o": "bind"
+ }
+ }
+ pod["volumes"][volume] = new_volume_spec
+
+
+@click.command()
+@click.option("--output", required=True, help="Write yaml spec file here")
+@click.pass_context
+def init(ctx, output):
+ yaml = _get_yaml()
+ stack = global_options(ctx).stack
+ verbose = global_options(ctx).verbose
+ spec_file_content = {"stack": stack}
+ if verbose:
+ print(f"Creating spec file for stack: {stack}")
+ named_volumes = _get_named_volumes(stack)
+ if named_volumes:
+ volume_descriptors = {}
+ for named_volume in named_volumes:
+ volume_descriptors[named_volume] = f"../data/{named_volume}"
+ spec_file_content["volumes"] = volume_descriptors
+ with open(output, "w") as output_file:
+ yaml.dump(spec_file_content, output_file)
+
+
+@click.command()
+@click.option("--spec-file", required=True, help="Spec file to use to create this deployment")
+@click.option("--deployment-dir", help="Create deployment files in this directory")
+@click.pass_context
+def create(ctx, spec_file, deployment_dir):
+ # This function fails with a useful error message if the file doens't exist
+ parsed_spec = get_parsed_deployment_spec(spec_file)
+ stack_name = parsed_spec['stack']
+ stack_file = get_stack_file_path(stack_name)
+ parsed_stack = get_parsed_stack_config(stack_name)
+ if global_options(ctx).debug:
+ print(f"parsed spec: {parsed_spec}")
+ if deployment_dir is None:
+ deployment_dir = _make_default_deployment_dir()
+ if os.path.exists(deployment_dir):
+ print(f"Error: {deployment_dir} already exists")
+ sys.exit(1)
+ os.mkdir(deployment_dir)
+ # Copy spec file and the stack file into the deployment dir
+ copyfile(spec_file, os.path.join(deployment_dir, os.path.basename(spec_file)))
+ copyfile(stack_file, os.path.join(deployment_dir, os.path.basename(stack_file)))
+ # Copy the pod files into the deployment dir, fixing up content
+ pods = parsed_stack['pods']
+ destination_compose_dir = os.path.join(deployment_dir, "compose")
+ os.mkdir(destination_compose_dir)
+ data_dir = Path(__file__).absolute().parent.joinpath("data")
+ yaml = _get_yaml()
+ for pod in pods:
+ pod_file_path = os.path.join(_get_compose_file_dir(), f"docker-compose-{pod}.yml")
+ parsed_pod_file = yaml.load(open(pod_file_path, "r"))
+ _fixup_pod_file(parsed_pod_file, parsed_spec, destination_compose_dir)
+ with open(os.path.join(destination_compose_dir, os.path.basename(pod_file_path)), "w") as output_file:
+ yaml.dump(parsed_pod_file, output_file)
+ # Copy the config files for the pod, if any
+ source_config_dir = data_dir.joinpath("config", pod)
+ if os.path.exists(source_config_dir):
+ copytree(source_config_dir, os.path.join(deployment_dir, "config", pod))
diff --git a/app/setup_repositories.py b/app/setup_repositories.py
index bfcbe0ec..db0bd779 100644
--- a/app/setup_repositories.py
+++ b/app/setup_repositories.py
@@ -69,8 +69,26 @@ def host_and_path_for_repo(fully_qualified_repo):
return repo_host_split[0], "/".join(repo_host_split[1:]), repo_branch
+# See: https://stackoverflow.com/questions/18659425/get-git-current-branch-tag-name
+def _get_repo_current_branch_or_tag(full_filesystem_repo_path):
+ current_repo_branch_or_tag = "***UNDETERMINED***"
+ is_branch = False
+ try:
+ current_repo_branch_or_tag = git.Repo(full_filesystem_repo_path).active_branch.name
+ is_branch = True
+ except TypeError as error:
+ # This means that the current ref is not a branch, so possibly a tag
+ # Let's try to get the tag
+ current_repo_branch_or_tag = git.Repo(full_filesystem_repo_path).git.describe("--tags", "--exact-match")
+ # Note that git is assymetric -- the tag you told it to check out may not be the one
+ # you get back here (if there are multiple tags associated with the same commit)
+ return current_repo_branch_or_tag, is_branch
+
+
# TODO: fix the messy arg list here
def process_repo(verbose, quiet, dry_run, pull, check_only, git_ssh, dev_root_path, branches_array, fully_qualified_repo):
+ if verbose:
+ print(f"Processing repo: {fully_qualified_repo}")
repo_host, repo_path, repo_branch = host_and_path_for_repo(fully_qualified_repo)
git_ssh_prefix = f"git@{repo_host}:"
git_http_prefix = f"https://{repo_host}/"
@@ -78,9 +96,9 @@ def process_repo(verbose, quiet, dry_run, pull, check_only, git_ssh, dev_root_pa
repoName = repo_path.split("/")[-1]
full_filesystem_repo_path = os.path.join(dev_root_path, repoName)
is_present = os.path.isdir(full_filesystem_repo_path)
- current_repo_branch = git.Repo(full_filesystem_repo_path).active_branch.name if is_present else None
+ (current_repo_branch_or_tag, is_branch) = _get_repo_current_branch_or_tag(full_filesystem_repo_path) if is_present else (None, None)
if not quiet:
- present_text = f"already exists active branch: {current_repo_branch}" if is_present \
+ present_text = f"already exists active {'branch' if is_branch else 'tag'}: {current_repo_branch_or_tag}" if is_present \
else 'Needs to be fetched'
print(f"Checking: {full_filesystem_repo_path}: {present_text}")
# Quick check that it's actually a repo
@@ -93,9 +111,12 @@ def process_repo(verbose, quiet, dry_run, pull, check_only, git_ssh, dev_root_pa
if verbose:
print(f"Running git pull for {full_filesystem_repo_path}")
if not check_only:
- git_repo = git.Repo(full_filesystem_repo_path)
- origin = git_repo.remotes.origin
- origin.pull(progress=None if quiet else GitProgress())
+ if is_branch:
+ git_repo = git.Repo(full_filesystem_repo_path)
+ origin = git_repo.remotes.origin
+ origin.pull(progress=None if quiet else GitProgress())
+ else:
+ print(f"skipping pull because this repo checked out a tag")
else:
print("(git pull skipped)")
if not is_present:
@@ -122,14 +143,15 @@ def process_repo(verbose, quiet, dry_run, pull, check_only, git_ssh, dev_root_pa
branch_to_checkout = repo_branch
if branch_to_checkout:
- if current_repo_branch is None or (current_repo_branch and (current_repo_branch != branch_to_checkout)):
+ if current_repo_branch_or_tag is None or (current_repo_branch_or_tag and (current_repo_branch_or_tag != branch_to_checkout)):
if not quiet:
print(f"switching to branch {branch_to_checkout} in repo {repo_path}")
git_repo = git.Repo(full_filesystem_repo_path)
+ # git checkout works for both branches and tags
git_repo.git.checkout(branch_to_checkout)
else:
if verbose:
- print(f"repo {repo_path} is already switched to branch {branch_to_checkout}")
+ print(f"repo {repo_path} is already on branch/tag {branch_to_checkout}")
def parse_branches(branches_string):
diff --git a/app/util.py b/app/util.py
index 127de213..69eda4af 100644
--- a/app/util.py
+++ b/app/util.py
@@ -30,10 +30,16 @@ def include_exclude_check(s, include, exclude):
return s not in exclude_list
-def get_parsed_stack_config(stack):
+def get_stack_file_path(stack):
# In order to be compatible with Python 3.8 we need to use this hack to get the path:
# See: https://stackoverflow.com/questions/25389095/python-get-path-of-root-project-structure
stack_file_path = Path(__file__).absolute().parent.joinpath("data", "stacks", stack, "stack.yml")
+ return stack_file_path
+
+
+# Caller can pass either the name of a stack, or a path to a stack file
+def get_parsed_stack_config(stack):
+ stack_file_path = stack if isinstance(stack, os.PathLike) else get_stack_file_path(stack)
try:
with stack_file_path:
stack_config = yaml.safe_load(open(stack_file_path, "r"))
@@ -48,3 +54,27 @@ def get_parsed_stack_config(stack):
print(f"Error: stack: {stack} does not exist")
print(f"Exiting, error: {error}")
sys.exit(1)
+
+
+def get_parsed_deployment_spec(spec_file):
+ spec_file_path = Path(spec_file)
+ try:
+ with spec_file_path:
+ deploy_spec = yaml.safe_load(open(spec_file_path, "r"))
+ return deploy_spec
+ except FileNotFoundError as error:
+ # We try here to generate a useful diagnostic error
+ print(f"Error: spec file: {spec_file_path} does not exist")
+ print(f"Exiting, error: {error}")
+ sys.exit(1)
+
+
+# TODO: this is fragile wrt to the subcommand depth
+# See also: https://github.com/pallets/click/issues/108
+def global_options(ctx):
+ return ctx.parent.parent.obj
+
+
+# TODO: hack
+def global_options2(ctx):
+ return ctx.parent.obj
diff --git a/cli.py b/cli.py
index b0d2f34c..e9a11d25 100644
--- a/cli.py
+++ b/cli.py
@@ -19,8 +19,9 @@ from dataclasses import dataclass
from app import setup_repositories
from app import build_containers
from app import build_npms
-from app import deploy_system
+from app import deploy
from app import version
+from app import deployment
CONTEXT_SETTINGS = dict(help_option_names=['-h', '--help'])
@@ -54,6 +55,7 @@ def cli(ctx, stack, quiet, verbose, dry_run, local_stack, debug, continue_on_err
cli.add_command(setup_repositories.command, "setup-repositories")
cli.add_command(build_containers.command, "build-containers")
cli.add_command(build_npms.command, "build-npms")
-cli.add_command(deploy_system.command, "deploy") # deploy is an alias for deploy-system
-cli.add_command(deploy_system.command, "deploy-system")
+cli.add_command(deploy.command, "deploy") # deploy is an alias for deploy-system
+cli.add_command(deploy.command, "deploy-system")
+cli.add_command(deployment.command, "deployment")
cli.add_command(version.command, "version")
diff --git a/requirements.txt b/requirements.txt
index 8249a55a..6264dcb6 100644
--- a/requirements.txt
+++ b/requirements.txt
@@ -4,3 +4,4 @@ tqdm>=4.64.0
python-on-whales>=0.58.0
click>=8.1.3
pyyaml>=6.0
+ruamel.yaml>=0.17.32