Files
blackroad/scripts/blackroad-agent-daemon.sh
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

271 lines
9.4 KiB
Bash

#!/bin/bash
# BlackRoad Agent Daemon — runs on Alice Pi
# Lightweight HTTP server that accepts commands from chat.blackroad.io
# Handles: shell exec, file read/write/edit, git ops, codebase search
# Uses socat for HTTP — no dependencies needed
# Usage: ./blackroad-agent-daemon.sh [port]
PORT="${1:-8080}"
SANDBOX_ROOT="/home/pi/projects"
MAX_OUTPUT=8000
TIMEOUT=30
mkdir -p "$SANDBOX_ROOT"
log() { echo "[$(date '+%H:%M:%S')] $*" >&2; }
handle_request() {
local method path body content_length=""
# Read request line
read -r method path _
method=$(echo "$method" | tr -d '\r')
path=$(echo "$path" | tr -d '\r')
# Read headers
while IFS= read -r header; do
header=$(echo "$header" | tr -d '\r')
[ -z "$header" ] && break
case "$header" in
Content-Length:*|content-length:*) content_length="${header#*: }" ;;
esac
done
# Read body
body=""
if [ -n "$content_length" ] && [ "$content_length" -gt 0 ] 2>/dev/null; then
body=$(head -c "$content_length")
fi
log "$method $path"
# CORS preflight
if [ "$method" = "OPTIONS" ]; then
echo "HTTP/1.1 200 OK"
echo "Access-Control-Allow-Origin: *"
echo "Access-Control-Allow-Methods: GET,POST,OPTIONS"
echo "Access-Control-Allow-Headers: Content-Type"
echo "Content-Length: 0"
echo ""
return
fi
respond() {
local status="$1" body="$2"
local len=${#body}
echo "HTTP/1.1 $status"
echo "Content-Type: application/json"
echo "Access-Control-Allow-Origin: *"
echo "Content-Length: $len"
echo ""
echo -n "$body"
}
# Health
if [ "$path" = "/health" ]; then
respond "200 OK" '{"status":"ok","node":"alice","pid":'$$'}'
return
fi
# Shell execution
if [ "$path" = "/exec" ] && [ "$method" = "POST" ]; then
local cmd lang code timeout_val
cmd=$(echo "$body" | python3 -c "import sys,json;d=json.load(sys.stdin);print(d.get('command',''))" 2>/dev/null)
lang=$(echo "$body" | python3 -c "import sys,json;d=json.load(sys.stdin);print(d.get('language','bash'))" 2>/dev/null)
code=$(echo "$body" | python3 -c "import sys,json;d=json.load(sys.stdin);print(d.get('code',''))" 2>/dev/null)
timeout_val=$(echo "$body" | python3 -c "import sys,json;d=json.load(sys.stdin);print(d.get('timeout',$TIMEOUT))" 2>/dev/null)
# Use command if provided, otherwise use code with language
if [ -n "$cmd" ]; then
code="$cmd"
lang="bash"
fi
if [ -z "$code" ]; then
respond "400 Bad Request" '{"error":"No command or code provided"}'
return
fi
local output exit_code
case "$lang" in
python|python3)
output=$(timeout "$timeout_val" python3 -c "$code" 2>&1)
exit_code=$?
;;
js|javascript|node)
output=$(timeout "$timeout_val" node -e "$code" 2>&1)
exit_code=$?
;;
bash|sh)
output=$(timeout "$timeout_val" bash -c "$code" 2>&1)
exit_code=$?
;;
*)
respond "400 Bad Request" "{\"error\":\"Unsupported language: $lang\"}"
return
;;
esac
# Truncate output
if [ ${#output} -gt $MAX_OUTPUT ]; then
output="${output:0:$MAX_OUTPUT}... (truncated)"
fi
# JSON-escape output
local json_output
json_output=$(python3 -c "import json,sys;print(json.dumps(sys.stdin.read()))" <<< "$output")
respond "200 OK" "{\"output\":$json_output,\"exitCode\":$exit_code,\"language\":\"$lang\"}"
return
fi
# File read
if [ "$path" = "/file/read" ] && [ "$method" = "POST" ]; then
local filepath
filepath=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('path',''))" 2>/dev/null)
if [ -z "$filepath" ] || [ ! -e "$filepath" ]; then
respond "404 Not Found" "{\"error\":\"File not found: $filepath\"}"
return
fi
if [ -d "$filepath" ]; then
local listing
listing=$(ls -la "$filepath" 2>&1 | head -50)
local json_listing
json_listing=$(python3 -c "import json,sys;print(json.dumps(sys.stdin.read()))" <<< "$listing")
respond "200 OK" "{\"type\":\"directory\",\"path\":\"$filepath\",\"listing\":$json_listing}"
return
fi
local content
content=$(head -c 50000 "$filepath" 2>&1)
local json_content
json_content=$(python3 -c "import json,sys;print(json.dumps(sys.stdin.read()))" <<< "$content")
local lines
lines=$(wc -l < "$filepath")
local size
size=$(stat -c%s "$filepath" 2>/dev/null || stat -f%z "$filepath" 2>/dev/null)
respond "200 OK" "{\"type\":\"file\",\"path\":\"$filepath\",\"content\":$json_content,\"lines\":$lines,\"size\":$size}"
return
fi
# File write
if [ "$path" = "/file/write" ] && [ "$method" = "POST" ]; then
local filepath content
filepath=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('path',''))" 2>/dev/null)
content=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('content',''))" 2>/dev/null)
if [ -z "$filepath" ]; then
respond "400 Bad Request" '{"error":"No path provided"}'
return
fi
mkdir -p "$(dirname "$filepath")"
echo "$content" > "$filepath"
respond "200 OK" "{\"written\":\"$filepath\",\"size\":$(wc -c < "$filepath")}"
return
fi
# File search (grep)
if [ "$path" = "/search" ] && [ "$method" = "POST" ]; then
local pattern dir max_results
pattern=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('pattern',''))" 2>/dev/null)
dir=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('directory','$SANDBOX_ROOT'))" 2>/dev/null)
max_results=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('max',20))" 2>/dev/null)
if [ -z "$pattern" ]; then
respond "400 Bad Request" '{"error":"No pattern provided"}'
return
fi
local results
results=$(grep -rn --include="*.{py,js,ts,sh,go,rs,md,json,yaml,yml,toml}" "$pattern" "$dir" 2>/dev/null | head -"$max_results")
local json_results
json_results=$(python3 -c "import json,sys;print(json.dumps(sys.stdin.read()))" <<< "$results")
local count
count=$(echo "$results" | grep -c . || echo 0)
respond "200 OK" "{\"pattern\":\"$pattern\",\"directory\":\"$dir\",\"count\":$count,\"results\":$json_results}"
return
fi
# File glob (find)
if [ "$path" = "/glob" ] && [ "$method" = "POST" ]; then
local pattern dir
pattern=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('pattern',''))" 2>/dev/null)
dir=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('directory','$SANDBOX_ROOT'))" 2>/dev/null)
local files
files=$(find "$dir" -name "$pattern" -type f 2>/dev/null | head -30)
local json_files
json_files=$(python3 -c "import json,sys;lines=sys.stdin.read().strip().split('\n');print(json.dumps([l for l in lines if l]))" <<< "$files")
respond "200 OK" "{\"pattern\":\"$pattern\",\"files\":$json_files}"
return
fi
# Git operations
if [ "$path" = "/git" ] && [ "$method" = "POST" ]; then
local repo op args_str
repo=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('repo',''))" 2>/dev/null)
op=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('op','status'))" 2>/dev/null)
args_str=$(echo "$body" | python3 -c "import sys,json;print(json.load(sys.stdin).get('args',''))" 2>/dev/null)
if [ -z "$repo" ] || [ ! -d "$repo/.git" ]; then
respond "400 Bad Request" "{\"error\":\"Not a git repo: $repo\"}"
return
fi
local output
case "$op" in
status) output=$(cd "$repo" && git status --short 2>&1) ;;
log) output=$(cd "$repo" && git log --oneline -20 2>&1) ;;
diff) output=$(cd "$repo" && git diff 2>&1 | head -200) ;;
branch) output=$(cd "$repo" && git branch -a 2>&1) ;;
add) output=$(cd "$repo" && git add $args_str 2>&1) ;;
commit) output=$(cd "$repo" && git commit -m "$args_str" 2>&1) ;;
pull) output=$(cd "$repo" && git pull 2>&1) ;;
push) output=$(cd "$repo" && git push 2>&1) ;;
clone) output=$(cd "$SANDBOX_ROOT" && git clone $args_str 2>&1) ;;
*) output="Unknown git op: $op" ;;
esac
local json_output
json_output=$(python3 -c "import json,sys;print(json.dumps(sys.stdin.read()))" <<< "$output")
respond "200 OK" "{\"op\":\"$op\",\"repo\":\"$repo\",\"output\":$json_output}"
return
fi
# List projects
if [ "$path" = "/projects" ]; then
local projects
projects=$(find "$SANDBOX_ROOT" -maxdepth 2 -name ".git" -type d 2>/dev/null | sed 's|/.git||' | sort)
local json_projects
json_projects=$(python3 -c "import json,sys;lines=sys.stdin.read().strip().split('\n');print(json.dumps([l for l in lines if l]))" <<< "$projects")
respond "200 OK" "{\"projects\":$json_projects}"
return
fi
respond "404 Not Found" '{"error":"Unknown endpoint"}'
}
log "BlackRoad Agent Daemon starting on port $PORT"
log "Sandbox: $SANDBOX_ROOT"
# Use socat if available, fall back to ncat/nc
if command -v socat &>/dev/null; then
socat TCP-LISTEN:$PORT,reuseaddr,fork SYSTEM:"bash -c 'handle_request'" &
# Export the function for socat
export -f handle_request log
log "Using socat"
socat TCP-LISTEN:$PORT,reuseaddr,fork EXEC:"bash -c handle_request"
else
log "Using while+nc loop"
while true; do
echo "" | nc -l -p $PORT -c "bash -c 'handle_request'" 2>/dev/null || \
{ handle_request | nc -l $PORT; } 2>/dev/null
done
fi