#!/usr/bin/env ruby # -*- coding: utf-8 -*- require 'rubygems' require 'sinatra' require 'sequel' require 'socket' require 'openssl' require 'cgi' require 'rufus/scheduler' require 'eventmachine' require 'sinatra/base' require 'yaml' ############################################################ ## Initilization Setup ############################################################ LIBDIR = File.expand_path(File.join(File.dirname(__FILE__), '..', 'lib')) ROOTDIR = File.expand_path(File.join(File.dirname(__FILE__), '..')) unless $LOAD_PATH.include?(LIBDIR) $LOAD_PATH << LIBDIR end unless File.exist?("#{Dir.pwd}/spns.yml") then puts 'create config file: spns.yml' system "cp #{ROOTDIR}/spns.yml #{Dir.pwd}/spns.yml" end unless File.exist?("#{Dir.pwd}/cron") then puts "create a demo 'cron' script" system "cp #{ROOTDIR}/cron #{Dir.pwd}/cron" end ############################################################ ## Configuration Setup ############################################################ env = ENV['SINATRA_ENV'] || "development" config = YAML.load_file("#{Dir.pwd}/spns.yml") $timer = "#{config['timer']}".to_i $cron = config['cron'] || 'cron' $port = config['port'] || 4567 $mode = config['mode'] || env $VERSION = File.open("#{ROOTDIR}/VERSION", "rb").read $apps = config['apps'] || [] ############################################################ ## Certificate Key Setup ############################################################ $certkey = {} def check_cert $apps.each { |app| unless File.exist?("#{Dir.pwd}/#{app}_#{$mode}.pem") then puts "Please provide #{app}_#{$mode}.pem under '#{Dir.pwd}/' directory" return false; else puts "'#{app}'s #{$mode} PEM: (#{app}_#{$mode}.pem)" certfile = File.read("#{Dir.pwd}/#{app}_#{$mode}.pem") openSSLContext = OpenSSL::SSL::SSLContext.new openSSLContext.cert = OpenSSL::X509::Certificate.new(certfile) openSSLContext.key = OpenSSL::PKey::RSA.new(certfile) $certkey["#{app}"] = openSSLContext end } return true end unless check_cert then puts "1: please provide certificate key pem file under current directory, name should be: appid_dev.pem for development and appid_prod.pem for production" puts "2: edit your spns.yml under current directory" puts "3: run spns" puts "4: iOS Client: in AppDelegate file, didRegisterForRemoteNotificationsWithDeviceToken method should access url below:" $apps.each { |app| puts "'#{app}'s registration url: http://serverIP:#{$port}/v1/apps/#{app}/DeviceToken" } puts "5: Server: cron should access 'curl http://localhost:#{$port}/v1/app/push/{messages}/{pid}' to send push message" exit else puts "*"*80 puts "Simple Push Notification Server(#{$VERSION}) is Running ..." puts "Mode: #{$mode}" puts "Port: #{$port}" puts "Cron Job: '#{Dir.pwd}/#{$cron}' script is running every #{$timer} #{($timer == 1) ? 'minute' : 'minutes'} " unless "#{$timer}".to_i == 0 puts "*"*80 end ############################################################ ## Sequel Database Setup ############################################################ unless File.exist?("#{Dir.pwd}/push.db") then $DB = Sequel.connect("sqlite://#{Dir.pwd}/push.db") $DB.create_table :tokens do primary_key :id String :app, :unique => true, :null => false String :token, :unique => true, :null => false, :size => 100 index [:app, :token] end $DB.create_table :pushes do primary_key :id String :pid, :unique => true, :null => false, :size => 100 index :pid end else $DB = Sequel.connect("sqlite://#{Dir.pwd}/push.db") end Token = $DB[:tokens] Push = $DB[:pushes] ############################################################ ## Timer Job Setup ############################################################ scheduler = Rufus::Scheduler.start_new unless $timer == 0 then scheduler.every "#{$timer}m" do puts "running job: '#{Dir.pwd}/#{$cron}' every #{$timer} #{($timer == 1) ? 'minute' : 'minutes'}" system "./#{$cron}" end else puts "1: How to register notification? (Client Side)" puts puts "In AppDelegate file, inside didRegisterForRemoteNotificationsWithDeviceToken method access url below to register device token:" $apps.each { |app| puts "'#{app}'s registration url: http://serverIP:#{$port}/v1/apps/#{app}/DeviceToken" } puts puts "2: How to send push notification? (Server Side)" puts $apps.each { |app| puts "curl http://localhost:#{$port}/v1/apps/#{app}/push/{message}/{pid}" } puts puts "Note:" puts "param1 (message): push notification message you want to send, remember the message should be html escaped" puts "param2 (pid ): unique string to mark the message, for example current timestamp or md5/sha1 digest" puts puts "*"*80 end ############################################################ ## Simple Push Notification Server based on Sinatra ############################################################ class App < Sinatra::Base set :port, "#{$port}".to_i if "#{$mode}".strip == 'development' then set :show_exceptions, true set :dump_errors, true else set :show_exceptions, false set :dump_errors, false end get '/' do o = "Simple Push Notification Server #{$VERSION}

" + "author: Eiffel(Q)
email: eiffelqiu@gmail.com

" o += "1: How to register notification? (Client Side)

" o += "In AppDelegate file, inside didRegisterForRemoteNotificationsWithDeviceToken method access url below to register device token:

" $apps.each { |app| o += "'#{app}': http://serverIP:#{$port}/v1/apps/#{app}/DeviceToken
" } o += "
2: How to send push notification? (Server Side)

" $apps.each { |app| o += "curl http://localhost:#{$port}/v1/apps/#{app}/push/{message}/{pid}
" } o += "
Note:
" o += "param1 (message): push notification message you want to send, remember the message should be html escaped
" o += "param2 (pid ): unique string to mark the message, for example current timestamp or md5/sha1 digest
" o end $apps.each { |app| get "/v1/apps/#{app}/:token" do puts "[#{params[:token]}] was added to '#{app}'" o = Token.first(:token => params[:token]) unless o Token.insert( :app => app, :token => params[:token] ) end end get "/v1/apps/#{app}/push/:message/:pid" do message = CGI::unescape(params[:message]) pid = params[:pid] puts "'#{message}' was sent to (#{app}) with pid: [#{pid}]" @push = Token.where(:app => app) @exist = Push.first(:pid => pid) unless @exist openSSLContext = $certkey["#{app}"] # Connect to port 2195 on the server. sock = nil if $mode == 'production' then sock = TCPSocket.new('gateway.push.apple.com', 2195) else sock = TCPSocket.new('gateway.sandbox.push.apple.com', 2195) end # do our SSL handshaking sslSocket = OpenSSL::SSL::SSLSocket.new(sock, openSSLContext) sslSocket.connect #Push.create( :pid => pid ) Push.insert(:pid => pid) # write our packet to the stream @push.each do |o| tokenText = o[:token] # pack the token to convert the ascii representation back to binary tokenData = [tokenText].pack('H*') # construct the payload payload = "{\"aps\":{\"alert\":\"#{message}\", \"badge\":1}}" # construct the packet packet = [0, 0, 32, tokenData, 0, payload.length, payload].pack("ccca*cca*") # read our certificate and set up our SSL context sslSocket.write(packet) end # cleanup sslSocket.close sock.close end end } end