mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-03 23:04:08 +01:00
improve rematch test
This commit is contained in:
parent
c726477878
commit
c31bc3c0eb
2 changed files with 76 additions and 42 deletions
|
@ -1692,6 +1692,9 @@ on_incoming_signal( GSocketService* XP_UNUSED(service),
|
||||||
if ( success ) {
|
if ( success ) {
|
||||||
addGIDToObject( &response, newGameID, "newGid" );
|
addGIDToObject( &response, newGameID, "newGid" );
|
||||||
}
|
}
|
||||||
|
} else if ( 0 == strcmp( cmdStr, "getStates" ) ) {
|
||||||
|
cJSON* gids = getGamesStateForArgs( aGlobals, args );
|
||||||
|
addObjectToObject( &response, "states", gids );
|
||||||
} else {
|
} else {
|
||||||
success = XP_FALSE;
|
success = XP_FALSE;
|
||||||
XP_ASSERT(0);
|
XP_ASSERT(0);
|
||||||
|
|
|
@ -5,6 +5,7 @@ import argparse, datetime, json, os, random, shutil, signal, \
|
||||||
|
|
||||||
g_NAMES = ['Brynn', 'Ariela', 'Kati', 'Eric']
|
g_NAMES = ['Brynn', 'Ariela', 'Kati', 'Eric']
|
||||||
gDone = False
|
gDone = False
|
||||||
|
gGamesMade = 0
|
||||||
|
|
||||||
def log(args, msg):
|
def log(args, msg):
|
||||||
if args.VERBOSE:
|
if args.VERBOSE:
|
||||||
|
@ -31,22 +32,33 @@ def chooseNames(nPlayers):
|
||||||
result.append(players.pop(indx))
|
result.append(players.pop(indx))
|
||||||
return result
|
return result
|
||||||
|
|
||||||
class GuestGameInfo():
|
class GameInfo():
|
||||||
def __init__(self, gid):
|
def __init__(self, gid, rematchLevel):
|
||||||
self.gid = gid
|
self.gid = gid
|
||||||
|
self.state = {}
|
||||||
# Should be subclass of GuestGameInfo
|
assert isinstance(rematchLevel, int)
|
||||||
class HostedInfo():
|
self.rematchLevel = rematchLevel
|
||||||
def __init__(self, guestNames, **kwargs):
|
|
||||||
self.guestNames = guestNames
|
|
||||||
self.gid = kwargs.get('gid')
|
|
||||||
self.needsInvite = kwargs.get('needsInvite', True)
|
|
||||||
|
|
||||||
def setGid(self, gid):
|
def setGid(self, gid):
|
||||||
# set only once!
|
# set only once!
|
||||||
assert 8 == len(gid) and not self.gid
|
assert 8 == len(gid) and not self.gid
|
||||||
self.gid = gid
|
self.gid = gid
|
||||||
|
|
||||||
|
def gameOver(self):
|
||||||
|
return self.state.get('gameOver', False)
|
||||||
|
|
||||||
|
class GuestGameInfo(GameInfo):
|
||||||
|
def __init__(self, gid, rematchLevel):
|
||||||
|
super().__init__(gid, rematchLevel)
|
||||||
|
|
||||||
|
class HostGameInfo(GameInfo):
|
||||||
|
def __init__(self, guestNames, **kwargs):
|
||||||
|
super().__init__(kwargs.get('gid'), kwargs.get('rematchLevel'))
|
||||||
|
self.guestNames = guestNames
|
||||||
|
self.needsInvite = kwargs.get('needsInvite', True)
|
||||||
|
global gGamesMade;
|
||||||
|
gGamesMade += 1
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return 'gid: {}, guests: {}'.format(self.gid, self.guestNames)
|
return 'gid: {}, guests: {}'.format(self.gid, self.guestNames)
|
||||||
|
|
||||||
|
@ -102,13 +114,13 @@ class GameStatus():
|
||||||
dev = Device._devs.get(player)
|
dev = Device._devs.get(player)
|
||||||
initial = player[0]
|
initial = player[0]
|
||||||
arg2 = -1
|
arg2 = -1
|
||||||
status = dev.gameStates.get(gid)
|
gameState = dev.gameFor(gid).state
|
||||||
if status:
|
if gameState:
|
||||||
if status.get('gameOver', False):
|
if gameState.get('gameOver', False):
|
||||||
initial = initial.lower()
|
initial = initial.lower()
|
||||||
arg2 = status.get('nPending', 0)
|
arg2 = gameState.get('nPending', 0)
|
||||||
else:
|
else:
|
||||||
arg2 = status.get('nTiles')
|
arg2 = gameState.get('nTiles')
|
||||||
lineTxt += '{}{: 3}'.format(initial, arg2)
|
lineTxt += '{}{: 3}'.format(initial, arg2)
|
||||||
results.append(lineTxt)
|
results.append(lineTxt)
|
||||||
return ' '.join(results)
|
return ' '.join(results)
|
||||||
|
@ -136,13 +148,12 @@ class Device():
|
||||||
self.mqttDevID = None
|
self.mqttDevID = None
|
||||||
self.smsNumber = args.WITH_SMS and '{}_phone'.format(host) or None
|
self.smsNumber = args.WITH_SMS and '{}_phone'.format(host) or None
|
||||||
self.host = host
|
self.host = host
|
||||||
self.hostedGames = [] # array of HostedInfo for each game I host
|
self.hostedGames = [] # array of HostGameInfo for each game I host
|
||||||
self.guestGames = []
|
self.guestGames = []
|
||||||
self.script = '{}/{}.sh'.format(Device._logdir, host)
|
self.script = '{}/{}.sh'.format(Device._logdir, host)
|
||||||
self.dbName = '{}/{}.db'.format(Device._logdir, host)
|
self.dbName = '{}/{}.db'.format(Device._logdir, host)
|
||||||
self.logfile = '{}/{}_log.txt'.format(Device._logdir, host)
|
self.logfile = '{}/{}_log.txt'.format(Device._logdir, host)
|
||||||
self.cmdSocketName = '{}/{}.sock'.format(Device._logdir, host)
|
self.cmdSocketName = '{}/{}.sock'.format(Device._logdir, host)
|
||||||
self.gameStates = {}
|
|
||||||
self._keyCur = 10000 * (1 + g_NAMES.index(host))
|
self._keyCur = 10000 * (1 + g_NAMES.index(host))
|
||||||
|
|
||||||
def init(self):
|
def init(self):
|
||||||
|
@ -251,7 +262,7 @@ class Device():
|
||||||
if not self.endTime:
|
if not self.endTime:
|
||||||
self.launchIfNot()
|
self.launchIfNot()
|
||||||
elif datetime.datetime.now() > self.endTime:
|
elif datetime.datetime.now() > self.endTime:
|
||||||
self.quit()
|
self.rematchOrQuit()
|
||||||
elif self.moveOne():
|
elif self.moveOne():
|
||||||
pass
|
pass
|
||||||
else:
|
else:
|
||||||
|
@ -261,16 +272,20 @@ class Device():
|
||||||
|
|
||||||
# I may be a guest or a host in this game. Rematch works either
|
# I may be a guest or a host in this game. Rematch works either
|
||||||
# way. But how I figure out the other players differs.
|
# way. But how I figure out the other players differs.
|
||||||
def rematch(self, gid):
|
def rematch(self, game):
|
||||||
|
gid = game.gid
|
||||||
newGid = self._sendWaitReply('rematch', gid=gid).get('newGid')
|
newGid = self._sendWaitReply('rematch', gid=gid).get('newGid')
|
||||||
if newGid:
|
if newGid:
|
||||||
guests = Device.playersIn(gid)
|
guests = Device.playersIn(gid)
|
||||||
guests.remove(self.host)
|
guests.remove(self.host)
|
||||||
self._log('rematch: new host: {}; new guest[s]: {}, gid: {}'.format(self.host, guests, newGid))
|
self._log('rematch: new host: {}; new guest[s]: {}, gid: {}'.format(self.host, guests, newGid))
|
||||||
|
|
||||||
self.hostedGames.append(HostedInfo(guests, needsInvite=False, gid=newGid))
|
rematchLevel = game.rematchLevel - 1
|
||||||
|
assert rematchLevel >= 0 # fired
|
||||||
|
self.hostedGames.append(HostGameInfo(guests, needsInvite=False, gid=newGid,
|
||||||
|
rematchLevel=rematchLevel))
|
||||||
for guest in guests:
|
for guest in guests:
|
||||||
Device.getFor(guest).expectInvite(newGid)
|
Device.getFor(guest).expectInvite(newGid, rematchLevel)
|
||||||
|
|
||||||
def invite(self, game):
|
def invite(self, game):
|
||||||
failed = False
|
failed = False
|
||||||
|
@ -284,13 +299,13 @@ class Device():
|
||||||
channel=ii+1, addr=addr)
|
channel=ii+1, addr=addr)
|
||||||
|
|
||||||
if response['success']:
|
if response['success']:
|
||||||
guestDev.expectInvite(game.gid)
|
guestDev.expectInvite(game.gid, game.rematchLevel)
|
||||||
else:
|
else:
|
||||||
failed = True
|
failed = True
|
||||||
if not failed: game.needsInvite = False
|
if not failed: game.needsInvite = False
|
||||||
|
|
||||||
def expectInvite(self, gid):
|
def expectInvite(self, gid, rematchLevel):
|
||||||
self.guestGames.append(GuestGameInfo(gid))
|
self.guestGames.append(GuestGameInfo(gid, rematchLevel))
|
||||||
self.launchIfNot()
|
self.launchIfNot()
|
||||||
|
|
||||||
# Return true only if all games I host are finished on all games.
|
# Return true only if all games I host are finished on all games.
|
||||||
|
@ -311,7 +326,7 @@ class Device():
|
||||||
def gameOver(self, gid):
|
def gameOver(self, gid):
|
||||||
result = False
|
result = False
|
||||||
"""Is the game is over for *this* device"""
|
"""Is the game is over for *this* device"""
|
||||||
gameState = self.gameStates.get(gid, None)
|
gameState = self.gameFor(gid).state
|
||||||
if gameState:
|
if gameState:
|
||||||
result = gameState.get('gameOver', False) and 0 == gameState.get('nPending', 1)
|
result = gameState.get('gameOver', False) and 0 == gameState.get('nPending', 1)
|
||||||
# if result: self._log('gameOver({}) => {}'.format(gid, result))
|
# if result: self._log('gameOver({}) => {}'.format(gid, result))
|
||||||
|
@ -329,6 +344,32 @@ class Device():
|
||||||
withGid = [game for game in self._allGames() if gid == game.gid]
|
withGid = [game for game in self._allGames() if gid == game.gid]
|
||||||
return 0 < len(withGid)
|
return 0 < len(withGid)
|
||||||
|
|
||||||
|
def gameFor(self, gid):
|
||||||
|
for game in self._allGames():
|
||||||
|
if gid == game.gid:
|
||||||
|
return game
|
||||||
|
|
||||||
|
def rematchOrQuit(self):
|
||||||
|
if self.endTime:
|
||||||
|
gids = [game.gid for game in self._allGames() if not self.gameOver(game.gid)]
|
||||||
|
response = self._sendWaitReply('getStates', gids=gids)
|
||||||
|
|
||||||
|
anyRematched = False
|
||||||
|
for obj in response.get('states', []):
|
||||||
|
game = self.gameFor(obj.get('gid'))
|
||||||
|
game.state = obj
|
||||||
|
|
||||||
|
if game.gameOver() and 0 < game.rematchLevel:
|
||||||
|
self.rematch(game)
|
||||||
|
game.rematchLevel = 0 # so won't be used again
|
||||||
|
anyRematched = True
|
||||||
|
|
||||||
|
if not anyRematched:
|
||||||
|
response = self._sendWaitReply('quit')
|
||||||
|
self.watcher.join()
|
||||||
|
self.watcher = None
|
||||||
|
assert not self.endTime
|
||||||
|
|
||||||
def quit(self):
|
def quit(self):
|
||||||
if self.endTime:
|
if self.endTime:
|
||||||
allGames = self._allGames()
|
allGames = self._allGames()
|
||||||
|
@ -336,8 +377,7 @@ class Device():
|
||||||
response = self._sendWaitReply('quit', gids=gids)
|
response = self._sendWaitReply('quit', gids=gids)
|
||||||
|
|
||||||
for obj in response.get('states', []):
|
for obj in response.get('states', []):
|
||||||
gid = obj.get('gid')
|
self.gameFor(obj.get('gid')).state = obj
|
||||||
self.gameStates[gid] = obj
|
|
||||||
|
|
||||||
# wait for the thing to actually die
|
# wait for the thing to actually die
|
||||||
self.watcher.join()
|
self.watcher.join()
|
||||||
|
@ -422,7 +462,8 @@ class Device():
|
||||||
return result
|
return result
|
||||||
|
|
||||||
def addGameWith(self, guests):
|
def addGameWith(self, guests):
|
||||||
self.hostedGames.append(HostedInfo(guests, needsInvite=True))
|
self.hostedGames.append(HostGameInfo(guests, needsInvite=True,
|
||||||
|
rematchLevel=self.args.REMATCH_LEVEL))
|
||||||
for guest in guests:
|
for guest in guests:
|
||||||
Device.deviceFor(self.args, guest) # in case this device never hosts
|
Device.deviceFor(self.args, guest) # in case this device never hosts
|
||||||
|
|
||||||
|
@ -450,22 +491,11 @@ def openOnExit(args):
|
||||||
subprocess.Popen([str(arg) for arg in appargs], stdout = subprocess.DEVNULL,
|
subprocess.Popen([str(arg) for arg in appargs], stdout = subprocess.DEVNULL,
|
||||||
stderr = subprocess.DEVNULL, universal_newlines = True)
|
stderr = subprocess.DEVNULL, universal_newlines = True)
|
||||||
|
|
||||||
# Pick a game that's joined -- all invites accepted -- and call
|
|
||||||
# rematch on it. Return True if successful
|
|
||||||
def testRematch():
|
|
||||||
for dev in Device.getAll():
|
|
||||||
for gid, status in dev.gameStates.items():
|
|
||||||
if 2 < status.get('nMoves', 0):
|
|
||||||
dev.rematch(gid)
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
def mainLoop(args, devs):
|
def mainLoop(args, devs):
|
||||||
startCount = len(devs)
|
startCount = len(devs)
|
||||||
|
|
||||||
startTime = datetime.datetime.now()
|
startTime = datetime.datetime.now()
|
||||||
nextStallCheck = startTime + datetime.timedelta(seconds = 20)
|
nextStallCheck = startTime + datetime.timedelta(seconds = 20)
|
||||||
rematchTested = False
|
|
||||||
|
|
||||||
while 0 < len(devs):
|
while 0 < len(devs):
|
||||||
if gDone:
|
if gDone:
|
||||||
|
@ -478,8 +508,6 @@ def mainLoop(args, devs):
|
||||||
devs.remove(dev)
|
devs.remove(dev)
|
||||||
log(args, 'removed dev for {}; {} devs left'.format(dev.host, len(devs)))
|
log(args, 'removed dev for {}; {} devs left'.format(dev.host, len(devs)))
|
||||||
|
|
||||||
if not rematchTested: rematchTested = testRematch()
|
|
||||||
|
|
||||||
now = datetime.datetime.now()
|
now = datetime.datetime.now()
|
||||||
if devs and now > nextStallCheck:
|
if devs and now > nextStallCheck:
|
||||||
nextStallCheck = now + datetime.timedelta(seconds = 10)
|
nextStallCheck = now + datetime.timedelta(seconds = 10)
|
||||||
|
@ -601,8 +629,8 @@ def mkParser():
|
||||||
parser.add_argument('--board-size', dest = 'BOARD_SIZE', type = int, default = 15,
|
parser.add_argument('--board-size', dest = 'BOARD_SIZE', type = int, default = 15,
|
||||||
help = 'Use <n>x<n> size board')
|
help = 'Use <n>x<n> size board')
|
||||||
|
|
||||||
# parser.add_argument('--rematch-limit-secs', dest = 'REMATCH_SECS', type = int, default = 0,
|
parser.add_argument('--rematch-level', dest = 'REMATCH_LEVEL', type = int, default = 0,
|
||||||
# help = 'rematch games that end within this many seconds of script launch')
|
help = 'rematch games down to this ancestry/depth')
|
||||||
|
|
||||||
# envpat = 'DISCON_COREPAT'
|
# envpat = 'DISCON_COREPAT'
|
||||||
# parser.add_argument('--core-pat', dest = 'CORE_PAT', default = os.environ.get(envpat),
|
# parser.add_argument('--core-pat', dest = 'CORE_PAT', default = os.environ.get(envpat),
|
||||||
|
@ -666,6 +694,9 @@ def main():
|
||||||
dev.quit()
|
dev.quit()
|
||||||
mainLoop(args, devs)
|
mainLoop(args, devs)
|
||||||
|
|
||||||
|
elapsed = datetime.datetime.now() - startTime
|
||||||
|
print('played {} games in {}'.format(gGamesMade, elapsed))
|
||||||
|
|
||||||
if args.OPEN_ON_EXIT: openOnExit(args)
|
if args.OPEN_ON_EXIT: openOnExit(args)
|
||||||
|
|
||||||
##############################################################################
|
##############################################################################
|
||||||
|
|
Loading…
Reference in a new issue