# gorush A push notification micro server using [Gin](https://github.com/gin-gonic/gin) framework written in Go (Golang). [![GoDoc](https://godoc.org/github.com/appleboy/gorush?status.svg)](https://godoc.org/github.com/appleboy/gorush) [![Build Status](http://drone.wu-boy.com/api/badges/appleboy/gorush/status.svg)](http://drone.wu-boy.com/appleboy/gorush) [![codecov](https://codecov.io/gh/appleboy/gorush/branch/master/graph/badge.svg)](https://codecov.io/gh/appleboy/gorush) [![Go Report Card](https://goreportcard.com/badge/github.com/appleboy/gorush)](https://goreportcard.com/report/github.com/appleboy/gorush) [![codebeat badge](https://codebeat.co/badges/0a4eff2d-c9ac-46ed-8fd7-b59942983390)](https://codebeat.co/projects/github-com-appleboy-gorush) [![](https://images.microbadger.com/badges/image/appleboy/gorush.svg)](https://microbadger.com/images/appleboy/gorush "Get your own image badge on microbadger.com") ## Contents - [Support Platform](#support-platform) - [Features](#features) - [Basic Usage](#basic-usage) - [Download a binary](#download-a-binary) - [Command Usage](#command-usage) - [Send Android notification](#send-android-notification) - [Send iOS notification](#send-ios-notification) - [Run gorush web server](#run-gorush-web-server) - [Web API](#web-api) - [GET /api/stat/go](#get-apistatgo) - [GET /api/stat/app](#get-apistatapp) - [GET /sys/stats](#get-sysstats) - [POST /api/push](#post-apipush) - [Request body](#request-body) - [iOS alert payload](#ios-alert-payload) - [Android notification payload](#android-notification-payload) - [iOS Example](#ios-example) - [Android Example](#Android-Example) - [Response body](#Response-body) - [Run gorush in Docker](#run-gorush-in-docker) - [License](#license) ## Support Platform * [APNS](https://developer.apple.com/library/ios/documentation/networkinginternet/conceptual/remotenotificationspg/Chapters/ApplePushService.html) * [GCM](https://developer.android.com/google/gcm/index.html) ## Features * Support [Google Cloud Message](https://developers.google.com/cloud-messaging/) using [go-gcm](https://github.com/google/go-gcm) library for Android. * Support [HTTP/2](https://http2.github.io/) Apple Push Notification Service using [apns2](https://github.com/sideshow/apns2) library. * Support [YAML](https://github.com/go-yaml/yaml) configuration. * Support command line to send single Android or iOS notification. * Support Web API to send push notification. * Support zero downtime restarts for go servers using [endless](https://github.com/fvbock/endless). * Support [HTTP/2](https://http2.github.io/) or HTTP/1.1 protocol. * Support notification queue and multiple workers. * Support `/api/stat/app` show notification success and failure counts. * Support `/api/config` show your [YAML](https://en.wikipedia.org/wiki/YAML) config. * Support store app stat to memory, [Redis](http://redis.io/), [BoltDB](https://github.com/boltdb/bolt), [BuntDB](https://github.com/tidwall/buntdb) or [LevelDB](https://github.com/syndtr/goleveldb). * Support `p12` or `pem` formtat of iOS certificate file. * Support `/sys/stats` show response time, status code count, etc. * Support for HTTP proxy to Google server (GCM). * Support retry send notification if server response is fail. See the [YAML config example](config/config.yml): ```yaml core: port: "8088" worker_num: 8 queue_num: 8192 max_notification: 100 mode: "release" ssl: false cert_path: "cert.pem" key_path: "key.pem" http_proxy: "" # only working for GCM server pid: enabled: true path: "gorush.pid" override: true api: push_uri: "/api/push" stat_go_uri: "/api/stat/go" stat_app_uri: "/api/stat/app" config_uri: "/api/config" sys_stat_uri: "/sys/stats" android: enabled: true apikey: "YOUR_API_KEY" max_retry: 0 # resend fail notification, default value zero is disabled ios: enabled: false key_path: "key.pem" password: "" # certificate password, default as empty string. production: false max_retry: 0 # resend fail notification, default value zero is disabled log: format: "string" # string or json access_log: "stdout" # stdout: output to console, or define log path like "log/access_log" access_level: "debug" error_log: "stderr" # stderr: output to console, or define log path like "log/error_log" error_level: "error" hide_token: true stat: engine: "memory" # support memory, redis, boltdb, buntdb or leveldb redis: addr: "localhost:6379" password: "" db: 0 boltdb: path: "bolt.db" bucket: "gorush" buntdb: path: "bunt.db" leveldb: path: "level.db" ``` ## Basic Usage How to send push notification using `gorush` command? (Android or iOS) ### Download a binary The pre-compiled binaries can be downloaded from [release page](https://github.com/appleboy/gorush/releases). With `Go` installed ``` $ go get -u github.com/appleboy/gorush ``` On linux ``` $ wget https://github.com/appleboy/gorush/releases/download/v1.6.5/gorush-v1.6.6-linux-amd64 -O gorush ``` On OS X ``` $ wget https://github.com/appleboy/gorush/releases/download/v1.6.5/gorush-v1.6.6-darwin-amd64 -O gorush ``` ### Command Usage ``` ________ .__ / _____/ ____ _______ __ __ ______| |__ / \ ___ / _ \\_ __ \| | \/ ___/| | \ \ \_\ \( <_> )| | \/| | /\___ \ | Y \ \______ / \____/ |__| |____//____ >|___| / \/ \/ \/ Usage: gorush [options] Server Options: -p, --port Use port for clients (default: 8088) -c, --config Configuration file -m, --message Notification message -t, --token Notification token --proxy Proxy URL (only for GCM) iOS Options: -i, --key certificate key file path -P, --password certificate key password --topic iOS topic --ios enabled iOS (default: false) --production iOS production mode (default: false) Android Options: -k, --apikey Android API Key --android enabled android (default: false) Common Options: -h, --help Show this message -v, --version Show version ``` ### Send Android notification Send single notification with the following command. ```bash $ gorush -android -m="your message" -k="API Key" -t="Device token" ``` * `-m`: Notification message. * `-k`: [Google cloud message](https://developers.google.com/cloud-messaging/) api key * `-t`: Device token. * `--proxy`: Set http proxy url. (only working for GCM) ### Send iOS notification Send single notification with the following command. ```bash $ gorush -ios -m="your message" -i="your certificate path" -t="device token" -topic="apns topic" ``` * `-m`: Notification message. * `-i`: Apple Push Notification Certificate path (`pem` or `p12` file). * `-t`: Device token. * `-topic`: The topic of the remote notification. * `-password`: The certificate password. The default endpoint is APNs development. Please add `-production` flag for APNs production push endpoint. ```bash $ gorush -ios -m="your message" -i="your certificate path" -t="device token" -production ``` ## Run gorush web server Please make sure your [config.yml](config/config.yml) exist. Default port is `8088`. ```bash $ gorush -c config.yml ``` Get go status of api server using [httpie](https://github.com/jkbrzt/httpie) tool: ```bash $ http -v --verify=no --json GET https://localhost:8088/api/stat/go ``` ## Web API Gorush support the following API. * **GET** `/api/stat/go` Golang cpu, memory, gc, etc information. Thanks for [golang-stats-api-handler](https://github.com/fukata/golang-stats-api-handler). * **GET** `/api/stat/app` show notification success and failure counts. * **GET** `/api/config` show server yml config file. * **POST** `/api/push` push ios and android notifications. ### GET /api/stat/go Golang cpu, memory, gc, etc information. Response with `200` http status code. ```json { "time": 1460686815848046600, "go_version": "go1.6.1", "go_os": "darwin", "go_arch": "amd64", "cpu_num": 4, "goroutine_num": 15, "gomaxprocs": 4, "cgo_call_num": 1, "memory_alloc": 7455192, "memory_total_alloc": 8935464, "memory_sys": 12560632, "memory_lookups": 17, "memory_mallocs": 31426, "memory_frees": 11772, "memory_stack": 524288, "heap_alloc": 7455192, "heap_sys": 8912896, "heap_idle": 909312, "heap_inuse": 8003584, "heap_released": 0, "heap_objects": 19654, "gc_next": 9754725, "gc_last": 1460686815762559700, "gc_num": 2, "gc_per_second": 0, "gc_pause_per_second": 0, "gc_pause": [ 0.326576, 0.227096 ] } ``` ### GET /api/stat/app Show success or failure counts information of notification. ```json { "version": "v1.6.2", "queue_max": 8192, "queue_usage": 0, "total_count": 77, "ios": { "push_success": 19, "push_error": 38 }, "android": { "push_success": 10, "push_error": 10 } } ``` ### GET /sys/stats Show response time, status code count, etc. ```json { "pid": 80332, "uptime": "1m42.428010614s", "uptime_sec": 102.428010614, "time": "2016-06-26 12:27:11.675973571 +0800 CST", "unixtime": 1466915231, "status_code_count": { }, "total_status_code_count": { "200": 5 }, "count": 0, "total_count": 5, "total_response_time": "10.422441ms", "total_response_time_sec": 0.010422441000000001, "average_response_time": "2.084488ms", "average_response_time_sec": 0.0020844880000000002 } ``` ### POST /api/push Simple send iOS notification example, the `platform` value is `1`: ```json { "notifications": [ { "tokens": ["token_a", "token_b"], "platform": 1, "message": "Hello World iOS!" } ] } ``` Simple send Android notification example, the `platform` value is `2`: ```json { "notifications": [ { "tokens": ["token_a", "token_b"], "platform": 2, "message": "Hello World Android!" } ] } ``` Send multiple notifications as below: ```json { "notifications": [ { "tokens": ["token_a", "token_b"], "platform": 1, "message": "Hello World iOS!" }, { "tokens": ["token_a", "token_b"], "platform": 2, "message": "Hello World Android!" }, { "tokens": ["token_a", "token_b"], "platform": 2, "message": "Hello World!" }, ..... ] } ``` See more example about [iOS](#ios-example) or [Android](#android-example). ### Request body Request body must has a notifications array. The following is a parameter table for each notification. | name | type | description | required | note | |-------------------------|--------------|---------------------------------------------------------------------------------------------------|----------|---------------------------------------------------------------| | tokens | string array | device tokens | o | | | platform | int | platform(iOS,Android) | o | 1=iOS, 2=Android | | message | string | message for notification | o | | | title | string | notification title | - | | | priority | string | Sets the priority of the message. | - | `normal` or `high` | | content_available | bool | data messages wake the app by default. | - | | | sound | string | sound type | - | | | data | string array | extensible partition | - | | | retry | int | retry send notification if fail response from server. Value must be small than `max_retry` field. | - | | | api_key | string | Android api key | - | only Android | | to | string | The value must be a registration token, notification key, or topic. | - | only Android | | collapse_key | string | a key for collapsing notifications | - | only Android | | delay_while_idle | bool | a flag for device idling | - | only Android | | time_to_live | uint | expiration of message kept on GCM storage | - | only Android | | restricted_package_name | string | the package name of the application | - | only Android | | dry_run | bool | allows developers to test a request without actually sending a message | - | only Android | | notification | string array | payload of a GCM message | - | only Android. See the [detail](#android-notification-payload) | | expiration | int | expiration for notification | - | only iOS | | apns_id | string | A canonical UUID that identifies the notification | - | only iOS | | topic | string | topic of the remote notification | - | only iOS | | badge | int | badge count | - | only iOS | | category | string | the UIMutableUserNotificationCategory object | - | only iOS | | alert | string array | payload of a iOS message | - | only iOS. See the [detail](#ios-alert-payload) | ### iOS alert payload | name | type | description | required | note | |----------------|------------------|--------------------------------------------------------------------------------------------------|----------|------| | subtitle | string | Apple Watch & Safari display this string as part of the notification interface. | - | | | action | string | The label of the action button. This one is required for Safari Push Notifications. | - | | | action-loc-key | string | If a string is specified, the system displays an alert that includes the Close and View buttons. | - | | | launch-image | string | The filename of an image file in the app bundle, with or without the filename extension. | - | | | loc-args | array of strings | Variable string values to appear in place of the format specifiers in loc-key. | - | | | loc-key | string | A key to an alert-message string in a Localizable.strings file for the current localization. | - | | | title-loc-args | array of strings | Variable string values to appear in place of the format specifiers in title-loc-key. | - | | | title-loc-key | string | The key to a title string in the Localizable.strings file for the current localization. | - | | See more detail about [APNs Remote Notification Payload](https://developer.apple.com/library/ios/documentation/NetworkingInternet/Conceptual/RemoteNotificationsPG/Chapters/TheNotificationPayload.html#//apple_ref/doc/uid/TP40008194-CH107-SW1). ### Android notification payload | name | type | description | required | note | |----------------|--------|-----------------------------------------------------------------------------------------------------------|----------|------| | icon | string | Indicates notification icon. | - | | | tag | string | Indicates whether each notification message results in a new entry on the notification center on Android. | - | | | color | string | Indicates color of the icon, expressed in #rrggbb format | - | | | click_action | string | The action associated with a user click on the notification. | - | | | body_loc_key | string | Indicates the key to the body string for localization. | - | | | body_loc_args | string | Indicates the string value to replace format specifiers in body string for localization. | - | | | title_loc_key | string | Indicates the key to the title string for localization. | - | | | title_loc_args | string | Indicates the string value to replace format specifiers in title string for localization. | - | | See more detail about [GCM server reference](https://developers.google.com/cloud-messaging/http-server-ref#send-downstream). ### iOS Example Send normal notification. ```json "notifications": [ { "tokens": ["token_a", "token_b"], "platform": 1, "message": "Hello World iOS!", "title": "You got message" } ] ``` The app icon be badged with the number `9` and that a bundled alert sound be played when the notification is delivered. ```json "notifications": [ { "tokens": ["token_a", "token_b"], "platform": 1, "message": "Hello World iOS!", "title": "You got message", "badge": 9, "sound": "bingbong.aiff" } ] ``` Add other fields which user defined via `data` field. ```json "notifications": [ { "tokens": ["token_a", "token_b"], "platform": 1, "message": "Hello World iOS!", "title": "You got message", "data": { "key1": "welcome", "key2": 2 } } ] ``` ### Android Example Send normal notification. ```json "notifications": [ { "tokens": ["token_a", "token_b"], "platform": 2, "message": "Hello World Android!", "title": "You got message" } ] ``` Add `notification` payload. ```json "notifications": [ { "tokens": ["token_a", "token_b"], "platform": 2, "message": "Hello World Android!", "title": "You got message", "notification" : { "icon": "myicon", "color": "#112244" } } ] ``` Add other fields which user defined via `data` field. ```json "notifications": [ { "tokens": ["token_a", "token_b"], "platform": 2, "message": "Hello World Android!", "title": "You got message", "data": { "Nick" : "Mario", "body" : "great match!", "Room" : "PortugalVSDenmark" } } ] ``` ### Response body Error response message table: | status code | message | |-------------|--------------------------------------------| | 400 | Missing `notifications` field. | | 400 | Notifications field is empty. | | 400 | Number of notifications(50) over limit(10) | Success response: ```json { "success": "ok" } ``` ## Run gorush in Docker Set up `gorush` in the cloud in under 5 minutes with zero knowledge of Golang or Linux shell using our [gorush Docker image](https://hub.docker.com/r/appleboy/gorush/). ```bash $ docker pull appleboy/gorush $ docker run --name gorush -p 80:8088 appleboy/gorush ``` Run `gorush` with your own config file. ```bash $ docker pull appleboy/gorush $ docker run --name gorush -v ${PWD}/config.yml:/config.yml -p 80:8088 appleboy/gorush ``` Testing your gorush server using [httpie](https://github.com/jkbrzt/httpie) command. ```bash $ http -v --verify=no --json GET http://your.docker.host/api/stat/go ``` ![statue screenshot](screenshot/status.png) ## License Copyright 2016 Bo-Yi Wu [@appleboy](https://twitter.com/appleboy). Licensed under the MIT License.