#!/usr/bin/env bash
set -euo pipefail
ROOT_DIR="$(cd " $(dirname "${BASH_SOURCE[0]}" )/../.." && pwd)"
source "$ROOT_DIR/scripts/lib/docker-e2e-image.sh"
IMAGE_NAME="$(docker_e2e_resolve_image " openclaw-config-reload-e2e" OPENCLAW_CONFIG_RELOAD_E2E_IMAGE)"
SKIP_BUILD="${OPENCLAW_CONFIG_RELOAD_E2E_SKIP_BUILD:-0}"
PORT="18789"
TOKEN="reload-e2e-token"
CONTAINER_NAME="openclaw-config-reload-e2e-$$"
cleanup() {
docker rm -f "$CONTAINER_NAME" >/dev/null 2>&1 || true
}
trap cleanup EXIT
docker_e2e_build_or_reuse "$IMAGE_NAME" config-reload "$ROOT_DIR/scripts/e2e/Dockerfile" "$ROOT_DIR" "" "$SKIP_BUILD"
echo "Starting gateway container..."
docker run -d \
--name "$CONTAINER_NAME" \
-e COREPACK_ENABLE_DOWNLOAD_PROMPT=0 \
-e GATEWAY_AUTH_TOKEN_REF="$TOKEN" \
-e OPENCLAW_SKIP_CHANNELS=1 \
-e OPENCLAW_SKIP_PROVIDERS=1 \
-e OPENCLAW_SKIP_GMAIL_WATCHER=1 \
-e OPENCLAW_SKIP_CRON=1 \
-e OPENCLAW_SKIP_CANVAS_HOST=1 \
"$IMAGE_NAME" \
bash -lc "set -euo pipefail
entry=dist/index.mjs
[ -f \"\$entry\" ] || entry=dist/index.js
mkdir -p \"\$HOME/.openclaw\"
cat > \"\$HOME/.openclaw/openclaw.json\" <<'JSON'
{
\"gateway\" : {
\"port\" : $PORT,
\"auth\" : {
\"mode\" : \"token\" ,
\"token\" : {
\"source\" : \"env\" ,
\"provider\" : \"default\" ,
\"id\" : \"GATEWAY_AUTH_TOKEN_REF\"
}
},
\"controlUi\" : {
\"enabled\" : false
},
\"reload\" : {
\"mode\" : \"hybrid\" ,
\"debounceMs\" : 0
}
},
\"plugins\" : {
\"installs\" : {
\"lossless-claw\" : {
\"source\" : \"npm\" ,
\"spec\" : \"@martian-engineering/lossless-claw\" ,
\"installPath\" : \"/tmp/lossless-claw\" ,
\"installedAt\" : \"2026-04-22T00:00:00.000Z\" ,
\"resolvedAt\" : \"2026-04-22T00:00:00.000Z\"
}
}
}
}
JSON
node \"\$entry\" gateway --port $PORT --bind loopback --allow-unconfigured > /tmp/config-reload-e2e.log 2>&1" >/dev/null
echo "Waiting for gateway..."
ready=0
for _ in $(seq 1 180); do
if [ "$(docker inspect -f '{{.State.Running}}' " $CONTAINER_NAME" 2>/dev/null || echo false)" != "true" ]; then
break
fi
if docker exec "$CONTAINER_NAME" bash -lc "node --input-type=module -e '
import net from \"node:net\" ;
const socket = net.createConnection({ host: \"127.0.0.1\" , port: $PORT });
const timeout = setTimeout(() => {
socket.destroy();
process.exit(1);
}, 400);
socket.on(\"connect\" , () => {
clearTimeout(timeout);
socket.end ();
process.exit(0);
});
socket.on(\"error\" , () => {
clearTimeout(timeout);
process.exit(1);
});
' >/dev/null 2>&1"; then
ready=1
break
fi
sleep 0.5
done
if [ "$ready" -ne 1 ]; then
echo "Gateway failed to start"
docker logs "$CONTAINER_NAME" 2>&1 | tail -n 120 || true
docker exec "$CONTAINER_NAME" bash -lc "tail -n 120 /tmp/config-reload-e2e.log" || true
exit 1
fi
echo "Checking initial RPC status..."
docker exec "$CONTAINER_NAME" bash -lc "
entry=dist/index.mjs
[ -f \"\$entry\" ] || entry=dist/index.js
node \"\$entry\" gateway status --url ws://127.0.0.1:$PORT --token '$TOKEN' --require-rpc --timeout 30000 >/tmp/config-reload-status-before.log
"
echo "Mutating plugin install timestamp metadata..."
docker exec "$CONTAINER_NAME" bash -lc "node --input-type=module - <<'NODE'
import fs from 'node:fs' ;
import os from 'node:os' ;
import path from 'node:path' ;
const configPath = path.join(os.homedir(), '.openclaw' , 'openclaw.json' );
const config = JSON.parse(fs.readFileSync(configPath, 'utf8' ));
config.plugins.installs['lossless-claw' ].installedAt = '2026-04-22T00:01:00.000Z' ;
config.plugins.installs['lossless-claw' ].resolvedAt = '2026-04-22T00:01:00.000Z' ;
fs.writeFileSync(configPath, JSON.stringify(config, null, 2) + '\n' , 'utf8' );
NODE"
sleep 2
if [ "$(docker inspect -f '{{.State.Running}}' " $CONTAINER_NAME" 2>/dev/null || echo false)" != "true" ]; then
echo "Gateway container exited after config metadata write"
docker logs "$CONTAINER_NAME" 2>&1 | tail -n 120 || true
exit 1
fi
echo "Checking post-write RPC status..."
docker exec "$CONTAINER_NAME" bash -lc "
entry=dist/index.mjs
[ -f \"\$entry\" ] || entry=dist/index.js
node \"\$entry\" gateway status --url ws://127.0.0.1:$PORT --token '$TOKEN' --require-rpc --timeout 30000 >/tmp/config-reload-status-after.log
"
echo "Checking reload log..."
docker exec "$CONTAINER_NAME" bash -lc "node --input-type=module - <<'NODE'
import fs from 'node:fs' ;
const log = fs.readFileSync('/tmp/config-reload-e2e.log' , 'utf8' );
const reloadLines = log
.split('\n' )
.filter((line) => line.includes('config change detected; evaluating reload' ));
const restartLines = log
.split('\n' )
.filter((line) => line.includes('config change requires gateway restart' ));
if (restartLines.length > 0) {
console.error(log.split('\n' ).slice(-160).join('\n' ));
throw new Error('unexpected restart-required reload line found' );
}
for (const line of reloadLines) {
for (const needle of ['gateway.auth.token' , 'plugins.entries.firecrawl.config.webFetch' ]) {
if (line.includes(needle)) {
console.error(log.split('\n' ).slice(-160).join('\n' ));
throw new Error('runtime-only path appeared in reload diff: ' + needle);
}
}
}
if (reloadLines.length === 0) {
console.error(log.split('\n' ).slice(-160).join('\n' ));
throw new Error('expected config reload detection log after metadata write' );
}
console.log('ok' );
NODE"
echo "Config reload Docker E2E passed."
Messung V0.5 in Prozent C=95 H=94 G=94
¤ Dauer der Verarbeitung: 0.12 Sekunden
(vorverarbeitet am 2026-04-27)
¤
*© Formatika GbR, Deutschland