mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-18 22:26:30 +01:00
843 lines
31 KiB
Python
Executable file
843 lines
31 KiB
Python
Executable file
#!/usr/bin/env python3
|
|
|
|
import argparse, datetime, glob, json, os, random, shutil, signal, \
|
|
socket, struct, subprocess, sys, threading, time
|
|
|
|
g_ROOT_NAMES = ['Brynn', 'Ariela', 'Kati', 'Eric']
|
|
# These must correspond to what the linux app is looking for in roFromStr()
|
|
g_ROS = ['same', 'low_score_first', 'high_score_first', 'juggle',]
|
|
gDone = False
|
|
gGamesMade = 0
|
|
g_LOGFILE = None
|
|
|
|
def log(args, msg):
|
|
now = datetime.datetime.strftime(datetime.datetime.now(), '%X.%f')
|
|
msg = '{} {}'.format(now, msg)
|
|
if args.VERBOSE:
|
|
print(msg)
|
|
global g_LOGFILE
|
|
if g_LOGFILE:
|
|
print(msg, file=g_LOGFILE)
|
|
g_LOGFILE.flush()
|
|
|
|
def pick_ndevs(args):
|
|
RNUM = random.randint(0, 99)
|
|
if RNUM > 95 and args.MAXDEVS >= 4:
|
|
NDEVS = 4
|
|
elif RNUM > 90 and args.MAXDEVS >= 3:
|
|
NDEVS = 3
|
|
else:
|
|
NDEVS = 2
|
|
if NDEVS < args.MINDEVS:
|
|
NDEVS = args.MINDEVS
|
|
return NDEVS
|
|
|
|
def makeNames(nDevs):
|
|
names = g_ROOT_NAMES[:nDevs]
|
|
for ii in range(len(names), nDevs):
|
|
newName = '{}{:02}'.format(g_ROOT_NAMES[ii%len(g_ROOT_NAMES)], ii//len(g_ROOT_NAMES))
|
|
names += [newName]
|
|
return names
|
|
|
|
def chooseNames(nPlayers):
|
|
global g_NAMES
|
|
players = g_NAMES[:]
|
|
result = []
|
|
for ii in range(nPlayers):
|
|
indx = random.randint(0, len(players)-1)
|
|
result.append(players.pop(indx))
|
|
return result
|
|
|
|
class GameInfo():
|
|
def __init__(self, device, gid, rematchLevel):
|
|
self.device = device
|
|
self.gid = gid
|
|
self.state = {}
|
|
assert isinstance(rematchLevel, int)
|
|
self.rematchLevel = rematchLevel
|
|
|
|
def setGid(self, gid):
|
|
# set only once!
|
|
assert 8 == len(gid) and not self.gid
|
|
self.gid = gid
|
|
|
|
def gameOver(self):
|
|
return self.state.get('gameOver', False)
|
|
|
|
def getDevice(self): return self.device
|
|
|
|
class GuestGameInfo(GameInfo):
|
|
def __init__(self, device, gid, rematchLevel):
|
|
super().__init__(device, gid, rematchLevel)
|
|
|
|
class HostGameInfo(GameInfo):
|
|
def __init__(self, device, guestNames, **kwargs):
|
|
super().__init__(device, kwargs.get('gid'), kwargs.get('rematchLevel'))
|
|
self.guestNames = guestNames
|
|
self.needsInvite = kwargs.get('needsInvite', True)
|
|
self.orderedPlayers = None
|
|
global gGamesMade;
|
|
gGamesMade += 1
|
|
|
|
def haveOrder(self): return self.orderedPlayers is not None
|
|
|
|
def setOrder(self, orderedPlayers): self.orderedPlayers = orderedPlayers
|
|
|
|
def __str__(self):
|
|
return 'gid: {}, guests: {}'.format(self.gid, self.guestNames)
|
|
|
|
class GameStatus():
|
|
_statuses = None
|
|
_prevLines = []
|
|
_lastChange = datetime.datetime.now()
|
|
|
|
def __init__(self, gid):
|
|
self.gid = gid
|
|
self.players = []
|
|
self.allOver = True
|
|
self.hostPlayerName = None
|
|
self.hostName = None
|
|
|
|
def harvest(self, dev):
|
|
self.players.append(dev.host)
|
|
self.allOver = self.allOver and dev.gameOver(self.gid)
|
|
|
|
def sortPlayers(self):
|
|
game = Device.getHostGame(self.gid)
|
|
orderedPlayers = game.orderedPlayers
|
|
if orderedPlayers:
|
|
assert len(orderedPlayers) == len(self.players)
|
|
self.players = orderedPlayers
|
|
self.hostName = game.getDevice().host
|
|
|
|
# Build a gid->status map for each game, querying each device in
|
|
# the game for details
|
|
@staticmethod
|
|
def makeAll():
|
|
statuses = {}
|
|
for dev in Device.getAll():
|
|
for game in dev._allGames():
|
|
gid = game.gid
|
|
assert 8 == len(gid)
|
|
if not gid in statuses: statuses[gid] = GameStatus(gid)
|
|
statuses[gid].harvest(dev)
|
|
|
|
for status in statuses.values():
|
|
status.sortPlayers()
|
|
|
|
GameStatus._statuses = statuses
|
|
|
|
@staticmethod
|
|
def summary():
|
|
now = datetime.datetime.strftime(datetime.datetime.now(), '%H:%M:%S')
|
|
count = sum([1 for one in GameStatus._statuses.values() if one.allOver])
|
|
return '{}; {} games; finished: {}'.format(now, len(GameStatus._statuses), count)
|
|
|
|
@staticmethod
|
|
def numLines():
|
|
maxPlayers = 0
|
|
for status in GameStatus._statuses.values():
|
|
nPlayers = len(status.players)
|
|
if nPlayers > maxPlayers: maxPlayers = nPlayers
|
|
return maxPlayers + 1
|
|
|
|
# For all games, print the proper line of status for that game in
|
|
# exactly 8 chars
|
|
@staticmethod
|
|
def line(indx):
|
|
results = []
|
|
for gid in sorted(GameStatus._statuses.keys()):
|
|
status = GameStatus._statuses[gid]
|
|
line = ''
|
|
if indx == 0:
|
|
line = gid
|
|
elif indx <= len(status.players) and not status.allOver:
|
|
player = status.players[indx-1]
|
|
hostMarker = status.hostName == player and '*' or ' '
|
|
initial = GameStatus._abbrev(player)
|
|
arg3 = -1
|
|
dev = Device._devs.get(player)
|
|
gameState = dev.gameFor(gid).state
|
|
if gameState:
|
|
if gameState.get('gameOver', False):
|
|
initial = initial.lower()
|
|
arg3 = gameState.get('nPending', 0)
|
|
else:
|
|
arg3 = gameState.get('nTiles')
|
|
line = '{}{:3}{: 3}'.format(hostMarker, initial, arg3)
|
|
results.append(line.center(len(gid)))
|
|
|
|
return ' '.join(results)
|
|
|
|
@staticmethod
|
|
def _abbrev(name):
|
|
for base in g_ROOT_NAMES:
|
|
if name.startswith(base):
|
|
return name[0] + name.strip(base)
|
|
|
|
class Device():
|
|
_devs = {}
|
|
_logdir = None
|
|
_nSteps = 0
|
|
|
|
@staticmethod
|
|
def setup(logdir):
|
|
Device._logdir = logdir
|
|
|
|
def __init__(self, args, host):
|
|
self.args = args
|
|
self.app = None
|
|
self.endTime = None
|
|
self.mqttDevID = None
|
|
self.smsNumber = args.WITH_SMS and '{}_phone'.format(host) or None
|
|
self.host = host
|
|
self.hostedGames = [] # array of HostGameInfo for each game I host
|
|
self.guestGames = []
|
|
self.script = '{}/{}.sh'.format(Device._logdir, host)
|
|
self.dbName = '{}/{}.db'.format(Device._logdir, host)
|
|
self.logfile = '{}/{}_log.txt'.format(Device._logdir, host)
|
|
self.cmdSocketName = '{}/{}.sock'.format(Device._logdir, host)
|
|
self._keyCur = 10000 * (1 + g_NAMES.index(host))
|
|
|
|
def init(self):
|
|
self._checkScript()
|
|
|
|
def _getApp(self):
|
|
# first time?
|
|
if not self.app:
|
|
pct = random.randint(0,99)
|
|
if pct < self.args.START_PCT:
|
|
self.app = self.args.APP_OLD
|
|
else:
|
|
self.app = self.args.APP_NEW
|
|
# upgrade time?
|
|
elif not self.app == self.args.APP_NEW:
|
|
pct = random.randint(0,99)
|
|
if pct < self.args.UPGRADE_PCT:
|
|
self._log('upgrading app')
|
|
self.app = self.args.APP_NEW
|
|
|
|
return self.app
|
|
|
|
# called by thread proc
|
|
def _launchProc(self):
|
|
assert not self.endTime
|
|
self.endTime = datetime.datetime.now() + datetime.timedelta(seconds = 5)
|
|
args = [ self.script, '--close-stdin' ]
|
|
if not self.args.USE_GTK: args.append('--curses')
|
|
|
|
env = os.environ.copy()
|
|
env['APP'] = self._getApp()
|
|
|
|
with open( self.logfile, 'a' ) as logfile:
|
|
subprocess.run(args, env=env, stdout = subprocess.DEVNULL,
|
|
stderr = logfile, universal_newlines = True)
|
|
self._log('_launchProc() (in thread): subprocess FINISHED')
|
|
os.unlink(self.cmdSocketName)
|
|
self.endTime = None
|
|
|
|
def launchIfNot(self):
|
|
if not self.endTime:
|
|
self.watcher = threading.Thread(target = Device.runnerStub, args=(self,))
|
|
self.watcher.isDaemon = True
|
|
self.watcher.start()
|
|
|
|
while not self.endTime or not os.path.exists(self.cmdSocketName):
|
|
time.sleep(0.2)
|
|
|
|
def moveOne(self):
|
|
moved = False
|
|
gids = [game.gid for game in self._allGames() if not self.gameOver(game.gid)]
|
|
random.shuffle(gids)
|
|
for gid in gids:
|
|
response = self._sendWaitReply('moveIf', gid=gid)
|
|
moved = response.get('success', False)
|
|
if moved: break
|
|
return moved
|
|
|
|
def _sendWaitReply(self, cmd, **kwargs):
|
|
self.launchIfNot()
|
|
|
|
client = socket.socket(socket.AF_UNIX, socket.SOCK_STREAM)
|
|
# print('connecting to: {}'.format(self.cmdSocketName))
|
|
|
|
client.connect(self.cmdSocketName);
|
|
|
|
key = self._nextKey()
|
|
params = [{'cmd': cmd, 'key': key, 'args': {**kwargs}}]
|
|
payload = json.dumps(params).encode()
|
|
client.send(struct.pack('!h', len(payload)))
|
|
client.sendall(payload)
|
|
|
|
# # Receive a response from the server
|
|
# self._log('_sendWaitReply({}): calling recv()'.format(cmd))
|
|
reslen = struct.unpack('!h', client.recv(2))[0]
|
|
response = client.recv(reslen).decode()
|
|
# self._log('_sendWaitReply({}): recv => str: {}'.format(cmd, response))
|
|
response = json.loads(response)
|
|
self._log('_sendWaitReply({}, {}): recv => {}'.format(cmd, kwargs, response))
|
|
assert 1 == len(response)
|
|
response = response[0]
|
|
assert response.get('key', 0) == key
|
|
assert response.get('cmd') == cmd
|
|
response = response.get('response')
|
|
|
|
client.close()
|
|
return response
|
|
|
|
def _nextKey(self):
|
|
self._keyCur += 1
|
|
return self._keyCur
|
|
|
|
def setDevID(self):
|
|
response = self._sendWaitReply('getMQTTDevID')
|
|
mqttDevID = response and response.get('mqtt')
|
|
if mqttDevID:
|
|
self.mqttDevID = mqttDevID
|
|
else:
|
|
printError('no mqtt or no response')
|
|
|
|
def makeGames(self):
|
|
args = self.args
|
|
for remote in self.hostedGames:
|
|
nPlayers = 1 + len(remote.guestNames)
|
|
hostPosn = random.randint(0, nPlayers-1)
|
|
traySize = 0 == args.TRAY_SIZE and random.randint(7, 9) or args.TRAY_SIZE
|
|
boardSize = random.choice(range(args.BOARD_SIZE_MIN, args.BOARD_SIZE_MAX+1, 2))
|
|
|
|
response = self._sendWaitReply('makeGame', nPlayers=nPlayers, hostPosn=hostPosn,
|
|
dict=args.DICTS[0], boardSize=boardSize,
|
|
traySize=traySize)
|
|
newGid = response.get('newGid')
|
|
if newGid:
|
|
remote.setGid(newGid)
|
|
|
|
# This is the heart of things. Do something as long as we have a
|
|
# game that needs to run.
|
|
def step(self):
|
|
# self._log('step() called for {}'.format(self))
|
|
stepped = False
|
|
for game in self.hostedGames:
|
|
if game.needsInvite:
|
|
self.invite(game)
|
|
stepped = True
|
|
break
|
|
|
|
if not stepped:
|
|
if not self.endTime:
|
|
self.launchIfNot()
|
|
elif datetime.datetime.now() > self.endTime:
|
|
self.rematchOrQuit()
|
|
elif self.moveOne():
|
|
pass
|
|
else:
|
|
# self._log('sleeping with {} to go'.format(self.endTime-now))
|
|
time.sleep(0.5)
|
|
stepped = True;
|
|
|
|
# I may be a guest or a host in this game. Rematch works either
|
|
# way. But how I figure out the other players differs.
|
|
def rematch(self, game):
|
|
gid = game.gid
|
|
rematchOrder = self.figureRematchOrder()
|
|
newGid = self._sendWaitReply('rematch', gid=gid, rematchOrder=rematchOrder) \
|
|
.get('newGid')
|
|
if newGid:
|
|
guests = Device.playersIn(gid)
|
|
guests.remove(self.host)
|
|
self._log('rematch: new host: {}; new guest[s]: {}, gid: {}'.format(self.host, guests, newGid))
|
|
|
|
rematchLevel = game.rematchLevel - 1
|
|
assert rematchLevel >= 0 # fired
|
|
self.hostedGames.append(HostGameInfo(self, guests, needsInvite=False, gid=newGid,
|
|
rematchLevel=rematchLevel))
|
|
for guest in guests:
|
|
Device.getForPlayer(guest).expectInvite(newGid, rematchLevel)
|
|
|
|
def invite(self, game):
|
|
failed = False
|
|
for ii in range(len(game.guestNames)):
|
|
guestDev = Device.getForPlayer(game.guestNames[ii])
|
|
|
|
addr = {}
|
|
if self.args.WITH_MQTT: addr['mqtt'] = guestDev.mqttDevID
|
|
if self.args.WITH_SMS: addr['sms'] = guestDev.smsNumber
|
|
response = self._sendWaitReply('invite', gid=game.gid,
|
|
channel=ii+1, addr=addr)
|
|
|
|
if response['success']:
|
|
guestDev.expectInvite(game.gid, game.rematchLevel)
|
|
else:
|
|
failed = True
|
|
if not failed: game.needsInvite = False
|
|
|
|
def expectInvite(self, gid, rematchLevel):
|
|
self.guestGames.append(GuestGameInfo(self, gid, rematchLevel))
|
|
self.launchIfNot()
|
|
|
|
def figureRematchOrder(self):
|
|
ro = self.args.REMATCH_ORDER
|
|
if not ro: ro = random.choice(g_ROS)
|
|
return ro
|
|
|
|
# Return true only if all games I host are finished on all games.
|
|
# But: what about games I don't host? For now, let's make it all
|
|
# games!
|
|
def finished(self):
|
|
allGames = self._allGames()
|
|
result = 0 < len(allGames)
|
|
for game in allGames:
|
|
if not result: break
|
|
peers = Device.devsWith(game.gid)
|
|
for dev in peers:
|
|
result = dev.gameOver(game.gid)
|
|
if not result: break
|
|
# if result: self._log('finished() => {}'.format(result))
|
|
return result
|
|
|
|
def gameOver(self, gid):
|
|
result = False
|
|
"""Is the game is over for *this* device"""
|
|
gameState = self.gameFor(gid).state
|
|
if gameState:
|
|
result = gameState.get('gameOver', False) and 0 == gameState.get('nPending', 1)
|
|
# if result: self._log('gameOver({}) => {}'.format(gid, result))
|
|
return result
|
|
|
|
# this device is stalled if none of its unfinshed games has
|
|
# changed state in some interval
|
|
def stalled(self):
|
|
return False
|
|
|
|
def _allGames(self):
|
|
return self.hostedGames + self.guestGames
|
|
|
|
def haveGame(self, gid):
|
|
withGid = [game for game in self._allGames() if gid == game.gid]
|
|
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)]
|
|
|
|
orders = []
|
|
for gid in gids:
|
|
game = Device.getHostGame(gid)
|
|
if game and not game.haveOrder():
|
|
orders.append(gid)
|
|
|
|
response = self._sendWaitReply('getStates', gids=gids, orders=orders)
|
|
|
|
for order in response.get('orders', []):
|
|
gid = order.get('gid')
|
|
game = Device.getHostGame(gid)
|
|
assert isinstance(game, HostGameInfo) # firing
|
|
game.setOrder(order.get('players'))
|
|
|
|
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):
|
|
if self.endTime:
|
|
allGames = self._allGames()
|
|
gids = [game.gid for game in allGames if not self.gameOver(game.gid)]
|
|
response = self._sendWaitReply('quit', gids=gids)
|
|
|
|
for obj in response.get('states', []):
|
|
self.gameFor(obj.get('gid')).state = obj
|
|
|
|
# wait for the thing to actually die
|
|
self.watcher.join()
|
|
self.watcher = None
|
|
assert not self.endTime
|
|
|
|
def _checkScript(self):
|
|
if not os.path.exists(self.script):
|
|
scriptArgs = ['exec'] # without exec means terminate() won't work
|
|
if self.args.VALGRIND:
|
|
scriptArgs += ['valgrind']
|
|
# args += ['--leak-check=full']
|
|
# args += ['--track-origins=yes']
|
|
|
|
scriptArgs.append('"${APP}"')
|
|
|
|
scriptArgs += '--db', self.dbName, '--skip-confirm'
|
|
if self.args.SEND_CHAT:
|
|
scriptArgs += '--send-chat', self.args.SEND_CHAT
|
|
scriptArgs += '--localName', self.host
|
|
scriptArgs += '--cmd-socket-name', self.cmdSocketName
|
|
|
|
if self.args.WITH_MQTT:
|
|
scriptArgs += [ '--mqtt-port', self.args.MQTT_PORT, '--mqtt-host', self.args.MQTT_HOST ]
|
|
|
|
if self.args.WITH_SMS:
|
|
scriptArgs += [ '--sms-number', self.smsNumber ]
|
|
|
|
scriptArgs += ['--board-size', '15', '--sort-tiles']
|
|
|
|
# useDupeMode = random.randint(0, 100) < self.args.DUP_PCT
|
|
# if not useDupeMode: scriptArgs += ['--trade-pct', self.args.TRADE_PCT]
|
|
|
|
# if self.devID: args.extend( ' '.split(self.devID))
|
|
scriptArgs += [ '$*' ]
|
|
|
|
with open( self.script, 'w' ) as fil:
|
|
fil.write('#!/bin/bash\n' )
|
|
fil.write('APP="${{APP:-{}}}"\n'.format(self.args.APP_NEW))
|
|
fil.write(' '.join([str(arg) for arg in scriptArgs]) + '\n')
|
|
os.chmod(self.script, 0o755)
|
|
|
|
@staticmethod
|
|
def printStatus(statusSteps):
|
|
Device._nSteps += 1
|
|
print('.', end='', flush=True)
|
|
if 0 == Device._nSteps % statusSteps:
|
|
GameStatus.makeAll()
|
|
print(' ' + GameStatus.summary())
|
|
|
|
lines = [GameStatus.line(ii) for ii in range(GameStatus.numLines())]
|
|
now = datetime.datetime.now()
|
|
if lines == GameStatus._prevLines:
|
|
print('no change in {}'.format(now - GameStatus._lastChange))
|
|
else:
|
|
for line in lines: print(line)
|
|
GameStatus._prevLines = lines
|
|
GameStatus._lastChange = now
|
|
|
|
@staticmethod
|
|
def deviceFor(args, host):
|
|
dev = Device._devs.get(host)
|
|
if not dev:
|
|
dev = Device(args, host)
|
|
Device._devs[host] = dev
|
|
return dev
|
|
|
|
@staticmethod
|
|
def playersIn(gid):
|
|
return [dev.host for dev in Device.devsWith(gid)]
|
|
|
|
@staticmethod
|
|
# return all devices (up to 4 of them) that are host or guest in a
|
|
# game with <gid>"""
|
|
def devsWith(gid):
|
|
result = [dev for dev in Device._devs.values() if dev.haveGame(gid)]
|
|
return result
|
|
|
|
@staticmethod
|
|
def getAll():
|
|
return [dev for dev in Device._devs.values()]
|
|
|
|
@staticmethod
|
|
def getForPlayer(player):
|
|
result = None
|
|
for dev in Device.getAll():
|
|
if dev.host == player:
|
|
result = dev
|
|
break;
|
|
assert result
|
|
return result
|
|
|
|
@staticmethod
|
|
def getHostGame(gid):
|
|
devs = Device.devsWith(gid)
|
|
for dev in devs:
|
|
game = dev.gameFor(gid)
|
|
if isinstance(game, HostGameInfo):
|
|
return game
|
|
assert False
|
|
|
|
def addGameWith(self, guests):
|
|
self.hostedGames.append(HostGameInfo(self, guests, needsInvite=True,
|
|
rematchLevel=self.args.REMATCH_LEVEL))
|
|
for guest in guests:
|
|
Device.deviceFor(self.args, guest) # in case this device never hosts
|
|
|
|
def _log(self, msg):
|
|
log(self.args, '{}: {}'.format(self.host, msg))
|
|
|
|
def __str__(self):
|
|
result = 'host: {}, devID: {}, with {} games: ' \
|
|
.format(self.host, self.mqttDevID,
|
|
len(self.hostedGames)+len(self.guestGames))
|
|
result += '{' + ', '.join(['{}'.format(game) for game in self._allGames()]) + '}'
|
|
result += ' running={}'.format(self.endTime is not None)
|
|
return result
|
|
|
|
@staticmethod
|
|
def runnerStub(self):
|
|
self._launchProc()
|
|
|
|
def openOnExit(args):
|
|
devs = Device.getAll()
|
|
for dev in devs:
|
|
appargs = [args.APP_NEW, '--db', dev.dbName]
|
|
if args.WITH_MQTT:
|
|
appargs += [ '--mqtt-port', args.MQTT_PORT, '--mqtt-host', args.MQTT_HOST ]
|
|
subprocess.Popen([str(arg) for arg in appargs], stdout = subprocess.DEVNULL,
|
|
stderr = subprocess.DEVNULL, universal_newlines = True)
|
|
|
|
|
|
def countCores(args):
|
|
count = 0
|
|
if args.CORE_PAT:
|
|
count = len( glob.glob(args.CORE_PAT) )
|
|
return count
|
|
|
|
def mainLoop(args, devs):
|
|
startCount = len(devs)
|
|
nCores = countCores(args)
|
|
|
|
startTime = datetime.datetime.now()
|
|
nextStallCheck = startTime + datetime.timedelta(seconds = 20)
|
|
|
|
while 0 < len(devs):
|
|
if gDone:
|
|
print('gDone set; exiting loop')
|
|
break
|
|
elif nCores < countCores(args):
|
|
print('core file count increased; exiting')
|
|
break
|
|
|
|
dev = random.choice(devs)
|
|
dev.step()
|
|
if dev.finished():
|
|
dev.quit()
|
|
devs.remove(dev)
|
|
log(args, 'removed dev for {}; {} devs left'.format(dev.host, len(devs)))
|
|
|
|
now = datetime.datetime.now()
|
|
if devs and now > nextStallCheck:
|
|
nextStallCheck = now + datetime.timedelta(seconds = 10)
|
|
allStalled = True
|
|
for dev in devs:
|
|
if not dev.stalled():
|
|
allStalled = False
|
|
break
|
|
if allStalled:
|
|
log(args, 'exiting mainLoop with {} left (of {}) because all stalled' \
|
|
.format(len(devs), startCount))
|
|
break
|
|
|
|
if False and endTime < datetime.datetime.now():
|
|
log(args, 'exiting mainLoop with {} left (of {}) because out of time' \
|
|
.format(len(devs), startCount))
|
|
break
|
|
|
|
if not args.VERBOSE:
|
|
Device.printStatus(args.STATUS_STEPS)
|
|
|
|
for dev in devs:
|
|
print('killing {}'.format(dev.host))
|
|
dev.quit()
|
|
|
|
# We will build one Device for each player in the set of games, and
|
|
# prime each with enough information that when we start running them
|
|
# they can invite each other.
|
|
def build_devs(args):
|
|
global g_NAMES
|
|
g_NAMES = makeNames(args.NDEVS)
|
|
for ii in range(args.NGAMES):
|
|
nPlayers = pick_ndevs(args)
|
|
players = chooseNames(nPlayers)
|
|
host = players[0]
|
|
guests = players[1:]
|
|
|
|
Device.deviceFor(args, host).addGameWith(guests)
|
|
|
|
return Device.getAll()
|
|
|
|
def mkParser():
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument('--status-steps', dest = 'STATUS_STEPS', type = int, default = 20,
|
|
help = 'how many steps between status dumps (matters only if not --debug)')
|
|
|
|
parser.add_argument('--send-chat', dest = 'SEND_CHAT', type = str, default = None,
|
|
help = 'the message to send')
|
|
|
|
parser.add_argument('--app-new', dest = 'APP_NEW', default = './obj_linux_memdbg/xwords',
|
|
help = 'the app we\'ll use')
|
|
parser.add_argument('--app-old', dest = 'APP_OLD', default = './obj_linux_memdbg/xwords',
|
|
help = 'the app we\'ll upgrade from')
|
|
parser.add_argument('--start-pct', dest = 'START_PCT', default = 50, type = int,
|
|
help = 'odds of starting with the new app, 0 <= n < 100')
|
|
parser.add_argument('--upgrade-pct', dest = 'UPGRADE_PCT', default = 0, type = int,
|
|
help = 'odds of upgrading at any launch, 0 <= n < 100')
|
|
|
|
parser.add_argument('--num-games', dest = 'NGAMES', type = int, default = 1, help = 'number of games')
|
|
parser.add_argument('--num-devs', dest = 'NDEVS', type = int, default = len(g_ROOT_NAMES), help = 'number of devices')
|
|
parser.add_argument('--timeout-mins', dest = 'TIMEOUT_MINS', default = 10000, type = int,
|
|
help = 'minutes after which to timeout')
|
|
# parser.add_argument('--nochange-secs', dest = 'NO_CHANGE_SECS', default = 30, type = int,
|
|
# help = 'seconds without change after which to timeout')
|
|
# parser.add_argument('--log-root', dest='LOGROOT', default = '.', help = 'where logfiles go')
|
|
# parser.add_argument('--dup-packets', dest = 'DUP_PACKETS', default = False, action = 'store_true',
|
|
# help = 'send all packet twice')
|
|
# parser.add_argument('--phonies', dest = 'PHONIES', default = -1, type = int,
|
|
# help = '0 (ignore), 1 (warn)) or 2 (lose turn); default is pick at random')
|
|
# parser.add_argument('--make-phony-pct', dest = 'PHONY_PCT', default = 20, type = int,
|
|
# help = 'how often a robot should play a phony (only applies when --phonies==2')
|
|
parser.add_argument('--use-gtk', dest = 'USE_GTK', default = False, action = 'store_true',
|
|
help = 'run games using gtk instead of ncurses')
|
|
|
|
# parser.add_argument('--dup-pct', dest = 'DUP_PCT', default = 0, type = int,
|
|
# help = 'this fraction played in duplicate mode')
|
|
|
|
# #
|
|
# # echo " [--clean-start] \\" >&2
|
|
parser.add_argument('--game-dict', dest = 'DICTS', action = 'append', default = [])
|
|
# # echo " [--help] \\" >&2
|
|
# # echo " [--max-devs <int>] \\" >&2
|
|
parser.add_argument('--min-devs', dest = 'MINDEVS', type = int, default = 2,
|
|
help = 'No game will have fewer devices than this')
|
|
parser.add_argument('--max-devs', dest = 'MAXDEVS', type = int, default = 4,
|
|
help = 'No game will have more devices than this')
|
|
|
|
# parser.add_argument('--robots-all-same-iq', dest = 'IQS_SAME', default = False,
|
|
# action = 'store_true', help = 'give all robots the same IQ')
|
|
|
|
# parser.add_argument('--min-run', dest = 'MINRUN', type = int, default = 2,
|
|
# help = 'Keep each run alive at least this many seconds')
|
|
# # echo " [--new-app <path/to/app] \\" >&2
|
|
# # echo " [--new-app-args [arg*]] # passed only to new app \\" >&2
|
|
# # echo " [--num-rooms <int>] \\" >&2
|
|
# # echo " [--old-app <path/to/app]* \\" >&2
|
|
# parser.add_argument('--one-per', dest = 'ONEPER', default = False,
|
|
# action = 'store_true', help = 'force one player per device')
|
|
# parser.add_argument('--resign-pct', dest = 'RESIGN_PCT', default = 0, type = int, \
|
|
# help = 'Odds of resigning [0..100]')
|
|
parser.add_argument('--seed', type = int, dest = 'SEED', default = 0)
|
|
# # echo " [--send-chat <interval-in-seconds> \\" >&2
|
|
# # echo " [--udp-incr <pct>] \\" >&2
|
|
# # echo " [--udp-start <pct>] # default: $UDP_PCT_START \\" >&2
|
|
# # echo " [--undo-pct <int>] \\" >&2
|
|
|
|
# parser.add_argument('--undo-pct', dest = 'UNDO_PCT', default = 0, type = int)
|
|
# parser.add_argument('--trade-pct', dest = 'TRADE_PCT', default = 10, type = int)
|
|
|
|
parser.add_argument('--with-sms', dest = 'WITH_SMS', action = 'store_true')
|
|
parser.add_argument('--without-sms', dest = 'WITH_SMS', default = False, action = 'store_false')
|
|
# parser.add_argument('--sms-fail-pct', dest = 'SMS_FAIL_PCT', default = 0, type = int)
|
|
|
|
parser.add_argument('--with-mqtt', dest = 'WITH_MQTT', default = True, action = 'store_true')
|
|
parser.add_argument('--without-mqtt', dest = 'WITH_MQTT', action = 'store_false')
|
|
parser.add_argument('--mqtt-port', dest = 'MQTT_PORT', default = 1883 )
|
|
parser.add_argument('--mqtt-host', dest = 'MQTT_HOST', default = 'localhost' )
|
|
|
|
parser.add_argument('--force-tray', dest = 'TRAY_SIZE', default = 0, type = int,
|
|
help = 'Always this many tiles per tray')
|
|
parser.add_argument('--board-size-min', dest = 'BOARD_SIZE_MIN', type = int, default = 11,
|
|
help = 'give boards at least this many rows and columns')
|
|
parser.add_argument('--board-size-max', dest = 'BOARD_SIZE_MAX', type = int, default = 23,
|
|
help = 'give boards no more than this many rows and columns')
|
|
|
|
parser.add_argument('--rematch-level', dest = 'REMATCH_LEVEL', type = int, default = 0,
|
|
help = 'rematch games down to this ancestry/depth')
|
|
parser.add_argument('--rematch-order', dest = 'REMATCH_ORDER', type = str, default = None,
|
|
help = 'order rematched games one of these ways: {}'.format(g_ROS))
|
|
|
|
envpat = 'DISCON_COREPAT'
|
|
parser.add_argument('--core-pat', dest = 'CORE_PAT', default = os.environ.get(envpat),
|
|
help = "pattern for core files that should stop the script " \
|
|
+ "(default from env {})".format(envpat) )
|
|
|
|
parser.add_argument('--with-valgrind', dest = 'VALGRIND', default = False,
|
|
action = 'store_true')
|
|
|
|
parser.add_argument('--debug', dest='VERBOSE', default=False, action='store_true',
|
|
help='log stuff')
|
|
parser.add_argument('--open-on-exit', dest = 'OPEN_ON_EXIT', default = False,
|
|
action = 'store_true', help='Open devs in gtk app when finished')
|
|
|
|
return parser
|
|
|
|
def parseArgs():
|
|
args = mkParser().parse_args()
|
|
assignDefaults(args)
|
|
print(args)
|
|
return args
|
|
# print(options)
|
|
|
|
def assignDefaults(args):
|
|
if len(args.DICTS) == 0: args.DICTS.append('CollegeEng_2to8.xwd')
|
|
assert 1 == (args.BOARD_SIZE_MAX % 2)
|
|
assert 1 == (args.BOARD_SIZE_MIN % 2)
|
|
assert args.BOARD_SIZE_MAX >= args.BOARD_SIZE_MIN
|
|
assert args.BOARD_SIZE_MIN >= 11
|
|
assert args.BOARD_SIZE_MAX <= 23
|
|
|
|
def termHandler(signum, frame):
|
|
global gDone
|
|
print('termHandler() called')
|
|
gDone = True
|
|
|
|
def printError(msg): print( 'ERROR: {}'.format(msg))
|
|
|
|
def initLogs():
|
|
scriptName = os.path.splitext(os.path.basename(sys.argv[0]))[0]
|
|
logdir = scriptName + '_logs'
|
|
if os.path.exists(logdir):
|
|
shutil.move(logdir, '/tmp/{}_{}'.format(logdir, os.getpid()))
|
|
os.mkdir(logdir)
|
|
|
|
logfilepath = '{}/{}_log.txt'.format(logdir, scriptName)
|
|
global g_LOGFILE
|
|
g_LOGFILE = open(logfilepath, 'w')
|
|
|
|
return logdir
|
|
|
|
def main():
|
|
startTime = datetime.datetime.now()
|
|
signal.signal(signal.SIGINT, termHandler)
|
|
|
|
logdir = initLogs()
|
|
|
|
args = parseArgs()
|
|
|
|
if args.SEED: random.seed(args.SEED)
|
|
# Hack: old files confuse things. Remove is simple fix good for now
|
|
if args.WITH_SMS:
|
|
try: rmtree('/tmp/xw_sms')
|
|
except: None
|
|
|
|
Device.setup(logdir) # deletes old logdif
|
|
devs = build_devs(args)
|
|
for dev in devs:
|
|
dev.init()
|
|
dev.launchIfNot()
|
|
time.sleep(1)
|
|
for dev in devs:
|
|
if args.WITH_MQTT: dev.setDevID()
|
|
for dev in devs:
|
|
dev.makeGames()
|
|
dev.quit()
|
|
mainLoop(args, devs)
|
|
|
|
elapsed = datetime.datetime.now() - startTime
|
|
print('played {} games in {}'.format(gGamesMade, elapsed))
|
|
|
|
if g_LOGFILE: g_LOGFILE.close()
|
|
|
|
if args.OPEN_ON_EXIT: openOnExit(args)
|
|
|
|
##############################################################################
|
|
if __name__ == '__main__':
|
|
main()
|