#!/usr/bin/python # Meant to be run on the server that's hosting the relay, loops, # checking the relay for new messages whose target devices have GCM # ids and sending GCM notifications to them. # # Depends on the gcm module import getpass, sys, gcm, psycopg2, time, signal, shelve from time import gmtime, strftime # I'm not checking my key in... import mykey # Backoff strategy # # A message is considered in need of delivery as long as its in the # msgs table, and the expected behavior is that as soon as a device # receives a GCM notification it fetches all messages so that they're # deleted. But when a device is offline we don't get any errors, so # while a message remains in that table we need to be sure we don't # ask GCM to contact the device too often. # # But it's devices we contact, not messages. A device is in the # contact list if it is the target of at least one message in the msgs # table. k_shelfFile = "gcm_loop.shelf" k_SENT = 'SENT' g_con = None g_sent = None g_debug = False g_skipSend = False # for debugging DEVTYPE_GCM = 3 # 3 == GCM LINE_LEN = 76 def init(): global g_sent try: con = psycopg2.connect(database='xwgames', user=getpass.getuser()) except psycopg2.DatabaseError, e: print 'Error %s' % e sys.exit(1) shelf = shelve.open( k_shelfFile ) if k_SENT in shelf: g_sent = shelf[k_SENT] else: g_sent = {} shelf.close(); if g_debug: print 'g_sent:', g_sent return con # WHERE stime IS NULL def getPendingMsgs( con, typ ): cur = con.cursor() query = """SELECT id, devid FROM msgs WHERE devid IN (SELECT id FROM devices WHERE devtype=%d and NOT unreg) AND NOT connname IN (SELECT connname FROM games WHERE dead); """ cur.execute(query % typ) result = cur.fetchall() if g_debug: print "getPendingMsgs=>", result return result def unregister( gcmid ): global g_con print "unregister(", gcmid, ")" query = "UPDATE devices SET unreg=TRUE WHERE devid = '%s'" % gcmid g_con.cursor().execute( query ) def asGCMIds(con, devids, typ): cur = con.cursor() query = "SELECT devid FROM devices WHERE devtype = %d AND id IN (%s)" \ % (typ, ",".join([str(y) for y in devids])) cur.execute( query ) return [elem[0] for elem in cur.fetchall()] def notifyGCM( devids, typ ): if typ == DEVTYPE_GCM: instance = gcm.GCM( mykey.myKey ) data = { 'getMoves': True, } response = instance.json_request( registration_ids = devids, data = data, ) if 'errors' in response: response = response['errors'] if 'NotRegistered' in response: for gcmid in response['NotRegistered']: unregister( gcmid ) else: print "got some kind of error" else: if g_debug: print 'no errors:', response else: print "not sending to", len(devids), "devices because typ ==", typ def shouldSend(val): return val == 1 # pow = 1 # while pow < val: # pow *= 3 # return pow == val # given a list of msgid, devid lists, figure out which messages should # be sent/resent now and mark them as sent. Backoff is based on # msgids: if the only messages a device has pending have been seen # before, backoff applies. def targetsAfterBackoff( msgs ): global g_sent targets = {} for row in msgs: msgid = row[0] devid = row[1] if not msgid in g_sent: g_sent[msgid] = 0 g_sent[msgid] += 1 if shouldSend( g_sent[msgid] ): targets[devid] = True return targets.keys() # devids is an array of (msgid, devid) tuples def pruneSent( devids ): global g_sent if g_debug: print "pruneSent: before:", g_sent lenBefore = len(g_sent) msgids = [] for row in devids: msgids.append(row[0]) for msgid in g_sent.keys(): if not msgid in msgids: del g_sent[msgid] if g_debug: print "pruneSent: after:", g_sent def cleanup(): global g_con, g_sent if g_con: g_con.close() g_con = None shelf = shelve.open( k_shelfFile ) shelf[k_SENT] = g_sent shelf.close(); def handleSigTERM( one, two ): print 'handleSigTERM called: ', one, two cleanup() def usage(): print "usage:", sys.argv[0], "[--loop ] [--type typ] [--verbose]" sys.exit(); def main(): global g_con, g_sent, g_debug loopInterval = 0 g_con = init() emptyCount = 0 typ = DEVTYPE_GCM ii = 1 while ii < len(sys.argv): arg = sys.argv[ii] if arg == '--loop': ii += 1 loopInterval = float(sys.argv[ii]) elif arg == '--type': ii += 1 typ = int(sys.argv[ii]) elif arg == '--verbose': g_debug = True else: usage() ii = ii + 1 signal.signal( signal.SIGTERM, handleSigTERM ) signal.signal( signal.SIGINT, handleSigTERM ) while g_con: if g_debug: print devids = getPendingMsgs( g_con, typ ) if 0 < len(devids): targets = targetsAfterBackoff( devids ) if 0 < len(targets): if 0 < emptyCount: print "" emptyCount = 0 print strftime("%Y-%m-%d %H:%M:%S", time.localtime()), print "devices needing notification:", targets notifyGCM( asGCMIds( g_con, targets, typ ), typ ) pruneSent( devids ) elif g_debug: print "no targets after backoff" else: emptyCount += 1 if (0 == (emptyCount%5)) and not g_debug: sys.stdout.write('.') sys.stdout.flush() if 0 == (emptyCount % (LINE_LEN*5)): print "" if 0 == loopInterval: break time.sleep( loopInterval ) cleanup() ############################################################################## if __name__ == '__main__': main()