Prism sits between your services and your phone. Services send HTTP notifications; Prism delivers them to Signal, Telegram or WebPush. Prism is ntfy-compatible, so existing integrations work without changes. It can optionally monitor a Proton Mail inbox and send a notification for new emails.
Android companion app: prism-android
curl -L -O https://raw.githubusercontent.com/lone-cloud/prism/master/.env.example
mv .env.example .env
nano .env # Set API_KEY=your-secret-key-here
curl -L -O https://raw.githubusercontent.com/lone-cloud/prism/master/docker-compose.yml
docker compose up -dcurl -L -O https://github.com/lone-cloud/prism/releases/latest/download/prism-linux-amd64
chmod +x prism-linux-amd64
mv prism-linux-amd64 prism
curl -L -O https://raw.githubusercontent.com/lone-cloud/prism/master/.env.example
mv .env.example .env
nano .env # Set API_KEY=your-secret-key-here
./prismPrism is now running at http://localhost:8080.
Deploy behind HTTPS. Every API request sends your API_KEY in the Authorization header. Over plain HTTP that header is transmitted in cleartext — anyone who can observe the traffic between your callers and the server can read the key and make authenticated requests. Use a reverse proxy with TLS termination (Caddy, nginx, Traefik) or a tunnel service like Cloudflare Tunnel in front of Prism.
Only use http URLs when callers run on the same host and traffic never leaves the machine.
All integrations are configured through the web UI. Authenticate with your API_KEY as the password (username can be anything).
Send notifications through Signal Messenger.
Setup:
- Visit http://localhost:8080 and authenticate with your API_KEY
- Expand the Signal integration card
- Click "Link Device"
- Scan the QR code with Signal on your phone:
- Open Signal → Settings → Linked Devices → Link New Device
- Scan the displayed QR code
- Your device will link automatically
Note: Binary installs require signal-cli in your PATH. Docker includes it automatically.
Send notifications through a Telegram bot.
Setup:
-
Create a bot:
- Message @BotFather on Telegram
- Send
/newbotand follow the prompts - Copy the bot token
-
Get your Chat ID:
- Message @userinfobot on Telegram
- Copy your Chat ID from the response
-
Configure in Prism:
- Visit http://localhost:8080 and authenticate with your API_KEY
- Expand the Telegram integration card
- Enter your bot token and chat ID
- Click "Configure"
Monitor a Proton Mail account and forward new emails as notifications through Signal or Telegram.
Setup:
- Visit http://localhost:8080 and authenticate with your API_KEY
- Configure Signal or Telegram first (required for routing)
- Expand the Proton Mail integration card
- Enter your Proton Mail credentials:
- Email address
- Password
- 2FA code (if enabled)
- Click "Link"
New emails appear as notifications from the "Proton Mail" app. Credentials are encrypted (AES-256-GCM) and tokens refresh automatically.
Send notifications directly to your browser.
Setup:
- Visit http://localhost:8080 and authenticate with your API_KEY
- Allow browser notifications when prompted
- Apps without Signal or Telegram configured will automatically use WebPush
All API endpoints require authentication with your API key:
Authorization: Bearer YOUR_API_KEYPublish notifications to POST /{appName}.
{appName} is the target app/topic name. Messages are routed to all subscriptions configured for that app (Signal, Telegram, WebPush).
JSON payload:
curl -X POST http://localhost:8080/my-app \
-H "Authorization: Bearer YOUR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"title": "Alert", "message": "Something happened"}'JSON payload with image:
curl -X POST http://localhost:8080/my-app \
-H "Authorization: Bearer YOUR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"title": "Motion Detected", "message": "Front door", "attach": "https://example.com/snapshot.jpg"}'Plain text payload:
curl -X POST http://localhost:8080/my-app \
-H "Authorization: Bearer YOUR_API_KEY" \
-d "Simple message text"Register and remove WebPush subscriptions for an app.
Creates a WebPush subscription.
Required fields:
appNamepushEndpoint
Optional encrypted payload fields (must be provided together):
p256dhauthvapidPrivateKey
curl -X POST http://localhost:8080/api/v1/webpush/subscriptions \
-H "Authorization: Bearer YOUR_API_KEY" \
-H "Content-Type: application/json" \
-d '{
"appName": "my-app",
"pushEndpoint": "https://example.push.service/send/abc123",
"p256dh": "BASE64URL_P256DH",
"auth": "BASE64URL_AUTH",
"vapidPrivateKey": "BASE64URL_VAPID_PRIVATE_KEY"
}'Minimal registration (without encrypted payload fields):
curl -X POST http://localhost:8080/api/v1/webpush/subscriptions \
-H "Authorization: Bearer YOUR_API_KEY" \
-H "Content-Type: application/json" \
-d '{
"appName": "my-app",
"pushEndpoint": "http://localhost:9001/mock-push"
}'Removes a WebPush subscription by ID.
curl -X DELETE http://localhost:8080/api/v1/webpush/subscriptions/SUBSCRIPTION_ID \
-H "Authorization: Bearer YOUR_API_KEY"Add to configuration.yaml:
notify:
- platform: rest
name: Prism
resource: "http://<Your Prism server network IP>/Home Assistant"
method: POST_JSON
headers:
Authorization: !secret prism_api_key
data_template:
title: "{{ title }}"
message: "{{ message }}"
image: "{{ data.image | default('') }}"Add to secrets.yaml:
prism_api_key: "Bearer YOUR_API_KEY_HERE"Then use the notify.prism action in automations.
Sending an image from a camera snapshot:
Take a snapshot first, save it to HA's local static file server, then send the URL:
actions:
- action: camera.snapshot
target:
entity_id: camera.front_door
data:
filename: /config/www/snapshot_front.jpg
- action: notify.prism
data:
title: "Motion Detected"
message: "Front door camera triggered"
data:
image: https://<your-ha-domain>/local/snapshot_front.jpgThe /config/www/ directory is served as /local/ by Home Assistant's built-in HTTP server. Use your HA's external HTTPS URL so Prism can fetch the image when delivering the notification.
In Beszel's Settings → Notifications, add:
ntfy://:YOUR_API_KEY@<prism-host>:<port>/Beszel?disableTLS=yes
disableTLS=yes is only needed for local HTTP. The app name (Beszel) can be anything.
Public. Returns 200 OK when running.
curl http://localhost:8080/healthAuthenticated. Returns uptime and integration status:
curl http://localhost:8080/api/v1/health \
-H "Authorization: Bearer YOUR_API_KEY"{
"version": "1.2.0",
"uptime": "2h15m",
"signal": {"linked": true, "account": "+1234567890"},
"telegram": {"linked": true, "account": "123456789"},
"proton": {"linked": true, "account": "user@proton.me"}
}

