Notir is a lightweight WebSocket server built with Rust using the Salvo web framework and Tokio. It allows users to connect via WebSockets, subscribe to a real-time message feed, and publish messages to other connected clients.
- WebSocket communication for real-time messaging.
- Simple publish/subscribe model.
- Containerized with Docker for easy deployment.
It has been deployed on the public server, you can try it out right away:
http://notir.fornecode.com:5800?id=${uuid}
Please change uuid to whatever you want, and now you can publish
messages to the server like this:
# Single mode - Point-to-point messaging
curl -X POST http://notir.fornecode.com:5800/single/pub?id=${uuid} \
-H 'Content-Type: application/json' \
-d '{"msg": "hello world"}'
# Single mode with PingPong - Two-way communication
curl -X POST http://notir.fornecode.com:5800/single/pub?id=${uuid}&mode=ping_pong \
-H 'Content-Type: application/json' \
-d '{"msg": "hello world"}'
# Broadcast mode - Message to all subscribers of a channel
curl -X POST http://notir.fornecode.com:5800/broad/pub?id=${uuid} \
-H 'Content-Type: application/json' \
-d '{"msg": "broadcast message"}'The easiest way to run Notir is by using the pre-built Docker image available
on GitHub Container Registry.
docker run -d -p 5800:5800 --name notir ghcr.io/timzaak/notir:latest-
WS /single/sub?id=<user_id>:- Establishes a WebSocket connection for a user to subscribe to messages.
- Query Parameters:
id(required): A unique string identifier for the client. Cannot be empty.
- Upgrades the connection to WebSocket. Messages from other users will be pushed to this WebSocket connection.
- Supports bidirectional communication and heartbeat mechanism.
-
POST /single/pub?id=<user_id>&mode=<Mode>:- Publishes a message to a specific connected client.
- Query Parameters:
id(required): The unique string identifier of the target client. Cannot be empty.mode(optional): The mode of communication. Can beshotorping_pong, defaults toshot.shot: One-way message delivery, no response expected.ping_pong: Two-way communication, waits for client response within 5 seconds.
- Request Body: The message content.
- If the
Content-Typeheader isapplication/jsonor starts withtext/(e.g.,text/plain), the message is treated as aUTF-8text message. - Otherwise, the message is treated as binary.
- If the
- Responses:
200 OK: If the message was successfully sent to the target user's channel.400 Bad Request: If theidquery parameter is missing or empty, or if atext/*body contains invalid UTF-8.404 Not Found: If the specifieduser_idis not currently connected.408 Request Timeout: If usingping_pongmode and no response received within 5 seconds.
-
WS /broad/sub?id=<broadcast_id>:- Establishes a WebSocket connection to subscribe to broadcast messages for a specific channel.
- Query Parameters:
id(required): The broadcast channel identifier. Cannot be empty.
- Multiple clients can subscribe to the same broadcast channel.
- Only receives messages from
broad/pub, ignores client-sent messages (except pong responses). - Supports heartbeat mechanism for connection health monitoring.
-
POST /broad/pub?id=<broadcast_id>:- Broadcasts a message to all clients subscribed to the specified channel.
- Query Parameters:
id(required): The broadcast channel identifier. Cannot be empty.
- Request Body: The message content.
- If the
Content-Typeheader isapplication/jsonor starts withtext/(e.g.,text/plain), the message is treated as aUTF-8text message. - Otherwise, the message is treated as binary.
- If the
- Responses:
200 OK: Always returns success, regardless of whether there are active subscribers.400 Bad Request: If theidquery parameter is missing or empty, or if atext/*body contains invalid UTF-8.
GET /health: Health check endpoint, returns200 OKif the service is running.GET /version: Returns the current version of the service.
This project is dual-licensed under either:
- Apache License 2.0 (LICENSE-APACHE or http://www.apache.org/licenses/LICENSE-2.0)
- MIT License (LICENSE-MIT or http://opensource.org/licenses/MIT)
You may choose either license at your option.