flipdot-gschichtler
allows to display text on the OpenLab’s
flipdot panel in an user- and machine-friendly way without
fear of interfering with others and without requiring direct
connectivity to the panel.
flipdot-gschichtler
consists of the following three components:
-
warteraum
manages a queue of text snippets which is accessible and manipulable via a public REST API. Users will directly or indirectly use it to send text to the flipdot panel. -
anzeigetafel
is a daemon which runs on a machine with direct connectivity to the flipdot panel. It monitors the queue ofwarteraum
and displays new incoming text on the panel and deletes it from the queue afterwards. -
bahnhofshalle
is a web frontend which allows users to view and manipulate the queue ofwarteraum
from their browser.
-
The
v2
API may supportapplication/json
request bodies in the future. This is not yet the case. -
Any endpoint may also return different status codes than the ones listed here in unusual cases. These would typically be codes like 500 or 502.
-
ids are not guaranteed to never be reassigned. However, ids only start to get reassigned after the queue has been fully emptied.
Request Content-Type |
none |
Response Content-Type |
|
Authentication |
no |
This endpoint returns the current queue in the
following JSON format. length
corresponds with
the length of the queue array. queue
is ordered
by time of each item’s addition, i. e. the first
item is next up to be displayed. The supplied id
may be used to delete the entries.
{ "queue": [ { "id": 0, "text": "hello" }, { "id": 1, "text": "world" } ], "length": 2 }
HTTP Status |
Meaning |
200 |
Success |
Request Content-Type |
|
Response Content-Type |
|
Authentication |
no |
This endpoints allows you to add text to the queue, it requires no authentication as it is a feature available to the public. There is also no rate limiting as of yet, but it may be implemented should it become necessary (hopefully not).
The endpoint expects an urlencoded form as request body with the following field.
|
text to be added to the queue |
The response contains the queue entry consisting
of its text
and id
:
{ "id": 3, "text": "hello" }
HTTP Status |
Meaning |
200 |
Success, text added |
400 |
Illegal method or malformed request |
415 |
Request body too big or text field longer allowed (usually 512 bytes) |
503 |
The queue is full, i. e. the max id has been reached |
Request Content-Type |
|
Response Content-Type |
none |
Authentication |
yes (see below) |
This endpoint can be used to delete queue entries, e. g. after they have been displayed on the panel. The request should send a form with the following fields as request body in order to authenticate itself:
|
API token of the application |
HTTP Status |
Meaning |
204 |
Success, queue entry deleted |
400 |
Illegal method or malformed request |
401 |
Unauthorized API token |
404 |
No queue entry with given |
415 |
Request body too big |
Request Content-Type |
|
Response Content-Type |
|
Authentication |
no |
This is the legacy endpoint to add text to the queue. It enabled
interacting with it via a <form>
in the old web app. The form
sent as part of the request should have the following fields:
|
text to be added to the queue |
The response format has been changed since the previous implementation. I sincerly hope that nobody scraped the resulting page.
HTTP Status |
Meaning |
200 |
Success, text added |
400 |
Illegal method or malformed request |
415 |
Request body too big or text field longer allowed (usually 512 bytes) |
I'll gift anyone who finds serious
bugs or exploits in flipdot-gschichtler
a crate of Mate (or another
beverage if you don’t like it) as a reward. Such issues would be remotely
triggering crashes or segfaults, bypassing authentication, remote code
execution etc. I won’t bother making an exhaustive list or a precise
list of criteria in the hopes it doesn’t come back to haunt me…
Help is welcome! Some things that remain to be done:
-
More “funny” bits for the web frontend (hint: see
const subjects
inmain.es6
) -
Important: Documentation. Annoying sterni into doing it is also helping.
-
Make
warteraum
acceptapplication/json
request bodies for thev2
API using microjson (?). -
Polish the web frontend, test across browsers
-
Refresh queue regularly in the web frontend
-
Write more tests
-
Full Unicode support by using Unifont on the flipdots
-
A completely new feature you thought of
To build bahnhofshalle
you need GNU make and esbuild.
One way to obtain both is to run nix-shell -A bahnhofshalle
from the repository’s
root. To build, use the following commands:
cd bahnhofshalle make firefox index.html # for local development where only js needs to be rebuilt make dist firefox dist/index.html # properly minified distribution
Note that all requests are sent using a same-origin
policy,
so you need to configure a reverse proxy to serve the web
frontend and API simuntaneously for testing. You may take
inspiration from the nginx
configuration in
nixos/flipdot-gschichtler.nix
.
To ease the submodule hassle, dependencies that are inconvenient to
handle via a package manager are vendored or added as a
git subtree.
To avoid confusion these are located under third_party
exclusively.
Also be aware that different licensing terms may apply to code under this directory.
default.nix
provides the following nix derivations which are
ready to be installed:
-
warteraum
: standard clang/glibc build of warteraum -
warteraum-static
: statically linked build of warteraum using gcc and musl (used for the systemd service so we can restrict file system access) -
bahnhofshalle
-
anzeigetafel
nixos/flipdot-gschichtler.nix
provides a NixOS module which
defines services.flipdot-gschichtler
to conveniently set up
the server side with warteraum
and bahnhofshalle
behind
a nginx reverse proxy. A minimal configuration.nix
utilizing
it could look like this:
{ pkgs, ... }: { imports = [ /path/to/flipdot-gschichtler/nixos/flipdot-gschichtler.nix ]; services.flipdot-gschichtler = { enable = true; virtualHost = "flipdot.openlab-augsburg.de"; tokensFile = "/var/secrets/flipdot-gschichtler/tokens"; saltFile = "/var/secrets/flipdot-gschichtler/salt"; # if you want to change the derivations to use # packages = { # warteraum = …; # bahnhofshalle = …; # }; }; services.nginx.enable = true; security.acme = { .... }; }
warteraum
is configured via environment variables (which the NixOS
module utilizes):
-
WARTERAUM_SALT_FILE
: A file containing random data to use as salt -
WARTERAUM_TOKENS_FILE
: API tokens hashed usingscrypt
To generate the tokens file, warteraum
ships a utility tool.
Setting up auth works like this:
$ head -c 512 /dev/urandom > $WARTERAUM_SALT_FILE $ hashtoken $WARTERAUM_SALT_FILE token1 >> $WARTERAUM_TOKENS_FILE $ hashtoken $WARTERAUM_SALT_FILE token2 >> $WARTERAUM_TOKENS_FILE
Now warteraum
would accept “token1” and “token2” when authenticating.
Note that hashtoken
only supports appending tokens in a
convenient fashion at the moment. Removing tokens is quite cumbersome
and only possible with a knowledge of warteraum
internals.
-
warteraum
-
Limit size of request bodies to prevent DoS attacks
-
Trim whitespace on input text
-
Instead of compiling in salt and tokens, read them from the files specified via the
WARTERAUM_SALT_FILE
andWARTERAUM_TOKENS_FILE
environment variables.
-
-
NixOS module
-
Reflect change to
warteraum
by usingsaltFile
andtokensFile
respectively over the previoussalt
andtokens
. -
Fix sandboxing in
nixos/flipdot-gschichtler.nix
: Now only the secret files and the nix store will be readable to thewarteraum
process. -
Allow changing
bahnhofshalle
andwarteraum
derivation to use viapackages
. -
Don’t require
flipdot-gschichtler
to be passed as module argument, instead import directly from file system (unless a non-default derivation is configured).
-
-
bahnhofshalle
-
Switch to
esbuild
, requiring ES6 support in the browser as a result.
-
-
Replace
admin
andweb
frontends with pure EcmaScript frontendbahnhofshalle
-
Replace
web
API implementation withwarteraum
-
Rename
flipper
toanzeigetafel
, port to Python 3 -
API:
-
Move endpoints from
/
to/api/v1/
-
/api/v1/queue/add
HTML response changes, since no longer used by the frontend (except when no JavaScript is available) -
Add cleaned up version of the API as
/api/v2
. This one is used bybahnhofshalle
andanzeigetafel
and should be utilized by clients going forward.
-
-
Deployment:
-
Implement API/Frontend deployment as a NixOS service
-