Use Umbrel DNS for Datum polling

This commit is contained in:
Dorian
2026-05-09 16:41:33 +01:00
parent 99bbd83c34
commit 4b28f760c5
5 changed files with 23 additions and 159 deletions

View File

@@ -13,11 +13,9 @@ LOG_LEVEL=info
# STATIC_DIR=
# ---- Datum gateway (the Umbrel app we're polling) ----
# Fallback URL. In Docker deployment, DATUM_DOCKER_* below lets the API discover
# Datum's current container IP before polling.
DATUM_URL=http://datum_datum_1:21000
DATUM_DOCKER_CONTAINER=datum_datum_1
DATUM_DOCKER_NETWORK=umbrel_main_network
# Datum's service alias on Umbrel's shared app network. Override if your Umbrel
# install exposes a different DNS name.
DATUM_URL=http://datum:21000
DATUM_ADMIN_USER=admin
DATUM_ADMIN_PASSWORD=
# How often to scrape /clients (ms). Datum updates per-worker hashrate every

View File

@@ -47,11 +47,11 @@ Allowlist of npubs is set via `NOSTR_ALLOWED_NPUBS`. Anything else is rejected b
- Compose path: `docker-compose.yml`
- Add env vars (see `.env.example`)
- Remove any old `DATUM_URL=http://10.21...` value from the stack env vars.
By default the stack reads Docker metadata for `datum_datum_1` on
`umbrel_main_network` and polls Datum at its current container IP, so Datum
can be recreated without changing a stale IP address.
- If your Umbrel install uses a different Datum container or network name,
set `DATUM_DOCKER_CONTAINER` or `DATUM_DOCKER_NETWORK` accordingly.
By default the stack uses `http://datum:21000` on Umbrel's
`umbrel_main_network`, so Datum can be recreated without changing a stale
IP address.
- If your Umbrel install exposes Datum under a different DNS name, set
`DATUM_URL=http://<datum-dns-name>:21000`.
4. Open the dashboard, log in with one of the allowed npubs, watch your boards lose at hashing in style.

View File

@@ -15,8 +15,6 @@ const RawEnv = z.object({
DATUM_ADMIN_USER: z.string().default("admin"),
DATUM_ADMIN_PASSWORD: z.string().min(1),
DATUM_POLL_INTERVAL_MS: z.coerce.number().int().min(1000).default(5000),
DATUM_DOCKER_CONTAINER: z.string().min(1).optional(),
DATUM_DOCKER_NETWORK: z.string().min(1).optional(),
MEMPOOL_API_URL: z.string().url().default("https://tx1138.com/api"),
NOSTR_ALLOWED_NPUBS: z.string().min(1),
@@ -55,12 +53,10 @@ export const config = {
corsOrigin: parsed.CORS_ORIGIN,
staticDir: parsed.STATIC_DIR,
datum: {
url: (parsed.DATUM_URL ?? "http://datum_datum_1:21000").replace(/\/$/, ""),
url: (parsed.DATUM_URL ?? "http://datum:21000").replace(/\/$/, ""),
adminUser: parsed.DATUM_ADMIN_USER,
adminPassword: parsed.DATUM_ADMIN_PASSWORD,
pollIntervalMs: parsed.DATUM_POLL_INTERVAL_MS,
dockerContainer: parsed.DATUM_DOCKER_CONTAINER,
dockerNetwork: parsed.DATUM_DOCKER_NETWORK,
},
mempool: {
url: parsed.MEMPOOL_API_URL.replace(/\/$/, ""),

View File

@@ -3,10 +3,8 @@ import { logger } from "../logger.js";
import { digestFetch } from "./digest.js";
import { parseClientsHtml, parseThreadsHtml } from "./parse.js";
import type { CurrentJob, DatumSnapshot, NetworkStat, PoolStat } from "./types.js";
import http from "node:http";
const DEFAULT_TIMEOUT_MS = 10_000;
const DOCKER_SOCKET = "/var/run/docker.sock";
const EMPTY_POOL: PoolStat = {
combinedHashrateThs: 0,
@@ -100,124 +98,6 @@ type MempoolHashrate = {
currentDifficulty?: number;
};
type DockerInspect = {
NetworkSettings?: {
Networks?: Record<string, { IPAddress?: string }>;
};
};
type DockerContainerSummary = {
Id: string;
Names?: string[];
Image?: string;
Labels?: Record<string, string>;
State?: string;
};
let lastResolvedDatumUrl: string | null = null;
function dockerGetJson<T>(path: string): Promise<T> {
return new Promise((resolve, reject) => {
const req = http.request(
{ socketPath: DOCKER_SOCKET, path, method: "GET", timeout: DEFAULT_TIMEOUT_MS },
(res) => {
const chunks: Buffer[] = [];
res.on("data", (chunk: Buffer) => chunks.push(chunk));
res.on("end", () => {
const body = Buffer.concat(chunks).toString("utf8");
if (!res.statusCode || res.statusCode < 200 || res.statusCode >= 300) {
reject(new Error(`Docker API returned ${res.statusCode ?? "unknown"} for ${path}: ${body}`));
return;
}
try {
resolve(JSON.parse(body) as T);
} catch (err) {
reject(err);
}
});
},
);
req.on("error", reject);
req.on("timeout", () => req.destroy(new Error("docker_socket_timeout")));
req.end();
});
}
async function resolveDatumUrl(): Promise<string> {
const container = config.datum.dockerContainer;
const network = config.datum.dockerNetwork;
if (!container || !network) return config.datum.url;
try {
const inspected = await inspectDatumContainer(container);
const address = selectNetworkAddress(inspected, network);
const port = new URL(config.datum.url).port || "21000";
lastResolvedDatumUrl = `http://${address}:${port}`;
} catch (err) {
logger.warn(
{ reason: formatErr(err), container, network, fallbackUrl: lastResolvedDatumUrl },
"datum_docker_discovery_failed",
);
if (!lastResolvedDatumUrl) throw err;
}
return lastResolvedDatumUrl ?? config.datum.url;
}
async function inspectDatumContainer(container: string): Promise<DockerInspect> {
try {
return await dockerGetJson<DockerInspect>(`/containers/${encodeURIComponent(container)}/json`);
} catch (err) {
logger.warn({ reason: formatErr(err), container }, "datum_named_container_inspect_failed");
}
const containers = await dockerGetJson<DockerContainerSummary[]>("/containers/json?all=1");
const candidates = containers
.filter(isLikelyDatumContainer)
.sort((a, b) => Number(b.State === "running") - Number(a.State === "running"));
for (const candidate of candidates) {
try {
return await dockerGetJson<DockerInspect>(`/containers/${candidate.Id}/json`);
} catch (err) {
logger.warn({ reason: formatErr(err), id: candidate.Id, names: candidate.Names }, "datum_candidate_inspect_failed");
}
}
throw new Error(
`Could not find a Datum container through Docker; checked ${containers.length} containers and expected ${container}`,
);
}
function isLikelyDatumContainer(container: DockerContainerSummary): boolean {
const names = (container.Names ?? []).join(" ").toLowerCase();
const image = (container.Image ?? "").toLowerCase();
const labels = Object.values(container.Labels ?? {}).join(" ").toLowerCase();
const haystack = `${names} ${image} ${labels}`;
if (haystack.includes("gashboard")) return false;
return (
image.includes("retropex/datum") ||
image.includes("datum_gateway") ||
names.includes("datum") ||
labels.includes("datum")
);
}
function selectNetworkAddress(inspected: DockerInspect, preferredNetwork: string): string {
const networks = inspected.NetworkSettings?.Networks ?? {};
const preferred = networks[preferredNetwork]?.IPAddress;
if (preferred) return preferred;
const entries = Object.entries(networks).filter(([, value]) => value.IPAddress);
const umbrel = entries.find(([name]) => name.toLowerCase().includes("umbrel"));
const first = umbrel ?? entries[0];
if (first?.[1].IPAddress) return first[1].IPAddress;
const networkNames = Object.keys(networks).join(", ") || "none";
throw new Error(`Datum container has no Docker network IP; networks: ${networkNames}`);
}
async function fetchNetworkStats(): Promise<NetworkStat> {
const timeout = abortableSignal(DEFAULT_TIMEOUT_MS);
try {
@@ -246,19 +126,7 @@ async function fetchNetworkStats(): Promise<NetworkStat> {
async function pollOnce(): Promise<DatumSnapshot> {
const fetchedAt = Date.now();
let datumUrl: string;
try {
datumUrl = await resolveDatumUrl();
} catch (err) {
const reason = formatErr(err);
logger.warn({ reason }, "datum_discovery_failed");
return {
...lastSnapshot,
ok: false,
fetchedAt,
error: { code: "DATUM_DISCOVERY_FAILED", message: reason },
};
}
const datumUrl = config.datum.url;
// /clients (admin Digest-gated) and /threads (public) in parallel.
const clientsTimeout = abortableSignal(DEFAULT_TIMEOUT_MS);

View File

@@ -1,7 +1,7 @@
# gashboard — deploy as a Portainer Stack on the same Umbrel host that runs Datum.
#
# Uses the Docker socket to discover Datum's current IP on Umbrel's shared
# Docker network, avoiding stale hard-coded container IPs.
# Joins Umbrel's shared Docker network so the API can reach Datum through
# Docker DNS instead of a changing container IP.
services:
gashboard:
@@ -21,11 +21,9 @@ services:
PORT: "1337"
LOG_LEVEL: "${LOG_LEVEL:-info}"
CORS_ORIGIN: "${CORS_ORIGIN:-}"
# Fallback URL. With DATUM_DOCKER_* set below, the API resolves Datum's
# current container IP before polling.
DATUM_URL: "${DATUM_URL:-http://datum_datum_1:21000}"
DATUM_DOCKER_CONTAINER: "${DATUM_DOCKER_CONTAINER:-datum_datum_1}"
DATUM_DOCKER_NETWORK: "${DATUM_DOCKER_NETWORK:-umbrel_main_network}"
# Datum's service alias on Umbrel's shared app network. Override this if
# your Umbrel install exposes a different DNS name.
DATUM_URL: "${DATUM_URL:-http://datum:21000}"
DATUM_ADMIN_USER: "${DATUM_ADMIN_USER:-admin}"
DATUM_ADMIN_PASSWORD: "${DATUM_ADMIN_PASSWORD?must be set}"
DATUM_POLL_INTERVAL_MS: "${DATUM_POLL_INTERVAL_MS:-5000}"
@@ -33,13 +31,17 @@ services:
NOSTR_ALLOWED_NPUBS: "${NOSTR_ALLOWED_NPUBS?must be set}"
JWT_SECRET: "${JWT_SECRET?must be set}"
JWT_TTL_SECONDS: "${JWT_TTL_SECONDS:-86400}"
network_mode: host
user: "0:0"
volumes:
- /var/run/docker.sock:/var/run/docker.sock:ro
ports:
- "${PORT:-1337}:1337"
networks:
- umbrel_main_network
healthcheck:
test: ["CMD", "wget", "-qO-", "http://127.0.0.1:1337/healthz"]
interval: 30s
timeout: 5s
retries: 3
start_period: 15s
networks:
umbrel_main_network:
external: true