2017-10-17 21:32:11 -07:00
|
|
|
#!/usr/bin/python
|
|
|
|
|
2017-10-18 21:18:30 -07:00
|
|
|
import base64, json, mod_python, socket, struct, sys
|
2017-11-07 07:33:04 -08:00
|
|
|
import psycopg2, random
|
2017-10-18 21:18:30 -07:00
|
|
|
|
|
|
|
PROTOCOL_VERSION = 0
|
2017-10-31 20:05:07 -07:00
|
|
|
PRX_DEVICE_GONE = 3
|
2017-10-18 21:18:30 -07:00
|
|
|
PRX_GET_MSGS = 4
|
2017-10-17 21:32:11 -07:00
|
|
|
|
2017-10-31 20:05:07 -07:00
|
|
|
# try:
|
|
|
|
# from mod_python import apache
|
|
|
|
# apacheAvailable = True
|
|
|
|
# except ImportError:
|
|
|
|
# apacheAvailable = False
|
2017-10-17 21:32:11 -07:00
|
|
|
|
2017-11-08 08:04:57 -08:00
|
|
|
# Joining a game. Basic idea is you have stuff to match on (room,
|
|
|
|
# number in game, language) and when somebody wants to join you add to
|
|
|
|
# an existing matching game if there's space otherwise create a new
|
|
|
|
# one. Problems are the unreliablity of transport: if you give a space
|
|
|
|
# and the device doesn't get the message you can't hold it forever. So
|
|
|
|
# device provides a seed that holds the space. If it asks again for a
|
|
|
|
# space with the same seed it gets the same space. If it never asks
|
|
|
|
# again (app deleted, say), the space needs eventually to be given to
|
|
|
|
# somebody else. I think that's done by adding a timestamp array and
|
|
|
|
# treating the space as available if TIME has expired. Need to think
|
|
|
|
# about this: what if app fails to ACK for TIME, then returns with
|
|
|
|
# seed to find it given away. Let's do a 30 minute reservation for
|
|
|
|
# now? [Note: much of this is PENDING]
|
|
|
|
|
|
|
|
def join(req, devID, room, seed, hid = 0, lang = 1, nInGame = 2, nHere = 1, inviteID = None):
|
2017-11-09 06:47:33 -08:00
|
|
|
assert hid <= 4
|
|
|
|
seed = int(seed)
|
|
|
|
assert seed != 0
|
2017-11-10 21:34:02 -08:00
|
|
|
nInGame = int(nInGame)
|
|
|
|
nHere = int(nHere)
|
|
|
|
assert nHere <= nInGame
|
|
|
|
assert nInGame <= 4
|
|
|
|
|
|
|
|
devID = int(devID, 16)
|
2017-11-09 06:47:33 -08:00
|
|
|
|
2017-11-07 07:33:04 -08:00
|
|
|
connname = None
|
2017-11-09 06:47:33 -08:00
|
|
|
logs = [] # for debugging
|
2017-11-10 21:34:02 -08:00
|
|
|
# logs.append('vers: ' + platform.python_version())
|
|
|
|
|
2017-11-07 07:33:04 -08:00
|
|
|
con = psycopg2.connect(database='xwgames')
|
|
|
|
cur = con.cursor()
|
2017-11-09 06:47:33 -08:00
|
|
|
# cur.execute('LOCK TABLE games IN ACCESS EXCLUSIVE MODE')
|
2017-11-08 08:04:57 -08:00
|
|
|
|
|
|
|
# First see if there's a game with a space for me. Must match on
|
|
|
|
# room, lang and size. Must have room OR must have already given a
|
|
|
|
# spot for a seed equal to mine, in which case I get it
|
|
|
|
# back. Assumption is I didn't ack in time.
|
|
|
|
|
|
|
|
query = "SELECT connname, seeds, nperdevice FROM games "
|
|
|
|
query += "WHERE lang = %s AND nTotal = %s AND room = %s "
|
|
|
|
query += "AND (njoined + %s <= ntotal OR %s = ANY(seeds)) "
|
|
|
|
query += "LIMIT 1"
|
|
|
|
cur.execute( query, (lang, nInGame, room, nHere, seed))
|
|
|
|
for row in cur:
|
|
|
|
(connname, seeds, nperdevice) = row
|
|
|
|
print('found', connname, seeds, nperdevice)
|
|
|
|
break # should be only one!
|
|
|
|
|
|
|
|
# If we've found a match, we either need to UPDATE or, if the
|
|
|
|
# seeds match, remind the caller of where he belongs. If a hid's
|
|
|
|
# been specified, we honor it by updating if the slot's available;
|
|
|
|
# otherwise a new game has to be created.
|
|
|
|
if connname:
|
|
|
|
if seed in seeds and nHere == nperdevice[seeds.index(seed)]:
|
|
|
|
hid = seeds.index(seed) + 1
|
|
|
|
print('resusing seed case; outta here!')
|
|
|
|
else:
|
|
|
|
if hid == 0:
|
|
|
|
# Any gaps? Assign it
|
|
|
|
if None in seeds:
|
|
|
|
hid = seeds.index(None) + 1
|
|
|
|
else:
|
|
|
|
hid = len(seeds) + 1
|
|
|
|
print('set hid to', hid, 'based on ', seeds)
|
|
|
|
else:
|
|
|
|
print('hid already', hid)
|
|
|
|
query = "UPDATE games SET njoined = njoined + %s, "
|
2017-11-10 21:34:02 -08:00
|
|
|
query += "devids[%d] = %%s, " % hid
|
2017-11-08 08:04:57 -08:00
|
|
|
query += "seeds[%d] = %%s, " % hid
|
|
|
|
query += "jtimes[%d] = 'now', " % hid
|
|
|
|
query += "nperdevice[%d] = %%s " % hid
|
|
|
|
query += "WHERE connname = %s "
|
|
|
|
print(query)
|
2017-11-10 21:34:02 -08:00
|
|
|
params = (nHere, devID, seed, nHere, connname)
|
2017-11-08 08:04:57 -08:00
|
|
|
cur.execute(query, params)
|
|
|
|
|
|
|
|
# If nothing was found, add a new game and add me. Honor my hid
|
|
|
|
# preference if specified
|
2017-11-07 07:33:04 -08:00
|
|
|
if not connname:
|
2017-11-10 21:34:02 -08:00
|
|
|
# This requires python3, which likely requires mod_wsgi
|
|
|
|
# ts = datetime.datetime.utcnow().timestamp()
|
|
|
|
# connname = '%s:%d:1' % (xwconfig.k_HOSTNAME, int(ts * 1000))
|
|
|
|
connname = '%s:%d:1' % (xwconfig.k_HOSTNAME, random.randint(0, 10000000000))
|
2017-11-08 08:04:57 -08:00
|
|
|
useHid = hid == 0 and 1 or hid
|
|
|
|
print('not found case; inserting using hid:', useHid)
|
2017-11-10 21:34:02 -08:00
|
|
|
query = "INSERT INTO games (connname, room, lang, ntotal, njoined, " + \
|
|
|
|
"devids[%d], seeds[%d], jtimes[%d], nperdevice[%d]) " % (4 * (useHid,))
|
|
|
|
query += "VALUES (%s, %s, %s, %s, %s, %s, %s, 'now', %s) "
|
2017-11-08 08:04:57 -08:00
|
|
|
query += "RETURNING connname, array_length(seeds,1); "
|
2017-11-10 21:34:02 -08:00
|
|
|
cur.execute(query, (connname, room, lang, nInGame, nHere, devID, seed, nHere))
|
2017-11-07 07:33:04 -08:00
|
|
|
for row in cur:
|
2017-11-08 08:04:57 -08:00
|
|
|
connname, gothid = row
|
|
|
|
break
|
|
|
|
if hid == 0: hid = gothid
|
|
|
|
|
2017-11-07 07:33:04 -08:00
|
|
|
con.commit()
|
|
|
|
con.close()
|
|
|
|
|
2017-11-09 06:47:33 -08:00
|
|
|
result = {'connname': connname, 'hid' : hid, 'log' : ':'.join(logs)}
|
2017-11-07 07:33:04 -08:00
|
|
|
|
|
|
|
return json.dumps(result)
|
|
|
|
|
2017-10-31 20:05:07 -07:00
|
|
|
def kill(req, params):
|
|
|
|
print(params)
|
|
|
|
params = json.loads(params)
|
|
|
|
count = len(params)
|
|
|
|
|
|
|
|
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
|
|
|
sock.connect(('127.0.0.1', 10998))
|
|
|
|
|
|
|
|
header = struct.Struct('!BBh')
|
|
|
|
strLens = 0
|
|
|
|
for ii in range(count):
|
|
|
|
strLens += len(params[ii]['relayID']) + 1
|
|
|
|
size = header.size + (2*count) + strLens
|
|
|
|
sock.send(struct.Struct('!h').pack(size))
|
|
|
|
sock.send(header.pack(PROTOCOL_VERSION, PRX_DEVICE_GONE, count))
|
|
|
|
|
|
|
|
for ii in range(count):
|
|
|
|
elem = params[ii]
|
|
|
|
asBytes = bytes(elem['relayID'])
|
|
|
|
sock.send(struct.Struct('!H%dsc' % (len(asBytes))).pack(elem['seed'], asBytes, '\n'))
|
|
|
|
sock.close()
|
|
|
|
|
|
|
|
result = {'err': 0}
|
|
|
|
return json.dumps(result)
|
2017-11-07 07:33:04 -08:00
|
|
|
|
|
|
|
# winds up in handle_udp_packet() in xwrelay.cpp
|
2017-10-28 20:12:05 -07:00
|
|
|
def post(req, params, timeoutSecs = 1.0):
|
2017-10-18 06:53:15 -07:00
|
|
|
err = 'none'
|
2017-10-17 21:32:11 -07:00
|
|
|
jobj = json.loads(params)
|
2017-10-18 06:53:15 -07:00
|
|
|
data = base64.b64decode(jobj['data'])
|
|
|
|
|
2017-11-07 07:33:04 -08:00
|
|
|
udpSock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
|
|
|
udpSock.settimeout(float(timeoutSecs)) # seconds
|
2017-10-18 06:53:15 -07:00
|
|
|
addr = ("127.0.0.1", 10997)
|
2017-11-07 07:33:04 -08:00
|
|
|
udpSock.sendto(data, addr)
|
2017-10-18 06:53:15 -07:00
|
|
|
|
2017-10-18 07:09:05 -07:00
|
|
|
responses = []
|
|
|
|
while True:
|
|
|
|
try:
|
2017-11-07 07:33:04 -08:00
|
|
|
data, server = udpSock.recvfrom(1024)
|
2017-10-18 07:09:05 -07:00
|
|
|
responses.append(base64.b64encode(data))
|
|
|
|
except socket.timeout:
|
|
|
|
#If data is not received back from server, print it has timed out
|
|
|
|
err = 'timeout'
|
|
|
|
break
|
2017-10-18 06:53:15 -07:00
|
|
|
|
2017-10-18 07:09:05 -07:00
|
|
|
jobj = {'err' : err, 'data' : responses}
|
2017-10-17 21:32:11 -07:00
|
|
|
return json.dumps(jobj)
|
|
|
|
|
2017-10-28 20:12:05 -07:00
|
|
|
def query(req, ids, timeoutSecs = 5.0):
|
2017-10-22 15:47:01 -07:00
|
|
|
print('ids', ids)
|
2017-10-18 22:02:14 -07:00
|
|
|
ids = json.loads(ids)
|
|
|
|
|
2017-10-18 21:18:30 -07:00
|
|
|
idsLen = 0
|
|
|
|
for id in ids: idsLen += len(id)
|
|
|
|
|
2017-11-02 07:16:11 -07:00
|
|
|
tcpSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
|
|
|
tcpSock.settimeout(float(timeoutSecs))
|
|
|
|
tcpSock.connect(('127.0.0.1', 10998))
|
2017-10-18 21:18:30 -07:00
|
|
|
|
2017-10-22 21:29:16 -07:00
|
|
|
lenShort = 2 + idsLen + len(ids) + 2
|
2017-10-22 15:47:01 -07:00
|
|
|
print(lenShort, PROTOCOL_VERSION, PRX_GET_MSGS, len(ids))
|
2017-10-22 21:29:16 -07:00
|
|
|
header = struct.Struct('!hBBh')
|
2017-10-22 15:47:01 -07:00
|
|
|
assert header.size == 6
|
2017-11-02 07:16:11 -07:00
|
|
|
tcpSock.send(header.pack(lenShort, PROTOCOL_VERSION, PRX_GET_MSGS, len(ids)))
|
2017-10-18 21:18:30 -07:00
|
|
|
|
2017-11-02 07:16:11 -07:00
|
|
|
for id in ids: tcpSock.send(id + '\n')
|
2017-10-18 21:18:30 -07:00
|
|
|
|
|
|
|
msgsLists = {}
|
2017-11-02 07:16:11 -07:00
|
|
|
try:
|
|
|
|
shortUnpacker = struct.Struct('!H')
|
|
|
|
resLen, = shortUnpacker.unpack(tcpSock.recv(shortUnpacker.size)) # not getting all bytes
|
|
|
|
nameCount, = shortUnpacker.unpack(tcpSock.recv(shortUnpacker.size))
|
|
|
|
resLen -= shortUnpacker.size
|
|
|
|
print('resLen:', resLen, 'nameCount:', nameCount)
|
|
|
|
if nameCount == len(ids) and resLen > 0:
|
|
|
|
print('nameCount', nameCount)
|
|
|
|
for ii in range(nameCount):
|
|
|
|
perGame = []
|
|
|
|
countsThisGame, = shortUnpacker.unpack(tcpSock.recv(shortUnpacker.size)) # problem
|
|
|
|
print('countsThisGame:', countsThisGame)
|
|
|
|
for jj in range(countsThisGame):
|
|
|
|
msgLen, = shortUnpacker.unpack(tcpSock.recv(shortUnpacker.size))
|
|
|
|
print('msgLen:', msgLen)
|
|
|
|
msgs = []
|
|
|
|
if msgLen > 0:
|
|
|
|
msg = tcpSock.recv(msgLen)
|
|
|
|
print('msg len:', len(msg))
|
|
|
|
msg = base64.b64encode(msg)
|
|
|
|
msgs.append(msg)
|
|
|
|
perGame.append(msgs)
|
|
|
|
msgsLists[ids[ii]] = perGame
|
|
|
|
except:
|
|
|
|
None
|
2017-10-18 21:18:30 -07:00
|
|
|
|
|
|
|
return json.dumps(msgsLists)
|
|
|
|
|
2017-10-17 21:32:11 -07:00
|
|
|
def main():
|
2017-10-31 20:05:07 -07:00
|
|
|
result = None
|
|
|
|
if len(sys.argv) > 1:
|
|
|
|
cmd = sys.argv[1]
|
|
|
|
args = sys.argv[2:]
|
2017-11-08 08:04:57 -08:00
|
|
|
if cmd == 'query':
|
|
|
|
result = query(None, json.dumps(args))
|
|
|
|
elif cmd == 'post':
|
|
|
|
# Params = { 'data' : 'V2VkIE9jdCAxOCAwNjowNDo0OCBQRFQgMjAxNwo=' }
|
|
|
|
# params = json.dumps(params)
|
|
|
|
# print(post(None, params))
|
|
|
|
pass
|
|
|
|
elif cmd == 'join':
|
|
|
|
if len(args) == 6:
|
|
|
|
result = join(None, 1, args[0], int(args[1]), int(args[2]), int(args[3]), int(args[4]), int(args[5]))
|
|
|
|
elif cmd == 'kill':
|
|
|
|
result = kill( None, json.dumps([{'relayID': args[0], 'seed':int(args[1])}]) )
|
2017-10-31 20:05:07 -07:00
|
|
|
|
|
|
|
if result:
|
|
|
|
print '->', result
|
|
|
|
else:
|
|
|
|
print 'USAGE: query [connname/hid]*'
|
2017-11-10 21:34:02 -08:00
|
|
|
print ' join <roomName> <seed> <hid> <lang> <nTotal> <nHere>'
|
|
|
|
print ' query [connname/hid]*'
|
2017-10-31 20:05:07 -07:00
|
|
|
# print ' post '
|
|
|
|
print ' kill <relayID> <seed>'
|
2017-11-08 08:04:57 -08:00
|
|
|
print ' join <roomName> <seed> <hid> <lang> <nTotal> <nHere>'
|
2017-10-17 21:32:11 -07:00
|
|
|
|
|
|
|
##############################################################################
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|