Files
blackroad/bin/blackroad-logs
Alexa Amundson 78fbe80f2a Initial monorepo — everything BlackRoad in one place
bin/       230 CLI tools (ask-*, br-*, agent-*, roadid, carpool)
scripts/   99 automation scripts
fleet/     Node configs and deployment
workers/   Cloudflare Worker sources (roadpay, road-search, squad webhooks)
roadc/     RoadC programming language
roadnet/   Mesh network (5 APs, WireGuard)
operator/  Memory system scripts
config/    System configs
dotfiles/  Shell configs
docs/      Documentation

BlackRoad OS — Pave Tomorrow.

RoadChain-SHA2048: d1a24f55318d338b
RoadChain-Identity: alexa@sovereign
RoadChain-Full: d1a24f55318d338b24b60bad7be39286379c76ae5470817482100cb0ddbbcb97e147d07ac7243da0a9f0363e4e5c833d612b9c0df3a3cd20802465420278ef74875a5b77f55af6fe42a931b8b635b3d0d0b6bde9abf33dc42eea52bc03c951406d8cbe49f1a3d29b26a94dade05e9477f34a7d4d4c6ec4005c3c2ac54e73a68440c512c8e83fd9b1fe234750b898ef8f4032c23db173961fe225e67a0432b5293a9714f76c5c57ed5fdf35b9fb40fd73c03ebf88b7253c6a0575f5afb6a6b49b3bda310602fb1ef676859962dad2aebbb2875814b30eee0a8ba195e482d4cbc91d8819e7f38f6db53e8063401649c77bb994371473cabfb917fb53e8cbe73d60
2026-03-14 17:08:41 -05:00

189 lines
6.4 KiB
Bash
Executable File

#!/usr/bin/env bash
# br-logs - Centralized Logging CLI
PINK='\033[38;5;205m'
AMBER='\033[38;5;214m'
GREEN='\033[38;5;82m'
BLUE='\033[38;5;69m'
RED='\033[38;5;196m'
YELLOW='\033[38;5;226m'
NC='\033[0m'
LOG_DIR="$HOME/.blackroad/logs"
LOG_URL="http://localhost:5141"
cmd="${1:-help}"
shift 2>/dev/null
# Color by level
color_level() {
case "$1" in
error|critical) echo -e "${RED}$1${NC}" ;;
warn|warning) echo -e "${YELLOW}$1${NC}" ;;
info) echo -e "${GREEN}$1${NC}" ;;
debug) echo -e "${BLUE}$1${NC}" ;;
*) echo "$1" ;;
esac
}
case "$cmd" in
start)
echo -e "${PINK}Starting Log Server...${NC}"
nohup python3 "$LOG_DIR/log_server.py" > "$LOG_DIR/server.log" 2>&1 &
echo $! > "$LOG_DIR/log.pid"
sleep 1
echo -e "${GREEN}Log Server started (PID: $(cat "$LOG_DIR/log.pid"))${NC}"
echo " HTTP: $LOG_URL"
;;
stop)
if [ -f "$LOG_DIR/log.pid" ]; then
kill $(cat "$LOG_DIR/log.pid") 2>/dev/null
rm "$LOG_DIR/log.pid"
echo -e "${AMBER}Log Server stopped${NC}"
fi
;;
status)
if [ -f "$LOG_DIR/log.pid" ] && kill -0 $(cat "$LOG_DIR/log.pid") 2>/dev/null; then
echo -e "${GREEN}${NC} Log Server running"
curl -s "$LOG_URL/stats" | python3 -c "
import sys, json
data = json.load(sys.stdin)
print(f\" Total: {data['total_logs']:,} logs\")
print(f\" Today: {data['logs_today']:,} logs\")
levels = data.get('by_level', {})
if levels:
print(f\" Levels: \" + ', '.join(f'{k}:{v}' for k,v in sorted(levels.items())))
" 2>/dev/null
else
echo -e "${RED}${NC} Log Server not running"
fi
;;
search)
query="$*"
curl -s "$LOG_URL/search?q=$query&limit=50" | python3 -c "
import sys, json
from datetime import datetime
data = json.load(sys.stdin)
for log in data.get('logs', []):
ts = datetime.fromisoformat(log['@timestamp']).strftime('%H:%M:%S')
level = log['level'].upper()[:4]
host = log.get('host', '')[:10]
svc = log.get('service', '')[:12]
msg = log['message'][:80]
# Color by level
if 'error' in level.lower() or 'crit' in level.lower():
level = f'\033[38;5;196m{level}\033[0m'
elif 'warn' in level.lower():
level = f'\033[38;5;226m{level}\033[0m'
else:
level = f'\033[38;5;82m{level}\033[0m'
print(f'{ts} {level} [{host:<10}] [{svc:<12}] {msg}')
"
;;
tail)
service="${1:-*}"
limit="${2:-20}"
echo -e "${PINK}Last $limit logs${NC} (service: $service)"
echo ""
curl -s "$LOG_URL/search?service=$service&limit=$limit" | python3 -c "
import sys, json
from datetime import datetime
data = json.load(sys.stdin)
for log in reversed(data.get('logs', [])):
ts = datetime.fromisoformat(log['@timestamp']).strftime('%H:%M:%S')
level = log['level'].upper()[:4]
host = log.get('host', '')[:10]
svc = log.get('service', '')[:12]
msg = log['message'][:100]
if 'error' in level.lower():
level = f'\033[38;5;196m{level}\033[0m'
elif 'warn' in level.lower():
level = f'\033[38;5;226m{level}\033[0m'
else:
level = f'\033[38;5;82m{level}\033[0m'
print(f'{ts} {level} [{host:<10}] [{svc:<12}] {msg}')
"
;;
send)
level="${1:-info}"
message="${*:2}"
if [ -z "$message" ]; then
echo "Usage: br-logs send <level> <message>"
exit 1
fi
curl -s -X POST "$LOG_URL/ingest" \
-H "Content-Type: application/json" \
-d "{\"level\":\"$level\",\"message\":\"$message\",\"host\":\"$(hostname)\",\"service\":\"cli\"}" | python3 -m json.tool
;;
services)
echo -e "${PINK}Services with logs:${NC}"
curl -s "$LOG_URL/services" | python3 -c "
import sys, json
for svc in json.load(sys.stdin):
print(f' {svc}')
"
;;
hosts)
echo -e "${PINK}Hosts with logs:${NC}"
curl -s "$LOG_URL/hosts" | python3 -c "
import sys, json
for host in json.load(sys.stdin):
print(f' {host}')
"
;;
aggregate)
interval="${1:-1h}"
curl -s "$LOG_URL/aggregate?interval=$interval" | python3 -c "
import sys, json
from datetime import datetime
data = json.load(sys.stdin)
print(f'{\"TIME\":<20} {\"TOTAL\":<8} {\"BY LEVEL\"}')
for bucket in data.get('buckets', []):
ts = datetime.fromtimestamp(bucket['timestamp']).strftime('%Y-%m-%d %H:%M')
levels = ', '.join(f'{k}:{v}' for k,v in bucket.get('by_level', {}).items())
print(f'{ts:<20} {bucket[\"total\"]:<8} {levels}')
"
;;
fleet)
echo -e "${PINK}╭─ LOG FLEET STATUS ────────────────────────────────────────────────────────────╮${NC}"
for host in cecilia lucidia octavia aria; do
echo -n -e "${PINK}${NC} ${BLUE}$host:${NC} "
result=$(ssh -o ConnectTimeout=3 "$host" 'curl -s http://localhost:5141/stats 2>/dev/null' 2>/dev/null)
if [ -n "$result" ]; then
echo "$result" | python3 -c "
import sys, json
data = json.load(sys.stdin)
print(f\"{data['total_logs']:,} total, {data['logs_today']:,} today\")
" 2>/dev/null || echo "offline"
else
echo "offline"
fi
done
echo -e "${PINK}╰────────────────────────────────────────────────────────────────────────────────╯${NC}"
;;
help|*)
echo -e "${PINK}br-logs - Centralized Logging${NC}"
echo ""
echo "Server:"
echo " start Start log server"
echo " stop Stop log server"
echo " status Show status"
echo ""
echo "Querying:"
echo " search <query> Search logs"
echo " tail [svc] [n] Show last N logs"
echo " aggregate [interval] Time-based aggregation"
echo ""
echo "Info:"
echo " services List services"
echo " hosts List hosts"
echo " fleet Fleet-wide status"
echo ""
echo "Send:"
echo " send <level> <msg> Send log entry"
;;
esac