Minimal self-hosted link redirector with admin UI, stats, SQLite storage and API for automation
- Multiple domains
- Link expiration
- Support for links with hash parts
- Optional query string forwarding
- Basic stats overview (full stats recording)
- API for automation with source IP restrictions
- JSON Import/export in UI
- Importing from Rebrandly and Kutt
flowchart LR
client[Client Browser]
subgraph relinky [Relinky]
router[Edge Proxy / Router]
admin[Admin CP :8081]
redirect[Redirector :8082]
db[(SQLite DB)]
router --> admin
router --> redirect
admin --> db
redirect --> db
end
client --> router
Databases:
db/main.db— settings/defaults/api keysdb/redirectables.db— domains/links/target URLsdb/stats.db— redirect statsdb/logs.db— audit-like logs
Mode 1: Plain Docker (docker-compose.yml)
flowchart LR
client[Client Browser] --> edgeProxy[Your proxy / direct port mapping]
subgraph relinky [Relinky]
admin[Admin CP :8081]
redirect[Redirector :8082]
db[(SQLite DB)]
admin --> db
redirect --> db
end
edgeProxy --> admin
edgeProxy --> redirect
Use when:
- You already have your own reverse proxy/TLS setup
- Or local/dev usage where you do not need automatic TLS
Start:
docker compose up -dMode 2: Gateway, embedded Caddy (docker-compose.gateway.yml)
Use when:
- You run on a VPS and want Relinky to manage routing/certs itself
- Host ports
80/443are available (or intentionally remapped)
flowchart BT
client[Client Browser]
subgraph relinky [Gateway Container]
subgraph scripts [Startup/reload Scripts]
entrypoint{{entrypoint-gateway.sh}}
startScript[/start.js/]
genScript[/generate-caddyfile.mjs/]
reloadHelper[/gateway-reload.js/]
entrypoint -.-> startScript
entrypoint -.-> genScript
reloadHelper -.-> genScript
end
subgraph services [Services]
db[(SQLite DB)]
caddy[Caddy :80/443]
admin[Admin CP :8081]
redirect[Redirector :8082]
admin --> db
redirect --> db
caddy --> admin
caddy --> redirect
end
startScript -.-> admin
startScript -.-> redirect
admin -. Triggered on changes .-> reloadHelper
genScript -.-> caddy
end
client --> caddy
- Container entrypoint starts and sets gateway defaults (loopback binds + Caddyfile path).
- Entrypoint runs DB init and generates the initial Caddyfile from current DB domains.
- Entrypoint starts Node services (
start.js) and Caddy (caddy run ...). - When domains are created/removed in admin API, backend schedules a non-blocking gateway reload.
- Reload helper regenerates Caddyfile from DB and executes
caddy reload.
Script pointers for this flow:
docker/entrypoint-gateway.sh(startup order, process launch)start.js(spawns admin + redirector)scripts/generate-caddyfile.mjs(reads DB domains, writes Caddyfile)app/shared/gateway-reload.js(regenerate +caddy reloadon domain changes)app/admin/backend/api.js(callsscheduleGatewayReload()after domain mutations)
Required environment variables:
RELINKY_ADMIN_HOST(required)ACME_EMAIL(recommended for real HTTPS)
Optional:
ADMIN_PASSWORD_HASHorADMIN_PASSWORD_HASH_B64— if set, seeded into the DB on startup instead of using onboarding.
Start:
export RELINKY_ADMIN_HOST='admin.example.com'
export ACME_EMAIL='you@example.com'
# optional: export ADMIN_PASSWORD_HASH='...'
docker compose -f docker-compose.gateway.yml up -dAfter startup:
- Open
https://admin.example.com - Complete onboarding (first visit) or log in if a password was seeded
- Add redirect domains in Domains
- Ensure those domains resolve to the same server
- Relinky regenerates Caddy config and reloads Caddy automatically
Port/cert notes:
- Let’s Encrypt HTTP-01/TLS-ALPN needs public
80/443 - For non-standard public ports, use
RELINKY_CADDY_TLS_INTERNAL=1for self-signed/internal TLS RELINKY_CADDY_HTTP_PORT/RELINKY_CADDY_HTTPS_PORTcontrol Caddy bind ports inside containerRELINKY_GATEWAY_HOST_HTTP/RELINKY_GATEWAY_HOST_HTTPScontrol published host ports
Mode 3: Coolify / Traefik (docker-compose.coolify.yml)
Use when:
- Coolify/Traefik already terminates TLS and owns host
80/443
Do not use gateway mode here unless you intentionally want double proxy.
Why split services:
- Coolify domain mapping is per service
- Admin and redirector need separate upstream targets (
8081,8082)
flowchart LR
client[Client Browser]
subgraph coolify [Coolify]
traefik[Traefik :80/443]
traefik --> admin[relinky_admin :8081]
traefik --> redirect[relinky_redirect :8082]
admin --> SharedDb[(Shared db volume)]
redirect --> SharedDb
end
client --> traefik
Checklist:
- Create an app from a public Github repo or your private cloned one
- Build pack: Docker Compose, file
docker-compose.coolify.yml. Note that by default Coolify offers.yamlextension, so change the whole file name. - Optional: set
ADMIN_PASSWORD_HASH_B64on the relinky_migrate service (Base64-encoded hash). Coolify often mangles$in env vars — use B64 if login fails after deploy. If unset, complete onboarding on first admin visit instead. - Ensure the persistent storage for
./dbis attached to both services (should happen automatically) - Setup admin and redirect domains in Coolify and the admin UI:
- Admin service: one admin hostname, for example
https://admin.example.com:8081 - Redirect service: one or many redirect hostnames, for example
https://link.example.com:8082, https://dl.example.com:8082 - In the panel, add redirect hostnames under Domains
- Important: keep in mind Coolify expects full links with protocols and ports like shown above, don't enter only domains.
- Admin service: one admin hostname, for example
On first visit, if no admin password exists in the database, Relinky shows an onboarding screen where you set a password and your first redirect domain. After that, use the normal login page.
You can also pre-set a password before first visit:
npm run hash-password -- 'your-password'Alternative:
openssl passwd -6 'your-password'If your platform mangles $ values:
npm run hash-password -- 'your-password' --b64Then set ADMIN_PASSWORD_HASH or ADMIN_PASSWORD_HASH_B64. On every startup the migrator copies this hash into the database (overwriting any in-app password change). Remove the env var to manage the password only from the admin UI (Tools → Password).
If neither env nor onboarding has run yet, the admin UI stays on onboarding until a password is set.
The Domains page has two layers:
- Global defaults (
GET/PUT /api/domains/defaults) — default domain, link defaults (expired URL, redirect code, keep referrer/query), and global error redirect URLs (error_404_url,error_500_urlinmain.db). - Per-domain overrides (
GET/PUT /api/domains/:id) — optional values on each redirect hostname.nullmeans inherit from global. PartialPUTupdates only the fields you send;nullclears an override.
At redirect time the redirector resolves link → domain → global for link fields (redirect_code, keep_referrer, keep_query_params, expired URL). Unknown slugs use domain → global error URLs; an unknown hostname uses global error URLs only.
Links can store null on those fields to inherit (set Default in the link form). Existing links keep their stored values until edited.
Create keys in Tools → API keys.
Capabilities:
- Links: list/create/update/delete
- Stats: read
- Optional IP allowlist per key (exact IP and CIDR)
Link fields redirect_code, keep_referrer, and keep_query_params accept JSON null on create/update to inherit (link → domain → global). List responses return null for inherit; false/0 means explicitly off.
Domain defaults and per-domain overrides are admin-only (not exposed on external routes).
Auth format:
Authorization: Bearer rk_<keyId>.<secret>Endpoints:
- Get links:
GET /api/external/links?page=1&limit=100&search=... - Create link:
POST /api/external/links - Edit link:
PUT /api/external/links/:id - Delete link:
DELETE /api/external/links/:id - Get stats:
GET /api/external/stats?period=day|week|month|year|all&linkId=<id>
Examples:
API_KEY='rk_xxx.yyy'
BASE='https://admin.example.com'
# Get 50 links
curl -sS -H "Authorization: Bearer $API_KEY" "$BASE/api/external/links?page=1&limit=50"
# Create 'go.example.com/promo-2026' link leading to 'https://example.com/landing' with 303 HTTP code
curl -sS -X POST "$BASE/api/external/links" \
-H "Authorization: Bearer $API_KEY" \
-H "Content-Type: application/json" \
-d '{"domain":"go.example.com","slug":"promo-2026","url":"https://example.com/landing","redirect_code":303}'
# Create link that inherits redirect code and bool defaults from domain/global settings
curl -sS -X POST "$BASE/api/external/links" \
-H "Authorization: Bearer $API_KEY" \
-H "Content-Type: application/json" \
-d '{"domain":"go.example.com","slug":"inherit-settings","url":"https://example.com/x","redirect_code":null,"keep_referrer":null,"keep_query_params":null}'Check .env.example file
Optional (seeds the DB password on every migrator run; omit to use onboarding or Tools → Password):
ADMIN_PASSWORD_HASH— Raw sha512-crypt admin password hash ($6$...). Copied into theauthtable on startup (overwrites). Login always checks the database, not env directly.ADMIN_PASSWORD_HASH_B64— Base64 form of the same hash; use when your platform mangles$characters (e.g. Coolify).
Optional:
ADMIN_LOGIN_DEBUG— Enables verbose admin login diagnostics in logs (1,true,yes).ADMIN_PASSWORD_SHA512_ROUNDS— Hash rounds used by the local hash-generation script.ADMIN_IP— Bind address for admin HTTP server.ADMIN_PORT(default8081) — Listen port for admin HTTP server.REDIRECTOR_IP— Bind address for redirector HTTP server.REDIRECTOR_PORT(default8082) — Listen port for redirector HTTP server.RELINKY_DB_DIR— Override the SQLite database directory (defaults to the repo-localdb/). Useful for isolated test runs or non-standard layouts; in Docker thedb/volume is the persistent location.RELINKY_DB_BUSY_TIMEOUT_MS(default5000) — How long a SQLite connection waits for a busy database lock before erroring. The admin and redirector both run migrations on boot, so this lets the second writer wait instead of failing withSQLITE_BUSY.RELINKY_DB_BACKUP_KEEP(default10,0= keep all) — How many pre-migration backups to retain per database indb/backups/. Older snapshots beyond this count are pruned automatically.
Gateway mode only (docker-compose.gateway.yml)
Required:
RELINKY_ADMIN_HOST— Public hostname for the admin UI.
Recommended (production HTTPS):
ACME_EMAIL— Contact email used by Caddy/ACME for Let's Encrypt registration.
Optional:
RELINKY_HTTP_ONLY— Force HTTP-only mode (no TLS/cert issuance).RELINKY_ACME_STAGING— Use Let's Encrypt staging endpoint (safe for testing rate limits).RELINKY_CADDY_HTTP_PORT— Internal container HTTP port where Caddy listens.RELINKY_CADDY_HTTPS_PORT—Internal container HTTPS port where Caddy listens.RELINKY_GATEWAY_HOST_HTTP— Host port published toRELINKY_CADDY_HTTP_PORT.RELINKY_GATEWAY_HOST_HTTPS— Host port published toRELINKY_CADDY_HTTPS_PORT.RELINKY_CADDY_TLS_INTERNAL— Use Caddy internal CA/self-signed certs instead of ACME certs.CADDYFILE_PATH(default/app/caddy/Caddyfile) — Filesystem path where generated Caddyfile is written/read.
Coolify mode only (docker-compose.coolify.yml)
Required:
ADMIN_PASSWORD_HASH_B64— Base64-encoded password hash. Don't use the normalADMIN_PASSWORD_HASHwith Coolify! It mangles$symbols in env variables as of April 2026.
npm install
cp .env.example .env
npm run build
npm run dev
npm run test:specnpm run dev runs migrations/seed once (dev:prepare), then watches app/admin/backend/server.js and app/redirector/server.js (plus Vite). Do not watch start-dev.js / start.js — Node's supervisor + --watch + spawned children loops on macOS. dev:backend (no watch) still uses start-dev.js if you only need the API processes. Loads .env via --env-file-if-exists. For normal local work, uncomment the dev ADMIN_PASSWORD_HASH in .env (password dev). To test onboarding, leave it unset and start with an empty db/ directory.
Schema changes are applied automatically — no manual step. Each SQLite file tracks its own version (PRAGMA user_version) and only the missing migrations run, inside a transaction, so existing databases upgrade in place without data loss. Migrations live in app/shared/migrations/ and are run by app/shared/init-db.js; to add one, append it to the relevant file's list.
Migrations run once per deployment as a dedicated step, not inside each service: the all-in-one image runs them in start.js (and the gateway entrypoint) before launching the admin and redirector; the multi-container Coolify compose uses a one-shot relinky_migrate service that the others wait on (depends_on: condition: service_completed_successfully). The admin and redirector then only verify the schema is current on boot and exit with an error if a migration was skipped — they never migrate concurrently, which avoids SQLITE_BUSY on a shared volume.
Before upgrading a database that already has data, init-db takes a consistent snapshot (via SQLite's online backup) into a backups/ folder inside the database directory — db/backups/ by default, or $RELINKY_DB_DIR/backups/ when that override is set — named <db>.<timestamp>.v<from-version>.<pid>.db. Backups are only created when a migration is actually pending, never on a fresh install or an ordinary restart. If the backup cannot be written, startup aborts rather than migrating without a restore point. Retention is controlled by RELINKY_DB_BACKUP_KEEP.
To restore a snapshot: stop the services, replace the live file (e.g. copy db/backups/main.<…>.db over db/main.db), delete any leftover db/main.db-wal / db/main.db-shm, then start again.
After switching Node versions, rebuild the native SQLite binding: npm rebuild better-sqlite3.
- There's a lot of meta information recorded when links are redirected, all very useful for good stats and analytics. At the same time the stats view is still in its most basic form yet. That's what I wanted to focus on in the nearest future.
- It's being used in sorta 'production environment' by myself for my own personal needs, and works well, but I can't guarantee it'll work well under very heavy load, though why not — it's very simple. That's something I'd love to see some feedback on and potentially improve if needed. For example, a blazing fast front-end router could be introduced, as well as the redirector could be rewritten with something more efficient than JS. In any case I want to keep it simple, because we all know what happens to overcomplicated and overengineered software products.
- The rest depends on feedback.
- Feedback or even a worthy pull-request is priceless!
- If you use it and it helps you with your business or personal needs, I wouldn't say no to donations:
- DOGE:
D5x6svhkv63EebUSwL7DCgp1CHdX2pu1Js - BTC:
bc1qslefa243svmdnph6s53fj6u9l4aj8juhf2wrce - ETH:
0x87EdCDfD97Bd6F7D1ec2764081cd37E64127E7e9 - USDT:
0x87EdCDfD97Bd6F7D1ec2764081cd37E64127E7e9(ETH),TPRxb7XYM2szKUNgv2jNugrZ4WMYgLULPc(TRX) - Non-crypto with my music in exchange: Alvisk, Veell, Chaoskeeper @ Bandcamp
- Anything else? Get in touch.
- DOGE:
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
