Files
openclaw/scripts/e2e/plugins-docker.sh

225 lines
6.5 KiB
Bash
Raw Permalink Normal View History

2026-01-11 12:21:45 +00:00
#!/usr/bin/env bash
set -euo pipefail
ROOT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")/../.." && pwd)"
2026-01-30 03:15:10 +01:00
IMAGE_NAME="openclaw-plugins-e2e"
2026-01-11 12:21:45 +00:00
echo "Building Docker image..."
docker build -t "$IMAGE_NAME" -f "$ROOT_DIR/scripts/e2e/Dockerfile" "$ROOT_DIR"
echo "Running plugins Docker E2E..."
docker run --rm -t "$IMAGE_NAME" bash -lc '
set -euo pipefail
if [ -f dist/index.mjs ]; then
OPENCLAW_ENTRY="dist/index.mjs"
elif [ -f dist/index.js ]; then
OPENCLAW_ENTRY="dist/index.js"
else
echo "Missing dist/index.(m)js (build output):"
ls -la dist || true
exit 1
fi
export OPENCLAW_ENTRY
home_dir=$(mktemp -d "/tmp/openclaw-plugins-e2e.XXXXXX")
export HOME="$home_dir"
2026-02-02 13:08:52 +00:00
mkdir -p "$HOME/.openclaw/extensions/demo-plugin"
2026-01-11 12:21:45 +00:00
2026-02-02 13:08:52 +00:00
cat > "$HOME/.openclaw/extensions/demo-plugin/index.js" <<'"'"'JS'"'"'
2026-01-11 12:21:45 +00:00
module.exports = {
id: "demo-plugin",
name: "Demo Plugin",
description: "Docker E2E demo plugin",
register(api) {
api.registerTool(() => null, { name: "demo_tool" });
api.registerGatewayMethod("demo.ping", async () => ({ ok: true }));
api.registerCli(() => {}, { commands: ["demo"] });
api.registerService({ id: "demo-service", start: () => {} });
},
};
JS
2026-02-02 13:08:52 +00:00
cat > "$HOME/.openclaw/extensions/demo-plugin/openclaw.plugin.json" <<'"'"'JSON'"'"'
{
"id": "demo-plugin",
"configSchema": {
"type": "object",
"properties": {}
}
}
JSON
2026-01-11 12:21:45 +00:00
node "$OPENCLAW_ENTRY" plugins list --json > /tmp/plugins.json
2026-01-11 12:21:45 +00:00
node - <<'"'"'NODE'"'"'
const fs = require("node:fs");
const data = JSON.parse(fs.readFileSync("/tmp/plugins.json", "utf8"));
const plugin = (data.plugins || []).find((entry) => entry.id === "demo-plugin");
if (!plugin) throw new Error("plugin not found");
if (plugin.status !== "loaded") {
throw new Error(`unexpected status: ${plugin.status}`);
}
const assertIncludes = (list, value, label) => {
if (!Array.isArray(list) || !list.includes(value)) {
throw new Error(`${label} missing: ${value}`);
}
};
assertIncludes(plugin.toolNames, "demo_tool", "tool");
assertIncludes(plugin.gatewayMethods, "demo.ping", "gateway method");
assertIncludes(plugin.cliCommands, "demo", "cli command");
assertIncludes(plugin.services, "demo-service", "service");
const diagErrors = (data.diagnostics || []).filter((diag) => diag.level === "error");
if (diagErrors.length > 0) {
throw new Error(`diagnostics errors: ${diagErrors.map((diag) => diag.message).join("; ")}`);
}
2026-01-12 01:16:42 +00:00
console.log("ok");
NODE
echo "Testing tgz install flow..."
2026-01-30 03:15:10 +01:00
pack_dir="$(mktemp -d "/tmp/openclaw-plugin-pack.XXXXXX")"
2026-01-12 01:16:42 +00:00
mkdir -p "$pack_dir/package"
cat > "$pack_dir/package/package.json" <<'"'"'JSON'"'"'
{
2026-01-30 03:15:10 +01:00
"name": "@openclaw/demo-plugin-tgz",
2026-01-12 01:16:42 +00:00
"version": "0.0.1",
2026-01-30 03:15:10 +01:00
"openclaw": { "extensions": ["./index.js"] }
2026-01-12 01:16:42 +00:00
}
JSON
cat > "$pack_dir/package/index.js" <<'"'"'JS'"'"'
module.exports = {
id: "demo-plugin-tgz",
name: "Demo Plugin TGZ",
register(api) {
api.registerGatewayMethod("demo.tgz", async () => ({ ok: true }));
},
};
JS
2026-02-02 13:08:52 +00:00
cat > "$pack_dir/package/openclaw.plugin.json" <<'"'"'JSON'"'"'
{
"id": "demo-plugin-tgz",
"configSchema": {
"type": "object",
"properties": {}
}
}
JSON
2026-01-12 01:16:42 +00:00
tar -czf /tmp/demo-plugin-tgz.tgz -C "$pack_dir" package
node "$OPENCLAW_ENTRY" plugins install /tmp/demo-plugin-tgz.tgz
node "$OPENCLAW_ENTRY" plugins list --json > /tmp/plugins2.json
2026-01-12 01:16:42 +00:00
node - <<'"'"'NODE'"'"'
const fs = require("node:fs");
const data = JSON.parse(fs.readFileSync("/tmp/plugins2.json", "utf8"));
const plugin = (data.plugins || []).find((entry) => entry.id === "demo-plugin-tgz");
if (!plugin) throw new Error("tgz plugin not found");
if (plugin.status !== "loaded") {
throw new Error(`unexpected status: ${plugin.status}`);
}
if (!Array.isArray(plugin.gatewayMethods) || !plugin.gatewayMethods.includes("demo.tgz")) {
throw new Error("expected gateway method demo.tgz");
}
2026-01-11 12:21:45 +00:00
console.log("ok");
2026-01-12 01:27:08 +00:00
NODE
echo "Testing install from local folder (plugins.load.paths)..."
2026-01-30 03:15:10 +01:00
dir_plugin="$(mktemp -d "/tmp/openclaw-plugin-dir.XXXXXX")"
2026-01-12 01:27:08 +00:00
cat > "$dir_plugin/package.json" <<'"'"'JSON'"'"'
{
2026-01-30 03:15:10 +01:00
"name": "@openclaw/demo-plugin-dir",
2026-01-12 01:27:08 +00:00
"version": "0.0.1",
2026-01-30 03:15:10 +01:00
"openclaw": { "extensions": ["./index.js"] }
2026-01-12 01:27:08 +00:00
}
JSON
cat > "$dir_plugin/index.js" <<'"'"'JS'"'"'
module.exports = {
id: "demo-plugin-dir",
name: "Demo Plugin DIR",
register(api) {
api.registerGatewayMethod("demo.dir", async () => ({ ok: true }));
},
};
JS
2026-02-02 13:08:52 +00:00
cat > "$dir_plugin/openclaw.plugin.json" <<'"'"'JSON'"'"'
{
"id": "demo-plugin-dir",
"configSchema": {
"type": "object",
"properties": {}
}
}
JSON
2026-01-12 01:27:08 +00:00
node "$OPENCLAW_ENTRY" plugins install "$dir_plugin"
node "$OPENCLAW_ENTRY" plugins list --json > /tmp/plugins3.json
2026-01-12 01:27:08 +00:00
node - <<'"'"'NODE'"'"'
const fs = require("node:fs");
const data = JSON.parse(fs.readFileSync("/tmp/plugins3.json", "utf8"));
const plugin = (data.plugins || []).find((entry) => entry.id === "demo-plugin-dir");
if (!plugin) throw new Error("dir plugin not found");
if (plugin.status !== "loaded") {
throw new Error(`unexpected status: ${plugin.status}`);
}
if (!Array.isArray(plugin.gatewayMethods) || !plugin.gatewayMethods.includes("demo.dir")) {
throw new Error("expected gateway method demo.dir");
}
console.log("ok");
NODE
echo "Testing install from npm spec (file:)..."
2026-01-30 03:15:10 +01:00
file_pack_dir="$(mktemp -d "/tmp/openclaw-plugin-filepack.XXXXXX")"
2026-01-12 01:27:08 +00:00
mkdir -p "$file_pack_dir/package"
cat > "$file_pack_dir/package/package.json" <<'"'"'JSON'"'"'
{
2026-01-30 03:15:10 +01:00
"name": "@openclaw/demo-plugin-file",
2026-01-12 01:27:08 +00:00
"version": "0.0.1",
2026-01-30 03:15:10 +01:00
"openclaw": { "extensions": ["./index.js"] }
2026-01-12 01:27:08 +00:00
}
JSON
cat > "$file_pack_dir/package/index.js" <<'"'"'JS'"'"'
module.exports = {
id: "demo-plugin-file",
name: "Demo Plugin FILE",
register(api) {
api.registerGatewayMethod("demo.file", async () => ({ ok: true }));
},
};
JS
2026-02-02 13:08:52 +00:00
cat > "$file_pack_dir/package/openclaw.plugin.json" <<'"'"'JSON'"'"'
{
"id": "demo-plugin-file",
"configSchema": {
"type": "object",
"properties": {}
}
}
JSON
2026-01-12 01:27:08 +00:00
node "$OPENCLAW_ENTRY" plugins install "file:$file_pack_dir/package"
node "$OPENCLAW_ENTRY" plugins list --json > /tmp/plugins4.json
2026-01-12 01:27:08 +00:00
node - <<'"'"'NODE'"'"'
const fs = require("node:fs");
const data = JSON.parse(fs.readFileSync("/tmp/plugins4.json", "utf8"));
const plugin = (data.plugins || []).find((entry) => entry.id === "demo-plugin-file");
if (!plugin) throw new Error("file plugin not found");
if (plugin.status !== "loaded") {
throw new Error(`unexpected status: ${plugin.status}`);
}
if (!Array.isArray(plugin.gatewayMethods) || !plugin.gatewayMethods.includes("demo.file")) {
throw new Error("expected gateway method demo.file");
}
console.log("ok");
2026-01-11 12:21:45 +00:00
NODE
'
echo "OK"