4.3 KiB
Manual smoke test — @paperclipai/plugin-kubernetes
Manual sanity check that the plugin works end-to-end against a real paperclip-server instance and a real Kubernetes cluster (kind for local dev). Future work may automate this in CI.
Prerequisites
- A running kind cluster:
kind create cluster --name paperclip kubectl --context kind-paperclip get nodesreturns a node inReadystate.
Steps
1. Build the plugin
cd packages/plugins/sandbox-providers/kubernetes
pnpm install --ignore-workspace
pnpm build
Expected: dist/ populated with compiled .js and .d.ts files. No errors.
2. Start paperclip-server in dev mode
In a separate terminal:
cd /path/to/paperclip
export PAPERCLIP_HOME=/tmp/paperclip-smoke
export PAPERCLIP_INSTANCE_ID=smoke
export PAPERCLIP_DEPLOYMENT_MODE=local_trusted
pnpm --filter @paperclipai/server dev
Wait for Server listening on 127.0.0.1:3100.
3. Install the plugin via the CLI
pnpm paperclipai plugin install \
--local /path/to/paperclip/packages/plugins/sandbox-providers/kubernetes \
--api-base http://127.0.0.1:3100
Expected: ✓ Installed paperclip.kubernetes-sandbox-provider v0.1.0 (ready).
4. Create a company and a kubernetes sandbox environment
CO_ID=$(curl -s -X POST -H "Content-Type: application/json" \
-d '{"name":"SmokeCo"}' \
http://127.0.0.1:3100/api/companies | jq -r '.id')
KUBECONFIG_CONTENT=$(cat ~/.kube/config | jq -Rs .)
curl -s -X POST -H "Content-Type: application/json" \
-d "{
\"name\": \"k8s-sandbox\",
\"driver\": \"sandbox\",
\"config\": {
\"provider\": \"kubernetes\",
\"kubeconfig\": $KUBECONFIG_CONTENT,
\"companySlug\": \"smoke\",
\"adapterType\": \"claude_local\",
\"imageAllowList\": [\"ghcr.io/paperclipai/agent-runtime-claude:v1\"]
}
}" \
http://127.0.0.1:3100/api/companies/$CO_ID/environments | jq
Expected: HTTP 201 with the new environment row.
5. Probe the environment
ENV_ID=$(curl -s http://127.0.0.1:3100/api/companies/$CO_ID/environments | jq -r '.[0].id')
curl -s -X POST -d '{}' -H "Content-Type: application/json" \
http://127.0.0.1:3100/api/environments/$ENV_ID/probe | jq
Expected: {"ok": true, ...} with a summary mentioning the tenant namespace
(paperclip-smoke). On first probe the namespace may not yet exist —
the plugin treats a 404 on listNamespacedPod as a successful reachability
check.
6. Trigger an agent run
Use the UI or the API to dispatch a run against the k8s-sandbox environment.
The plugin's onEnvironmentAcquireLease will:
ensureTenant— provision thepaperclip-smokenamespace, SA, Role, RoleBinding, ResourceQuota, LimitRange, NetworkPoliciesbuildSandboxCrManifest— render the security-hardened Sandbox CR manifestcreateNamespacedCustomObject— submit toagents.x-k8s.io/v1alpha1createPerRunSecret— owned by the Sandbox CR for cascade-delete- Fast-upload workspace/config/skill payloads by collapsing adapter-utils chunked uploads into a single stdin-backed exec per file
7. Verify the tenant resources
kubectl --context kind-paperclip get namespace paperclip-smoke
kubectl --context kind-paperclip get all,networkpolicy,resourcequota,limitrange,sa,role,rolebinding -n paperclip-smoke
Expected:
- Namespace
paperclip-smokeexists with PSS labels (pod-security.kubernetes.io/enforce=restricted) - ServiceAccount
paperclip-tenant-sa - Role
paperclip-tenant-role, RoleBindingpaperclip-tenant-rb - ResourceQuota
paperclip-quota, LimitRangepaperclip-limits - NetworkPolicies
paperclip-deny-all+paperclip-egress-allow - Sandbox
pc-{ulid}and its managed Pod - Secret
pc-{ulid}-envwithownerReferencespointing at the Sandbox CR - Run logs or plugin metadata include
fastUpload: "flush"entries during workspace/config/skill upload
8. Tear down
kubectl --context kind-paperclip delete namespace paperclip-smoke
kill %1 # paperclip-server
9. Document the result
In the PR description (or appended to this file as a dated section), record:
- Date + git SHA
kubectl versionserver version- Output of
kubectl get all -n paperclip-smokeafter step 6 - Probe response from step 5
- Time-to-acquire-lease (target: <30s on kind for a cold tenant)