The official whiteboard app for Nextcloud. It allows users to create and share whiteboards with other users and collaborate in real-time.
You can create whiteboards in the files app and share and collaborate on them.
- 🎨 Drawing shapes, writing text, connecting elements
- đź“ť Real-time collaboration
- đź’Ş Strong foundation: We use Excalidraw as our base library
Running the whiteboard server is required for the whiteboard to work. The server will handle real-time collaboration events and broadcast them to all connected clients, which means that the server must be accessible from the users browser, so exposing it for example throuhg a reverse proxy is necessary. It is intended to be used as a standalone service that can be run in a container.
We require the following connectivity:
- The whiteboard server needs to be able to reach the Nextcloud server over HTTP(S)
- The Nextcloud server needs to be able to reach the whiteboard server over HTTP(S)
- The user's browser needs to be able to reach the whiteboard server over HTTP(S) in the browser
- Nextcloud and the whiteboard server share a secret key to sign and verify JWTs
On the Nextcloud side, the server must be configured through:
occ config:app:set whiteboard collabBackendUrl --value="http://nextcloud.local:3002"
occ config:app:set whiteboard jwt_secret_key --value="some-random"
This mode requires at least Node 20 and NPM 10 to be installed. You can clone this repository, checkout the release version matching your whiteboard app. The server can be run locally using the following command:
npm ci
JWT_SECRET_KEY="some-random" NEXTCLOUD_URL=http://nextcloud.local npm run server:start
The server requires the NEXTCLOUD_URL
environment variable to be set to the URL of the Nextcloud instance that the Whiteboard app is installed on. The server will connect to the Nextcloud instance and listen for whiteboard events.
The server can be run in a container using the following command:
docker run -e JWT_SECRET_KEY=some-random -e NEXTCLOUD_URL=https://nextcloud.local --rm ghcr.io/nextcloud-releases/whiteboard:release
Docker compose can also be used to run the server:
version: '3.7'
services:
nextcloud-whiteboard-server:
image: ghcr.io/nextcloud-releases/whiteboard:release
ports:
- 3002:3002
environment:
- NEXTCLOUD_URL=https://nextcloud.local
- JWT_SECRET_KEY=some-random-key
While we publish image on the GitHub container registry you can build the image locally using the following command:
docker build -t nextcloud-whiteboard-server -f Dockerfile .
ProxyPass /whiteboard http://localhost:3002/
RewriteEngine on
RewriteCond %{HTTP:Upgrade} websocket [NC]
RewriteCond %{HTTP:Connection} upgrade [NC]
RewriteRule ^/?whiteboard/(.*) "ws://localhost:3002/$1" [P,L]
ProxyPass /whiteboard http://localhost:3002/ upgrade=websocket
location /whiteboard/ {
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header Host $host;
proxy_pass http://localhost:3002/;
proxy_http_version 1.1;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "upgrade";
}
handle_path /whiteboard/* {
reverse_proxy http://127.0.0.1:3002
}
The whiteboard application supports two storage strategies: LRU (Least Recently Used) cache and Redis. Each strategy has its own characteristics and is suitable for different deployment scenarios.
The LRU cache strategy is an in-memory storage solution that keeps the most recently used items in memory while automatically removing the least recently used items when the cache reaches its capacity.
Advantages:
- Simple setup with no additional infrastructure required
- Fast read and write operations
- Suitable for single-node deployments
Limitations:
- Limited by available memory on the server
- Data is not persistent across server restarts
- Not suitable for multi-node deployments
Configuration:
To use the LRU cache strategy, set the following in your .env
file:
STORAGE_STRATEGY=lru
Resources:
Redis is an in-memory data structure store that can be used as a database, cache, and message broker. It provides persistence and supports distributed setups.
Advantages:
- Persistent storage
- Supports multi-node deployments
- Allows for horizontal scaling
Limitations:
- Requires additional infrastructure setup and maintenance
- Slightly higher latency compared to LRU cache for single-node setups
Configuration:
To use the Redis strategy, set the following in your .env
file:
STORAGE_STRATEGY=redis
REDIS_URL=redis://[username:password@]host[:port][/database_number]
Replace the REDIS_URL
with your actual Redis server details.
For small to medium-sized deployments, a single-node setup can be sufficient:
- Choose either LRU or Redis strategy based on your persistence needs.
- Configure the
.env
file with the appropriateSTORAGE_STRATEGY
. - If using Redis, ensure the Redis server is accessible and configure the
REDIS_URL
. - Start the whiteboard server.
For larger deployments requiring high availability and scalability, a multi-node setup is recommended:
- Use the Redis storage strategy.
- Set up a Redis cluster or a managed Redis service.
- Configure each node's
.env
file with:STORAGE_STRATEGY=redis REDIS_URL=redis://[username:password@]host[:port][/database_number]
- Set up a load balancer to distribute traffic across the nodes.
- Ensure all nodes can access the same Redis instance or cluster.
The whiteboard application uses the Redis Streams adapter for scaling WebSocket connections across multiple nodes. This adapter leverages Redis Streams, not the Redis Pub/Sub mechanism, for improved performance and scalability.
When using the Redis strategy, the application automatically sets up the Redis Streams adapter for WebSocket scaling. This allows multiple server instances to share WebSocket connections and real-time updates.
Resources:
- Load Balancing: Set up a load balancer to distribute incoming connections across your server nodes.
- Session Stickiness: While not strictly required for WebSocket transport, it's recommended to configure your load balancer to use session stickiness. This ensures that requests from a client are routed to the same server for the duration of a session, which can be beneficial if falling back to long polling.
- WebSocket Support: Ensure your load balancer is configured to support WebSocket connections and maintain long-lived connections.
- Redis Setup: The current implementation does not configure Redis Cluster. So if you need to use a Redis Cluster for high availability, you'll need to set up your own load balancer in front of your Redis Cluster nodes.
- Redis Connection: The application currently supports only one Redis connection for both the storage layer and streaming/scaling the WebSocket server.
- Redis Persistence: Configure Redis with appropriate persistence settings (e.g., RDB snapshots or AOF logs) to prevent data loss in case of Redis server restarts.
- Monitoring: Implement monitoring for both your application nodes and Redis servers to quickly identify and respond to issues.
- LRU Cache: Ideal for small deployments, development environments, or scenarios where data persistence across restarts is not critical.
- Redis: Recommended for production environments, especially when scaling horizontally or when data persistence is required.
By carefully considering your deployment needs and choosing the appropriate storage strategy, you can ensure optimal performance and scalability for your whiteboard application.
If the integration_whiteboard app was previously installed there might be a leftover non-standard mimetype configured. In this case opening the whiteboard may fail and a file is downloaded instead. Make sure to remove any entry in config/mimetypealiases.json mentioning whiteboard and run occ maintenance:mimetype:update-db
and occ maintenance:mimetype:update-js
.