Chanify alternatives and similar packages
Based on the "Messaging" category.
Alternatively, view chanify alternatives based on common mentions on social networks and blogs.
-
sarama
DISCONTINUED. Sarama is a Go library for Apache Kafka. [Moved to: https://github.com/IBM/sarama] -
Centrifugo
Scalable real-time messaging server in a language-agnostic way. Self-hosted alternative to Pubnub, Pusher, Ably. Set up once and forever. -
Benthos
DISCONTINUED. Fancy stream processing made operationally mundane [Moved to: https://github.com/redpanda-data/connect] -
APNs2
⚡ HTTP/2 Apple Push Notification Service (APNs) push provider for Go — Send push notifications to iOS, tvOS, Safari and OSX apps, using the APNs HTTP/2 protocol. -
Uniqush-Push
Uniqush is a free and open source software system which provides a unified push service for server side notification to apps on mobile devices. -
amqp
An AMQP 0-9-1 Go client maintained by the RabbitMQ team. Originally by @streadway: `streadway/amqp` -
PingMe
PingMe is a CLI which provides the ability to send messages or alerts to multiple messaging platforms & email. -
emitter
Emits events in Go way, with wildcard, predicates, cancellation possibilities and many other good wins -
Bus
🔊Minimalist message bus implementation for internal communication with zero-allocation magic on Emit -
go-mq
Declare AMQP entities like queues, producers, and consumers in a declarative way. Can be used to work with RabbitMQ. -
Ratus
Ratus is a RESTful asynchronous task queue server. It translated concepts of distributed task queues into a set of resources that conform to REST principles and provides a consistent HTTP API for various backends. -
RapidMQ
RapidMQ is a pure, extremely productive, lightweight and reliable library for managing of the local messages queue
CodeRabbit: AI Code Reviews for Developers
Do you think we are missing an alternative of Chanify or a related project?
README
Chanify
English | [简体中文](README-zh_CN.md)
Chanify is a safe and simple notification tools. For developers, system administrators, and everyone can push notifications with API.
Table of Contents Features Getting Started Installation Precompiled binary Docker From source Usage As Sender Client As Serverless node As Serverful node Add New Node Send message Command Line Python 3 Ruby NodeJS PHP HTTP API Send Text Send Link Send Image Send Audio Send File Send Actions Configuration Security Setting Registrable Token Lifetime Chrome Extension Windows Client Docker Compose Lua API Contributing License
Features
Chanify is include these features:
- Customize channel for notifications.
- Deploy your own node server.
- Distributed architecture design.
- Design for privacy protection.
- Support text/image/audio/file message format.
Getting Started
- Install iOS App(1.0.0 version and above) or macOS App(1.3.0 version and above).
- Get send token, more detail.
- Send message.
Installation
Precompiled binary
Download precompiled binary from here.
Docker
$ docker pull wizjin/chanify:latest
From source
$ git clone https://github.com/chanify/chanify.git
$ cd chanify
$ make install
Usage
As Sender Client
Use chanify to send message.
# Text message
$ chanify send --endpoint=http://<address>:<port> --token=<token> --text=<message>
# URL message
$ chanify send --endpoint=http://<address>:<port> --token=<token> --link=<web url>
# Image message
$ chanify send --endpoint=http://<address>:<port> --token=<token> --image=<image file path>
# Audio message
$ chanify send --endpoint=http://<address>:<port> --token=<token> --audio=<audio file path>
# File message
$ chanify send --endpoint=http://<address>:<port> --token=<token> --file=<file path> --text=<file description>
# Action message
$ chanify send --endpoint=http://<address>:<port> --token=<token> --text=<message> --title=<title> --action="<action name>|<action url>"
# Timeline message
$ chanify send --endpoint=http://<address>:<port> --token=<token> --timeline.code=<code> <item1>=<value1> <item2>=<value2> ...
endpoint
default value is https://api.chanify.net
, and notification will send by default server.
If you have own node server, please set endpoint
to your node server url.
As Serverless node
Chanify run in stateless mode, no device token will be stored in node.
All device token will be stored in api.chanify.net.
Message will send to apple apns server by api.chanify.net.
Send message workflow:
Start => node server => api.chanify.net => Apple server => iOS client
# Start chanify
$ chanify serve --host=<ip address> --port=<port> --secret=<secret key> --name=<node name> --endpoint=http://<address>:<port>
# Docker
$ docker run -it wizjin/chanify:latest serve --secret=<secret key> --name=<node name> --endpoint=http://<address>:<port>
As Serverful node
Chanify run in stateful mode, device token will be stored in node.
Message will send to apple apns server direct.
Send message workflow:
Start => node server => Apple server => iOS client
Start server
# Start chanify
$ chanify serve --host=<ip address> --port=<port> --name=<node name> --datapath=~/.chanify --endpoint=http://<address>:<port>
# Docker
$ docker run -it -v /my/data:/root/.chanify wizjin/chanify:latest serve --name=<node name> --endpoint=http://<address>:<port>
Use MySQL as a backend
--dburl=mysql://<user>:<password>@tcp(<ip address>:<port>)/<database name>?charset=utf8mb4&parseTime=true&loc=Local
Chanify will not create database.
Add New Node
- Start node server
- iOS client can scan QRCode(
http://<address>:<port>/
) to add node.
Send message
Command Line
# Text message
$ curl --form-string "text=hello" "http://<address>:<port>/v1/sender/<token>"
# Text file
$ cat message.txt | curl -H "Content-Type: text/plain" --data-binary @- "http://<address>:<port>/v1/sender/<token>"
Python 3
from urllib import request, parse
data = parse.urlencode({ 'text': 'hello' }).encode()
req = request.Request("http://<address>:<port>/v1/sender/<token>", data=data)
request.urlopen(req)
Ruby
require 'net/http'
uri = URI('http://<address>:<port>/v1/sender/<token>')
res = Net::HTTP.post_form(uri, 'text' => 'hello')
puts res.body
NodeJS
const https = require('https')
const querystring = require('querystring');
const data = querystring.stringify({ text: 'hello' })
const options = {
hostname: '<address>:<port>',
port: 80,
path: '/v1/sender/<token>',
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded',
'Content-Length': data.length
}
}
var req = https.request(options, (res) => {
res.on('data', (d) => {
process.stdout.write(d);
});
});
req.write(data);
req.end();
PHP
$curl = curl_init();
curl_setopt_array($curl, [
CURLOPT_URL => 'http://<address>:<port>/v1/sender/<token>',
CURLOPT_CUSTOMREQUEST => 'POST',
CURLOPT_POSTFIELDS => [ 'text' => 'hello' ],
]);
$response = curl_exec($curl);
curl_close($curl);
echo $response;
HTTP API
Send Text
GET
http://<address>:<port>/v1/sender/<token>/<message>
POST
http://<address>:<port>/v1/sender/<token>
Content-Type:
text/plain
: Body is text messagemultipart/form-data
: The block of data("text") is text messageapplication/x-www-form-urlencoded
:text=<url encoded text message>
application/json; charset=utf-8
: The fields are optionaljson { "token": "<token>", "title": "<message title>", "text": "<text message content>", "copy": "<copy text for text message>", "autocopy": 1, "sound": 1, "priority": 10, "interruptionlevel": 0, "actions": [ "ActionName1|http://<action host>/<action1>", "ActionName2|http://<action host>/<action2>", ... ], "timeline": { "code": "<timeline code>", "timestamp": 1620000000000, "items": { "key1": "value1", "key2": "value2", ... } } }
Additional params
Key | Default | Description |
---|---|---|
title | None | The title for notification message. |
copy | None | The copy text for text notification. |
autocopy | 0 |
Enable autocopy text for text notification. |
sound | 0 |
1 enable sound, otherwise disable sound. |
priority | 10 |
10 normal, 5 lower level. |
interruption-level | active |
Interruption level for timing of a notification. |
actions | None | Actions list. |
timeline | None | Timeline object. |
sound
:
- 1 enable default sound
- sound code, e.g. "bell"
interruption-level
:
active
: Lights up screen and may play a sound.passive
: Does not light up screen or play sound.time-sensitive
: Lights up screen and may play a sound; May be presented during Do Not Disturb.
timestamp
in milliseconds (timezone - UTC)
E.g.
http://<address>:<port>/v1/sender/<token>?sound=1&priority=10&title=hello©=123&autocopy=1
Overwrite Content-Type
http://<address>:<port>/v1/sender/<token>?content-type=<text|json>
Send Link
$ curl --form "link=@<web url>" "http://<address>:<port>/v1/sender/<token>"
{
"link": "<web url>",
"sound": 1,
"priority": 10,
}
Send Image
Send image only support POST method used serverful node.
- Content-Type:
image/png
ORimage/jpeg
cat <jpeg image path> | curl -H "Content-Type: image/jpeg" --data-binary @- "http://<address>:<port>/v1/sender/<token>"
- Content-Type:
multipart/form-data
$ curl --form "image=@<jpeg image path>" "http://<address>:<port>/v1/sender/<token>"
Send Audio
Send mp3 audio only support POST method used serverful node.
- Content-Type:
audio/mpeg
cat <mp3 audio path> | curl -H "Content-Type: audio/mpeg" --data-binary @- "http://<address>:<port>/v1/sender/<token>"
- Content-Type:
multipart/form-data
$ curl --form "audio=@<mp3 audio path>" "http://<address>:<port>/v1/sender/<token>"
Send File
Send file only support POST method used serverful node.
- Content-Type:
multipart/form-data
$ curl --form "file=@<file path>" "http://<address>:<port>/v1/sender/<token>"
Send Actions
Send Actions (Up to 4 actions).
- Content-Type:
multipart/form-data
$ curl --form "action=ActionName1|http://<action host>/<action1>" "http://<address>:<port>/v1/sender/<token>"
URL for script action:
chanify://action/run-script/<scipt name>?<arg name 1>=<arg value 1>&<arg name 2>=<arg value 2>
Configuration
Chanify can be configured with a yml format file, and the default path is ~/.chanify.yml
.
server:
host: 0.0.0.0 # Listen ip address
port: 8080 # Listen port
endpoint: http://my.server/path # Endpoint URL
name: Node name # Name for node server
secret: <secret code> # key for serverless node server
datapath: <data path> # data storage path for serverful node server
# pluginpath: <plugin path> # plugin path for lua
dburl: mysql://root:test@tcp(127.0.0.1:3306)/chanify?charset=utf8mb4&parseTime=true&loc=Local # database dsn for serverful node server
http:
- readtimeout: 10s # 10 seconds for http read timeout
- writetimeout: 10s # 10 seconds for http write timeout
register:
enable: false # Disable user register
whitelist: # whitelist for user register
- <user id 1>
- <user id 2>
# plugin:
# webhook:
# - name: github # POST http://my.server/path/v1/webhook/github/<token>
# file: webhook/github.lua # <pluginpath>/webhook/github.lua
# env:
# secret_token: "secret token"
client: # configuration for sender client
sound: 1 # enable sound
endpoint: <default node server endpoint>
token: <default token>
interruption-level: <interruption level>
Security
Setting Registrable
Node server can be disabled user registration and become a private server.
chanify serve --registerable=false --whitelist=<user1 id>,<user2 id>
--registerable=false
: used to disable user registrationwhitelist
: list users can be add into node server
Token Lifetime
- Token lifetime is about 90 days (default).
- Can configurable token lifetime (1 day ~ 5 years) in channel detail page.
If your token is leaked, add leaked token into the blocklist (iOS client settings).
Note: Please protect your token from leakage. The blockist need trusted node server (1.1.9 version and above).
Chrome Extension
Download the extension from Chrome web store.
Extension features:
- Send select
text/image/audio/url
message to Chanify - Send page url to Chanify
Windows Client
Get the Windows Client from here.
Windows Client features:
- Support Chanify to the Windows
Send To
Menu. - Support send message with
CLI
.
Docker Compose
- Install docker compose.
- Edit configuration file (
docker-compose.yml
). - Start docker compose:
docker-compose up -d
docker-compose.yml
:
version: "3"
services:
web:
image: nginx:alpine
restart: always
volumes:
- <workdir>/nginx.conf:/etc/nginx/nginx.conf
- <workdir>/ssl:/ssl
ports:
- 80:80
- 443:443
chanify:
image: wizjin/chanify:dev
restart: always
volumes:
- <workdir>/data:/data
- <workdir>/chanify.yml:/root/.chanify.yml
Key | Description |
---|---|
workdir | Work directory for node server. |
<workdir>/nginx.conf
:
user nginx;
worker_processes auto;
error_log /var/log/nginx/error.log warn;
pid /var/run/nginx.pid;
events {
worker_connections 1024;
}
http {
include /etc/nginx/mime.types;
default_type application/octet-stream;
log_format main '$remote_addr - $remote_user [$time_local] "$request" '
'$status $body_bytes_sent "$http_referer" '
'"$http_user_agent" "$http_x_forwarded_for"';
access_log /var/log/nginx/access.log main;
server_tokens off;
autoindex off;
sendfile on;
tcp_nopush on;
tcp_nodelay on;
keepalive_timeout 10;
server {
listen 80;
server_name <hostname or ip>;
access_log off;
return 301 https://$host$request_uri;
}
server {
listen 443 ssl http2;
server_name <hostname or ip>;
ssl_certificate /ssl/<ssl key>.crt;
ssl_certificate_key /ssl/<ssl key>.key;
ssl_protocols TLSv1.2 TLSv1.3;
ssl_ciphers HIGH:!aNULL:!MD5;
keepalive_timeout 30;
charset UTF-8;
access_log off;
location / {
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-Proto $scheme;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_pass http://chanify:8080/;
}
}
}
Key | Description |
---|---|
hostname or ip | Internet hostname or ip for node server. |
ssl key | SSL key file for node server. |
<workdir>/chanify.yml
:
server:
endpoint: https://<hostname or ip>
host: 0.0.0.0
port: 80
name: <node name>
datapath: /data
register:
enable: false
whitelist: # whitelist
- <user id>
Key | Description |
---|---|
hostname or ip | Internet hostname or ip for node server. |
node name | Name for node server. |
user id | User ids for whitelist. |
Lua API
Usage
local hex = require "hex"
local bytes = hex.decode(hex_string)
local text = hex.encode(bytes_data)
local json = require "json"
local obj = json.decode(json_string)
local str = json.encode(json_object)
local crypto = require "crypto"
local is_equal = crypto.equal(mac1, mac2)
local mac = crypto.hmac("sha1", key, message) -- Support md5 sha1 sha256
-- Http request
local req = ctx:request()
local token_string = req:token()
local http_url = req:url()
local body_string = req:body()
local query_value = req:query("key")
local header_value = req:header("key")
-- Send message
ctx:send({
title = "message title", -- Optional
text = "message body",
sound = "sound or not", -- Optional
copy = "copy", -- Optional
autocopy = "autocopy", -- Optional
})
Example: [Github webhook event](plugin/webhook/github.lua)
Contributing
Contributions are what make the open source community such an amazing place to be learn, inspire, and create. Any contributions you make are greatly appreciated.
- Fork the Project
- Change to dev Branch (
git checkout dev
) - Create your Feature Branch (
git checkout -b feature/AmazingFeature
) - Commit your Changes (
git commit -m 'Add some AmazingFeature'
) - Push to the Branch (
git push origin feature/AmazingFeature
) - Open a Pull Request (merge to
chanify:dev
branch)
License
Distributed under the MIT License. See [LICENSE
](LICENSE) for more information.
*Note that all licence references and agreements mentioned in the Chanify README section above
are relevant to that project's source code only.