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
271 lines
9.4 KiB
Bash
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
|