App Node plugins
Use the default App Node plugins, mount third-party plugins, and build your own using the mdk-plugin.json format
Overview
The App Node exposes HTTP routes through a declarative plugin system. Each plugin is a directory containing an
mdk-plugin.json manifest and one or more controller files. MDK ships a set of default plugins that load automatically; you can mount additional
plugins for your own site logic.
Plugins call into the ORK kernel through services.mdkClient — the same SDK you use anywhere else in MDK. No knowledge of the
MDK Protocol envelope or internal message shapes is required.
Default plugins
MDK ships plugins that load automatically on App Node startup:
- The
authplugin serves identity and token endpoints under/auth - The
telemetryplugin serves site metrics (hashrate, consumption, efficiency, temperature, and more) under/auth/metrics - The
site-hashrateplugin serves aggregated site hashrate history
The plugin reference lists every default route, its method, and whether it needs a token — those tables are generated
from each plugin's mdk-plugin.json. Plugins you mount yourself are documented by their own manifests.
Mount a plugin
Pass an extraPluginDirs array to startAppNode() to load additional plugins at boot alongside the default plugins:
const { startAppNode } = require('@tetherto/mdk')
await startAppNode({
ork,
port: 3000,
extraPluginDirs: [
path.join(__dirname, 'plugins/custom-metrics'),
path.join(__dirname, 'plugins/alerts')
]
})Each entry must be an absolute path to a directory containing an mdk-plugin.json. The plugin loader validates the
manifest and all handler files at startup — missing files or invalid manifests throw immediately before the server comes up.
Build a plugin
A plugin is a directory with two things: a manifest and controllers.
1.1 Create the manifest
mdk-plugin.json declares the plugin identity (name, version) and a routes array. Each route needs an id, a handler path, and an http
block with a method and path. Rather than copy a synthetic example, start from a real manifest and trim it:
examples/full-site/plugins/site/mdk-plugin.json— three routes including aGET, aPOSTwith arequestBody, and path parametersbackend/core/plugins/telemetry/mdk-plugin.json— auth, caching, query parameters, and named-export handlers
Path parameters use {param} syntax — the loader normalises them to Fastify's :param format. For named exports use "handler": "./controllers/foo.js#namedExport". The plugin reference explains what each field means and what the loader requires.
1.2 Write a controller
Every controller exports an async function (req, services):
// controllers/live.js — read live telemetry
module.exports = async function live (req, services) {
const deviceId = req.query.deviceId
const telemetry = await services.mdkClient.pullTelemetry(deviceId, 'metrics')
return { deviceId, ...telemetry }
}// controllers/command.js — dispatch a command
module.exports = async function command (req, services) {
const deviceId = req.params.deviceId
const { mode } = req.body
const result = await services.mdkClient.sendCommand(deviceId, 'setPowerMode', { mode })
return {
deviceId,
commandId: result.commandId,
status: result.status
}
}The req object
| Field | Type | Contains |
|---|---|---|
req.params | object | Path parameters (e.g. { deviceId: 'wm-001' }) |
req.query | object | Query string parameters |
req.body | object | Parsed JSON request body |
req.headers | object | HTTP headers |
req._info | object | Internal request metadata (rarely needed) |
The services object
| Field | Type | Use for |
|---|---|---|
services.mdkClient | MdkClient | Live reads and command dispatch — sendCommand, pullTelemetry, getCapabilities, listWorkers |
services.dataProxy | DataProxy | Historical and aggregated data from worker tail-logs — requestData, requestDataMap |
services.authLib | AuthLib | JWT and session helpers (needed only for advanced auth flows) |
services.conf | object | App Node runtime config |
Always guard services.mdkClient — it is null when the App Node starts without a live ORK connection:
if (!services.mdkClient) throw new Error('ERR_MDK_CLIENT_UNAVAILABLE')Read hardware data
For live device data use mdkClient:
// Pull a live metrics snapshot
const tel = await services.mdkClient.pullTelemetry(deviceId, 'metrics')
// Pull the declared capabilities (from the worker's mdk-contract.json)
const { capabilities } = await services.mdkClient.getCapabilities(deviceId)
// List all registered workers
const { workers } = await services.mdkClient.listWorkers()For historical or aggregated series from a worker's persisted tail-log use dataProxy:
const results = await services.dataProxy.requestData('tailLogRangeAggr', {
type: 'miner',
startDate: start,
endDate: end,
fields: { hashrate_sum: 1 }
})The default telemetry controllers show worked examples of both patterns.
Send a command
sendCommand dispatches via the ORK to the worker that owns the device. The command must be declared in
the worker's mdk-contract.json. It returns:
| Field | Type | Description |
|---|---|---|
commandId | string | Correlation ID generated by ORK. Echo this to the HTTP caller so they can track the operation. |
status | string | 'SUCCESS' or 'FAILED' |
result | object | Command-specific response payload (present when status is 'SUCCESS') |
error | string | Error message (present when status is 'FAILED') |
const result = await services.mdkClient.sendCommand(deviceId, 'reboot', {})
if (result.status === 'FAILED') throw new Error(result.error)
return { commandId: result.commandId, status: result.status }Auth, permissions, and caching
Auth — set "auth": true on a route to require a valid Bearer token. The adapter runs authCheck before the handler is called.
Permissions — add a "permissions" array to enforce RBAC:
{
"id": "site.miners.command",
"auth": true,
"permissions": ["device:write"],
"handler": "./controllers/command.js",
"http": { "method": "POST", "path": "/site/miners/{deviceId}/command" }
}Caching — add a "cache" array of dot-path strings to enable request-level caching. The cache key is composed
from the route ID and the resolved values of each path:
{
"id": "telemetry.hashrate",
"cache": ["query.start", "query.end", "query.groupBy"],
...
}Pass ?overwriteCache=true to any cached route to bypass and refresh.
Manifest validation errors
The plugin loader validates every manifest and handler at startup and throws if anything is wrong:
| Error | Cause |
|---|---|
ERR_PLUGIN_MANIFEST_MISSING | No mdk-plugin.json found in the plugin directory |
ERR_PLUGIN_MANIFEST_INVALID | JSON parse error, or missing required field (name, version, or routes) |
ERR_PLUGIN_ROUTE_DUPLICATE_ID | Two routes in the same manifest share the same id |
ERR_PLUGIN_HANDLER_NOT_FOUND | The handler file path does not exist or failed to load |
ERR_PLUGIN_HANDLER_NOT_FUNCTION | The handler file exports something other than a function |
Full example
The examples/full-site/plugins/site/ directory is a complete worked plugin with three routes: a live site overview, a
historical series, and a command endpoint. Read it alongside this guide.
Troubleshooting
Migrate from v0.2 to v0.3
In v0.3, metricsRoutes and devicesRoutes were removed from backend/core/app-node/workers/lib/server/index.js. The auth and telemetry
endpoints they registered are now delivered by the default plugins, which load automatically — no action needed for those.
If your v0.2 code patched or monkey-patched those registrations to inject custom logic:
- Create a plugin directory with an
mdk-plugin.jsonand controller files for the routes you were injecting. - Pass the directory to
startAppNode()viaextraPluginDirs.
// Before (v0.2 — no longer works)
const server = require('@tetherto/mdk-app-node/workers/lib/server')
server.metricsRoutes.push(myCustomRoute)
// After (v0.3+)
await startAppNode({
ork,
extraPluginDirs: [path.join(__dirname, 'plugins/my-metrics')]
})