mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-06 08:10:44 +00:00
* test(e2e): fix kitchen sink crabbox coverage * test(e2e): update kitchen sink expected diagnostics * fix(plugins): harden registry and package gates * fix(plugins): load lazy tool middleware snapshots * fix(ci): satisfy crabbox branch gates * fix(plugins): await guarded fetch cleanup
184 lines
6.3 KiB
YAML
184 lines
6.3 KiB
YAML
name: Crabbox Hydrate
|
|
|
|
on:
|
|
workflow_dispatch:
|
|
inputs:
|
|
crabbox_id:
|
|
description: "Crabbox lease ID"
|
|
required: true
|
|
type: string
|
|
ref:
|
|
description: "Git ref to hydrate"
|
|
required: false
|
|
type: string
|
|
crabbox_runner_label:
|
|
description: "Dynamic Crabbox runner label"
|
|
required: true
|
|
type: string
|
|
crabbox_job:
|
|
description: "Hydration job identifier expected by Crabbox"
|
|
required: false
|
|
default: "hydrate"
|
|
type: string
|
|
crabbox_keep_alive_minutes:
|
|
description: "Minutes to keep the hydrated job alive"
|
|
required: false
|
|
default: "90"
|
|
type: string
|
|
|
|
permissions:
|
|
contents: read
|
|
|
|
env:
|
|
FORCE_JAVASCRIPT_ACTIONS_TO_NODE24: "true"
|
|
|
|
jobs:
|
|
hydrate:
|
|
name: hydrate
|
|
runs-on: [self-hosted, "${{ inputs.crabbox_runner_label }}"]
|
|
timeout-minutes: 120
|
|
steps:
|
|
- uses: actions/checkout@v6
|
|
with:
|
|
ref: ${{ inputs.ref || github.ref }}
|
|
|
|
- name: Setup Node environment
|
|
uses: ./.github/actions/setup-node-env
|
|
with:
|
|
install-bun: "false"
|
|
|
|
- name: Prepare Crabbox shell
|
|
shell: bash
|
|
run: |
|
|
set -euo pipefail
|
|
|
|
git fetch --no-tags --depth=50 origin "+refs/heads/main:refs/remotes/origin/main"
|
|
|
|
node_bin="$(dirname "$(node -p 'process.execPath')")"
|
|
pnpm_bin="$(command -v pnpm)"
|
|
sudo ln -sf "$node_bin/node" /usr/local/bin/node
|
|
sudo ln -sf "$node_bin/npm" /usr/local/bin/npm
|
|
sudo ln -sf "$node_bin/npx" /usr/local/bin/npx
|
|
sudo ln -sf "$node_bin/corepack" /usr/local/bin/corepack
|
|
sudo ln -sf "$pnpm_bin" /usr/local/bin/pnpm
|
|
|
|
- name: Ensure Docker is available
|
|
shell: bash
|
|
run: |
|
|
set -euo pipefail
|
|
|
|
if ! command -v docker >/dev/null 2>&1; then
|
|
curl -fsSL https://get.docker.com | sudo sh
|
|
fi
|
|
|
|
if command -v systemctl >/dev/null 2>&1; then
|
|
sudo systemctl start docker
|
|
fi
|
|
|
|
if [ -S /var/run/docker.sock ]; then
|
|
sudo usermod -aG docker "$USER" || true
|
|
# The runner process keeps its original groups; grant this
|
|
# ephemeral runner session access without requiring a relogin.
|
|
sudo chmod 666 /var/run/docker.sock
|
|
fi
|
|
|
|
- name: Hydrate provider env helper
|
|
shell: bash
|
|
env:
|
|
ANTHROPIC_API_KEY: ${{ secrets.ANTHROPIC_API_KEY }}
|
|
ANTHROPIC_API_KEY_OLD: ${{ secrets.ANTHROPIC_API_KEY_OLD }}
|
|
ANTHROPIC_API_TOKEN: ${{ secrets.ANTHROPIC_API_TOKEN }}
|
|
CEREBRAS_API_KEY: ${{ secrets.CEREBRAS_API_KEY }}
|
|
DEEPINFRA_API_KEY: ${{ secrets.DEEPINFRA_API_KEY }}
|
|
FIREWORKS_API_KEY: ${{ secrets.FIREWORKS_API_KEY }}
|
|
GEMINI_API_KEY: ${{ secrets.GEMINI_API_KEY }}
|
|
GOOGLE_API_KEY: ${{ secrets.GOOGLE_API_KEY }}
|
|
GROQ_API_KEY: ${{ secrets.GROQ_API_KEY }}
|
|
KIMI_API_KEY: ${{ secrets.KIMI_API_KEY }}
|
|
MINIMAX_API_KEY: ${{ secrets.MINIMAX_API_KEY }}
|
|
MISTRAL_API_KEY: ${{ secrets.MISTRAL_API_KEY }}
|
|
MOONSHOT_API_KEY: ${{ secrets.MOONSHOT_API_KEY }}
|
|
OPENAI_API_KEY: ${{ secrets.OPENAI_API_KEY }}
|
|
OPENAI_BASE_URL: ${{ secrets.OPENAI_BASE_URL }}
|
|
OPENROUTER_API_KEY: ${{ secrets.OPENROUTER_API_KEY }}
|
|
QWEN_API_KEY: ${{ secrets.QWEN_API_KEY }}
|
|
TOGETHER_API_KEY: ${{ secrets.TOGETHER_API_KEY }}
|
|
XAI_API_KEY: ${{ secrets.XAI_API_KEY }}
|
|
ZAI_API_KEY: ${{ secrets.ZAI_API_KEY }}
|
|
Z_AI_API_KEY: ${{ secrets.Z_AI_API_KEY }}
|
|
run: bash scripts/ci-hydrate-testbox-env.sh
|
|
|
|
- name: Mark Crabbox ready
|
|
shell: bash
|
|
env:
|
|
CRABBOX_ID: ${{ inputs.crabbox_id }}
|
|
CRABBOX_JOB: ${{ inputs.crabbox_job }}
|
|
run: |
|
|
set -euo pipefail
|
|
job="${CRABBOX_JOB}"
|
|
if [ -z "$job" ]; then job=hydrate; fi
|
|
case "$CRABBOX_ID" in
|
|
''|*[!A-Za-z0-9._-]*)
|
|
echo "Invalid crabbox_id" >&2
|
|
exit 2
|
|
;;
|
|
esac
|
|
mkdir -p "$HOME/.crabbox/actions"
|
|
state="$HOME/.crabbox/actions/${CRABBOX_ID}.env"
|
|
env_file="$HOME/.crabbox/actions/${CRABBOX_ID}.env.sh"
|
|
services_file="$HOME/.crabbox/actions/${CRABBOX_ID}.services"
|
|
write_export() {
|
|
key="$1"
|
|
value="${!key-}"
|
|
if [ -n "$value" ]; then
|
|
printf 'export %s=%q\n' "$key" "$value"
|
|
fi
|
|
}
|
|
{
|
|
for key in CI GITHUB_ACTIONS GITHUB_WORKSPACE GITHUB_REPOSITORY GITHUB_RUN_ID GITHUB_RUN_NUMBER GITHUB_RUN_ATTEMPT GITHUB_REF GITHUB_REF_NAME GITHUB_SHA GITHUB_EVENT_NAME GITHUB_ACTOR RUNNER_OS RUNNER_ARCH RUNNER_TEMP RUNNER_TOOL_CACHE; do
|
|
write_export "$key"
|
|
done
|
|
} > "${env_file}.tmp"
|
|
mv "${env_file}.tmp" "$env_file"
|
|
{
|
|
echo "# Docker containers visible from the hydrated runner"
|
|
docker ps --format '{{.Names}}\t{{.Image}}\t{{.Ports}}' 2>/dev/null || true
|
|
} > "${services_file}.tmp"
|
|
mv "${services_file}.tmp" "$services_file"
|
|
tmp="${state}.tmp"
|
|
{
|
|
echo "WORKSPACE=${GITHUB_WORKSPACE}"
|
|
echo "RUN_ID=${GITHUB_RUN_ID}"
|
|
echo "JOB=${job}"
|
|
echo "ENV_FILE=${env_file}"
|
|
echo "SERVICES_FILE=${services_file}"
|
|
echo "READY_AT=$(date -u +%Y-%m-%dT%H:%M:%SZ)"
|
|
} > "$tmp"
|
|
mv "$tmp" "$state"
|
|
|
|
- name: Keep Crabbox job alive
|
|
shell: bash
|
|
env:
|
|
CRABBOX_ID: ${{ inputs.crabbox_id }}
|
|
CRABBOX_KEEP_ALIVE_MINUTES: ${{ inputs.crabbox_keep_alive_minutes }}
|
|
run: |
|
|
set -euo pipefail
|
|
case "$CRABBOX_ID" in
|
|
''|*[!A-Za-z0-9._-]*)
|
|
echo "Invalid crabbox_id" >&2
|
|
exit 2
|
|
;;
|
|
esac
|
|
minutes="${CRABBOX_KEEP_ALIVE_MINUTES}"
|
|
case "$minutes" in
|
|
''|*[!0-9]*) minutes=90 ;;
|
|
esac
|
|
stop="$HOME/.crabbox/actions/${CRABBOX_ID}.stop"
|
|
deadline=$(( $(date +%s) + minutes * 60 ))
|
|
while [ "$(date +%s)" -lt "$deadline" ]; do
|
|
if [ -f "$stop" ]; then
|
|
exit 0
|
|
fi
|
|
sleep 15
|
|
done
|