2026-04-28 15:00:58 -04:00
|
|
|
|
#!/usr/bin/env bats
|
|
|
|
|
|
# tests/lifecycle/bats/required-stack.bats
|
|
|
|
|
|
#
|
|
|
|
|
|
# Read-only release-gate checks for the required Bitcoin stack on .116.
|
|
|
|
|
|
#
|
|
|
|
|
|
# This suite is intentionally non-destructive and does not use RPC auth;
|
|
|
|
|
|
# it can run anytime as a health gate during long sync/reindex windows.
|
|
|
|
|
|
|
|
|
|
|
|
required_containers=(
|
|
|
|
|
|
"bitcoin-knots"
|
|
|
|
|
|
"electrumx"
|
|
|
|
|
|
"lnd"
|
2026-05-13 15:09:22 -04:00
|
|
|
|
"archy-mempool-db"
|
2026-04-28 15:00:58 -04:00
|
|
|
|
"mempool-api"
|
|
|
|
|
|
"mempool"
|
2026-05-13 15:09:22 -04:00
|
|
|
|
"filebrowser"
|
2026-04-28 15:00:58 -04:00
|
|
|
|
"archy-bitcoin-ui"
|
|
|
|
|
|
"archy-lnd-ui"
|
|
|
|
|
|
"archy-electrs-ui"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
podman_names() {
|
|
|
|
|
|
podman ps --format '{{.Names}}'
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
container_running() {
|
|
|
|
|
|
local name="$1"
|
|
|
|
|
|
podman inspect --format '{{.State.Running}}' "$name" 2>/dev/null
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-05-13 15:09:22 -04:00
|
|
|
|
bitcoin_rpc() {
|
|
|
|
|
|
curl -fsS --max-time 60 \
|
|
|
|
|
|
--user "archipelago:$(cat /var/lib/archipelago/secrets/bitcoin-rpc-password)" \
|
|
|
|
|
|
--data-binary '{"jsonrpc":"1.0","id":"required-stack","method":"getblockchaininfo","params":[]}' \
|
|
|
|
|
|
-H 'content-type: text/plain;' \
|
|
|
|
|
|
http://127.0.0.1:8332/
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bitcoin_json() {
|
|
|
|
|
|
python3 -c 'import json,sys; r=json.load(sys.stdin)["result"]; print(r[sys.argv[1]])' "$1"
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-28 15:00:58 -04:00
|
|
|
|
@test "required containers are present" {
|
2026-06-22 17:11:15 -04:00
|
|
|
|
# Under sustained 5× churn an app may still be mid-restart when this runs;
|
|
|
|
|
|
# wait for the whole required set rather than single-shot.
|
|
|
|
|
|
local deadline=$(( $(date +%s) + 180 )) names missing
|
|
|
|
|
|
while (( $(date +%s) < deadline )); do
|
|
|
|
|
|
names="$(podman_names)"; missing=""
|
|
|
|
|
|
for c in "${required_containers[@]}"; do
|
|
|
|
|
|
echo "$names" | grep -Fx "$c" >/dev/null || missing="$missing $c"
|
|
|
|
|
|
done
|
|
|
|
|
|
[[ -z "$missing" ]] && return 0
|
|
|
|
|
|
sleep 3
|
2026-04-28 15:00:58 -04:00
|
|
|
|
done
|
2026-06-22 17:11:15 -04:00
|
|
|
|
fail "required containers never all present; missing:$missing"
|
2026-04-28 15:00:58 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "required containers are running" {
|
2026-06-22 17:11:15 -04:00
|
|
|
|
local deadline=$(( $(date +%s) + 180 )) notrunning
|
|
|
|
|
|
while (( $(date +%s) < deadline )); do
|
|
|
|
|
|
notrunning=""
|
|
|
|
|
|
for c in "${required_containers[@]}"; do
|
|
|
|
|
|
[[ "$(container_running "$c" 2>/dev/null)" == "true" ]] || notrunning="$notrunning $c"
|
|
|
|
|
|
done
|
|
|
|
|
|
[[ -z "$notrunning" ]] && return 0
|
|
|
|
|
|
sleep 3
|
2026-04-28 15:00:58 -04:00
|
|
|
|
done
|
2026-06-22 17:11:15 -04:00
|
|
|
|
fail "required containers never all running; not-running:$notrunning"
|
2026-04-28 15:00:58 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "bitcoin-knots RPC responds" {
|
2026-05-13 15:09:22 -04:00
|
|
|
|
run bitcoin_rpc
|
|
|
|
|
|
[ "$status" -eq 0 ]
|
|
|
|
|
|
echo "$output" | python3 -c 'import json,sys; r=json.load(sys.stdin)["result"]; assert r["chain"] == "main" and r["blocks"] >= 0'
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "bitcoin backend is synced archival for electrumx/lnd gate" {
|
|
|
|
|
|
run bitcoin_rpc
|
2026-04-28 15:00:58 -04:00
|
|
|
|
[ "$status" -eq 0 ]
|
2026-05-13 15:09:22 -04:00
|
|
|
|
|
|
|
|
|
|
local pruned ibd blocks headers
|
|
|
|
|
|
pruned="$(echo "$output" | bitcoin_json pruned)"
|
|
|
|
|
|
ibd="$(echo "$output" | bitcoin_json initialblockdownload)"
|
|
|
|
|
|
blocks="$(echo "$output" | bitcoin_json blocks)"
|
|
|
|
|
|
headers="$(echo "$output" | bitcoin_json headers)"
|
|
|
|
|
|
|
|
|
|
|
|
if [ "$pruned" = "True" ] || [ "$pruned" = "true" ]; then
|
|
|
|
|
|
echo "bitcoin is pruned (blocks=$blocks headers=$headers); electrumx cannot index pruned historical blocks"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
|
|
|
|
|
if [ "$ibd" = "True" ] || [ "$ibd" = "true" ]; then
|
|
|
|
|
|
echo "bitcoin is still in initial block download (blocks=$blocks headers=$headers)"
|
|
|
|
|
|
return 1
|
|
|
|
|
|
fi
|
2026-04-28 15:00:58 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "electrumx TCP port accepts connections" {
|
|
|
|
|
|
run python3 - <<'PY'
|
|
|
|
|
|
import socket
|
|
|
|
|
|
s = socket.create_connection(("127.0.0.1", 50001), 3)
|
|
|
|
|
|
s.close()
|
|
|
|
|
|
print("ok")
|
|
|
|
|
|
PY
|
|
|
|
|
|
[ "$status" -eq 0 ]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "lnd CLI getinfo succeeds" {
|
2026-06-22 14:45:36 -04:00
|
|
|
|
# lnd RPC readiness lags the container "running" state (wallet auto-unlock on
|
|
|
|
|
|
# start), so retry until ready rather than single-shot. See lnd.bats note.
|
|
|
|
|
|
run sh -lc 'for i in $(seq 1 30); do
|
|
|
|
|
|
timeout 20 podman exec lnd lncli --tlscertpath /root/.lnd/tls.cert --macaroonpath /root/.lnd/data/chain/bitcoin/mainnet/readonly.macaroon --rpcserver localhost:10009 getinfo >/dev/null 2>&1 && exit 0
|
|
|
|
|
|
sleep 3
|
|
|
|
|
|
done; exit 1'
|
2026-05-13 15:09:22 -04:00
|
|
|
|
[ "$status" -eq 0 ]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "lnd REST port accepts connections" {
|
|
|
|
|
|
run python3 - <<'PY'
|
|
|
|
|
|
import socket
|
|
|
|
|
|
s = socket.create_connection(("127.0.0.1", 18080), 3)
|
|
|
|
|
|
s.close()
|
|
|
|
|
|
print("ok")
|
|
|
|
|
|
PY
|
2026-04-28 15:00:58 -04:00
|
|
|
|
[ "$status" -eq 0 ]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "mempool api endpoint responds" {
|
2026-06-22 17:11:15 -04:00
|
|
|
|
# mempool-api reconnects to electrumx after a stack restart — retry ~180s.
|
|
|
|
|
|
run sh -lc 'for i in $(seq 1 60); do curl -fsS -m 5 -o /dev/null "http://127.0.0.1:8999/api/v1/backend-info" && exit 0; sleep 3; done; exit 1'
|
2026-04-28 15:00:58 -04:00
|
|
|
|
[ "$status" -eq 0 ]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "mempool frontend responds" {
|
2026-06-22 17:11:15 -04:00
|
|
|
|
run sh -lc 'for i in $(seq 1 60); do curl -fsS -m 5 -o /dev/null "http://127.0.0.1:4080/" && exit 0; sleep 3; done; exit 1'
|
2026-04-28 15:00:58 -04:00
|
|
|
|
[ "$status" -eq 0 ]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "bitcoin ui responds" {
|
2026-06-22 15:43:51 -04:00
|
|
|
|
# The companion (archy-bitcoin-ui) may have just been recreated by an earlier
|
|
|
|
|
|
# companion-survives test; its nginx takes a moment to serve. Retry ~120s
|
|
|
|
|
|
# rather than single-shot.
|
|
|
|
|
|
run sh -lc 'for i in $(seq 1 40); do curl -fsS -o /dev/null "http://127.0.0.1:8334/" && exit 0; sleep 3; done; exit 1'
|
2026-04-28 15:00:58 -04:00
|
|
|
|
[ "$status" -eq 0 ]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "lnd ui responds" {
|
2026-05-13 15:09:22 -04:00
|
|
|
|
run curl -fsS "http://127.0.0.1:18083/"
|
|
|
|
|
|
[ "$status" -eq 0 ]
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@test "filebrowser responds" {
|
|
|
|
|
|
run curl -fsS "http://127.0.0.1:8083/"
|
2026-04-28 15:00:58 -04:00
|
|
|
|
[ "$status" -eq 0 ]
|
|
|
|
|
|
}
|