{"id":13394987,"url":"https://github.com/rpush/rpush","last_synced_at":"2025-05-14T22:06:34.438Z","repository":{"id":6177939,"uuid":"7408076","full_name":"rpush/rpush","owner":"rpush","description":"The push notification service for Ruby.","archived":false,"fork":false,"pushed_at":"2025-04-09T13:20:07.000Z","size":27727,"stargazers_count":2195,"open_issues_count":35,"forks_count":379,"subscribers_count":53,"default_branch":"master","last_synced_at":"2025-05-07T21:59:12.600Z","etag":null,"topics":["apns","apns2","fcm","firebase-cloud-messaging","hacktoberfest","notification-service","notifications","push-notifications","pushy","rails","rpush","ruby","webpush","windows-phone","wns"],"latest_commit_sha":null,"homepage":"","language":"Ruby","has_issues":true,"has_wiki":null,"has_pages":null,"mirror_url":null,"source_name":null,"license":"mit","status":null,"scm":"git","pull_requests_enabled":true,"icon_url":"https://github.com/rpush.png","metadata":{"files":{"readme":"README.md","changelog":"CHANGELOG.md","contributing":null,"funding":null,"license":"LICENSE","code_of_conduct":null,"threat_model":null,"audit":null,"citation":null,"codeowners":null,"security":null,"support":null,"governance":null,"roadmap":null,"authors":null,"dei":null,"publiccode":null,"codemeta":null,"zenodo":null}},"created_at":"2013-01-02T14:46:47.000Z","updated_at":"2025-04-21T05:02:41.000Z","dependencies_parsed_at":"2024-01-31T05:09:45.396Z","dependency_job_id":"fa870f22-76bb-4374-bcb8-7ebd1b197f29","html_url":"https://github.com/rpush/rpush","commit_stats":{"total_commits":1531,"total_committers":135,"mean_commits":11.34074074074074,"dds":"0.40431090790333113","last_synced_commit":"fe0decac1dcb2ef983773e3d8ceefb91bc455093"},"previous_names":[],"tags_count":47,"template":false,"template_full_name":null,"repository_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/rpush%2Frpush","tags_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/rpush%2Frpush/tags","releases_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/rpush%2Frpush/releases","manifests_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories/rpush%2Frpush/manifests","owner_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners/rpush","download_url":"https://codeload.github.com/rpush/rpush/tar.gz/refs/heads/master","host":{"name":"GitHub","url":"https://github.com","kind":"github","repositories_count":254235695,"owners_count":22036963,"icon_url":"https://github.com/github.png","version":null,"created_at":"2022-05-30T11:31:42.601Z","updated_at":"2022-07-04T15:15:14.044Z","host_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub","repositories_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repositories","repository_names_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/repository_names","owners_url":"https://repos.ecosyste.ms/api/v1/hosts/GitHub/owners"}},"keywords":["apns","apns2","fcm","firebase-cloud-messaging","hacktoberfest","notification-service","notifications","push-notifications","pushy","rails","rpush","ruby","webpush","windows-phone","wns"],"created_at":"2024-07-30T17:01:38.331Z","updated_at":"2025-05-14T22:06:34.414Z","avatar_url":"https://github.com/rpush.png","language":"Ruby","readme":"[![Gem Version](https://badge.fury.io/rb/rpush.svg)](http://badge.fury.io/rb/rpush)\n[![RPush Test](https://github.com/rpush/rpush/actions/workflows/test.yml/badge.svg)](https://github.com/rpush/rpush/actions/workflows/test.yml)\n[![Test Coverage](https://codeclimate.com/github/rpush/rpush/badges/coverage.svg)](https://codeclimate.com/github/rpush/rpush)\n[![Code Climate](https://codeclimate.com/github/rpush/rpush/badges/gpa.svg)](https://codeclimate.com/github/rpush/rpush)\n[![Join the chat at https://gitter.im/rpush/rpush](https://badges.gitter.im/Join%20Chat.svg)](https://gitter.im/rpush/rpush?utm_source=badge\u0026utm_medium=badge\u0026utm_campaign=pr-badge\u0026utm_content=badge)\n\n\u003cimg src=\"https://raw.github.com/rpush/rpush/master/logo.png\" align=\"right\" width=\"200px\" /\u003e\n\n### Rpush. The push notification service for Ruby.\n\nRpush aims to be the *de facto* gem for sending push notifications in Ruby. Its core goals are ease of use, reliability and a rich feature set. Rpush provides numerous advanced features not found in others gems, giving you greater control \u0026 insight as your project grows. These are a few of the reasons why companies worldwide rely on Rpush to deliver their notifications.\n\n#### Supported Services\n\n  * [**Apple Push Notification Service**](#apple-push-notification-service)\n    * Including Safari Push Notifications.\n  * [**Firebase Cloud Messaging**](#firebase-cloud-messaging) (used to be Google Cloud Messaging)\n  * [**Amazon Device Messaging**](#amazon-device-messaging)\n  * [**Windows Phone Push Notification Service**](#windows-phone-notification-service)\n  * [**Pushy**](#pushy)\n  * [**Webpush**](#webpush)\n\n#### Feature Highlights\n\n* Use [**ActiveRecord**](https://github.com/rpush/rpush/wiki/Using-ActiveRecord) or [**Redis**](https://github.com/rpush/rpush/wiki/Using-Redis) for storage.\n* Plugins for [**Bugsnag**](https://github.com/rpush/rpush-plugin-bugsnag),\n[**Sentry**](https://github.com/rpush/rpush-plugin-sentry), [**StatsD**](https://github.com/rpush/rpush-plugin-statsd). Third party plugins: [**Prometheus Exporter**](https://github.com/equinux/rpush-plugin-prometheus-exporter). Or [write your own](https://github.com/rpush/rpush/wiki/Writing-a-Plugin).\n* Seamless integration with your projects, including **Rails**.\n* Run as a [daemon](https://github.com/rpush/rpush#as-a-daemon), inside a [job queue](https://github.com/rpush/rpush/wiki/Push-API), on the [command-line](https://github.com/rpush/rpush#on-the-command-line) or [embedded](https://github.com/rpush/rpush/wiki/Embedding-API) in another process.\n* Scales vertically (threading) and horizontally (multiple processes).\n* Designed for uptime - new apps are loaded automatically, signal `HUP` to update running apps.\n* Hooks for fine-grained instrumentation and error handling ([Reflection API](https://github.com/rpush/rpush/wiki/Reflection-API)).\n* Tested with **MRI**\n\n\n### Getting Started\n\nAdd it to your Gemfile:\n\n```ruby\ngem 'rpush'\n```\n\nInitialize Rpush into your project. **Rails will be detected automatically.**\n\n```sh\n$ cd /path/to/project\n$ bundle\n$ bundle exec rpush init\n```\n\n### Create an App \u0026 Notification\n\n#### Apple Push Notification Service\n\nThere is a choice of two modes, using certificates or using tokens:\n\n* `Rpush::Apns2` This requires an annually renewable certificate. see https://developer.apple.com/documentation/usernotifications/setting_up_a_remote_notification_server/establishing_a_certificate-based_connection_to_apns\n* `Rpush::Apnsp8` This uses encrypted tokens and requires an encryption key id and encryption key (provide as a p8 file). (see https://developer.apple.com/documentation/usernotifications/setting_up_a_remote_notification_server/establishing_a_token-based_connection_to_apns)\n\nIf this is your first time using the APNs, you will need to generate either SSL certificates (for standard Apns) or an Encryption Key (p8) and an Encryption Key ID (for Apnsp8). See [Generating Certificates](https://github.com/rpush/rpush/wiki/Generating-Certificates) for instructions.\n\n##### Apnsp8\n\nTo use the p8 APNs Api:\n\n```ruby\napp = Rpush::Apnsp8::App.new\napp.name = \"ios_app\"\napp.apn_key = File.read(\"/path/to/sandbox.p8\")\napp.environment = \"development\" # APNs environment.\napp.apn_key_id = \"APN KEY ID\" # This is the Encryption Key ID provided by apple\napp.team_id = \"TEAM ID\" # the team id - e.g. ABCDE12345\napp.bundle_id = \"BUNDLE ID\" # the unique bundle id of the app, like com.example.appname\napp.connections = 1\napp.save!\n```\n\n```ruby\nn = Rpush::Apnsp8::Notification.new\nn.app = Rpush::Apnsp8::App.find_by_name(\"ios_app\")\nn.device_token = \"...\" # hex string\nn.alert = \"hi mom!\"\n# n.alert = { title: \"push title\", subtitle: \"more to say\", body: \"hi mom!\" }\nn.data = { foo: :bar }\nn.save!\n```\n\n##### Apns2\n\n(NB this uses the same protocol as Apnsp8, but authenticates with a certificate rather than tokens)\n\n```ruby\napp = Rpush::Apns2::App.new\napp.name = \"ios_app\"\napp.certificate = File.read(\"/path/to/sandbox.pem\")\napp.environment = \"development\"\napp.password = \"certificate password\"\napp.bundle_id = \"BUNDLE ID\" # the unique bundle id of the app, like com.example.appname\napp.connections = 1\napp.save!\n```\n\n```ruby\nn = Rpush::Apns2::Notification.new\nn.app = Rpush::Apns2::App.find_by_name(\"ios_app\")\nn.device_token = \"...\" # hex string\nn.alert = \"hi mom!\"\n# n.alert = { title: \"push title\", subtitle: \"more to say\", body: \"hi mom!\" }\nn.data = {\n  headers: { 'apns-topic': \"BUNDLE ID\" }, # the bundle id of the app, like com.example.appname. Not necessary if set on the app (see above)\n  foo: :bar\n}\nn.save!\n```\n\nYou should also implement the [ssl_certificate_will_expire](https://github.com/rpush/rpush/wiki/Reflection-API) reflection to monitor when your certificate is due to expire.\n\n##### Safari Push Notifications\n\nUsing one of the notifications methods above, the `url_args` attribute is available for Safari Push Notifications.\n\n##### Environment\n\nThe app `environment` for any Apns* option is \"development\" for XCode installs, and \"production\" for app store and TestFlight. Note that you can now use one (production + sandbox) certificate (you don't need a separate \"sandbox\" or development certificate), but if you do generate a development/sandbox certificate it can only be used for \"development\". With Apnsp8 tokens, you can target either \"development\" or \"production\" environments.\n\n#### Firebase Cloud Messaging\n\nYou will need two params to make use of FCM via Rpush.\n- `firebase_project_id` - The `Project number` in your Firebase Project Settings\n- `json_key` - The JSON key file for a service account with the `Firebase Admin SDK Administrator Service Agent` role.\n\nCreate service account in the google cloud account attached to your firebase account:\nhttps://console.cloud.google.com/iam-admin/serviceaccounts\nMake sure it has Role `Firebase Admin SDK Administrator Service Agent`\nAdd + Download the json key for the service account.\n\nOnce you have those two params, you can create an FCM app and send notifications.\n\n```ruby\nfcm_app = Rpush::Fcm::App.new\nfcm_app.name = \"fcm_app\"\nfcm_app.firebase_project_id = \"someapp-123456\"\nfcm_app.json_key = Rails.root.join(\"your/key/somewhere.json\").read # or from a ENV variable - just needs to be the whole json file\nfcm_app.connections = 30\nfcm_app.save!\n```\n\n```ruby\nn = Rpush::Fcm::Notification.new\nn.app = Rpush::Fcm::App.where(name: \"fcm_app\").first\nn.device_token = device_token # Note that device_token is used here instead of registration_ids\nn.notification = { title: \"push title\", body: \"hi mom!\" } # either title or body needs to be set, or nothing goes through\nn.data = {}.transform_values(\u0026:to_s) # All values going in here have to be strings, if you have anything else - nothing goes through\nn.save!\n```\n\nFCM also requires you to respond to [Canonical IDs](https://github.com/rpush/rpush/wiki/Canonical-IDs).\n\nCheck the [FCM reference](https://firebase.google.com/docs/cloud-messaging/http-server-ref#notification-payload-support) for what keys you can use and are available to you. **Note:** Not all are yet implemented in Rpush.\n\n#### Amazon Device Messaging\n\n```ruby\napp = Rpush::Adm::App.new\napp.name = \"kindle_app\"\napp.client_id = \"...\"\napp.client_secret = \"...\"\napp.connections = 1\napp.save!\n```\n\n```ruby\nn = Rpush::Adm::Notification.new\nn.app = Rpush::Adm::App.find_by_name(\"kindle_app\")\nn.registration_ids = [\"...\"]\nn.data = { message: \"hi mom!\"}\nn.collapse_key = \"Optional consolidationKey\"\nn.save!\n```\n\nFor more documentation on [ADM](https://developer.amazon.com/sdk/adm.html).\n\n#### Windows Phone Notification Service (Windows Phone 8.0 and 7.x)\n\nUses the older [Windows Phone 8 Toast template](https://msdn.microsoft.com/en-us/library/windows/apps/jj662938(v=vs.105).aspx)\n\n```ruby\napp = Rpush::Wpns::App.new\napp.name = \"windows_phone_app\"\napp.client_id = # Get this from your apps dashboard https://dev.windows.com\napp.client_secret = # Get this from your apps dashboard https://dev.windows.com\napp.connections = 1\napp.save!\n```\n\n```ruby\nn = Rpush::Wpns::Notification.new\nn.app = Rpush::Wpns::App.find_by_name(\"windows_phone_app\")\nn.uri = \"http://...\"\nn.data = {title:\"MyApp\", body:\"Hello world\", param:\"user_param1\"}\nn.save!\n```\n\n#### Windows Notification Service (Windows 8.1, 10 Apps \u0026 Phone \u003e 8.0)\n\nUses the more recent [Toast template](https://msdn.microsoft.com/en-us/library/windows/apps/xaml/mt631604.aspx)\n\nThe `client_id` here is the SID URL as seen [here](https://msdn.microsoft.com/en-us/library/windows/apps/hh465407.aspx#7-SIDandSecret). Do not confuse it with the `client_id` on dashboard.\n\nYou can (optionally) include a launch argument by adding a `launch` key to the notification data.\n\nYou can (optionally) include an [audio element](https://msdn.microsoft.com/en-us/library/windows/apps/xaml/br230842.aspx) by setting the sound on the notification.\n\n```ruby\napp = Rpush::Wns::App.new\napp.name = \"windows_phone_app\"\napp.client_id = YOUR_SID_URL\napp.client_secret = YOUR_CLIENT_SECRET\napp.connections = 1\napp.save!\n```\n\n```ruby\nn = Rpush::Wns::Notification.new\nn.app = Rpush::Wns::App.find_by_name(\"windows_phone_app\")\nn.uri = \"http://...\"\nn.data = {title:\"MyApp\", body:\"Hello world\", launch:\"launch-argument\"}\nn.sound = \"ms-appx:///mynotificationsound.wav\"\nn.save!\n```\n\n#### Windows Raw Push Notifications\n\nNote: The data is passed as `.to_json` so only this format is supported, although raw notifications are meant to support any kind of data.\nCurrent data structure enforces hashes and `.to_json` representation is natural presentation of it.\n\n```ruby\nn = Rpush::Wns::RawNotification.new\nn.app = Rpush::Wns::App.find_by_name(\"windows_phone_app\")\nn.uri = 'http://...'\nn.data = { foo: 'foo', bar: 'bar' }\nn.save!\n```\n\n#### Windows Badge Push Notifications\n\nUses the [badge template](https://msdn.microsoft.com/en-us/library/windows/apps/xaml/br212849.aspx) and the type `wns/badge`.\n\n```ruby\nn = Rpush::Wns::BadgeNotification.new\nn.app = Rpush::Wns::App.find_by_name(\"windows_phone_app\")\nn.uri = 'http://...'\nn.badge = 4\nn.save!\n```\n\n#### Pushy\n\n[Pushy](https://pushy.me/) is a highly-reliable push notification gateway, based on [MQTT](https://pushy.me/support#what-is-mqtt) protocol for cross platform push notification delivery that includes web, Android, and iOS. One of its advantages is it allows for reliable notification delivery to Android devices in China where Google Cloud Messaging and Firebase Cloud Messaging are blocked and to custom hardware devices that use Android OS but are not using Google Play Services.\n\nNote: current implementation of Pushy only supports Android devices and does not include [subscriptions](https://pushy.me/docs/android/subscribe-topics).\n\n```ruby\napp = Rpush::Pushy::App.new\napp.name = \"android_app\"\napp.api_key = YOUR_API_KEY\napp.connections = 1\napp.save!\n```\n\n```ruby\nn = Rpush::Pushy::Notification.new\nn.app = Rpush::Pushy::App.find_by_name(\"android_app\")\nn.registration_ids = [\"...\"]\nn.data = { message: \"hi mom!\"}\nn.time_to_live = 60 # seconds\nn.save!\n```\n\nFor more documentation on [Pushy](https://pushy.me/docs).\n\n#### Webpush\n\n[Webpush](https://tools.ietf.org/html/draft-ietf-webpush-protocol-10) is a\nprotocol for delivering push messages to desktop browsers. It's supported by\nall major browsers (except Safari, you have to use one of the Apns transports\nfor that).\n\nUsing [VAPID](https://tools.ietf.org/html/draft-ietf-webpush-vapid-01), there\nis no need for the sender of push notifications to register upfront with push\nservices (as was the case with the now legacy Mozilla or Google desktop push\nproviders).\n\nInstead, you generate a pair of keys and use the public key when subscribing\nusers in your web app. The keys are stored along with an email address (which,\naccording to the spec, can be used by push service providers to contact you in\ncase of problems) in the `certificates` field of the Rpush Application record:\n\n```ruby\nvapid_keypair = Webpush.generate_key.to_hash\napp = Rpush::Webpush::App.new\napp.name = 'webpush'\napp.certificate = vapid_keypair.merge(subject: 'mailto:user@example.org').to_json\napp.connections = 1\napp.save!\n```\n\nThe `subscription` object you obtain from a subscribed browser holds an\nendpoint URL and cryptographic keys. When sending a notification, simply pass\nthe whole subscription as sole member of the `registration_ids` collection:\n\n```ruby\nn = Rpush::Webpush::Notification.new\nn.app = Rpush::App.find_by_name(\"webpush\")\nn.registration_ids = [subscription]\nn.data = { message: \"hi mom!\" }\nn.save!\n```\n\nIn order to send the same message to multiple devices, create one\n`Notification` per device, as passing multiple subscriptions at once as\n`registration_ids` is not supported.\n\n\n### Running Rpush\n\nIt is recommended to run Rpush as a separate process in most cases, though embedding and manual modes are provided for low-workload environments.\n\nSee `rpush help` for all available commands and options.\n\n#### As a daemon\n\n```sh\n$ cd /path/to/project\n$ rpush start\n```\n\n#### As a foreground process\n\n```sh\n$ cd /path/to/project\n$ rpush start -f\n```\n\n#### On the command-line\n\n```sh\n$ rpush push\n```\n\nRpush will deliver all pending notifications and then exit.\n\n#### In a scheduled job\n\n```ruby\nRpush.push\n```\n\nSee [Push API](https://github.com/rpush/rpush/wiki/Push-API) for more details.\n\n#### Embedded inside an existing process\n\n```ruby\nif defined?(Rails)\n  ActiveSupport.on_load(:after_initialize) do\n    Rpush.embed\n  end\nelse\n  Rpush.embed\nend\n```\n\nCall this during startup of your application, for example, by adding it to the end of `config/rpush.rb`. See [Embedding API](https://github.com/rpush/rpush/wiki/Embedding-API) for more details.\n\n#### Using mina\n\nIf you're using [mina](https://github.com/mina-deploy/mina), there is a gem called [mina-rpush](https://github.com/d4rky-pl/mina-rpush) which helps you control rpush.\n\n### Cleanup\n\nRpush leaves delivered notifications in the database. If you do not clear them out, they will take up more and more space. This isn't great for any database, but is especially problematic if using Redis as the Rpush store. [Here](https://github.com/rpush/rpush/wiki/Using-Redis) is an example solution for cleaning up delivered notifications in Redis.\n\n### Configuration\n\nSee [Configuration](https://github.com/rpush/rpush/wiki/Configuration) for a list of options.\n\n### Updating Rpush\n\nYou should run `rpush init` after upgrading Rpush to check for configuration and migration changes.\n\n### From The Wiki\n\n### General\n* [Using Redis](https://github.com/rpush/rpush/wiki/Using-Redis)\n* [Using ActiveRecord](https://github.com/rpush/rpush/wiki/Using-ActiveRecord)\n* [Configuration](https://github.com/rpush/rpush/wiki/Configuration)\n* [Moving from Rapns](https://github.com/rpush/rpush/wiki/Moving-from-Rapns-to-Rpush)\n* [Deploying to Heroku](https://github.com/rpush/rpush/wiki/Heroku)\n* [Hot App Updates](https://github.com/rpush/rpush/wiki/Hot-App-Updates)\n* [Signals](https://github.com/rpush/rpush/wiki/Signals)\n* [Reflection API](https://github.com/rpush/rpush/wiki/Reflection-API)\n* [Push API](https://github.com/rpush/rpush/wiki/Push-API)\n* [Embedding API](https://github.com/rpush/rpush/wiki/Embedding-API)\n* [Writing a Plugin](https://github.com/rpush/rpush/wiki/Writing-a-Plugin)\n* [Implementing your own storage backend](https://github.com/rpush/rpush/wiki/Implementing-your-own-storage-backend)\n* [Upgrading from 2.x to 3.0](https://github.com/rpush/rpush/wiki/Upgrading-from-version-2.x-to-3.0)\n\n### Apple Push Notification Service\n* [Generating Certificates](https://github.com/rpush/rpush/wiki/Generating-Certificates)\n* [Advanced APNs Features](https://github.com/rpush/rpush/wiki/Advanced-APNs-Features)\n* [APNs Delivery Failure Handling](https://github.com/rpush/rpush/wiki/APNs-Delivery-Failure-Handling)\n* [Why open multiple connections to the APNs?](https://github.com/rpush/rpush/wiki/Why-open-multiple-connections-to-the-APNs%3F)\n* [Silent failures might be dropped connections](https://github.com/rpush/rpush/wiki/Dropped-connections)\n\n### Firebase Cloud Messaging\n* [Notification Options](https://github.com/rpush/rpush/wiki/GCM-Notification-Options)\n* [Canonical IDs](https://github.com/rpush/rpush/wiki/Canonical-IDs)\n* [Delivery Failures \u0026 Retries](https://github.com/rpush/rpush/wiki/Delivery-Failures-\u0026-Retries)\n\n### Contributing\n\n#### Running Tests\n\nRpush uses [Appraisal](https://github.com/thoughtbot/appraisal) to run tests against multiple versions of Ruby on Rails. This helps making sure that Rpush performs correctly with multiple Rails versions.\n\nRpush also uses RSpec for its tests.\n\n##### Bootstrapping your test suite:\n\nFirst, we need to setup a test database, `rpush_test`.\n\nE.g. (postgres): `psql -c 'create database rpush_test;' -U postgres \u003e/dev/null`\n\n```\nbundle install\nbundle exec appraisal install\n```\nThis will install all the required gems that requires to test against each version of Rails, which defined in `gemfiles/*.gemfile`.\n\n##### To run a full test suite:\n\n```\nbundle exec appraisal rake\n```\nThis will run RSpec against all versions of Rails.\n\n##### To run a single test\n\nYou need to specify a `BUNDLE_GEMFILE` pointing to the gemfile before running the normal test command:\n\n```\nBUNDLE_GEMFILE=gemfiles/rails_6.0.gemfile rspec spec/unit/apns_feedback_spec.rb\n```\n\n##### Multiple database adapter support\n\nWhen running specs, please note that the ActiveRecord adapter can be changed by setting the `ADAPTER` environment variable. For example: `ADAPTER=postgresql rake`.\n\nAvailable adapters for testing are `postgresql`, `jdbcpostgresql`, `mysql2`, `jdbcmysql`, `jdbch2`, and `sqlite3`.\n\nNote that the database username is changed at runtime to be the currently logged in user's name. So if you're testing\nwith mysql and you're using a user named 'bob', you will need to grant a mysql user 'bob' access to the 'rpush_test'\nmysql database.\n\nTo switch between ActiveRecord and Redis, set the `CLIENT` environment variable to either `active_record` or `redis`.\n","funding_links":[],"categories":["Ruby","Gems","Push Notification","Notifications","Mobile Development"],"sub_categories":["Misc"],"project_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Frpush%2Frpush","html_url":"https://awesome.ecosyste.ms/projects/github.com%2Frpush%2Frpush","lists_url":"https://awesome.ecosyste.ms/api/v1/projects/github.com%2Frpush%2Frpush/lists"}