Broadcast Box

May 21, 2026 · View on GitHub

License Discord

What is Broadcast Box

Broadcast Box lets you broadcast/screen-share to friends in sub-second time. It was designed to be simple to use and easily modifiable. Broadcast Box uses WebRTC, to learn more see the OBS WHIP Streaming Guide

Want to contribute to the development of Broadcast Box? See Contributing.

Using

A public instance of Broadcast Box is hosted at b.siobud.com. Feel free to use this as much as you want. Go to Getting Started for instructions on running it locally.

OBS Broadcasting

To use Broadcast Box with OBS you must set your output to WebRTC and set a proper URL + Stream Key. You may use any Stream Key you like. The same stream key is used for broadcasting and playback.

Go to Settings -> Stream and set the following values.

Your settings page should look like this:

OBS Stream settings example

OBS by default will have ~2 seconds of latency. If you want sub-second latency you can configure this in Settings -> Output. Set your encoder to x264 and set tune to zerolatency. Your Output page will look like this.

OBS Output settings example

When you are ready to broadcast press Start Streaming and now time to watch!

Browser Publishing

Open /publish/<streamKey> (for example https://b.siobud.com/publish/StreamTest) to publish directly from the browser. Broadcast Box can capture either your screen or webcam and uses the same stream key for playback.

If the supplied bearer token belongs to a reserved profile, the page also exposes profile settings so you can update the stream MOTD and toggle public/private visibility without leaving the publish flow.

FFmpeg Broadcasting

The following broadcasts a test feed to https://b.siobud.com with a Bearer Token of ffmpeg-test

ffmpeg \
  -re \
  -f lavfi -i testsrc=size=1280x720 \
  -f lavfi -i sine=frequency=440 \
  -pix_fmt yuv420p -vcodec libx264 -profile:v baseline -r 25 -g 50 \
  -acodec libopus -ar 48000 -ac 2 \
  -f whip -authorization "ffmpeg-test" \
  "https://b.siobud.com/api/whip"

Note that WHIP support and libx264 are required for this example. WHIP was added in version 8 of FFmpeg.

GStreamer Broadcasting

See the example script here.

Can broadcast gstreamer's test sources, or pulsesrc+v4l2src

Expects gstreamer-1.0, with good,bad,ugly plugins and gst-plugins-rs

Use of example scripts:

# testsrcs
./examples/gstreamer-broadcast.sh http://localhost:8080/api/whip testStream1
# v4l2src
./examples/gstreamer-broadcast.sh http://localhost:8080/api/whip testStream1 v4l2

Playback

If you are broadcasting to the Stream Key StreamTest your video will be available at https://b.siobud.com/StreamTest.

The player page also supports multi-view playback. Use the Add Stream button in the footer to add more active streams to the same page, and use ?cinemaMode=true to open the player in a chrome-free layout.

You can also go to the home page and enter StreamTest. The following is a screenshot of OBS broadcasting and the latency of 120 milliseconds observed.

Example have potential latency

Admin Portal

When FRONTEND_ADMIN_TOKEN is set Broadcast Box provides an Admin Portal at /admin. The same token is used to log in.

The current Admin Portal exposes:

  • Status to inspect active publishers/subscribers and metadata
  • Profiles to create stream profiles, rotate tokens, and remove profiles
  • Logging to read the current server log file

Stream Profiles let you reserve a stream key, so only authorized users can stream to that key.

Admin Portal

Statistics

Viewable at /statistics, this page shows stream uptime, track metrics, and WHEP session details.

This page relies on /api/status, so disabling the status API disables statistics.

Statistics

Examples

The repository includes a few small examples and integration helpers:

Getting Started

Broadcast Box is made up of two parts. The server is written in Go and is in charge of ingesting and broadcasting WebRTC. The frontend is in react and connects to the Go backend. The Go server can be used to serve the HTML/CSS/JS directly. Use the following instructions to build from source or utilize Docker / Docker Compose.

Configuring

The backend loads .env.production by default. Set APP_ENV=development to load .env.development instead.

If you only want the backend API or CLI helpers without serving the built frontend, set DISABLE_FRONTEND=TRUE.

Building From Source

Frontend

React dependencies are installed by running npm install in the web directory.

  • npm run build builds production assets into web/build
  • npm start runs the Vite dev server and proxies /api to the backend
  • npm run host runs the same Vite dev server on your local network

If everything is successful, you should see output similar to:

> broadcast-box@0.1.0 build
> vite build

[dotenv@17.2.3] injecting env (0) from ../.env.development,../.env -- tip: ⚙️  load multiple .env files with { path: ['.env.local', '.env'] }
Target Backend: http://localhost:8080
vite v6.4.1 building for production...
✓ 724 modules transformed.
build/index.html                       0.84 kB │ gzip:  0.49 kB
build/assets/index-BZVYZNKC.css       20.37 kB │ gzip:  4.82 kB
build/assets/index-BeQC1JnS.js         1.43 kB │ gzip:  0.69 kB
build/assets/components-BcOZaJ_1.js   63.11 kB │ gzip: 16.08 kB
build/assets/node-DpLsG32O.js        239.39 kB │ gzip: 75.76 kB
✓ built in 601ms

Backend

Go dependencies are automatically installed.

To run the Go server, run go run . in the root of this project, you should see the following:

2026/02/24 12:00:00 Environment: Loading `.env.production`
2026/02/24 12:00:00 Starting HTTP server at :8080

To use Broadcast Box navigate to: http://<YOUR_IP>:8080. In your broadcast tool of choice, you will broadcast to http://<YOUR_IP>:8080/api/whip.

Docker

A Docker image is also provided to make it easier to run locally and in production. The arguments you run the Dockerfile with depending on if you are using it locally or a server.

If you want to run locally execute docker run -e UDP_MUX_PORT=8080 -e NAT_1_TO_1_IP=127.0.0.1 -p 8080:8080 -p 8080:8080/udp seaduboi/broadcast-box. This will make broadcast-box available on http://localhost:8080. The UDPMux is needed because Docker on macOS/Windows runs inside a NAT.

If you are running on AWS (or other cloud providers) execute. docker run --net=host -e INCLUDE_PUBLIC_IP_IN_NAT_1_TO_1_IP=yes seaduboi/broadcast-box broadcast-box needs to be run in net=host mode. broadcast-box listens on random UDP ports to establish sessions.

Docker Compose

A Docker Compose is included that uses LetsEncrypt for automated HTTPS. It also includes Watchtower so your instance of Broadcast Box will be automatically updated every night. If you are running on a VPS/Cloud server this is the quickest/easiest way to get started.

export URL=my-server.com
docker-compose up -d

URL Parameters

The frontend can be configured by passing these URL Parameters.

Environment Variables

Server Configuration

VariableDescription
APP_ENVSet to development to load .env.development instead of .env.production.
HTTP_ADDRESSAddress for the main server to bind to. Used for HTTP, or HTTPS when certificates are configured.
ENABLE_HTTP_REDIRECTWhen set, enables automatic redirection from HTTP to HTTPS.
HTTPS_REDIRECT_PORTPort to listen on for the HTTP-to-HTTPS redirect server.
NETWORK_TEST_ON_STARTIf true, checks network connectivity on startup.
DISABLE_STATUSWhen set, disables /api/status. Stream discovery and /statistics rely on this endpoint.
ENABLE_PROFILINGIf true, enables PPROF profiling on localhost:6060.

SSL Configuration

VariableDescription
SSL_CERTPath to the SSL certificate file. When set together with SSL_KEY, the Go server serves HTTPS.
SSL_KEYPath to the SSL key file. When set together with SSL_CERT, the Go server serves HTTPS.

Authorization & Profiles

VariableDescription
STREAM_PROFILE_PATHPath to store stream profile configurations. Default is profiles.
STREAM_PROFILE_POLICYPolicy configuration for local reserved profiles. Default is ANYONE_WITH_RESERVED. See Stream Profile Policy.
WEBHOOK_URLURL for a webhook backend used to authorize/log publish (WHIP) and subscribe (WHEP) requests. See Webhooks.

Frontend Configuration

VariableDescription
DISABLE_FRONTENDDisables frontend asset serving and UI routes (/, /publish, /statistics, /admin).
FRONTEND_PATHPath to built frontend assets. Defaults to ./web/build.
FRONTEND_ADMIN_TOKENEnables /admin and defines the bearer token required to log in.

WebRTC & Networking

VariableDescription
INCLUDE_PUBLIC_IP_IN_NAT_1_TO_1_IPAutomatically includes public IPs in NAT configuration.
NAT_1_TO_1_IPManually specify IPs (like Public IP) to announce, delineated by |
INTERFACE_FILTERRestrict UDP traffic to a specific network interface.
NAT_ICE_CANDIDATE_TYPESet to srflx to append IPs instead of overriding with NAT_1_TO_1_IP.
NETWORK_TYPESList of network types to use delineated by | (e.g.,udp4 |udp6).
INCLUDE_LOOPBACK_CANDIDATEEnables WebRTC traffic on loopback interface.
UDP_MUX_PORTPort to multiplex all UDP traffic. Uses random port by default.
UDP_MUX_PORT_WHEPPort to multiplex WHEP traffic only.
UDP_MUX_PORT_WHIPPort to multiplex WHIP traffic only.
TCP_MUX_ADDRESSAddress to serve WebRTC traffic over TCP.
TCP_MUX_FORCEForces WebRTC traffic to use TCP only.
APPEND_CANDIDATEAppends ICE candidates not generated by the agent.

STUN Servers

VariableDescription
STUN_SERVERSList of STUN servers separated by |.

These values are parsed by the Go backend and applied to WHIP/WHEP PeerConnection configuration server-side. Clients do not fetch ICE server configuration from an API endpoint.

Debugging

VariableDescription
DEBUG_PRINT_OFFERPrints WebRTC offers received from clients.
DEBUG_PRINT_ANSWERPrints WebRTC answers sent to clients.
DEBUG_INCOMING_API_REQUESTLogs incoming API request paths.
DEBUG_PRINT_SSE_MESSAGESLogs Server-Sent Events messages.

Logging

VariableDescription
LOGGING_ENABLEDEnables logging system.
LOGGING_LEVELMinimum slog level: DEBUG, INFO, WARN, or ERROR. Defaults to INFO.
LOGGING_DIRECTORYDirectory to store log files.
LOGGING_SINGLEFILELogs everything into a single file called 'log'. Default is log files are stamped with current date.
LOGGING_NEW_FILE_ON_STARTUPCreates a new log file on each startup. Either a new 'log' file, or replaces the current dates log file.
LOGGING_API_ENABLEDEnables logging API to show current log entries on the backend. /api/log
LOGGING_API_KEYWhen set, the logging API requires a bearer token that uses this key.

Chat

VariableDescription
CHAT_MAX_HISTORYMaximum number of chat messages retained per stream in memory.
CHAT_DEFAULT_TTLHow long idle chat sessions stay alive before they expire.
CHAT_CLEANUP_INTERVALHow often expired chat sessions are cleaned up.

Broadcast Box attaches a WebRTC data channel (bb-chat-v1) to WHIP/WHEP peer connections for simple per-stream chat state. The bundled frontend does not currently expose a chat UI, but you can connect from your own client.

See CONNECTING.md for the message contract and a minimal standalone client example.

CLI Flags

The binary also supports a small local profile-management helper:

  • -createNewProfile -streamKey <stream-key> creates a new reserved profile in STREAM_PROFILE_PATH, prints the bearer token, and exits

Example:

go run . -createNewProfile -streamKey MyStream

Stream Profile Policy

The STREAM_PROFILE_POLICY environment variable controls who is allowed to initiate streaming sessions based on profile reservation status.

ValueDescription
ANYONE_WITH_RESERVEDReserved stream keys require a valid token. Unreserved stream keys may still be used by anyone.
RESERVEDOnly users with a valid token and a reserved stream key are allowed to stream. This is the most restrictive mode.

Any other value currently falls back to ANYONE_WITH_RESERVED behavior.

Webhooks

When WEBHOOK_URL is set Broadcast Box sends a webhook for every publish and subscribe. If this webhook is rejected the video session is disconnected.

The webhook payload includes:

  • action (whip-connect for publishers, whep-connect for viewers)
  • bearerToken
  • queryParams
  • ip
  • userAgent

The webhook must return HTTP 200 OK with a JSON body like { "streamKey": "YourResolvedStreamKey" }. Any other status code rejects the session.

This enables you to implement authorization or logging for broadcasting (WHIP) and subscribing (WHEP) independently.

See here. For an example Webhook Server that only allows the stream broadcastBoxRulez

For a more advanced example of a webhook server implementation making use of separating the key for streaming from the key for watching, see the broadcastbox-webhookserver repository.

Network Test on Start

When running in Docker Broadcast Box runs a network tests on startup. This tests that WebRTC traffic can be established against your server. If you server is misconfigured Broadcast Box will not start.

If the network test is enabled this will be printed on startup

NETWORK_TEST_ON_START is enabled. If the test fails Broadcast Box will exit.
See the README.md for how to debug or disable NETWORK_TEST_ON_START

If the test passed you will see

Network Test passed.
Have fun using Broadcast Box

If the test failed you will see the following. The middle sentence will change depending on the error.

Network Test failed.
Network Test client reported nothing in 30 seconds
Please see the README and join Discord for help

Join the Discord and we are ready to help! To debug check the following.

  • Have you allowed UDP traffic?
  • Do you have any restrictions on ports?
  • Is your server publicly accessible?

If you wish to disable the test set the environment variable NETWORK_TEST_ON_START to false.

Design

The backend exposes the following endpoints to support WebRTC streaming and server-side monitoring:

EndpointDescription
/api/whipInitiates a WHIP session for broadcasting via WebRTC. Requires an Authorization: Bearer <token> header.
/api/whip/{sessionID}PATCH handles WHIP trickle ICE for an existing session and DELETE closes it. Requires the same bearer token.
/api/whip/profileGET/POST endpoint for reading or updating the reserved profile (MOTD/privacy) associated with the supplied bearer token.
/api/whepInitiates a WHEP session for playback via WebRTC. Requires an Authorization: Bearer <streamKey> header.
/api/whep/{sessionID}PATCH handles WHEP trickle ICE for an existing playback session.
/api/sse/{sessionID}Server-sent events for stream status and available layers.
/api/layer/{sessionID}Switches audio/video layers for a WHEP session.
/api/statusReturns the status of all active public WHIP streams. Pass ?key=<streamKey> to fetch one active stream by key.
/api/logReturns the current log file when LOGGING_API_ENABLED=TRUE. If LOGGING_API_KEY is set, this endpoint also requires a bearer token.
/api/admin/loginValidates the admin bearer token configured in FRONTEND_ADMIN_TOKEN.
/api/admin/statusReturns full session state for the admin UI, including private streams.
/api/admin/profilesLists configured stream profiles for the admin UI.
/api/admin/profiles/add-profileCreates a new stream profile.
/api/admin/profiles/remove-profileRemoves an existing stream profile.
/api/admin/profiles/reset-tokenRotates the token for an existing stream profile.
/api/admin/loggingReturns the current log file for the admin UI.

All /api/admin/* endpoints require the FRONTEND_ADMIN_TOKEN bearer token.

The frontend ships the following browser routes:

RouteDescription
/Home page for joining an existing stream or navigating to browser publishing.
/publish/{streamKey}Browser publisher for screen/webcam streaming and reserved profile settings.
/statisticsLive stream and subscriber statistics derived from /api/status.
/adminAdmin portal for status, profiles, and log viewing when FRONTEND_ADMIN_TOKEN is set.
/{streamKey}Player page for a stream. The built-in UI can add more streams to create a multi-view layout.