The bridge to everything else
v0.5.3.1It's as simple as:
# Get current track info (just open in browser)
http://localhost:8080/nowplaying
# Get album artwork (returns image)
http://localhost:8080/nowplaying/artwork
# Browse all albums in library
http://localhost:8080/library/albums
# Skip to next track
curl -X POST http://localhost:8080/player/next
Try it now: /nowplaying • /nowplaying/artwork • /library/albums
Web-based remote control with integrated library search. All actions use POST-Redirect-GET pattern. Supports keyboard shortcuts: / (search), Space (play/pause), ←/→ (prev/next), ↑/↓ (volume ±1%), Shift+↑/↓ (volume ±5%), M (mute).
Full web dashboard with library search, now playing, controls, volume, ratings, shuffle/autodj/repeat toggles. Panel visibility, order, and collapsible grouping are configurable via dashboardLayout in settings.
Search: client-side fuzzy album/artist matching + server-side track search. Albums containing matching tracks are boosted (e.g. “wonderwall” surfaces its album). Inline track expansion on album results. Explore button seeds explore.html?album=&artist=.
volup/voldown - Adjust volume by 5%.
volup1/voldown1 - Adjust volume by 1% (fine control).
shuffle=enable, shuffle-off=disable both, autodj=start AutoDJ, repeat=cycle mode
love - Toggle love tag on current track.
rate/0 - Toggle bomb (don't play). Click on → click off.
rate/1-5 - Toggle star rating. Click to set, click same to clear.
ban - Ban track from shuffle, skip to next (undo: skip back within 10s).
setban - Set ban flag without skipping.
mood - Set AutoQ mood channel (form: mood=Energetic).
react/fire|heart|like|dislike - Submit reaction. Fire triggers queue refresh.
refresh-queue - Refresh vibe list and queue tracks.
influence/artist|genre/up|down - Thumbs up/down for artist or genre.
Play or queue a playlist (form: playlistUrl=...&action=now|next|last).
Dashboard layout, display toggles, and feature kill-switches are all configured in mbxhub.json. Full field list with defaults, types, and descriptions: GET /system/settings/schema or the config reference. Live feature state (except proxy) is exposed via GET /system/features. Section IDs for dashboardLayout.order and .hidden: status, search, nowplaying, rating, controls, volume, charms, mood, playlists, toggles, footer.
Now Playing Styles (nowPlayingStyle, default: "full"):
| Style | Description |
|---|---|
full | Standard layout with full-size artwork and metadata |
horizontal | Side-by-side artwork and metadata |
noart | No artwork, text-only display |
split | Two-column grid (art left, metadata right). Column ratio configurable via localStorage mbxh_split_ratio (default 55, range 40–70). Falls back to stacked below 480px |
immersive | Full-bleed album art with gradient overlay and blurred letterbox fill for non-square art. Metadata fades on hover/touch, fades out after dashboardLayout.immersiveFadeDelay (0 = always visible) |
Client-side override: localStorage key mbxh_np_style overrides the server default per-device. Header button cycles through all styles. All styles have zoom-level overrides (67%, 42%, 27%). The compact style was removed in v0.5.2.3 — existing configs gracefully fall back to full.
MBXHub can serve custom HTML pages from a configurable directory. This enables building custom web UIs that use the REST API.
/pages/ with correct Content-Type headers.
Default pages are extracted on first run and can be customized without losing changes on updates.
Set pagesPath in mbxhub.json to customize the pages directory:
{
"pagesPath": "C:\\MyCustomPages"
}
Default location: %APPDATA%\MusicBee\MBXHub\pages\
Set defaultPage to change the root URL redirect:
{
"defaultPage": "/pages/player.html"
}
Options: /dashboard (default), /pages/player.html, /pages/play.html, /pages/partymode/, /pages/nowplaying.html, or any custom page.
Kiosk Mode: Lock the display to a single page/app. All navigation redirects to the default page.
{
"defaultPage": "/pages/partymode/",
"kioskMode": true
}
Multi-page apps work: sub-paths are allowed (e.g., /pages/partymode/ allows /pages/partymode/guest.html). API calls and resources still work. Only editable in mbxhub.json (not exposed via API).
Serves index.html from the pages directory
Serves any file from the pages directory (HTML, CSS, JS, images, fonts)
/pages/index.html - Landing page listing available views/pages/player.html - Full-featured desktop player with:
/pages/nowplaying.html - Focused now-playing view with artwork, lyrics, and queue. Auto-stacks below 800px with accordion sections (Now Playing, Queue, Lyrics) — expanding one collapses the others/pages/browse.html - Library browser with 8 responsive tabs, drilldown navigation, fuzzy search with auto-search fallback (triggers server search when no inline matches), album art grid, video direct play, batch queue, and playlist picker/pages/config.html - Settings and configuration for dashboard layout and AutoQ parameters/pages/autoq.html - AutoQ Tuning Console with mixer-style sliders for scoring weights and normalization ranges/pages/mixer.html - Unified fader mixing surface: three independent faders for Player (MusicBee), Device (Windows audio), and Endpoint (network speaker) volume. Configurable default fader, mute controls, endpoint source selection/pages/explore.html - Album art explorer: browse albums as artwork with source filters, search, sort, image gallery, PDF booklets, play/queue. Accepts ?album=&artist= for seeding. Artist discography grid in expanded view. Expanded-view hero has paired close (×) and dashboard home (⌂) buttons — close stays on explore, home returns to /dashboard (home hidden in iframe-mode)/pages/play.html - Use MusicBee from a browser. Laid out like the MusicBee AMOLED skin: artist picker (left) with infinite-scroll + jump-on-letter typeahead, pluggable middle pane (Albums / Library / Now-Playing), upcoming-queue + now-playing card (right), full-width transport footer.
/influences/currentmbxhub.log via /system/client-log (no console-only logging)/pages/components/dashboard.css — Dashboard stylesheet (~2,900 lines). Served as an external linked resource by /dashboard. Theme variables are injected separately in an inline <style> block ahead of this link; this file holds the bulk of dashboard styling (layout, transport, charm bar, command palette, theme designer, party banner, search results). Linked with ?v={version} for release-driven cache invalidation./pages/components/dashboard.js — Dashboard client script (~2,800 lines). WebSocket lifecycle, transport handlers, theme designer, charm bar, partial-section reload, Cmd+K wiring. Loaded with defer by /dashboard after an inline <script> that defines the per-request _themeData JSON. Same ?v={version} cache-busting pattern./pages/components/cmdk.html — v0.5.3.0 Cmd+K command palette. Self-mounting <style> + <dialog> + <script> fragment. XSS-safe (textContent only, no untrusted innerHTML). Federated search via /search, recents via /search/history with localStorage fallback. Actions are POSTs (Play/Pause/Skip/Volume, Start AutoQ radio) or navigations (Charms / Settings / Mixer / Player / Browse / Explore / ARiA). Settings nested under dashboardLayout.commandPalette.{enabled, openShortcut, showChip, recentLimit, bucketLimit, actions, enabledOnPages}./pages/components/cmdk-bootstrap.js — v0.5.3.0 palette loader. Drop-in <script> include — loads search-shared.js then injects cmdk.html via DOMParser so embedded scripts execute. Self-mounting; degrades gracefully when search-shared.js is unreachable. Wired into dashboard, play.html, explore.html, nowplaying.html, and browse.html./pages/search-shared.js — Search lifecycle helper. Provides MBXSearch.attachSearch for debounced typeahead with abort-on-keystroke (exposed as .abort() on the returned handle for v0.5.3.0+ callers that need to cancel in-flight fetches before navigation). Required by cmdk; usable standalone./pages/components/shared.css — v0.5.3.0 shared frontend core. Cross-page styles consolidated from per-page duplication: CSS reset + box-sizing, prefers-reduced-motion rules, focus / focus-visible defaults, dual theme palettes (default + theme-quiet, light + dark), scrollbar styling, .album-art-placeholder, .hl highlight class, .empty-state. Linked first in each shell page's <head> so per-page styles can override./pages/components/shared.js — v0.5.3.0 shared frontend core. Cross-page helpers under the MBXShared global namespace: setPageName / getPageName, clientLog (batched + sendBeacon-on-pagehide) + flushClientLog, esc (HTML-encode), DIACRITICS (char map ported from MusicBee CharMap.cs — union of browse / explore / player local maps, 1:1 char invariant preserved), normalize (iter-based: DIACRITICS lookup → lowercase → strip apostrophes → collapse non-alphanumerics to single spaces → trim; NFD form removed in the Option C foundation), findHighlightSpans(text, query) (returns Array<{start, end}> half-open ranges in ORIGINAL-text index space — walks normalized text, finds full-phrase + per-word occurrences, leaves HTML rendering to the caller so pages can wrap spans their own way), fmtDuration (ms → m:ss or h:mm:ss), connectWebSocket (subscribe + JSON parse + dispatch-by-event + auto-reconnect with capped backoff). Loaded with defer by every shell page./manifest.webmanifest — v0.5.3.0 PWA manifest. Returns application/manifest+json. The name field carries the literal token __HOST_NAME__ in source, replaced at serve time with Environment.MachineName so the installed PWA shows "MBXHub - <hostname>" in browser install dialogs and “Manage apps” UI (the OS-level desktop / drawer label uses short_name, which stays plain "MBXHub" so renames don't orphan shortcuts). Standalone display, theme color matches the dashboard surface, icons array references /icons/icon-{192,512}.png plus maskable variants. Served with Cache-Control: no-cache (Chromium's installed-PWA “update on reload” path otherwise heuristically caches the manifest indefinitely — install never picks up the dynamic hostname on app updates)./icons/{name} — v0.5.3.0 PWA icon set. Resolves the embedded MBXHub.Resources.icons.{name} and serves with image/png. Standard set: icon-192.png, icon-512.png, icon-maskable-192.png, icon-maskable-512.png, apple-touch-icon-180.png. Generated from MBXS\MBXHub.Shell\brand\mbxhub-halard.png.To customize pages:
player.html or create new HTML filesMBXHub serves /llms.txt - an AI-friendly API reference. Use it with Claude or any AI to generate custom pages:
/nowplaying, /player/play) that work on any MBXHub instance%APPDATA%\MusicBee\MBXHub\pages\http://localhost:8080/pages/partyon.htmlThe generated code uses relative URLs, so it works on your local MBXHub without modification.
MBXHub advertises itself on the local network via three protocols so clients can find it automatically.
DnsServiceRegister API.
Requires Windows 10 1809+; gracefully skipped on older systems. Registered as MBXHub (Name)._http._tcp.local with TXT records for path and version.
UPnP device description XML. Contains device info, service URLs, and presentation URL.
WS-Discovery metadata exchange endpoint. Windows sends a SOAP GetMetadata request after discovering MBXHub via UDP Probe; response includes PresentationUrl pointing to /dashboard.
| Field | Description |
|---|---|
| deviceType | urn:halrad-com:device:MBXHub:1 |
| friendlyName | MBXHub instance identifier |
| presentationURL | Dashboard URL (/dashboard) |
| controlURL | REST API base (/api) |
| eventSubURL | WebSocket endpoint (/ws) |
| Port | Protocol | Owner | Purpose | When needed |
|---|---|---|---|---|
| 8080 | TCP | Plugin | REST API + WebSocket (restPort) | Always — the plugin's main HTTP listener |
| 8081 | TCP | Shell | SMTC control routes /meta/smtc/* (smtc.port, convention is REST port + 1) | Only when MBXHub.exe is running. Open it for remote SMTC retargeting from the dashboard, or for Bluetooth/lock-screen control on a different machine. Skip it for plugin-only / NAS / headless installs. |
| 1900 | UDP | Plugin | SSDP (UPnP discovery) | For Windows Network folder + SSDP browsers |
| 3702 | UDP | Plugin | WS-Discovery (Windows Network folder) | For Windows Explorer Network integration |
| 5353 | UDP | Plugin | mDNS/DNS-SD (device discovery via raw UDP multicast) | For Bonjour-style zero-conf discovery (Win10 1809+) |
Plugin-managed (REST port + Shell SMTC port + UDP discovery) — one rule covers everything the plugin can see. Use the Settings → Firewall panel or CLI:
MBXHub.exe firewall add --name MBXHub --tcp 8080,8081 --udp 1900,3702,5353 --urlacl 8080,8081
Shell-only (SMTC port, added by Shell itself) — MBXHub.exe --install writes a MBXHub-Shell rule and URL ACL for smtc.port automatically; --uninstall removes it. You only need to run firewall commands manually if you skip --install.
Override the Shell SMTC port by editing smtc.port in mbxhub-shell.json. The plugin's firewall helper auto-adopts restPort + 1 as the Shell port; if you change smtc.port from that convention, open the new value manually.
discoveryEnabled (default: true), discoveryName (default: empty = machine name).
The discovery name is used as the friendly name across all protocols and in GET /system/version.
HTML status dashboard with live stats: system info (version, uptime, host, modules), library counts (tracks, albums, artists, genres, playlists, podcasts), AutoQ state (status, vibe list, mood cache, auto scan), and feature flags. All data fetched client-side from existing API endpoints.
Returns system status and enabled modules (JSON)
Diagnostic surface — a live “Geiger counter” for steady-state activity. Off by default; gated on diagnostics.diagEndpointEnabled = true in mbxhub.json. Returns 404 when disabled, with explanatory body.
/diag serves a self-contained HTML page that polls /diag/perf at 1 Hz from the browser and renders inline-SVG sparklines for: process CPU %, network bytes in/out, working-set / private bytes, thread count, handle count, GC gen0/1/2 collection rate. Last 120 samples (~2 min) held client-side. Server is stateless — no background timer, no ring buffer, no allocation while the page is closed.
/diag/perf returns an instant snapshot of public process counters via System.Diagnostics.Process, System.Net.NetworkInformation, and GC.CollectionCount. Fields: ts, processorMs, workingSetMB, privateMB, threadCount, handleCount, gen0/gen1/gen2, bytesIn, bytesOut, logicalProcessors. Cumulative counters; the page diffs them client-side to derive per-second rates.
/diag/mbxhub returns a per-MBXHub instrumentation snapshot — counters specific to this plugin (request rates, broadcaster fan-out, lock waits, etc.) rather than the OS-level metrics in /diag/perf. The /diag page renders this as a second tile row. Same diagEndpointEnabled gate.
/diag/snapshot (v0.5.2.6+) takes a JSON body containing the page’s client-side ring (samples + derived rates + small header) and appends a formatted text block to diag-snapshots.log next to mbxhub.log. Each snapshot is delimited by “=” rules with a per-metric min/avg/max/latest table and the raw samples JSON below for forensic re-processing. Triggered by the Snapshot button on the /diag page; same diagEndpointEnabled gate. Response: { success, path, bytes, at }.
/diag/search (v0.5.3.0) returns a rolling p50/p90/p95/p99 summary over the last 256 search calls (in-memory ring, ~28 KB, server-wide singleton). Summary is segregated by endpoint: library (calls into /library/search), federated (calls into /search), and combined. Each block carries totalMs {p50,p90,p95,p99,max,mean}, candidates {p50,p95,max}, and mbCalls {p50,p95,max}. The recent array (default 50, override with ?recent=N, capped at ring capacity) lists the newest entries with per-stage timings (mb / cue / am / pf / sort / pg) and per-family MB-call counts (qf / ql / qlr / gt / gts / gp / gb / pl). POST /diag/search/reset clears the ring (operator-triggered flush; empty body, send Content-Length: 0). Same diagEndpointEnabled gate. Returns 403 FORBIDDEN when ApiReadOnlyMode is set — kiosk deployments can’t have their perf history wiped by remote clients.
// Enable in mbxhub.json:
{
"diagnostics": {
"diagEndpointEnabled": true
}
}
// Then visit:
// http://host:8080/diag → live page
// http://host:8080/diag/perf → JSON snapshot
Returns API version information. Includes host field with the configured discovery name (or machine name if not set).
Returns this node's capabilities for cross-channel SSDP discovery. Used by Shell and other MBXHub nodes to identify what this instance offers.
// Response:
{
"node": "plugin",
"version": "0.5.3.1",
"capabilities": ["rest-api", "websocket", "player-control", "autoq", "discovery"],
"endpoints": {
"rest": "http://host:8080",
"ws": "ws://host:8080/ws"
}
}
Health check endpoint (alias: /system/ping)
Returns enabled feature flags: banlist, ratings, loved, reactions, streaming, autoq. Clients use these to show/hide UI elements.
Get or update hub configuration (ports, enabled modules, log verbosity, library behavior). restPort and restEnabled changes require localhost (403 for remote). All changes blocked for remote clients during party mode.
Beyond the network fields (restPort, restEnabled, wsEnabled, defaultPage), GET and PUT also surface (MBRC was cut; mbrcPort / mbrcEnabled no longer exist):
{
"logLevel": "Info", // trace/debug/info/warn/error, case-insensitive; propagates to Shell
"library": { "disableStrictSearch": false }, // v0.5.2.4+: disables phrase-per-field strict search
"search": { "live": { "minQueryLength": 2 } } // v0.5.3.0: server-side minimum-length backstop for /search and /library/search
}
logLevel is validated against the same vocabulary as HubLogger.ParseLevel/ShellLog.ParseLevel; invalid values are silently ignored (same soft-skip pattern used for restPort range checks). library.disableStrictSearch takes a nested object — the parent object is created if missing. search.live.minQueryLength defaults to 2; queries below this return 400 QUERY_TOO_SHORT. None of these require localhost; all are UX-level settings safe to change remotely.
Returns schema for all configurable settings. Each entry includes:
| Field | Description |
|---|---|
key | Dotted key path (e.g. autoQ.batchSize) |
type | bool, int, double, string, enum |
category | Grouping: General, AutoQ, Scoring, Dashboard, API, etc. |
tier | Standard, Advanced, or Expert |
description | Human-readable explanation |
default | Default value |
current | Current live value |
min, max, step | Range constraints (numeric types only) |
options | Valid values (enum types only) |
requiresRestart | Whether a restart is needed for the change to take effect |
Security: blocked when disableRemoteConfig is true (403) or during party mode (403). The /pages/settings.html page consumes this endpoint.
Update configurable settings via dotted key paths. Only properties decorated with [ConfigSetting] can be modified.
Body: JSON object with dotted keys:
{"autoQ.batchSize": 10, "apiReadOnlyMode": true}
Security: blocked by apiReadOnlyMode (403), disableRemoteConfig (403), and party mode (403). POST is an alias for PUT.
Get or set the default redirect page (body: {"defaultPage":"/pages/player.html"})
Get or update the unified theme configuration. Two configurable mode slots (mode1, mode2) each with 13 HSL fields: accentHue (0–360), accentSaturation (0–100), accentLightness (0–100), bgHue (0–360), bgSaturation (0–100), bgLightness (0–100), surfaceHue (0–360), surfaceSaturation (0–100), surfaceLightness (0–100), textHue (0–360), textSaturation (0–100), textLightness (0–100), intensity (0–100, saturation multiplier: 0 = grayscale, 100 = full color). Partial updates supported — only include the fields you want to change.
// GET response:
{
"activeMode": 1,
"mode1": {
"accentHue": 197, "accentSaturation": 80, "accentLightness": 55,
"bgHue": 203, "bgSaturation": 30, "bgLightness": 94,
"surfaceHue": 203, "surfaceSaturation": 30, "surfaceLightness": 96,
"textHue": 203, "textSaturation": 50, "textLightness": 13,
"intensity": 100
},
"mode2": {
"accentHue": 197, "accentSaturation": 80, "accentLightness": 55,
"bgHue": 203, "bgSaturation": 30, "bgLightness": 7,
"surfaceHue": 203, "surfaceSaturation": 30, "surfaceLightness": 10,
"textHue": 203, "textSaturation": 50, "textLightness": 90,
"intensity": 100
},
"active": {
"accentHue": 197, "accentSaturation": 80, "accentLightness": 55,
"bgHue": 203, "bgSaturation": 30, "bgLightness": 94,
"surfaceHue": 203, "surfaceSaturation": 30, "surfaceLightness": 96,
"textHue": 203, "textSaturation": 50, "textLightness": 13,
"intensity": 100
},
"disablePinchZoomLock": false
}
// PUT examples:
{"activeMode": 2} // Switch to mode 2
{"mode1": {"accentHue": 180}} // Update just accent hue on mode 1
{"mode1": {"bgLightness": 10, "textLightness": 90}} // Make mode 1 dark
{"mode1": {"intensity": 0}} // Desaturate mode 1 to grayscale
PUT broadcasts a ThemeChanged WebSocket event and sets the mbxh_theme cookie for dashboard SSR.
Process metrics for the MusicBee host process. Used by MBXHVAL for remote monitoring.
// Response:
{
"success": true,
"data": {
"process": {
"name": "MusicBee",
"cpuPercent": 2.15,
"memoryMB": 185.3,
"privateMemoryMB": 210.5,
"threadCount": 42,
"handleCount": 1250
},
"gc": {
"gen0": 150,
"gen1": 30,
"gen2": 5,
"totalMemoryMB": 45.2
},
"timestamp": "2026-03-10T12:00:00.0000000Z"
}
}
Server start time and uptime. Used by dashboard to detect restarts.
// Response:
{
"success": true,
"data": {
"startedAt": "2026-03-10T20:30:00.0000000Z",
"uptimeSeconds": 3600,
"uptime": "1h 0m"
}
}
Generate QR code PNG image for the MBXHub base URL. Optional ?url= for custom target.
Returns the client's IP address as seen by the server. Used by the SMTC Link Charm to discover the client's local Shell when accessing a remote dashboard.
Push a named event into the WebSocket broadcaster from sub-process components (Shell, scripts, automations). The request body is forwarded as the event payload; dashboards subscribed to {name} receive it. Used to fan out signals that originate outside the plugin process.
Force re-extract all embedded resources (pages and charms) to disk. Overwrites existing files.
// Response:
{
"success": true,
"data": {
"pagesUpdated": 8,
"charmsUpdated": 3,
"userModified": []
}
}
Accepts browser-side log events and writes them to mbxhub.log. Use this instead of console.log for operational telemetry from dashboard pages so problems in guest browsers land in the server-side log where the operator can see them. Accepts either a single event or a batch via events[]. Silent on empty body (returns {success:true, data:{accepted:0}}); accepted counts events that passed validation and reached the log. Invalid JSON returns 400 INVALID_REQUEST (not INVALID_JSON — this endpoint predates the topology handler’s specific code). Access: ActionCategory.System — admin-gated. PartyMode guests are rejected (they lack System) so guest-sourced log entries can’t mix into operator telemetry.
// Single event:
POST /system/client-log
{ "level": "warn", "msg": "Retry #3 on /queue/add" }
// With page context (prefixes the log line as "[page] msg"):
POST /system/client-log
{ "level": "info", "msg": "Charm bar rendered", "page": "dashboard" }
// Batch:
POST /system/client-log
{ "events": [
{ "level": "info", "msg": "Charm bar rendered" },
{ "level": "error", "msg": "ThemeChanged handler threw" }
] }
// Response:
{ "success": true, "data": { "accepted": 2 } }
Recognized fields: msg (required; empty or missing drops the event), level (optional, defaults to info; free-form string mapped to server log levels, unknown values log as info), and page (optional; wraps the output as [page] msg). Any other fields on the event object are silently dropped — if you need to capture a stack trace or structured data, concatenate it into msg. Both msg and page are sanitized: CR/LF/NUL stripped (blocks log-line forgery) and length-capped (4096 chars for msg, 256 for page).
Runtime SMTC target management — discover MBXHub endpoints on the network and switch which one the Shell mirrors to Windows Media Transport Controls (taskbar overlay, lock screen, Bluetooth headsets).
Served by the Shell (MBXHub.exe), not the plugin. Listens on smtc.port from mbxhub-shell.json, default 8081 (REST port + 1). The listener emits Access-Control-Allow-Origin: * and handles OPTIONS preflight, so any browser page served by the plugin on 8080 can call this port directly cross-origin.
Current SMTC target (host:port) and connection state. Response shape: { "target": "127.0.0.1:8080", "state": "connected", "controller": { "nodeId": "...", "name": "...", "restPort": 8080 } }.
Switch SMTC target. Body: { "target": "host:port" }. Tears down the existing WS connection, updates Shell config, reconnects to the new endpoint. 422 if the target probe fails.
Cached list of MBXHub endpoints discovered on the network. Each entry: address, name, active (boolean), capabilities (e.g. ["rest-api", "websocket", "player-control", "autoq", "discovery"]). Returns 503 if the SMTC bridge is unavailable.
Clear the cache and re-scan the network via SSDP for MBXHub endpoints. Returns the fresh endpoint list.
Returns current track info with full metadata
// Response:
{
"success": true,
"data": {
"playing": true,
"url": "file:///C:/Music/Artist/Album/track.mp3",
"title": "Love Don't Live Here",
"artist": "Breaking Rust",
"album": "Greatest Hits",
"albumArtist": "Breaking Rust",
"year": "2024",
"genre": "Rock",
"trackNo": "3",
"discNo": "1",
"rating": "4.5",
"love": true,
"duration": 234000,
"position": 45000
}
}
Tag fields: TrackTitle, Artist, Album, AlbumArtist, Year, Genre, Rating, RatingLove, Comment, Composer, Conductor, TrackNo, DiscNo, Lyrics, Publisher
Properties: Bitrate, SampleRate, Channels, Duration, Size, DateAdded, DateModified, PlayCount, SkipCount, LastPlayed
Examples:
GET /nowplaying/tag?field=Artist → {"value": "Breaking Rust"}
GET /nowplaying/tag?field=Album → {"value": "Love Don't Live Here"}
GET /nowplaying/tag?field=Genre → {"value": "Rock"}
GET /nowplaying/tag?field=TrackNo → {"value": "3"}
GET /nowplaying/property?type=Bitrate → {"value": "320"}
GET /nowplaying/property?type=Duration → {"value": "234000"} (ms)
GET /nowplaying/property?type=PlayCount → {"value": "42"}
Returns album artwork as binary image or URL
Returns the current track's lyrics. Three response shapes:
{ hasLyrics: false, lyrics: null } — no lyrics and no fallback.{ hasLyrics: true, lyrics, source: "lyrics" } — real MusicBee lyrics.{ hasLyrics: true, lyrics, source: "comment", label } — fallback from the track Comment tag (great for concert setlists, album liner notes). label is the chip text shown above the body in the UI.Fallback is configured via the LyricsFallback section (Enabled, MaxDisplayChars, Label) and can be hard-killed via ApiDisableLyricsFallback.
/nowplaying/artist-picture-urls — returns a pictures array with src URLs (serveable via /nowplaying/artist-pictures/{index}) in addition to raw file paths.
/nowplaying/artist-pictures/{index} — serve current artist’s Nth picture as binary image (0-based). Content-Type auto-detected; Cache-Control: max-age=3600. Query: ?localOnly=true (default).
FFT spectrum and waveform data for visualizations
Current stereo peak and RMS levels (0.0–1.0). Returns {peak: [L, R], rms: [L, R]}. Requires MusicBee 3.6+ (API rev 58+).
Returns the now playing list. Supports ?offset=0&limit=50
// Response:
{
"success": true,
"data": {
"currentIndex": 5,
"total": 150,
"offset": 0,
"limit": 50,
"tracks": [
{
"index": 0,
"url": "file:///C:/Music/track1.mp3",
"title": "First Track",
"artist": "Artist Name",
"album": "Album Name",
"duration": 234000
},
// ... more tracks
]
}
}
// POST /queue/add - Add tracks to queue (position: "next" or "last")
{"urls": ["file:///C:/Music/song1.mp3", "file:///C:/Music/song2.mp3"], "position": "last"}
// or single track:
{"url": "file:///C:/Music/song.mp3", "position": "next"}
// Response: {"success": true, "data": {"result": true, "added": 2, "position": "last"}}
// POST /queue/playnow - Play track immediately
{"url": "file:///C:/Music/song.mp3"}
// POST /queue/play - Play track at index
{"index": 3}
// POST /queue/move - Move track in queue
{"from": 5, "to": 2}
Query library with ?query=, ?artist=, ?albumArtist=, ?album=, ?genre=, ?sort=
// GET /library/files?artist=Breaking%20Rust&limit=10
{
"success": true,
"data": {
"total": 42,
"offset": 0,
"limit": 10,
"tracks": [
{
"url": "file:///C:/Music/Breaking Rust/track.mp3",
"title": "Love Don't Live Here",
"artist": "Breaking Rust",
"album": "Greatest Hits",
"duration": 234000
},
// ... more tracks
]
}
}
Full-text search: ?q=search+term&sort=alpha (searches title, artist, album, genre; diacritic and punctuation normalized so "cafe" matches "Café" and "acdc" matches "AC/DC"). Two matching modes:
The default mode is controlled by the Library.DisableStrictSearch setting (false by default = strict). Override per-call with ?substring=true|false. The response payload includes a mode field echoing which mode was used.
v0.5.3.0: DSL auto-detection is on by default (library.search.dsl.enabled = true). Qualifier syntax (artist:, album:, genre:, year:, rating:, fmt:, range / boolean / grouping operators) routes through SearchDslParser automatically when detected; plain free-text queries take the strict/substring path as before. Force per-call with ?dsl=true; set library.search.dsl.enabled = false to require the explicit opt-in. Cheat sheet at GET /library/search/syntax.
v0.5.3.0 backstop: queries shorter than search.live.minQueryLength (default 2) return 400 QUERY_TOO_SHORT with body { "error": "Query must be at least N characters (got M)." }. Single-char walks visit every track on a 200k library (~4 s) and stall every other MB-API consumer for the duration of the cursor lock; the backstop protects fleet-wide responsiveness from runaway non-conforming clients (curl, scripts). Set search.live.minQueryLength = 0 to disable.
v0.5.3.0. Returns the DSL grammar as JSON (qualifiers, operators, worked examples). Authoritative source — the reference below is rendered from the same data shape. Used by Cmd+K's cheatsheet and any other client surfacing DSL syntax to users.
Response shape: { version, qualifiers: [{ name, type, allowsRange, allowsOperators, mapping, enumValues }], operators: [{ symbol, description }], examples: [{ query, description }] }.
The DSL layers on top of free-text search: every plain word still matches normally, qualifiers narrow the result, operators compose. Auto-detected on /library/search and /search when library.search.dsl.enabled = true (default); force per-call with ?dsl=true.
| Qualifier | Type | Range / Ops | Maps to | Notes |
|---|---|---|---|---|
artist: | string | — | ArtistPeople | Includes featured / album artists. |
album: | string | — | Album | |
albumartist: | string | — | AlbumArtist | Album-level artist (compilation-aware). |
genre: | string | — | Genre | Multi-value genre tags split client-side. |
year: | int | range + ops | Year | e.g. year:1985, year:1985..1990, year:>=2000. |
decade: | enum | — | derived from Year | Values: 60s, 70s, 80s, 90s, 00s, 10s, 20s. |
rating: | int | range + ops | Rating (0–5) | e.g. rating:>=4, rating:3..5. |
loved: | bool | — | Loved tag | loved:true / loved:false. |
bpm: | int | range + ops | Tempo (Truedat / Essentia) | Requires fingerprint or mood-cache data. |
mood: | string | — | AutoQ mood channel | Channel name (e.g. mood:chill). Post-filter — runs after candidate selection. |
vibe: | float | range + ops | AutoQ vibe score | 0–1. e.g. vibe:>=0.7. |
source: | enum | — | MB Source Type | Values: library, inbox, audiobooks, videos, podcasts. Default from library.search.dsl.defaultSource. |
playlist: | string | — | playlist filter | Post-process — intersects with named playlist membership. |
added: | date | range + ops | DateAdded | Accepts absolute dates (2025-01-01) and relative (now-7d); user aliases via library.search.dsl.dateAliases. |
played: | date | range + ops | DateLastPlayed | Same date format as added:. |
playcount: | int | range + ops | PlayCount | e.g. playcount:>10, playcount:0 (never played). |
duration: | int (seconds) | range + ops | Duration | e.g. duration:>3600 (over an hour). |
path: | string | — | FilePath substring | Case-insensitive substring match on the full path. |
lyric: | string | — | lyrics body | Post-process — fetches lyrics per candidate. Expensive; gated on library.search.dsl.allowLyricSearch = true (default off). |
| Symbol | Meaning | Where it applies |
|---|---|---|
> | greater than | Qualifiers with allowsOperators. |
>= | greater than or equal | Same. |
< | less than | Same. |
<= | less than or equal | Same. |
.. | inclusive range | Qualifiers with allowsRange. low..high — both ends inclusive. |
- (prefix) | exclude | Negates a free-text term or qualifier (e.g. -genre:metal, -live). |
OR / or | boolean OR | Between sibling expressions. Default between terms is AND. |
( ... ) | grouping | Forces precedence inside a larger expression. |
| Query | Result |
|---|---|
artist:radiohead | Tracks by Radiohead. |
year:1985..1990 | Tracks released between 1985 and 1990 inclusive. |
year:1965 rating:>4 | 1965 tracks rated above 4 — combines two range/operator qualifiers (implicit AND). |
mood:chill rating:>=4 | Chill-mood tracks rated 4 or higher — mood is a post-filter applied after the rating cut. |
rock -genre:metal | Rock tracks excluding the metal genre — free-text plus exclusion. |
(rock or metal) -live | Rock or metal, but not live recordings — boolean OR plus exclusion. |
source:audiobooks duration:>3600 | Audiobooks longer than one hour — source scope plus duration operator. |
played:<now-30d playcount:>5 | Favorites you haven't played in the last 30 days — relative date plus playcount. |
decade:80s -genre:disco | 80s, no disco — decade enum plus genre exclusion. |
OR (or lowercase or) for disjunction; parentheses for grouping.radiohead year:>=2000 filters Radiohead tracks from 2000 onward.mood, playlist, lyric) run after the candidate set is fetched, so they're capped by library.search.dsl.maxPostFilterCandidates (default 5000).422 INVALID_DSL with a parse position and Levenshtein-based “did you mean” suggestions.2025-01-01), relative (now-7d, now-30d), or named alias from library.search.dsl.dateAliases (defaults: lastweek, lastmonth, thisyear).v0.5.3.0 federated search. Runs typed buckets (tracks / albums / artists / playlists / saved) in parallel and returns a unified response with cursor pagination on the tracks bucket, facet counts, and a top-hit. Backs the Cmd+K palette and (v0.5.3.0+) the dashboard search bar, player.html, browse.html, and explore.html.
DSL routing (v0.5.3.0 post-launch): /search auto-detects DSL queries the same way /library/search does — qualifier colons (year:, rating:, etc.), .. ranges, >/< comparisons, OR , leading - exclusion. When detected and Search.Dsl.Enabled=true (default), the query routes through the DSL pipeline and the response carries mode: "dsl". Per-call ?dsl=true still works as an explicit opt-in. Plain free-text queries continue to use strict/substring mode as before. Tracks bucket is sorted by match-strength score (title > album > artist, with prefix-match bonus) so the most-relevant result appears first regardless of mode.
Params: ?q=&buckets=tracks,albums,artists,playlists,saved&limit=N&cursor=opaque. Cursors are HMAC-stamped + TTL-checked; stale → 410, malformed → 400, query-mismatch → 400. Same QUERY_TOO_SHORT backstop as /library/search. v0.5.3.0: per-bucket ?limit= hard cap raised from 200 to 500 (browse.html aggregates client-side over the tracks bucket and needs the headroom).
v0.5.3.0. Live engine + index state snapshot. Currently reports engine: "mb" (FTS5 substrate dropped during the post-decouple harvest; seam preserved for future engines).
Persist a query under a name and re-run on a schedule. Backed by mbxhub-search.json next to mbxhub.json. Disabled by default; enable via library.searchDsl.savedSearch.enabled. Disabled endpoints return 404 NOT_FOUND. Background SavedSearchScheduler ticks each saved search's interval, evaluates, diffs against last-match URLs, and broadcasts the SearchMatched WebSocket event when matches change.
Validation: name required (1–80 chars, unique per host case-insensitive → 409 on duplicate); query parsed via the DSL (422 INVALID_DSL on parse error with parse position).
Mutation routes (POST / PUT / DELETE / run) return 403 FORBIDDEN when ApiReadOnlyMode is set — kiosk deployments can’t have saved searches mutated by remote clients.
Server-side recent-search log shared by Cmd+K and any other search bar. Falls through to client localStorage if the endpoint is missing or returns 404.
GET ?limit=N — newest-first. limit defaults 20, caps at 200. POST body {query, source} — empty body clears (returns {cleared:true}). DELETE — wipe all entries.
POST and DELETE return 403 FORBIDDEN when ApiReadOnlyMode is set.
/library/albums/detailed returns albums with firstTrackUrl, year, and dateAdded for artwork lookups and sorting. Eliminates per-album /library/files?limit=1 round-trips. Params: ?offset=&limit=
/library/album-artists returns distinct album artists. Params: ?offset=&limit=
/library/albums/unheard returns albums where all tracks have playCount=0. /library/albums/with-pdf returns albums containing PDF booklets. Both support ?offset=&limit=
/library/inbox, /library/audiobooks, and /library/videos query MusicBee library categories (Source Types 4, 32, 64). Browse page shows these tabs only when non-empty (progressive reveal).
Evaluate a MusicBee expression: ?expression=<Artist> - <Title>&fileUrl=C:\Music\track.mp3. If fileUrl is omitted, evaluates against the currently playing track. Returns {expression, fileUrl, result}. Supports MusicBee template syntax (<Artist>, $If(), virtual tags). Requires MusicBee 3.4.1+ (API rev 55+).
MusicBee’s built-in placeholder image for tracks with no artwork. Returns binary image data with appropriate content type. Cache-Control: 24h. Requires MusicBee 3.5+ (API rev 57+).
Albums for an artist with year, track count, and firstTrackUrl for artwork. Params: ?albumArtist= or ?artist= (one required), ?sort=alpha|year|year-asc. ?artist= queries ArtistPeople (broader match), ?albumArtist= queries AlbumArtist (exact album credit).
// GET /library/albums/by-artist?albumArtist=Pink%20Floyd&sort=year
{
"success": true,
"data": {
"albumArtist": "Pink Floyd",
"total": 3,
"sort": "year",
"albums": [
{
"name": "The Dark Side of the Moon",
"year": "1973",
"count": 10,
"firstTrackUrl": "file:///C:/Music/Artist/Album/track.mp3"
}
]
}
}
Note: URL-encode the file path in the URL (e.g., /library/file/file%3A%2F%2F%2FC%3A%2FMusic%2Ftrack.mp3)
// GET /library/file/{url} - Extended track metadata (includes playCount, lastPlayed, etc.)
{
"success": true,
"data": {
"url": "file:///C:/Music/Artist/Album/track.mp3",
"title": "Love Don't Live Here",
"artist": "Breaking Rust",
"album": "Greatest Hits",
"duration": 234000,
"albumArtist": "Breaking Rust",
"genre": "Rock",
"year": "2024",
"trackNo": "3",
"discNo": "1",
"rating": "4.5",
"composer": "J. Smith",
"bitrate": "320",
"format": "MPEG Audio",
"sampleRate": "44100",
"playCount": 42,
"dateAdded": "2024-01-15",
"lastPlayed": "2024-01-20"
}
}
// PUT /library/file/{url} - Update metadata (fields: title, artist, album, albumArtist, genre, year, trackNo, discNo, composer, comment, rating)
{"rating": "5", "comment": "Great track!"}
// Response: {"success": true, "data": {"result": true, "updated": ["rating", "comment"]}}
// POST /library/artwork/batch - Batch artwork fetch (max 50 URLs per request)
// Request: {"urls": ["D:\\Music\\track1.mp3", "D:\\Music\\track2.flac"]}
// Response: {"success": true, "data": {"D:\\Music\\track1.mp3": "data:image/jpeg;base64,/9j/...", "D:\\Music\\track2.flac": null}}
// POST /library/commit - Commit pending tag changes to file
// Use after batching Library_SetFileTag RPC calls
// Request: {"file": "D:\\Music\\track.mp3"}
// Response: {"success": true, "data": {"result": true}}
/artwork-count returns 0 if the file has no embedded artwork, 1 otherwise. Internally probes up to 20 embedded-artwork locations (MusicBee returns the same cover at multiple locations — EmbedInFile, LinkToSource, FolderThumb —) and picks the largest byte-size as the canonical image; subsequent /artwork?index=0 requests serve that best variant. Cached per fileUrl. /pdf serves the PDF booklet from the track's album folder. /has-pdf checks existence without downloading.
List all images in the track's album folder and one level of subfolders. Returns folder path, image paths, and count. Excludes: canonical primary-cover filenames (folder.jpg, cover.jpg, front.jpg, album.jpg + .jpeg/.png siblings — these are duplicates of the primary artwork served by /artwork); all Windows Media Player cache files (AlbumArtSmall*, AlbumArt_*); thumbnail artifacts (<5KB); Thumbs.db; desktop.ini.
Security: track must be in the MusicBee library.
Serve a single fan art image by absolute path (binary). Returns image with appropriate content-type. Cache-Control: max-age=3600.
Security: image must reside in a directory that contains at least one MusicBee library file.
/library/artist/{name}/pictures — returns a pictures array with src URLs (serveable via /library/artist/{name}/pictures/{index}) in addition to raw file paths. Query: ?localOnly=false.
/library/artist/{name}/pictures/{index} — serve artist picture by 0-based index as binary image. Content-Type auto-detected (falls back to image/jpeg for MusicBee cache files); Cache-Control: max-age=3600. Query: ?localOnly=true (default).
Recently played tracks sorted by last played descending. Params: limit (1–200, default 50), offset (default 0), days (1–365, default 30). Returns tracks with lastPlayed, playCount, skipCount fields.
Video files from MusicBee’s Video library node (Source Type 64). Returns {total, videos: [{url, title, artist, album, kind, duration}]}. Title falls back to filename when tag is empty. Stream video files via /stream/{url}.
/library/files/raw — raw MusicBee file data without CUE processing. Optional ?album= filter. Max 50 results.
/library/cuetest — test CUE track resolution for a query. Param: ?query=
List radio stations from MusicBee’s Radio node.
// Response:
{
"success": true,
"data": {
"total": 5,
"stations": [
{"url": "http://stream.example.com/radio", "name": "Jazz FM"},
// ...
]
}
}
// GET /playlists - List all playlists
{
"success": true,
"data": {
"total": 5,
"playlists": [
{"url": "playlist://Favorites", "name": "Favorites", "trackCount": 120},
{"url": "playlist://Workout", "name": "Workout", "trackCount": 45},
// ... more playlists
]
}
}
// POST /playlists - Create new playlist
{"name": "New Playlist", "folder": "", "files": ["file:///C:/Music/song1.mp3"]}
// Response: {"success": true, "data": {"url": "playlist://New Playlist", "name": "New Playlist", "trackCount": 1}}
// PUT /playlists/{url} - Replace playlist contents
{"files": ["file:///C:/Music/song1.mp3", "file:///C:/Music/song2.mp3"]}
// GET /playlists/{url}/files - Get playlist tracks (supports pagination: ?offset=0&limit=50)
{
"success": true,
"data": {
"total": 120,
"offset": 0,
"limit": 50,
"tracks": [
{"index": 0, "url": "file:///...", "title": "Song", "artist": "Artist", "album": "Album", "duration": 234000},
// ... more tracks
]
}
}
// POST /playlists/{url}/files - Add tracks to playlist
{"urls": ["C:\\Music\\song.mp3"]}
List podcast subscriptions. Query: ?query=
/podcasts/{id}/episodes/{index} — get a specific episode by numeric index.
Flat episode listing. Query: ?id= (feed URL). Use this when the subscription ID is a URL rather than a simple ID.
Read-only passthroughs to MusicBee’s own settings API. For MBXHub’s configurable hub settings, see /system/settings above.
Overview bundle of the most common settings in one response.
// Response:
{
"success": true,
"data": {
"storagePath": "C:\\Users\\...\\MusicBee",
"skin": "Default",
"windowBordersSkinned": false,
"lastFmUserId": "scott365",
"webProxy": null
}
}
GET /settings/storage-path — Persistent storage pathGET /settings/skin — Current skin nameGET /settings/skin-element-color — Skin color. Query: ?element=SkinSubPanel&state=ElementStateDefault&component=ComponentBackgroundGET /settings/window-borders-skinned — Whether window borders are skinnedGET /settings/lastfm-user — Last.fm user IDGET /settings/web-proxy — Web proxy configurationGET /settings/field-name?field={MetaDataType} — Display name for a metadata field (e.g. TrackTitle, Custom1)GET /settings/data-type?field={MetaDataType} — Data type for a metadata fieldGET /settings/value?id={SettingId} — Raw MusicBee setting by ID (e.g. CompactPlayerFlickrEnabled). 400 with the full valid-ID list if unknown.GET /settings/convert-command?codec=Mp3&quality=HighQuality — File-conversion command line for a codec/quality pairGracefully close MusicBee. Requires allowRemoteExit=true in settings.
Optional body to schedule a restart via Windows Task Scheduler before closing:
// Request (optional):
{
"restart": true, // schedule restart before closing (default: false)
"delay": 22 // seconds to wait before restarting (default: 22, range: 1-300)
}
// Response:
{
"success": true,
"data": {
"message": "MusicBee restarting in 10s",
"restart": true,
"delay": 22
}
}
Restart MusicBee. Convenience alias for /app/exit with restart=true. Requires allowRemoteExit=true.
// Request (optional):
{
"delay": 22 // seconds to wait before restarting (default: 22, range: 1-300)
}
Turn MusicBee into a social jukebox. Guests scan a QR code, enter a PIN, and request songs from their phone. The DJ controls playback while a TV display shows artwork, lyrics, and a live feed of requests and joins.
Roles: Guest (browse/request), DJ (full control), Display (TV mode)
Kill switch: set ApiDisablePartyMode = true in mbxhub.json
(Features section) to disable party mode entirely. All /partymode/* routes return 404 NOT_FOUND
and /pages/partymode/* static pages return 404. Live state is exposed at
/system/features as partymode: false.
Get current party state (active, request count).
Start a party session. Host-only: caller must be on the host machine
(loopback / request.IsLocal). Tablets, phones, and other LAN devices become DJ via
/partymode/verify-dj after the host has started the party.
// Request body:
{"guestPin": "1234", "djPin": "5678"}
// djPin is optional - defaults to guestPin
// Errors:
// 403 PARTY_START_FORBIDDEN — caller is not on the host machine
// 409 PARTY_ALREADY_ACTIVE — a party is already running; stop it first
End the current party session.
Validate PIN and get role. If nickname provided, announces join in feed.
// Response:
{"success": true, "data": {"valid": true, "role": "guest"}}
// role: "guest" or "dj"
Verify if a PIN grants DJ access. Used by DJ page login.
// Request body:
{"pin": "5678"}
// Response:
{"success": true, "data": {"valid": true}}
// valid is true only if PIN matches DJ PIN
Submit a vote (thumbs up/down) with guest attribution. Records in feed and forwards to influences if AutoQ available.
// Request body:
{"type": "++", "target": "Artist", "value": "Daft Punk", "nickname": "Haro"}
// type: "++"=more, "--"=less; target: "Artist" or "Genre"
Submit a song request (adds to queue).
// Request body:
{"url": "C:\\Music\\song.mp3", "nickname": "Haro"}
// Response includes requestId, title, artist
Get party feed (joins, requests, votes, reactions - newest first).
// Response:
{"success": true, "data": {
"items": [
{"type": "join", "nickname": "Haro", "timestamp": "..."},
{"type": "request", "nickname": "Haro", "title": "Song", "artist": "Artist", "timestamp": "..."},
{"type": "voteup", "nickname": "Haro", "artist": "Daft Punk", "timestamp": "..."},
{"type": "votedown", "nickname": "Haro", "artist": "Nickelback", "timestamp": "..."},
{"type": "reaction", "nickname": "Haro", "emoji": "🔥", "title": "Song", "artist": "Artist", "timestamp": "..."}
]
}}
Get recent song requests only (for DJ page).
Generate QR code PNG image. Auto-includes PIN if party is active.
Returns the caller's role based on IP: host (loopback), dj (verified DJ PIN), or guest.
Party Mode settings are configured in MusicBee via Settings → Network → Party Mode...
| Setting | Default | Description |
|---|---|---|
protectMetadata | true | Block metadata writes (love, rate, tag edits) for ALL users including DJ |
rateLimitEnabled | true | Enable per-IP rate limiting on party endpoints |
rateLimitRequestsPerMinute | 5 | Max song requests per minute per IP (when rate limiting enabled) |
rateLimitVotesPerMinute | 5 | Max votes per minute per IP (when rate limiting enabled) |
rateLimitPinAttemptsPerMinute | 5 | Max PIN validation attempts per minute per IP |
trustForwardedFor | false | Trust X-Forwarded-For header for client IP. Enable only behind a reverse proxy. |
Protect Metadata: When enabled, blocks POST /dashboard/love, POST /dashboard/rate/{N},
and PUT /library/file/* tag edits for everyone during the party. Use this to prevent guests from permanently modifying
your library ratings. The setting also blocks the DJ—toggle it off temporarily if you need to make edits.
Built-in pages at /pages/partymode/:
index.html - PIN + nickname entry (guest join page)guest.html - Browse 7 tabs (Albums, Artists, Genres, Playlists, Podcasts, Radio, Moods), fuzzy search, request songs, vote on vibesdj.html - Start/stop party, set PINs, manage queue, see requests & vibesdisplay.html - TV mode: artwork, lyrics, request feed, QR code, floating reactionsleaderboard.html - Party stats: guest activity, top tracks, reaction countsIntelligent queue system. AutoQ combines TrueShuffle rules, mood analysis, reactions, and influences to automatically queue tracks that match the room's energy.
TrueShuffle manages the shuffle cycle — play rules, cycle tracking, and queue constraints.
Returns 503 SERVICE_UNAVAILABLE if TrueShuffle/AutoQ not enabled.
Get shuffle cycle status
// Response:
{
"success": true,
"data": {
"enabled": true,
"totalTracks": 1000,
"playedCount": 250,
"remainingCount": 750,
"percentComplete": 25.0,
"cycleStarted": "2024-01-01T00:00:00Z"
}
}
Reset the shuffle cycle. All tracks become unplayed.
Tracks played/remaining in shuffle cycle. Query: ?offset=&limit=
Permanently excluded tracks. Banned tracks are never queued by AutoQ.
Returns 503 SERVICE_UNAVAILABLE if TrueShuffle/AutoQ not enabled.
Get list of banned tracks. Query: ?offset=&limit=
// Response:
{
"success": true,
"data": {
"total": 5,
"offset": 0,
"limit": 50,
"tracks": [
{
"url": "file:///C:/Music/corrupted.mp3",
"reason": "Audio corruption detected at 2:30",
"bannedAt": "2024-01-01T12:00:00Z"
}
]
}
}
// POST /banlist - Ban a track
{"url": "file:///C:/Music/track.mp3", "reason": "Corrupted audio at 2:30"}
// DELETE /banlist/{url} - Unban a track (URL-encode the file path)
Influence rules shape AutoQ scoring — Pandora-style thumbs up/down on artists and genres.
Unlike bans (permanent, track-specific), influences are resettable metadata preferences.
Returns 503 if TrueShuffle/AutoQ not enabled.
Negative (--): Hard exclude matching tracks. Positive (++): Preference boost (future).
Get list of all influences. Query: ?offset=&limit=
// Response:
{
"success": true,
"data": {
"total": 2,
"offset": 0,
"limit": 50,
"influences": [
{
"type": "--",
"target": "Genre",
"value": "Audiobook",
"timestamp": "2024-01-15T10:30:00Z"
},
{
"type": "++",
"target": "Artist",
"value": "The Beatles",
"timestamp": "2024-01-15T10:35:00Z"
}
]
}
}
Get current track's genre/artist and any matching influences (for UI state).
// Response:
{
"success": true,
"data": {
"genre": "Rock",
"artist": "Pink Floyd",
"genreInfluence": null,
"artistInfluence": "++"
}
}
// POST /influences - Add an influence
{"type": "--", "target": "Genre", "value": "Audiobook"}
// type: "++" (more) or "--" (less/exclude)
// target: "Genre" or "Artist"
// DELETE /influences/Genre/Audiobook - Remove specific influence
// POST /influences/clear - Clear all influences (reset preferences)
Get AutoQ status and configuration.
// Response:
{
"success": true,
"data": {
"enabled": true,
"mode": "autopilot",
"soloMode": false,
"vibeListCount": 10,
"vibeListPreview": [
{ "title": "Uptown Funk", "artist": "Mark Ronson", "score": 12.5 }
]
}
}
Start AutoQ. Begins monitoring queue and adding tracks when needed.
// Optional request body:
{ "mode": "autopilot" }
// Modes: "autopilot" (default), "djassist" (aliases: "dj", "assist", "auto")
Stop AutoQ. Queue continues playing but no new tracks are added automatically.
Reset AutoQ session state (clears reactions, taste vector, ban list). DJ-only in party mode.
Force refresh vibe list. Returns updated track count.
// Response:
{
"success": true,
"data": { "message": "Vibe list refreshed", "count": 100 }
}
Refresh the vibe list and immediately enqueue picked tracks in a single call (vibe-list/refresh + queue action combined). Used by the dashboard’s refresh button.
Pick the next track from the vibe list without queueing it.
// Response:
{
"success": true,
"data": { "url": "C:\\Music\\track.mp3", "title": "Uptown Funk", "artist": "Mark Ronson" }
}
Unban a track, allowing it back into the vibe list.
// Request:
{ "url": "C:\\Music\\track.mp3" }
// Response:
{
"success": true,
"data": { "result": true, "url": "C:\\Music\\track.mp3" }
}
Check if the currently playing track is banned.
// Response:
{
"success": true,
"data": { "url": "C:\\Music\\track.mp3", "isBanned": false }
}
Get available mood channels with arousal/valence coordinates. Channels are customizable via autoQ.moodChannels in mbxhub.json.
// Response:
{
"success": true,
"data": {
"currentMood": "Energetic",
"channels": [
{ "name": "Energetic", "emoji": "🔥", "arousal": 0.90, "valence": 0.80 },
{ "name": "Chill", "emoji": "😌", "arousal": 0.35, "valence": 0.65 }
]
}
}
Browse tracks matching a mood channel. Returns scored tracks sorted by mood similarity.
channel | Mood channel name (e.g. "Energetic", "Chill") |
limit | Max results (default: 200, max: 500) |
// GET /autoq/moods/browse?channel=Energetic&limit=50
// Response:
{
"success": true,
"data": {
"channel": "Energetic",
"emoji": "🔥",
"total": 50,
"tracks": [
{ "url": "...", "title": "...", "artist": "...", "album": "...", "trackNo": "3", "moodMatch": 0.92 }
]
}
}
Hash-keyed counterpart to /autoq/track-mood for cross-system lookup — resolve mood data without knowing the local file URL. {hash} is hex (audioStreamSha256 = 64 chars, fileMd5 = 32 chars). Returns the same mood payload shape as /autoq/track-mood.
Raw mood data for the current track (or any track via ?url=). Returns file, album, Essentia features, percentile ranks, computed valence/arousal, and best mood channel match.
// Response (Essentia-analyzed track):
{
"success": true,
"data": {
"file": "C:\\Music\\Artist\\Album\\Track.mp3",
"album": "Album",
"valence": 0.5003,
"arousal": 0.5794,
"source": "essentia",
"hasEssentiaData": true,
"moodChannel": "Upbeat",
"moodEmoji": "😊",
"raw": {
"bpm": 119.95,
"mode": "major",
"loudness": -10.23,
"spectralCentroid": 1284.56,
"spectralFlux": 0.000342,
"spectralRms": 0.001245,
"spectralFlatness": 0.000089,
"danceability": 1.45,
"onsetRate": 2.87,
"zeroCrossingRate": 0.058,
"dissonance": 0.4123,
"pitchSalience": 0.3856,
"chordsChangesRate": 0.0312,
"mfcc": 142.56
},
"percentiles": {
"bpm": 0.52, "loudness": 0.61, "centroid": 0.48,
"flux": 0.35, "dance": 0.67, "onset": 0.55,
"zcr": 0.42, "rms": 0.58, "dissonance": 0.44,
"salience": 0.39, "chords": 0.51, "mfcc": 0.63
},
"confidence": 0.82,
"confidenceLabel": "high",
"genreProfile": "electronic"
}
}
// Response (fallback — no Essentia data):
{
"success": true,
"data": {
"file": "C:\\Music\\Artist\\Album\\Track.mp3",
"album": "Album",
"valence": 0.65,
"arousal": 0.70,
"source": "fallback",
"hasEssentiaData": false,
"moodChannel": "Energetic",
"moodEmoji": "🔥",
"confidence": 0.45,
"confidenceLabel": "medium",
"genreProfile": null
}
}
Set target mood for Mood mode. Pass empty channel to disable mood mode.
// Request:
{ "channel": "Energetic" }
// Response:
{
"success": true,
"data": {
"message": "Mood set to Energetic",
"mode": "mood",
"mood": { "name": "Energetic", "emoji": "🔥", "arousal": 0.90, "valence": 0.80 }
}
}
Reload the mood channel cache from disk (re-reads Essentia data).
Export mood file (seed another node). Streams the raw mbxmoods.json with the same byte content as on disk so you can save it and drop it into another MBXHub's plugin data folder. Disabled by default. Returns 404 unless “Disable mood export” is unchecked in Settings → Configuration → Services. Also returns 404 when no moods file is resolved on this node, and 503 when the file is busy. Supports ETag + If-None-Match (weak validator) and Last-Modified for cheap re-pulls; response carries Content-Disposition: attachment; filename="mbxmoods.json".
Local mood-cache totals. Returns current cache totals (Essentia-backed and fallback). warmupInProgress is always false and lastWarmup is always null — the warm-up shape is preserved for client compatibility but no longer fires.
// Response:
{
"success": true,
"data": {
"total": 12480,
"essentia": 9812,
"fallback": 2668,
"warmupInProgress": false,
"lastWarmup": null
}
}
Bulk write mood tags to a MusicBee custom field. Only processes Essentia-analyzed tracks. Requires autoQ.moodTagField to be configured (e.g. "Custom1").
// Response:
{
"success": true,
"data": { "updated": 342 }
}
Get current vibe list (candidate tracks with scores). Query: ?count=50
// Response:
{
"success": true,
"data": {
"count": 50,
"tracks": [
{
"url": "C:\\Music\\track.mp3",
"title": "Uptown Funk",
"artist": "Mark Ronson",
"genre": "Funk",
"score": 12.5
}
]
}
}
Discover tracks adjacent to the current taste profile, grouped by category. Query: ?limit=100&groupBy=auto|genre|artist|mood
// Response:
{
"success": true,
"data": {
"profile": {
"topGenres": [{ "name": "Rock", "weight": 1.0 }],
"topArtists": [{ "name": "Foo Fighters", "weight": 0.85 }],
"bpmRange": [90, 160],
"mood": "Energetic",
"influenceCount": 3,
"reactionCount": 12
},
"groupBy": "genre",
"groups": [{
"label": "Rock",
"reason": "genre",
"count": 15,
"tracks": [{
"url": "C:\\Music\\track.mp3",
"title": "Everlong",
"artist": "Foo Fighters",
"genre": "Rock",
"score": 8.2,
"reasons": ["genre match", "artist match"]
}]
}],
"total": 45
}
}
Find tracks similar to a seed track by metadata and feature distance. Query: ?url={trackUrl}&limit=20
// Response:
{
"success": true,
"data": {
"seed": {
"url": "C:\\Music\\seed.mp3",
"title": "Everlong",
"artist": "Foo Fighters",
"genre": "Rock"
},
"tracks": [{
"url": "C:\\Music\\similar.mp3",
"title": "Learn to Fly",
"artist": "Foo Fighters",
"genre": "Rock",
"similarity": 0.6,
"reasons": ["same genre", "same artist", "similar BPM"]
}]
}
}
Tiered reactions for the now playing track. Each emoji has a different score weight.
| Emoji | Name | Score | Description |
|---|---|---|---|
| 🔥 | fire | +3 | This track is fire! (triggers queue refresh) |
| ❤️ | heart | +2 | Love this song |
| 👍 | like | +1 | Good choice |
| 👎 | dislike | -1 | Not feeling it |
| 🚫 | ban | -100 | Skip and exclude from queue |
Submit a reaction for a track. Reactions appear as floating emojis on the Display page.
// Request body (always reacts to currently playing track):
{
"emoji": "🔥",
"nickname": "Mike"
}
// emoji: "🔥", "❤️", "👍", "👎", "🚫" or "fire", "heart", "like", "dislike", "ban"
// Response:
{
"success": true,
"data": {
"recorded": true,
"emoji": "🔥",
"trackUrl": "C:\\Music\\track.mp3",
"trackTitle": "Uptown Funk",
"trackArtist": "Mark Ronson",
"nickname": "Mike"
}
}
Get reaction history. Query: ?trackUrl= for specific track, ?limit=100
// Response:
{
"success": true,
"data": {
"count": 25,
"trackUrl": null,
"reactions": [
{
"emoji": "🔥",
"type": "fire",
"trackUrl": "C:\\Music\\track.mp3",
"trackTitle": "Uptown Funk",
"trackArtist": "Mark Ronson",
"nickname": "Haro",
"timestamp": "2026-02-01T20:45:00Z"
}
]
}
}
Get party leaderboard data: guest activity, top tracks, reaction breakdown.
// Response:
{
"success": true,
"data": {
"totalReactions": 156,
"topTracks": [
{ "url": "...", "title": "Uptown Funk", "artist": "Mark Ronson", "score": 15 }
],
"guests": [
{ "nickname": "Haro", "totalReactions": 42, "fire": 10, "heart": 15, "like": 12, "dislike": 3, "ban": 2 }
]
}
}
Get all tunable AutoQ parameters: scoring weights, reaction scores, influence scores, estimation weights, and normalization ranges. Use with Tuning Console.
Partial update of AutoQ parameters. Only provided fields are changed. Changes take effect on the next scoring pass — no restart needed. Nested objects are merged, not replaced.
// Example: change just the BPM arousal weight
PUT /autoq/settings
{"estimation": {"arousalWeightBpm": 0.30}}
AutoQ settings in mbxhub.json under the autoQ section:
| Setting | Default | Description |
|---|---|---|
enabled | false | Enable AutoQ feature |
pickMode | "weighted" | Track selection mode: off (disabled, shuffle fill only), favorites (always pick highest-scored), weighted (score-proportional random from top candidates), random (uniform random, diversity caps still apply) |
queueThreshold | 3 | Add tracks when queue drops below this |
batchSize | 5 | Tracks to add per batch |
vibeListSize | 100 | Size of candidate track pool |
moodMatchWeight | 0.4 | Weight for mood matching in scoring (0-1) |
recencyDecayLambda | 0.1 | Decay rate for recency boost on reacted tracks |
recencyPenaltyDecay | 0.1 | Decay rate for recently-played penalty |
minReplayMinutes | 30 | Minimum minutes before a track can be replayed |
diversityWindowSize | 10 | Recent tracks considered for diversity calculations |
minSessionEntropy | 0.5 | Entropy threshold before boosting diversity (0-5) |
vibeListRefreshMinutes | 30 | Minutes between automatic vibe list refreshes |
moodChannels | null | Custom mood channels (array). Uses defaults if null. |
genreQuota | 3 | Max consecutive same-genre tracks (0=disabled) |
artistQuota | 1 | Max tracks from same artist in batch (0=disabled) |
moodTagField | "Custom1" | MusicBee custom tag field for mood labels (null=disabled) |
moodTagFieldName | "AutoQ Mood" | Expected display name for the tag field (must match MusicBee config) |
Scoring weights under autoQ.scoringWeights: featureSimilarity (0.5), trackSentiment (0.25), artistSentiment (0.15), recencyPenalty (0.3), diversityPenalty (0.6), explorationBonus (0.1), influenceWeight (0.3).
Reaction scores under autoQ.reactionScores: fire (3), heart (2), like (1), dislike (-1), ban (-100).
Estimation, valence/arousal weights, normalization ranges, confidence thresholds, and genre profiles — full field list with defaults and descriptions is in the config reference. Live values: GET /autoq/settings. Tune interactively at /pages/autoq.html.
Mood Quadrants — Arousal (energy) is the vertical axis, valence (positivity) is the horizontal. Each mood channel targets a point in this space.
| Quadrant | Profile | Typical Genres | Acoustic Traits |
|---|---|---|---|
| High arousal + high valence | Energetic, upbeat | EDM, pop, funk | Fast tempo, bright timbre, strong beats |
| High arousal + low valence | Tense, aggressive | Metal, hard rock, industrial | Distortion, high energy, dissonance |
| Low arousal + low valence | Sad, subdued | Ambient drone, slow blues, lo-fi | Slow tempo, dark timbre, soft dynamics |
| Low arousal + high valence | Calm, pleasant | Chillhop, acoustic folk, soft jazz | Warm timbre, consonance, smooth textures |
Simulate keyboard and mouse input to wake or control the host PC. Useful for remote wake scenarios. Full ARiA Documentation →
ariaEnabled: false). Returns 403 ARIA_DISABLED when disabled.
Check if ARiA input simulation is enabled
Quick wake: move mouse + send Shift key to wake sleeping/locked PC
Send keyboard input. Body: {"keys": "^a"} (Ctrl+A). Optional: {"keys": "%{F4}", "window": "Notepad"} to focus window first. Prefix keys with ! to send to the current foreground window without refocusing MusicBee.
SendKeys format: ^=Ctrl, %=Alt, +=Shift. Special keys: {ENTER}, {TAB}, {ESC}, {F1}-{F12}, {UP}, {DOWN}, etc.
DuckyScript format: CTRL ALT V, SHIFT F1, ALT TAB. Modifiers: CTRL, ALT, SHIFT. Special: WIN/GUI (opens Start Menu, standalone only - not a modifier).
Focus a window by title. Body: {"window": "Notepad"} (partial match, case-insensitive)
Move: {"x":100,"y":100} (absolute) or {"dx":10,"dy":0} (relative)
Click: {"button":"left"} or {"x":500,"y":300,"button":"right"}
List available presets
Execute a preset by name (e.g., /aria/preset/RIA3)
List allowed programs for the run() command. Returns names only (paths not exposed).
%APPDATA%\MusicBee\mbxhub.json to add/modify presets:
"ariaPresets": [
{"name": "RIA1", "script": "sndkeys(^%a)", "icon": "1"},
{"name": "DuckyDemo", "script": "sndkeys(CTRL ALT V)", "icon": "D"},
{"name": "Notify", "script": "toast(MBXHub,Hello World!)", "icon": "N"}
]
Script commands:sndkeys(keys) - SendKeys or DuckyScript: sndkeys(^a) or sndkeys(CTRL A). Prefix ! to skip refocus: sndkeys(!{ENTER})delay(ms) - Wait milliseconds (max 30000)click(x,y[,button]) - Mouse click: click(100,200) or click(100,200,right)volume(action) - Volume control: up, down, mute, or steps like +5/-3run(name[,extraArgs]) - Launch a pre-configured program: run(amp-on) or run(visualizer,--fullscreen). Programs must be defined in ariaAllowedPrograms in mbxhub.jsonwebhook(url[,method,body]) - HTTP request: webhook(http://example.com) or webhook(!http://...,POST,{}) (prefix ! for fire-and-forget)toast(msg) or toast(title,msg) - Show notificationrestart(target) - Restart: mb (MusicBee), system, or shutdownsndkeys(^a);delay(100);sndkeys(^c)
run() command only launches programs defined in ariaAllowedPrograms. Empty by default.
"ariaAllowedPrograms": [
{"name": "amp-on", "path": "C:\\Tools\\amp-control.exe", "args": "--power on", "hidden": true},
{"name": "visualizer", "path": "C:\\Program Files\\ProjectM\\projectm.exe"}
]
Each program has: name (used in scripts), path (executable), args (default arguments, optional), hidden (no console window, optional).
Extra arguments can be appended: run(visualizer,--fullscreen).
Publish MusicBee as a Windows RemoteApp, allowing the full desktop UI to be accessed from other machines via RDP. Requires Windows Pro, Enterprise, or Server — Home edition is not supported.
Check RemoteApp status. Always accessible.
Response:
{
"configured": false,
"supported": true,
"rdpEnabled": true,
"edition": "Professional",
"enabled": false,
"apiDisabled": false,
"message": "RemoteApp not configured. Run 'MBXHub.exe remoteapp setup' to configure."
}
Download a .rdp file for connecting to MusicBee as a RemoteApp. Blocked when remoteAppApiDisabled is true.
Query parameters:
hostname - Override the hostname in the .rdp file (defaults to request Host header)audioqualitymode=0, redirectprinters=1)Response: application/x-rdp file download (MusicBee.rdp)
403 when remoteAppApiDisabled is true.
Visibility: Dashboard footer link requires remoteAppEnabled and remoteAppApiDisabled: false. Footer links are configurable via dashboardFooterLinks in settings.
App program: On Windows Client (Pro/Enterprise), the .rdp file uses the full executable path from the registry.
On Windows Server, it uses the ||AppName alias for RDS published app lookup.
MBXHub.exe remoteapp setup --path "C:\MusicBee\MusicBee.exe" (requires elevation)MBXHub.exe remoteapp setup --detect (auto-detect MusicBee)MBXHub.exe remoteapp remove (remove configuration)MBXHub.exe remoteapp status (check current state)
Install-WindowsFeature RDS-Connection-Broker, RDS-Web-Access, RDS-RD-Server -IncludeManagementTools
Generic HTTP proxy for controlling LAN devices (speakers, receivers, home automation) from the browser.
Browsers enforce CORS on all cross-origin requests, and LAN devices don’t serve CORS headers —
direct fetch() from the dashboard to a device IP will silently fail. The proxy solves this by
forwarding requests server-side.
Forward an HTTP request to a LAN device. Only private IPs are allowed (RFC 1918).
Request body:
{
"method": "GET",
"url": "http://192.168.10.100/ipcontrol/v1/systems/current/sources",
"body": {}
}
method — HTTP method to use: GET, POST, or PUTurl — Full URL of the target device endpoint. Must be a private IP address.body — Optional JSON body to forward with POST/PUT requests.Response: The target device’s response is passed through verbatim (status code and body).
Errors:
POST /api/proxy → MBXHub → LAN deviceThe proxy can be disabled in Settings → API → Feature Toggles (apiDisableProxy).
When disabled, POST /api/proxy returns 404 and charm webapps that depend on it will not be able to reach LAN devices.
Serves audio files from the MusicBee library over HTTP with Range support.
Enables “Listen Here” mode in the player — the browser plays audio locally
via <audio> while MusicBee acts as the library manager.
Stream an audio or video file. The path must be URL-encoded and must be a file in the MusicBee library.
Path parameter: URL-encoded absolute file path (e.g. /stream/C%3A%5CMusic%5Csong.mp3).
Range support: Send Range: bytes=N-M header for partial content (required for seeking).
The server responds with 206 Partial Content and Content-Range header.
Supported formats: mp3, flac, m4a, mp4, ogg, oga, wav, opus, aac, wma, aiff, aif. Actual browser playback depends on codec support — FLAC works in Firefox/Chrome/Edge, WMA is not supported by any browser.
Security:
..) is blockedErrors:
Streaming can be disabled in Settings → API → Feature Toggles (disableStreaming).
When disabled, GET /stream/* returns 404 and the Listen Here button is hidden in the player.
Serves images and videos from configured root directories. Categories are subfolders under each root. No MusicBee library integration — purely a file-serving feature for slideshows and ambient display.
/media/* endpoint listed below is DJ-only. Guest and Anonymous callers get 403 PARTY_LOCKED with the message “Media browsing is locked during PartyMode (DJ-only).” The host’s Projector charm runs as DJ on localhost so the big-screen flow keeps working. Outside party mode, every caller is allowed (media is browse-only and not covered by the existing ApiReadOnly* gates).
Configure in mbxhub.json under media:
"media": {
"imageRoot": "C:\\Users\\...\\Pictures\\Wallpapers",
"videoRoot": "C:\\Users\\...\\Videos",
"intervalSeconds": 30,
"shuffle": true
}
Returns timer config: {intervalSeconds, shuffle}
List subfolder names under imageRoot. Files directly in root appear as _root.
List subfolder names under videoRoot.
List filenames in category. Returns {name, files:[], count}. Filenames only, no paths.
List video filenames in category.
Serve the next image (rotation state per category, sequential or shuffled). Returns image binary.
Serve the next video with HTTP Range support for seeking.
Serve a specific image by filename.
Serve a specific video by filename. Supports HTTP Range requests (206 Partial Content) for seeking.
Images: .jpg, .jpeg, .png, .bmp, .webp, .gif
Videos: .mp4, .webm, .mkv, .avi, .mov, .asf, .wmv (.mp4/.webm play inline; others open externally)
Minimum file size: 5 KB (skips thumbnail artifacts).
All paths validated under configured root — no directory traversal. Category names mapped to actual subfolders. Filenames validated against directory contents. Unconfigured roots return empty categories.
/pages/media — Full-bleed media viewer with auto-rotation,
crossfade transitions, video playback, chrome overlay with category/mode selectors,
keyboard (arrows, space, F) and touch/swipe navigation.
Control audio volume across three layers: MusicBee player volume (via /player/volume),
the Windows audio device, and network endpoints (e.g. Devialet Phantom speakers).
The Mixer charm provides a unified fader mixing surface for all three.
List all active Windows audio render devices (Core Audio ground truth)
// Response:
[
{ "name": "Speakers (HD Audio)", "id": "{0.0.0...}", "isDefault": true },
{ "name": "HDMI Output", "id": "{0.0.0...}", "isDefault": false }
]
Get Windows audio device volume and mute state
// Response:
{
"device": "Speakers (HD Audio)",
"volume": 75,
"muted": false
}
Set Windows audio device volume (0–100)
PUT /devices/audio/volume
Content-Type: application/json
{ "volume": 50 }
Set Windows audio device mute state
PUT /devices/audio/mute
Content-Type: application/json
{ "mute": true }
Manage network audio endpoints (speakers, receivers). Endpoints are saved in settings and controlled via REST. Currently supports Devialet Phantom speakers.
List all configured network endpoints
// Response:
[
{
"id": "devialet-1",
"name": "Living Room",
"type": "Devialet",
"ip": "192.168.1.50"
}
]
Add a new network endpoint
POST /devices/endpoints
Content-Type: application/json
{
"ip": "192.168.1.50",
"type": "devialet",
"name": "Living Room"
}
Scan the LAN for network speakers via mDNS/DNS-SD. Bodyless POST
(send Content-Length: 0). Blocks for ~4 seconds while the multicast
discovery completes. Concurrent calls are rate-limited internally and return an
empty list with a warning.
// Response:
[
{
"name": "Living Room Phantom",
"ip": "192.168.1.50",
"port": 80,
"hostName": "phantom-abc123.local",
"serviceType": "_devialet._tcp.local.",
"alreadyConfigured": false,
"existingId": null
}
]
Remove a saved endpoint
Get endpoint volume and mute state
// Response:
{
"volume": 40,
"muted": false
}
Set endpoint volume (0–100)
PUT /devices/endpoint/{id}/volume
Content-Type: application/json
{ "volume": 50 }
Set endpoint mute state
PUT /devices/endpoint/{id}/mute
Content-Type: application/json
{ "mute": true }
List available input sources on the endpoint
// Response:
[
{ "sourceId": "upnp", "name": "UPnP", "type": "upnp", "active": true },
{ "sourceId": "optical", "name": "Optical", "type": "optical", "active": false },
{ "sourceId": "analog", "name": "Analog", "type": "analog", "active": false }
]
Select an input source on the endpoint
PUT /devices/endpoint/{id}/source
Content-Type: application/json
{ "sourceId": "upnp" }
Configure which fader the dashboard volume controls target by default.
Get mixer settings
// Response:
{
"defaultFader": "player"
}
Set mixer settings. defaultFader controls which fader the dashboard volume slider and keyboard shortcuts target.
PUT /mixer/settings
Content-Type: application/json
// Values: "player", "device", "endpoint"
{ "defaultFader": "device" }
Get current volume from the active default fader (player or device). Returns fader type and volume (0-100).
// Response (player fader):
{ "fader": "player", "volume": 75 }
// Response (device fader):
{ "fader": "device", "volume": 80, "muted": false, "device": "Speakers" }
Set volume on the active default fader. Also accepts POST.
PUT /mixer/volume
Content-Type: application/json
{ "volume": 80 }
Charms are configurable action buttons on the dashboard. They can open webapps, fire HTTP requests to LAN devices, or call MBXHub endpoints. The charm bar appears as a dashboard section and can be reordered/hidden like any other panel.
Each charm is a .json file in the charms/ folder inside the MBXHub data directory.
MBXHub seeds built-in charms (mixer.json, browse.json) on first run.
Simple charm (single button):
{
"id": "my-charm",
"icon": "\uD83D\uDD0A",
"label": "My Charm",
"action": "webapp /pages/my-charm.html",
"display": "both",
"msg": "My Charm"
}
Expand charm (grouped buttons, e.g. Mixer):
{
"id": "mixer",
"label": "Mixer",
"expand": [
{ "icon": "\uD83C\uDFA8", "label": "Mixer", "action": "webapp /pages/mixer.html", "display": "both" },
{ "icon": "+", "label": "Volume Up", "action": "iframe-cmd volumeUp", "msg": "Vol +" },
{ "icon": "\u2013", "label": "Volume Down", "action": "iframe-cmd volumeDown", "msg": "Vol \u2013" },
{ "icon": "\uD83D\uDD07", "label": "Mute", "action": "iframe-cmd toggleMute", "msg": "Mute" }
]
}
id — Unique identifier (matches filename without extension).icon — Emoji or character displayed on the button.label — Tooltip and display name.action — What happens on click:
webapp /path — Opens an HTML page (standalone tab or inline iframe).iframe-cmd <command> — Sends a postMessage command to the charm iframe. The iframe webapp listens for { charmCmd: "command" } messages. If the iframe isn’t loaded yet, the dashboard auto-loads it from the sibling webapp action.http://... — Fires an HTTP request. Same-origin requests go direct; cross-origin LAN requests are routed through /api/proxy automatically.display — How webapp charms open:
standalone — Always opens in a new tab.inline — Always opens in an iframe below the charm bar.both — Click opens inline; Shift+click opens standalone.action-menu — Single trigger + popover of expand[] items (e.g. ARiA presets). Popover auto-orients above/below trigger; keyboard nav and Esc-to-close.msg — Status message shown after execution.context — Optional. library-only or stream-only to restrict when the charm appears.expand — Optional array of sub-actions. Each sub-action has icon, label, action, display, and msg. Renders as a grouped button row with a connection status indicator.Charm bar order, visibility, sizing, and per-charm overrides are stored in charmBar in mbxhub.json:
{
"charmBar": {
"order": ["mixer"],
"hidden": [],
"buttonSize": "M",
"sizeOverrides": { "mixer": "XL" },
"breakBefore": ["mixer"],
"displayOverrides": { "mixer": "inline" }
}
}
order — Charm IDs in display order. New charms not in the list are appended.hidden — Charm IDs to hide from the dashboard.buttonSize — Bar-level button size preset: S (36px), M (44px, default), L (52px), XL (64px), XXL (76px). S and M were bumped up in v0.5.2.1 (was 32/40); XL and XXL were added in v0.5.2.3 for comfortable touch on high-density displays (11.6″ FHD @ 150% DPI needs ~76px to hit a 0.6″ physical target).sizeOverrides — Per-charm size overrides (charm ID → S/M/L/XL/XXL). Individual charms can break from the bar-level size; the override wins via CSS specificity.breakBefore — Array of charm IDs that force a new row in the charm bar. Combined with sizeOverrides, one charm (e.g. the mixer) can sit on its own XL row while the rest stay compact.displayOverrides — Per-charm display mode (charm ID → inline or standalone). Overrides the charm manifest's default.The charm bar also appears as “Charm Bar” in the Dashboard Layout panel ordering, so it can be repositioned or collapsed like any other section.
Optional. The plugin (mb_MBXHub.dll) is fully functional on its own — REST, WebSocket, dashboard, AutoQ, charms, and discovery all work without the Shell. The Shell adds Windows-side conveniences: SMTC integration, a system-tray UI for switching SMTC targets and opening dashboards across the fleet, Windows app identity (AUMID, Start Menu shortcut), and a firewall helper that adds / removes the Windows Firewall rule and URL ACL on --install / --uninstall.
The Shell binds the SMTC bridge on restPort + 1 (default 8081). Local use (Shell and plugin on the same box) needs no firewall config — loopback traffic passes through. Remote use (Shell on one machine bridging to MBXHub on another) needs the SMTC port open on the machine hosting the Shell, so the remote dashboard can reach /meta/smtc/*. MBXHub.exe --install opens it for you.
Key commands for the MBXHub Shell (MBXHub.exe):
| Command | Description |
|---|---|
MBXHub.exe status | Show system health and tool discovery status |
MBXHub.exe --no-smtc | Run without SMTC bridge (headless/NAS mode) |
MBXHub.exe --install | Register AUMID, create Start Menu shortcut, report tool discovery, print Send To setup tip |
MBXHub.exe --uninstall | Remove AUMID registration and shortcut |
Direct access to all 137 MusicBee API methods. Use this for operations not exposed via REST endpoints or for scripting/automation.
Invoke any MusicBee API method by name
POST /rpc/Library_GetFileTag
Content-Type: application/json
{
"fileUrl": "C:\\Music\\song.mp3",
"field": "TrackTitle"
}
// Response:
{
"success": true,
"data": {
"method": "Library_GetFileTag",
"result": "Song Title"
}
}
| Method | Parameters | Returns |
|---|---|---|
| Player_PlayPause | - | boolean |
| Player_Stop | - | boolean |
| Player_StopAfterCurrent | - | boolean |
| Player_PlayNextTrack | - | boolean |
| Player_PlayPreviousTrack | - | boolean |
| Player_PlayNextAlbum | - | boolean |
| Player_PlayPreviousAlbum | - | boolean |
| Player_StartAutoDj | - | boolean |
| Player_EndAutoDj | - | boolean |
| Player_GetPosition | - | int (ms) |
| Player_SetPosition | position: int | boolean |
| Player_GetVolume | - | float (0-1) |
| Player_SetVolume | volume: float | boolean |
| Player_GetMute | - | boolean |
| Player_SetMute | mute: boolean | boolean |
| Player_GetShuffle | - | boolean |
| Player_SetShuffle | shuffle: boolean | boolean |
| Player_GetRepeat | - | RepeatMode |
| Player_SetRepeat | mode: RepeatMode | boolean |
| Player_GetPlayState | - | PlayState |
| Player_GetEqualiserEnabled | - | boolean |
| Player_SetEqualiserEnabled | enabled: boolean | boolean |
| Player_GetDspEnabled | - | boolean |
| Player_SetDspEnabled | enabled: boolean | boolean |
| Player_GetCrossfade | - | boolean |
| Player_SetCrossfade | enabled: boolean | boolean |
| Player_GetReplayGainMode | - | ReplayGainMode |
| Player_SetReplayGainMode | mode: ReplayGainMode | boolean |
| Player_GetScrobbleEnabled | - | boolean |
| Player_SetScrobbleEnabled | enabled: boolean | boolean |
| Player_QueueRandomTracks | count: int | int |
| Player_GetOutputDevices | - | {devices, activeDevice} |
| Player_SetOutputDevice | deviceName: string | boolean |
| Method | Parameters | Returns |
|---|---|---|
| NowPlaying_GetFileUrl | - | string |
| NowPlaying_GetDuration | - | int (ms) |
| NowPlaying_GetFileTag | field: MetaDataType | string |
| NowPlaying_GetFileTags | fields: MetaDataType[] | string[] |
| NowPlaying_GetFileProperty | type: FilePropertyType | string |
| NowPlaying_GetArtwork | - | string (base64/path) |
| NowPlaying_GetArtworkUrl | - | string |
| NowPlaying_GetLyrics | - | string |
| NowPlaying_GetDownloadedLyrics | - | string |
| NowPlaying_GetArtistPicture | fadingPercent: int | string |
| NowPlaying_GetArtistPictureThumb | - | string |
| NowPlaying_GetArtistPictureUrls | localOnly: boolean | string[] |
| NowPlaying_IsSoundtrack | - | boolean |
| NowPlaying_GetSpectrumData | - | float[] |
| NowPlaying_GetSoundGraph | - | float[] |
| Method | Parameters | Returns |
|---|---|---|
| NowPlayingList_GetCurrentIndex | - | int |
| NowPlayingList_GetNextIndex | offset: int | int |
| NowPlayingList_IsAnyPriorTracks | - | boolean |
| NowPlayingList_IsAnyFollowingTracks | - | boolean |
| NowPlayingList_GetListFileUrl | index: int | string |
| NowPlayingList_GetFileTag | index: int, field: MetaDataType | string |
| NowPlayingList_GetFileTags | index: int, fields: MetaDataType[] | string[] |
| NowPlayingList_GetFileProperty | index: int, type: FilePropertyType | string |
| NowPlayingList_Clear | - | boolean |
| NowPlayingList_PlayNow | fileUrl: string | boolean |
| NowPlayingList_QueueNext | fileUrl: string | boolean |
| NowPlayingList_QueueLast | fileUrl: string | boolean |
| NowPlayingList_QueueFilesNext | fileUrls: string[] | boolean |
| NowPlayingList_QueueFilesLast | fileUrls: string[] | boolean |
| NowPlayingList_RemoveAt | index: int | boolean |
| NowPlayingList_MoveFiles | fromIndices: int[], toIndex: int | boolean |
| NowPlayingList_PlayLibraryShuffled | - | boolean |
| NowPlayingList_QueryFilesEx | query: string | string[] |
| Method | Parameters | Returns |
|---|---|---|
| Library_GetFileTag | fileUrl: string, field: MetaDataType | string |
| Library_GetFileTags | fileUrl: string, fields: MetaDataType[] | string[] |
| Library_GetFileProperty | fileUrl: string, type: FilePropertyType | string |
| Library_SetFileTag | fileUrl: string, field: MetaDataType, value: string | boolean |
| Library_CommitTagsToFile | fileUrl: string | boolean |
| Library_GetLyrics | fileUrl: string, type: LyricsType | string |
| Library_GetArtwork | fileUrl: string, index: int | string |
| Library_GetArtworkUrl | fileUrl: string, index: int | string |
| Library_GetArtistPicture | artistName: string, fadingPercent: int | string |
| Library_GetArtistPictureThumb | artistName: string | string |
| Library_GetArtistPictureUrls | artistName: string, localOnly: boolean | string[] |
| Library_QueryFilesEx | query: string | string[] |
| Library_QuerySimilarArtists | artistName: string, minimumSimilarity: double | string |
| Library_AddFileToLibrary | fileUrl: string, category: LibraryCategory | string |
| Method | Parameters | Returns |
|---|---|---|
| Playlist_QueryPlaylists | - | boolean |
| Playlist_QueryGetNextPlaylist | - | string |
| Playlist_GetName | playlistUrl: string | string |
| Playlist_GetType | playlistUrl: string | PlaylistFormat |
| Playlist_IsInList | playlistUrl: string, filename: string | boolean |
| Playlist_QueryFilesEx | playlistUrl: string | string[] |
| Playlist_CreatePlaylist | folderName: string, playlistName: string, filenames: string[] | string |
| Playlist_DeletePlaylist | playlistUrl: string | boolean |
| Playlist_SetFiles | playlistUrl: string, filenames: string[] | boolean |
| Playlist_AppendFiles | playlistUrl: string, filenames: string[] | boolean |
| Playlist_RemoveAt | playlistUrl: string, index: int | boolean |
| Playlist_MoveFiles | playlistUrl: string, fromIndices: int[], toIndex: int | boolean |
| Playlist_PlayNow | playlistUrl: string | boolean |
| Method | Parameters | Returns |
|---|---|---|
| Podcasts_QuerySubscriptions | query: string | string[] |
| Podcasts_GetSubscription | id: string | string[] |
| Podcasts_GetSubscriptionArtwork | id: string, index: int | string (base64) |
| Podcasts_GetSubscriptionEpisodes | id: string | string[] |
| Podcasts_GetSubscriptionEpisode | id: string, index: int | string[] |
| Method | Parameters | Returns |
|---|---|---|
| Setting_GetPersistentStoragePath | - | string |
| Setting_GetSkin | - | string |
| Setting_GetSkinElementColour | element: SkinElement, state: ElementState, component: ElementComponent | int |
| Setting_IsWindowBordersSkinned | - | boolean |
| Setting_GetFieldName | field: MetaDataType | string |
| Setting_GetDataType | field: MetaDataType | string |
| Setting_GetLastFmUserId | - | string |
| Setting_GetWebProxy | - | string |
| Setting_GetValue | settingId: SettingId | object |
| Method | Parameters | Returns |
|---|---|---|
| MB_GetWindowHandle | - | long |
| MB_RefreshPanels | - | true |
| MB_GetLocalisation | id: string, defaultText: string | string |
| MB_ShowNowPlayingAssistant | - | boolean |
| MB_InvokeCommand | command: Command, parameter: object | boolean |
| MB_SetWindowSize | width: int, height: int | boolean |
| MB_GetVisualiserInformation | - | {visualiserNames, defaultState, currentState} |
| MB_ShowVisualiser | visualiserName: string, state: WindowState | boolean |
"rpcEnabled": falsePlayer_PlayPause, Library_SetFileTag) require the same permissions as their REST equivalents.
Real-time event streaming. Connect once, receive updates automatically - no polling.
ws://localhost:8080/ws| Step | Description |
|---|---|
| 1. Connect | Open WebSocket to ws://localhost:8080/ws |
| 2. Receive | Immediately starts receiving ALL events (default behavior) |
| 3. Subscribe (optional) | Send subscribe message to filter to specific events only |
| 4. Disconnect | Close the WebSocket connection when done |
Note: New clients receive ALL events by default. Once you send a subscribe message, you only receive those specific events. Use unsubscribe to stop receiving events without disconnecting.
| Event | Description | Frequency |
|---|---|---|
| TrackChanged | New track started playing | On track change |
| PlayStateChanged | Play/pause/stop state changed | On state change |
| VolumeChanged | Volume level or mute state changed | On volume change |
| PositionChanged | Playback position update (milliseconds) | ~1 per second while playing |
| QueueChanged | Now playing list modified (add/remove/clear) | On queue change |
| ShuffleChanged | Shuffle mode toggled on/off | On shuffle change |
| RepeatChanged | Repeat mode changed (none/all/one) | On repeat change |
| MetadataChanged | Rating or love tag changed on current track | On tag/rating change |
| Reaction | User reacted to now playing track (emoji, nickname, track info) | On reaction submit |
| TasteChanged | AutoQ taste vector updated | On taste update |
| ThemeChanged | Theme configuration updated (active mode HSL values) | On theme change via PUT /system/theme or dashboard toggle |
| SearchMatched | v0.5.3.0. Saved-search match set changed. Payload: {id, name, added:[urls], removed:[urls], total, evaluatedAt} | Periodic SavedSearchScheduler tick when the diff is non-empty |
// Subscribe to specific events (filters to only these events)
{"subscribe": ["TrackChanged", "PlayStateChanged"]}
// Unsubscribe from events (stop receiving them)
{"unsubscribe": ["PositionChanged"]}
// Subscribe to all events (equivalent to no subscriptions)
{"subscribe": ["TrackChanged", "PlayStateChanged", "VolumeChanged",
"PositionChanged", "QueueChanged", "ShuffleChanged",
"RepeatChanged", "MetadataChanged", "Reaction",
"TasteChanged", "ThemeChanged", "SearchMatched"]}
// TrackChanged
// cueTrack + cueStartMs are present only when the current track is a
// CUE-backed virtual track (one physical file, multiple logical tracks).
// Clients that stream audio locally use cueStartMs to seek <audio>.currentTime.
{
"event": "TrackChanged",
"timestamp": "2024-01-03T12:00:00.000Z",
"data": {
"fileUrl": "C:\\Music\\song.mp3",
"title": "Track Title",
"artist": "Artist Name",
"album": "Album Name",
"duration": 245000,
"artworkUrl": "/nowplaying/artwork",
"cueTrack": 3, // optional
"cueStartMs": 184000 // optional
}
}
// PlayStateChanged
{
"event": "PlayStateChanged",
"timestamp": "2024-01-03T12:00:00.000Z",
"data": {
"state": "playing" // "playing", "paused", "stopped"
}
}
// VolumeChanged
{
"event": "VolumeChanged",
"timestamp": "2024-01-03T12:00:00.000Z",
"data": {
"volume": 75, // 0 to 100
"muted": false
}
}
// PositionChanged
{
"event": "PositionChanged",
"timestamp": "2024-01-03T12:00:00.000Z",
"data": {
"position": 45000, // Current position in milliseconds
"duration": 245000 // Total duration in milliseconds
}
}
// QueueChanged
{
"event": "QueueChanged",
"timestamp": "2024-01-03T12:00:00.000Z",
"data": {
"action": "add", // "add", "remove", "clear", "move"
"index": 5,
"totalTracks": 42
}
}
// ShuffleChanged
{
"event": "ShuffleChanged",
"timestamp": "2024-01-03T12:00:00.000Z",
"data": {
"enabled": true
}
}
// RepeatChanged
{
"event": "RepeatChanged",
"timestamp": "2024-01-03T12:00:00.000Z",
"data": {
"mode": "all" // "none", "all", "one"
}
}
// MetadataChanged
{
"event": "MetadataChanged",
"timestamp": "2024-01-03T12:00:00.000Z",
"data": {
"fileUrl": "C:\\Music\\song.mp3",
"rating": 3, // -1 (unrated) to 5
"love": "L" // "L" (loved), "B" (banned), or "" (neither)
}
}
// Reaction
{
"event": "Reaction",
"timestamp": "2024-01-03T12:00:00.000Z",
"data": {
"emoji": "fire",
"type": "fire", // fire, heart, like, dislike, ban
"nickname": "Guest",
"trackTitle": "Song Name",
"trackArtist": "Artist"
}
}
// TasteChanged (debounced, fires after reactions/influences/mood changes)
{
"event": "TasteChanged",
"timestamp": "2024-01-03T12:00:05.000Z",
"data": {
"topGenres": [{ "name": "Rock", "weight": 1.0 }],
"topArtists": [{ "name": "Foo Fighters", "weight": 0.85 }],
"bpmRange": [90, 160],
"mood": "Energetic",
"moodConfidence": 0.88,
"influenceCount": 3,
"reactionCount": 12
}
}
// ThemeChanged (fires on PUT /system/theme or dashboard mode toggle)
{
"event": "ThemeChanged",
"timestamp": "2024-01-03T12:00:06.000Z",
"data": {
"activeMode": 1,
"accentHue": 197, "accentSaturation": 80, "accentLightness": 55,
"bgHue": 203, "bgSaturation": 30, "bgLightness": 94,
"surfaceHue": 203, "surfaceSaturation": 30, "surfaceLightness": 96,
"textHue": 203, "textSaturation": 50, "textLightness": 13,
"intensity": 100
}
}
// Connect to WebSocket
const ws = new WebSocket('ws://localhost:8080/ws');
ws.onopen = function() {
console.log('Connected to MBXHub');
// Optional: Subscribe to specific events only
// Without this, you receive ALL events
ws.send(JSON.stringify({
subscribe: ['TrackChanged', 'PlayStateChanged', 'PositionChanged']
}));
};
ws.onmessage = function(event) {
const msg = JSON.parse(event.data);
switch (msg.event) {
case 'TrackChanged':
console.log('Now playing:', msg.data.title, '-', msg.data.artist);
break;
case 'PlayStateChanged':
console.log('State:', msg.data.state);
break;
case 'PositionChanged':
const pct = (msg.data.position / msg.data.duration * 100).toFixed(1);
console.log('Position:', pct + '%');
break;
}
};
ws.onclose = function() {
console.log('Disconnected from MBXHub');
};
ws.onerror = function(err) {
console.error('WebSocket error:', err);
};
// Later: Unsubscribe from position updates (too frequent)
ws.send(JSON.stringify({ unsubscribe: ['PositionChanged'] }));
// Clean disconnect
ws.close();
{
"success": false,
"error": {
"code": "ERROR_CODE",
"message": "Human-readable description"
}
}
| Code | HTTP | Description |
|---|---|---|
| NOT_FOUND | 404 | Endpoint or resource not found |
| INVALID_REQUEST | 400 | Invalid parameters or malformed request |
| ARIA_DISABLED | 403 | ARiA input simulation is disabled |
| FORBIDDEN | 403 | Operation not allowed (e.g., RPC disabled) |
| METHOD_NOT_ALLOWED | 405 | Wrong HTTP method |
| SERVICE_UNAVAILABLE | 503 | Required service not available (e.g., TrueShuffle/AutoQ) |
| INTERNAL_ERROR | 500 | Server error. Response body contains the full exception dump (type, message, stack trace, inner-exception chain) when debugMode is on and logLevel is Debug or lower; otherwise body is the generic fallback string. The dump always lands in mbxhub.log via _log.Warn(ex, ...) regardless of the gate. See Logging section below for how to enable. |
Common enum values used in API parameters and responses.
| Value | Code | Description |
|---|---|---|
| undefined | 0 | Unknown state |
| loading | 1 | Track is loading |
| playing | 3 | Playing |
| paused | 6 | Paused |
| stopped | 7 | Stopped |
| Value | Code | Description |
|---|---|---|
| none | 0 | No repeat |
| all | 1 | Repeat all tracks |
| one | 2 | Repeat current track |
| Value | Code | Description |
|---|---|---|
| off | 0 | Disabled |
| track | 1 | Track-based gain |
| album | 2 | Album-based gain |
| smart | 3 | Automatic selection |
| Field | Code | Description |
|---|---|---|
| TrackTitle | 65 | Track title |
| Album | 30 | Album name |
| AlbumArtist | 31 | Album artist |
| Artist | 32 | Track artist |
| Composer | 43 | Composer |
| Genre | 59 | Genre |
| Rating | 75 | Star rating (0-5) |
| RatingLove | 76 | Love rating |
| TrackNo | 86 | Track number |
| DiscNo | 52 | Disc number |
| Year | 88 | Year |
| Lyrics | 114 | Lyrics text |
| Comment | 44 | Comment |
| Publisher | 73 | Publisher/Label |
| Conductor | 45 | Conductor |
| Property | Code | Description |
|---|---|---|
| Url | 2 | File path/URL |
| Kind | 4 | File type (Music, Video, etc.) |
| Format | 5 | Audio format (MP3, FLAC, etc.) |
| Size | 7 | File size in bytes |
| Channels | 8 | Audio channels |
| SampleRate | 9 | Sample rate (Hz) |
| Bitrate | 10 | Bitrate (kbps) |
| Duration | 16 | Duration (ms) |
| PlayCount | 14 | Play count |
| SkipCount | 15 | Skip count |
| LastPlayed | 13 | Last played date |
| DateAdded | 12 | Date added to library |
| DateModified | 11 | File modification date |
| Format | Code | Description |
|---|---|---|
| Unknown | 0 | Unknown format |
| M3u | 1 | M3U playlist |
| Xspf | 2 | XSPF (XML Shareable Playlist) |
| Asx | 3 | ASX (Windows Media) |
| Wpl | 4 | WPL (Windows Media) |
| Pls | 5 | PLS playlist |
| Auto | 7 | Auto-detect format |
| Type | Code | Description |
|---|---|---|
| NotSpecified | 0 | Any lyrics type |
| Synchronised | 1 | Time-synced lyrics (LRC) |
| UnSynchronised | 2 | Plain text lyrics |
| Category | Code | Description |
|---|---|---|
| Music | 0 | Music files |
| Audiobook | 1 | Audiobooks |
| Video | 2 | Video files |
| Inbox | 4 | Inbox (new files) |
MBXHub includes comprehensive logging for debugging and monitoring. Logs are written using NLog to a log file in the MBXHub folder.
Logging is controlled by two settings in mbxhub.json:
| Setting | Type | Description |
|---|---|---|
debugMode | boolean | Master switch - enables/disables all logging. Also gates exception dumps in 500 response bodies (see below). |
logLevel | string | Minimum log level when debug mode is on |
500 response body gate: when debugMode is true AND logLevel is Debug or Trace, INTERNAL_ERROR responses include the full exception dump in the body (type, message, stack trace, inner-exception chain) — useful when debugging a self-hosted LAN deployment where the operator is also the consumer. Otherwise the body is the generic fallback string and the dump only lands in mbxhub.log. Default is fail-secure (debugMode: false) so production / shared exposure does not leak internals.
| Level | What Gets Logged |
|---|---|
| Trace | Everything including request/response bodies, WebSocket message content. Very verbose. |
| Debug | Route matching, handler selection, subscription changes, internal decisions. |
| Info | Startup/shutdown, HTTP requests (method/path/status/timing), connections, track changes. |
| Warning | Recoverable errors, timeouts, retries, unexpected but handled situations. |
| Error | Failures, exceptions, service unavailable. Always logged even with debug mode off. |
Log files are stored in the MBXHub subfolder of MusicBee's persistent storage:
%AppData%\MusicBee\MBXHub\mbxhub.log
mbxhub.1.log, mbxhub.2.log, etc.To enable debug logging:
// In mbxhub.json:
{
"debugMode": true,
"logLevel": "Trace" // or "Debug", "Info", "Warning", "Error"
}
HTTP requests are logged at Info level with timing:
2025-01-25 14:32:15 [INFO ] [REST] HTTP GET /player/status -> 200 (12ms)
2025-01-25 14:32:16 [INFO ] [REST] HTTP POST /player/playpause -> 200 (8ms)
2025-01-25 14:32:17 [INFO ] [Plugin] Track changed: Artist Name - Track Title
2025-01-25 14:32:17 [INFO ] [WebSocket] WS client abc12345 connected from 192.168.1.50:54321
MBXHub operates on a trusted local network model with multiple security layers.
PartyMode uses PIN-based authentication with three roles:
| Role | Access | Authentication |
|---|---|---|
DJ | Full control: player, queue, start/stop party | DJ PIN via X-Party-PIN header |
Guest | Browse library, request songs, vote | Guest PIN via X-Party-PIN header |
Anonymous | Read-only: now playing, artwork, status | No PIN required |
PartyMode endpoints authenticate via the X-Party-PIN header:
# Guest request example
curl -X POST http://localhost:8080/partymode/request \
-H "X-Party-PIN: 1234" \
-H "Content-Type: application/json" \
-d '{"url":"C:\\Music\\Track.mp3","nickname":"Haro"}'
# Validate PIN and get role
GET /partymode/validate?pin=1234&nickname=Haro
Invalid or missing PIN returns 401 Unauthorized.
MBXHub supports three protection levels for different deployment scenarios:
| Level | Description | Use Case |
|---|---|---|
| Default | Full API access, no restrictions | Personal use, trusted networks |
| Kiosk | All requests redirect to defaultPage | Party displays, public screens |
| Restricted | Read-only mode with granular controls | Shared access, limited control |
Configure via kioskMode and apiReadOnlyMode in settings.
PartyMode includes built-in limits and optional per-IP rate limiting:
Built-in limits (always active):
Per-IP rate limiting (configurable via Settings → Party Mode...):
Returns 429 Too Many Requests when limits exceeded.
| Origin | Access |
|---|---|
localhost, 127.0.0.1 | Always allowed |
192.168.x.x | Allowed when allowRemoteConnections is enabled |
10.x.x.x | Allowed when allowRemoteConnections is enabled |
172.16.x.x - 172.31.x.x | Allowed when allowRemoteConnections is enabled |
| External origins | Blocked (prevents cross-site attacks) |
MBXHub can restrict write operations via settings with granular per-operation controls:
Restriction Hierarchy:
Settings cascade: master OR category OR operation = blocked
Granular Operations:
| Category | Operation | Endpoints Affected |
|---|---|---|
| Library | Tag edits | PUT /library/file/*, POST /library/commit |
| Queue | Add tracks | POST /queue/add, /queue/playnow, /queue/play |
| Remove tracks | DELETE /queue/* | |
| Reorder tracks | POST /queue/move | |
| Player | Playback | POST /player/play, /pause, /stop, /next, /previous |
| Volume | POST /player/volume, /mute | |
| Seek | POST /player/position | |
| Playlists | Create | POST /playlists |
| Delete | DELETE /playlists/* | |
| Modify | PUT /playlists/*, POST /playlists/*/files |
Default Settings:
The philosophy: player and queue are open, destructive operations are locked. Playlists, tag edits, and file deletion default to read-only.
| Setting | Default | Effect |
|---|---|---|
apiReadOnlyMode | false | Master switch — off |
apiReadOnlyPlayer | false | Player controls allowed |
apiReadOnlyQueue | false | Queue modifications allowed |
apiReadOnlyLibrary | false | Library allowed (but see granular) |
apiReadOnlyPlaylists | true | Playlists blocked by default |
apiReadOnlyLibraryTags | true | Tag edits blocked by default |
apiReadOnlyLibraryDelete | true | File deletion blocked by default |
apiReadOnlyPlayerPlayback | false | Playback allowed |
apiReadOnlyPlayerVolume | false | Volume allowed |
apiReadOnlyPlayerSeek | false | Seek allowed |
apiReadOnlyQueueAdd | false | Queue add allowed |
apiReadOnlyQueueRemove | false | Queue remove allowed |
apiReadOnlyQueueReorder | false | Queue reorder allowed |
apiReadOnlyPlaylistsCreate | false | (moot — parent is true) |
apiReadOnlyPlaylistsDelete | false | (moot — parent is true) |
apiReadOnlyPlaylistsModify | false | (moot — parent is true) |
Settings cascade: master OR category OR operation = blocked. The granular playlist settings default to false but are moot because their parent apiReadOnlyPlaylists is true.
Action Categories (v0.5.2.6+):
| Category | Endpoints | Notes |
|---|---|---|
| MediaHandler party-mode | /media/* | When a party is active, all /media/* endpoints are DJ-only. Guest/Anonymous ⇒ 403 PARTY_LOCKED. The host’s Projector charm runs as DJ on localhost. Outside party mode, every caller is allowed. |
Always Permitted (exempt from restrictions):
Blocked requests return 403 Forbidden:
{"success":false,"error":{"code":"READ_ONLY","message":"API is in read-only mode"}}
{"success":false,"error":{"code":"PARTY_LOCKED","message":"Media browsing is locked during PartyMode (DJ-only)."}}
allowRemoteConnections disabled unless neededrpcEnabled: false) if not neededariaEnabled: false) unless specifically needed for PC wake scenariosrun() command only launches programs defined in ariaAllowedPrograms. Do not add shell interpreters (cmd.exe, powershell.exe) to the allowlist
All file URLs are Windows paths. URL-encode when passing in path parameters:
// Original: C:\Music\Artist\Track.mp3
// Encoded: C%3A%5CMusic%5CArtist%5CTrack.mp3
// Example: GET /library/file/C%3A%5CMusic%5CArtist%5CTrack.mp3
Most list endpoints support offset and limit query parameters:
GET /library/files?offset=100&limit=50Library endpoints support ?sort= parameter for server-side sorting:
alpha (default) - Alphabetical by title+artistartist - By artist name, then titlealbum - By album name, then track numbertitle - By title onlydate - By date added (newest first)track - By disc number, then track number (natural album order)name - By display nameyear-asc - By year ascending (chronological, oldest first)Example: GET /library/files?artist=Pink+Floyd&sort=album
Shuffle, banlist, and influence endpoints require TrueShuffle or AutoQ to be enabled. Check availability:
GET /shuffle/status
// Returns 503 SERVICE_UNAVAILABLE if TrueShuffle/AutoQ not enabled
For real-time updates (track changes, state changes), use WebSocket instead of polling:
By default, MBXHub only accepts localhost connections. For network access:
allowRemoteConnections in settingshttp://192.168.1.100:8080)MBXHub automatically detects CUE-backed audio files and resolves per-track metadata across all surfaces:
/player/status) - Overlays title, artist, trackNo, album from CUE sheet/nowplaying/position) - Includes cueTrack and cueTitle/nowplaying/tag) - Returns CUE track data for TrackTitle, Artist, Album, TrackNotrack events include cueTrack field when CUE activeEncoding detection: BOM check → UTF-8 validation → Windows-1252 fallback. Built-in regex parser; no external CUE library required.
| Use REST when... | Use RPC when... |
|---|---|
| Building a client app | Writing automation scripts |
| Need clean, discoverable URLs | Need direct MusicBee API access |
| Want resource-oriented design | Familiar with MusicBee plugin API |
| Working with standard HTTP clients | Need parameter flexibility |