Install
openclaw skills install bohrium-nodeManage Bohrium dev nodes (containers/VMs) via bohr CLI or open.bohrium.com API. Use when: user asks about creating/starting/stopping/deleting dev machines on Bohrium, checking available resources and pricing, or managing node lifecycle. NOT for: job submission, image management, or project management.
openclaw skills install bohrium-nodeManage dev nodes (container/VM instances) on the Bohrium platform. Prefer bohr CLI; fall back to the API for unsupported operations.
Dev nodes are used for data preparation, compilation, debugging, and post-processing. They support Web Shell and SSH connections.
"bohrium-node": {
"enabled": true,
"apiKey": "YOUR_ACCESS_KEY",
"env": { "ACCESS_KEY": "YOUR_ACCESS_KEY" }
}
# macOS
/bin/bash -c "$(curl -fsSL https://dp-public.oss-cn-beijing.aliyuncs.com/bohrctl/1.0.0/install_bohr_mac_curl.sh)"
# Linux
/bin/bash -c "$(curl -fsSL https://dp-public.oss-cn-beijing.aliyuncs.com/bohrctl/1.0.0/install_bohr_linux_curl.sh)"
source ~/.bashrc && export PATH="$HOME/.bohrium:$PATH"
bohr node list # All nodes (table)
bohr node list --json # JSON
bohr node list -s # Running only
bohr node list -p # Paused only
bohr node list -d # Pending only
bohr node list -w # Waiting only
bohr node list -q # ID and name only
JSON fields: nodeId, nodeName, status (Started/Paused/Pending/Waiting), cpu/memory/gpu, ip, imageName, cost
bohr node create # Interactive: Project -> Image -> Machine -> Name -> Disk
bohr node createis interactive. For automation, use the API (see below).
Recommended images:
| Scenario | Image |
|---|---|
| CPU basic | registry.dp.tech/dptech/ubuntu:20.04-py3.10 |
| CPU + Intel MPI | registry.dp.tech/dptech/ubuntu:20.04-py3.10-intel2022 |
| GPU basic | registry.dp.tech/dptech/ubuntu:20.04-py3.10-cuda11.6 |
| GPU + Intel MPI | registry.dp.tech/dptech/ubuntu:20.04-py3.10-intel2022-cuda11.6 |
bohr node connect 1431145 # Passwordless SSH via nodeId
Alternative: Web Shell via the Bohrium web UI (auto-login as root), or manual SSH using credentials from the API.
bohr node stop 1431145 # Stop (pause billing, data preserved)
bohr node delete 1431145 # Delete (irreversible)
Important: Nodes are billed continuously while running. Stop or delete when not in use.
| Item | Details |
|---|---|
| System disk | Selected at creation (max 100GB); stores OS packages |
| Personal disk /personal | 500GB per user per project; persists after node release |
| Shared disk /share | 1TB per project; read/write for all members |
| Public ports | 50001-50005 open by default |
| GPU driver | v525 default; cannot upgrade |
| Docker | Not supported inside container nodes (security) |
Mount datasets when creating a container node; access via path (e.g. /bohr/my-dataset/v1).
df -a | grep bohr to view mount points| Scenario | Boot time |
|---|---|
| Cached CPU machine | ~20s |
| Cached GPU machine | ~40s |
| GPU under resource pressure | 1-5 min |
| No cache (new/expired image) | 10-30 min (image pull) |
Cache rules:
import os, requests
AK = os.environ.get("ACCESS_KEY", "")
BASE = "https://open.bohrium.com/openapi/v1/node"
HEADERS = {"accessKey": AK}
HEADERS_JSON = {**HEADERS, "Content-Type": "application/json"}
# Programmatic node creation (non-interactive)
r = requests.post(f"{BASE}/add", headers=HEADERS_JSON, json={
"projectId": 154, "name": "my-node", "imageId": 2168,
"machineConfig": {"type": 0, "value": 388, "label": "c2_m4_cpu"},
"diskSize": 20,
})
# Returns: {"code": 0, "data": {"machineId": 1427300}}
# Available resources
r = requests.get(f"{BASE}/resources", headers=HEADERS)
# Returns: {disks, cpuList, gpuList} — value = skuId
# Resource pricing
r = requests.get(f"{BASE}/resources/price", headers=HEADERS,
params={"skuId": 388, "projectId": 154})
# Returns: {"data": {"price": "0.4"}} (CNY/hour)
# Node details (includes SSH password)
r = requests.get(f"{BASE}/{machine_id}", headers=HEADERS)
# Returns: {nodeId, nodeName, status, ip, nodeUser, nodePwd, domainName, ...}
# Restart (must stop first)
requests.post(f"{BASE}/restart/{machine_id}", headers=HEADERS)
# Rename
requests.post(f"{BASE}/modify/{machine_id}", headers=HEADERS_JSON,
json={"name": "new-name"})
# View/bind datasets
r = requests.get(f"{BASE}/ds", headers=HEADERS, params={"nodeId": node_id})
requests.post(f"{BASE}/ds/bind", headers=HEADERS_JSON,
json={"nodeId": node_id, "datasetId": dataset_id})
| status | Meaning | CLI Display |
|---|---|---|
| 2 | Running | Started |
| -1 | Stopped/Released | Paused |
| Resource | Limit |
|---|---|
| Nodes | 4 per user per project |
| System disk | Max 100GB |
| Personal disk | 500GB per user per project |
| Shared disk | 1TB per project |
| Method | Env var source |
|---|---|
| Web Shell | System env + /root/.bashrc |
| SSH | /root/.bashrc only (overwrites globals) |
| Problem | Cause | Solution |
|---|---|---|
No resource for selected machine | Out of stock | Try another spec or retry later |
record not found | Invalid machineId | Verify with bohr node list --json |
| Restart fails | Node not stopped | bohr node stop first, wait for Paused |
nodeId vs machineId | Two different IDs | CLI uses nodeId; API uses machineId; dataset API uses nodeId |
| SSH fails | Image lacks SSH | DockerHub images need manual sshd install |
| Domain not resolving | Stopped >7 days | Restart; wait 10-30 min for DNS; use Web Shell meanwhile |
| Slow terminal | VPN/network or browser memory | Disable VPN; refresh page |
| Cannot run Docker | Container security | Use VM image LBG_Common_v2 |
| Image pulling | Cache not ready or expired | Wait 10-30 min after build |