DIY server in PHP/Laravel for self hosted screen generation.
Find a file
2025-11-06 15:36:27 +01:00
.devcontainer chore(docker): update to PHP 8.4 2025-08-26 11:59:52 +02:00
.github ci: update action 2025-11-05 14:12:41 +01:00
app Wrapping text in json object with 'data' as key 2025-11-04 09:08:25 +01:00
bootstrap add features 2025-03-03 21:24:22 +01:00
config feat: initial implementation of recipe catalog 2025-09-01 23:56:42 +02:00
database fix(#103): add recipe options to remove bleed margin and enable dark mode 2025-10-30 15:13:50 +01:00
docker/prod fix: docker-compose for production 2025-05-15 20:41:37 +02:00
docs chore: minor fixes for documentation 2025-08-21 15:38:37 +02:00
lang/de feat: add TRMNL custom Liquid filters 2025-07-24 18:57:18 +02:00
public feat: add favicon 2025-04-26 14:59:49 +02:00
resources fix(#103): add recipe options to remove bleed margin and enable dark mode 2025-10-30 15:13:50 +01:00
routes fix(#71): device specific sleep and setup images 2025-09-26 20:07:34 +02:00
screenshots chore: replace files with cdn links 2025-05-13 21:39:17 +02:00
storage fix(#71): device specific sleep and setup images 2025-09-26 20:07:34 +02:00
tests Wrapping text in json object with 'data' as key 2025-11-04 09:08:25 +01:00
.dockerignore ignore more things 2025-03-28 12:55:28 +01:00
.editorconfig Set up a fresh Laravel app 2025-02-08 15:05:41 +01:00
.env.example Move DB_DATABASE to docker-compose 2025-05-11 18:25:17 +02:00
.env.example.local chore: updates 2025-03-14 18:26:29 +01:00
.gitattributes Set up a fresh Laravel app 2025-02-08 15:05:41 +01:00
.gitignore chore: update dependencies 2025-10-29 13:34:46 +01:00
artisan Set up a fresh Laravel app 2025-02-08 15:05:41 +01:00
boost.json chore: update dependencies 2025-10-02 22:25:12 +02:00
composer.json fix(#103): apply dithering if requested by markup 2025-10-30 14:26:59 +01:00
composer.lock chore: update dependencies 2025-11-06 15:36:27 +01:00
CONTRIBUTING.md Create CONTRIBUTING.md 2025-05-05 21:53:03 +02:00
docker-compose.yml Move DB_DATABASE to docker-compose 2025-05-11 18:25:17 +02:00
Dockerfile chore(docker): update to PHP 8.4 2025-08-26 11:59:52 +02:00
example.png initial commit 2025-02-12 22:15:57 +01:00
LICENSE.md Create LICENSE.md 2025-02-12 22:35:04 +01:00
package-lock.json chore: update js dependencies 2025-11-03 12:21:55 +01:00
package.json feat: add codemirror 2025-10-10 16:05:42 +02:00
phpstan.neon chore: add larastan 2025-06-17 22:18:20 +02:00
phpunit.xml feat: update to Laravel 12 starter kit (#1) 2025-02-25 12:15:35 +01:00
pint.json chore: stricter pint rules 2025-06-17 21:30:59 +02:00
README.md Update README.md 2025-10-23 20:04:40 +02:00
README_byos-devices.jpeg chore: refresh Readme 2025-05-13 21:34:41 +02:00
README_byos-screenshot-dark.png chore: refresh Readme 2025-05-13 21:30:38 +02:00
README_byos-screenshot.png chore: refresh Readme 2025-05-13 21:30:38 +02:00
rector.php refactor: apply rector 2025-09-24 20:35:48 +02:00
vite.config.js feat: add codemirror 2025-10-10 16:05:42 +02:00

TRMNL BYOS (PHP/Laravel)

tests

TRMNL BYOS Laravel is a self-hostable implementation of a TRMNL server, built with Laravel. It allows you to manage TRMNL devices, generate screens using native plugins, recipes (55+ from the community catalog), or the API, and can also act as a proxy for the native cloud service (Core). With over 20k downloads and 100+ stars, its the most popular community-driven BYOS.

Screenshot Screenshot

Key Features

  • 📡 Device Information Display battery status, WiFi strength, firmware version, and more.
  • 🔍 Auto-Join Automatically detects and adds devices from your local network.
  • 🖥️ Screen Generation Supports Plugins (including Mashups), Recipes, API, Markup, or updates via Code.
    • Support for TRMNL Design Framework
    • Over 45 compatible open-source recipes are available in the community catalog
    • Supported Devices
      • TRMNL OG (1-bit & 2-bit)
      • SeeedStudio TRMNL 7,5" (OG) DIY Kit
      • Seeed Studio (XIAO 7.5" ePaper Panel)
      • reTerminal E1001 Monochrome ePaper Display
      • Custom ESP32 with TRMNL firmware
      • E-Reader Devices
      • Android Devices with trmnl-android
      • Raspberry Pi (HDMI output) trmnl-display
  • 🔄 TRMNL API Proxy Can act as a proxy for the native cloud service (requires TRMNL Developer Edition).
    • This enables a hybrid setup for example, you can update your custom Train Monitor every 5 minutes in the morning, while displaying native TRMNL plugins throughout the day.
  • 🌙 Dark Mode Switch between light and dark mode.
  • 🐳 Deployment Dockerized setup for easier hosting (Dockerfile, docker-compose).
  • 💾 Flexible Database configuration uses SQLite by default, also compatible with MySQL or PostgreSQL
  • 🛠️ Devcontainer support for easier development.

Devices

Support ❤️

This repo is maintained voluntarily by @bnussbau.

Support the development of this package by purchasing a TRMNL device through the referral link: https://usetrmnl.com/?ref=laravel-trmnl. At checkout, use the code laravel-trmnl to receive a $15 discount on your purchase.

or

"Buy Me A Coffee"

GitHub Sponsors

Hosting

Run everywhere, where Docker is supported: Raspberry Pi, VPS, NAS, Container Cloud Service (Cloud Run, ...). For production use, generate a new APP_KEY (php artisan key:generate --show) and set the environment variable APP_KEY=. For personal use, you can disable registration (see section Environment Variables).

Docker Compose

Docker Compose file located at: docker/prod/docker-compose.yml.

Backup Database
docker ps #find container id of byos_laravel container
docker cp {{CONTAINER_ID}}:/var/www/html/database/storage/database.sqlite database_backup.sqlite
Updating via Docker Compose
docker compose pull
docker compose down
docker compose up -d

VPS

If youre using a VPS (e.g., Hetzner) and prefer an alternative to native Docker, you can install Dokploy and deploy BYOS Laravel using the integrated Template. Its a quick way to get started without having to manually manage Docker setup.

PikaPods

You can vote for TRMNL BYOS Laravel to be included as PikaPods Template here: feedback.pikapods.com

Umbrel

Umbrel is supported through a community store, see.

Other Hosting Options

Laravel Forge, or bare metal PHP server with Nginx or Apache is also supported.

Requirements

Local Development

see docs/DEVELOPMENT.md

Demo Plugins

Run the ExampleRecipesSeeder to seed the database with example plugins:

php artisan db:seed --class=ExampleRecipesSeeder
  • Zen Quotes
  • This Day in History
  • Weather
  • Train Departure Monitor
  • Home Assistant
  • Sunrise/Sunset

Usage

Environment Variables

Environment Variable Description Default
TRMNL_PROXY_BASE_URL Base URL of the native TRMNL service https://trmnl.app
TRMNL_PROXY_REFRESH_MINUTES How often should the server fetch new images from native service 15
REGISTRATION_ENABLED Allow user registration via Webinterface 1
SSL_MODE SSL Mode, if not using a Reverse Proxy (docs) off
FORCE_HTTPS If your server handles SSL termination, enforce HTTPS. 0
PHP_OPCACHE_ENABLE Enable PHP Opcache 0
TRMNL_IMAGE_URL_TIMEOUT How long TRMNL waits for a response on the display endpoint. (sec) 30
APP_TIMEZONE Default timezone, which will be used by the PHP date functions UTC

Login

If your environment is local, you can access the server at http://localhost:4567 and login with user / password admin@example.com / admin@example.com, otherwise register. With environment variable REGISTRATION_ENABLED you can control, if registration is allowed.

Add Your TRMNL Device

Auto-Join (Local Network)
  1. Switch on the “Permit Auto-Join” toggle in the header. For that to work only one user can be registered.
  2. New devices on your local network will be detected and added automatically when connecting to the server.

This is the easiest way to connect your devices with minimal effort.

Manually
  1. Open the Devices page: 👉 http://localhost:4567/devices
  2. Click “Add New Device”.
  3. Retrieve your TRMNL MAC Address and API Key:
  • You can grab the TRMNL Mac Address and API Key from the TRMNL Dashboard
  • Alternatively, debug incoming requests to /api/setup to determine them

⚙️ Configure Server for Device

📌 Firmware Version 1.4.6 or Newer

  • Setup device
  • After entering Wifi credentials, choose "Custom Server"
  • Point to the URL of your server

Firmware Older Than 1.4.6

If your device firmware is older than 1.4.6, you need to flash a new firmware version to point it to your server.

See this YouTube guide: https://www.youtube.com/watch?v=3xehPW-PCOM

☁️ Activate fresh TRMNL Device with Cloud Proxy

  1. Setup the TRMNL as in the official docs with the cloud service (connect one of the plugins to later verify it works)
  2. Setup Laravel BYOS, create a user and login
  3. In Laravel BYOS in the header bar, activate the toggle "Permit Auto-Join"
  4. Press and hold the button on the back of your TRMNL for 5 seconds to reactivate the captive portal (or reflash).
  5. Go through the setup process again, in the screen where you provide the Wi-Fi credentials there is also option to set the Server URL. Use the local address of your Laravel BYOS
  6. The device should automatically appear in the device list; you can deactivate the "Permit Auto-Join" toggle again.
  7. In the devices list, activate the toggle "☁️ Proxy" for your device. (Make sure that the queue worker is active. In the docker image it should be running automatically.)
  8. As long as no Laravel BYOS plugin is scheduled, the device will show your cloud plugins.
Troubleshooting

Make sure that your device has a Developer license, you should be able to verify by calling the https://trmnl.app/api/display endpoint.

🖥️ Generate Screens

Markup via Web Interface

  1. Navigate to Plugins > Markup in the Web Interface.
  2. Enter your markup manually or select from the available templates.
  3. Save and apply the changes.

🎨 Blade View

php artisan trmnl:screen:generate

Generate via API

You can dynamically update screens by sending a POST request.

  • Send a POST request to /api/screen with the following payload
Header

Authorization Bearer <TOKEN>

Body
{
    "markup": "<h1>Hello World</h1>"
}

Releated Work

🤝 Contribution

Contributions are welcome! See CONTRIBUTING.md for details.

License

MIT