WBO is an online collaborative whiteboard that allows many users to draw simultaneously on a large virtual board. The board is updated in real time for all connected users, and its state is always persisted. It can be used for many different purposes, including art, entertainment, design, teaching.
A demonstration server is available at wbo.ophir.dev
| The anonymous board |
| collaborative diagram editing |
|
| teaching math on WBO |
| drawing art |
|
If you have your own web server, and want to run a private instance of WBO on it, you can. It should be very easy to get it running on your own server.
If you use the docker containerization service, you can easily run WBO as a container.
An official docker image for WBO is hosted on dockerhub as lovasoa/wbo: .
You can run the following bash command to launch WBO on port 5001, while persisting the boards outside of docker:
mkdir wbo-boards # Create a directory that will contain your whiteboards
chown -R 1000:1000 wbo-boards # Make this directory accessible to WBO
docker run -it --publish 5001:80 --volume "$(pwd)/wbo-boards:/opt/app/server-data" lovasoa/wbo:latest # run wboYou can then access WBO at http://localhost:5001.
The official Docker image does not force an IP source. By default the application uses WBO_IP_SOURCE=remoteAddress. If you run the container behind a trusted proxy or CDN, set -e WBO_IP_SOURCE=... explicitly, for example X-Forwarded-For, Forwarded, or CF-Connecting-IP.
Alternatively, you can run the code with node.js directly, without docker.
First, download the sources:
git clone https://github.com/lovasoa/whitebophir.git
cd whitebophir
Then install node.js (v22 or superior) if you don't have it already, then install WBO's dependencies:
npm install --production
Finally, you can start the server:
PORT=5001 npm start
This will run WBO directly on your machine, on port 5001, without any isolation from the other services. You can also use an invokation like
PORT=5001 HOST=127.0.0.1 npm start
to make whitebophir only listen on the loopback device. This is useful if you want to put whitebophir behind a reverse proxy.
By default, WBO launches its own web server and serves all of its content at the root of the server (on /).
If you want to make the server accessible with a different path like https://your.domain.com/wbo/ you have to setup a reverse proxy.
See instructions on our Wiki about how to setup a reverse proxy for WBO.
WBO is available in multiple languages. The translations are stored in server/translations.json.
If you feel like contributing to this collaborative project, you can translate WBO into your own language.
WBO supports authentication using Json Web Tokens. Pass the token as a token query parameter, for example http://myboard.com/boards/test?token={token}.
The AUTH_SECRET_KEY variable in configuration.js should be filled with the secret key for the JWT.
WBO recognizes two privileged roles:
editor: can modify accessible boards.moderator: can modify accessible boards and use the Clear tool.
Roles are declared in the JWT payload:
{
"iat": 1516239022,
"exp": 1516298489,
"roles": ["editor"]
}Moderators have access to the Clear tool, which wipes all content from the board.
If AUTH_SECRET_KEY is not set, boards are visible to anyone who knows the URL.
If AUTH_SECRET_KEY is set, opening a board requires a valid token. You can then restrict which board names a token may open by adding :<boardName> to a claim:
{
"roles": ["editor:board-a", "moderator:board-b", "reader:board-c"]
}editor:<boardName>allows editing that board.moderator:<boardName>allows moderating that board.reader:<boardName>allows opening that board without granting editor or moderator privileges.
For example, http://myboard.com/boards/mySecretBoardName?token={token} with:
{
"iat": 1516239022,
"exp": 1516298489,
"roles": ["moderator:mySecretBoardName"]
}If a token contains any board-scoped claims, it can only open the boards named in those claims.
Board visibility and board editability are separate.
- A writable board accepts writes from users who can access it.
- A read-only board can be opened by users who have access to it.
- On a read-only board, only
editorandmoderatorclaims may write. - On instances without JWT authentication, a read-only board blocks all writes because there is no authenticated editor or moderator identity.
Read-only state is stored in the board JSON file itself under the reserved key __wbo_meta__:
{
"__wbo_meta__": {
"readonly": true
}
}- Without JWT auth: visibility is controlled by sharing or not sharing the board URL.
- With JWT auth: visibility is controlled by the token you issue. Add or remove board-scoped claims to decide which boards a token may open.
- Use
editorormoderatorclaims for users who should write. - Use
reader:<boardName>for users who should only view a read-only board.
- Find the board file in
WBO_HISTORY_DIR. The filename isboard-${encodeURIComponent(boardName)}.json. - Add or update the
__wbo_meta__.readonlyflag in that file. - Reload the board after it is unloaded from memory, or restart the server, so the new state is picked up.
- Remove the flag or set it to
falseto make the board writable.
When you start a WBO server, it loads its configuration from several environment variables.
You can see a list of these variables in configuration.js.
Some important environment variables are :
WBO_HISTORY_DIR: configures the directory where the boards are saved. Defaults to./server-data/.WBO_MAX_EMIT_COUNT: the general socket write limit profile. Use compact entries such as*:250/5s anonymous:125/5s. Increase this if you want smoother drawings, at the expense of making denial-of-service bursts cheaper for clients. The default is*:250/5s.WBO_MAX_CONSTRUCTIVE_ACTIONS_PER_IP: the constructive per-IP write limit profile. Use compact entries such as*:40/10s anonymous:20/10s.WBO_MAX_DESTRUCTIVE_ACTIONS_PER_IP: the destructive per-IP write limit profile. Use compact entries such as*:190/60s anonymous:95/60s.WBO_IP_SOURCE: which request attribute to trust for client IP based limits and logs. SupportsremoteAddress,X-Forwarded-For,Forwarded, or a custom header such asCF-Connecting-IP. The default isremoteAddress.AUTH_SECRET_KEY: If you would like to authenticate your boards using jwt, this declares the secret key.
If you experience an issue or want to propose a new feature in WBO, please open a github issue.
If you are self-hosting a WBO instance, you may want to monitor its load, the number of connected users, request latency, and board lifecycle events.
WBO now uses OpenTelemetry for metrics and logs on the server side.
Configure a standard OTLP exporter with the usual OTEL_* environment variables.
Example:
docker run \
-e OTEL_SERVICE_NAME=whitebophir-server \
-e OTEL_EXPORTER_OTLP_ENDPOINT=http://otel-collector:4318 \
lovasoa/wboCommon settings:
OTEL_SERVICE_NAMEOTEL_RESOURCE_ATTRIBUTESOTEL_EXPORTER_OTLP_ENDPOINTOTEL_EXPORTER_OTLP_METRICS_ENDPOINTOTEL_EXPORTER_OTLP_LOGS_ENDPOINTOTEL_EXPORTER_OTLP_HEADERS
If no OTLP endpoint is configured, WBO still emits canonical server log lines to stdout/stderr but does not attempt remote export.
To download a preview of a board in SVG format you can got to /preview/{boardName}, e.g. change https://wbo.ophir.dev/board/anonymous to https://wbo.ophir.dev/preview/anonymous. The renderer is not 100% faithful, but it's often good enough.



