-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathgameserver.py
executable file
·762 lines (709 loc) · 30.6 KB
/
gameserver.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
#!/usr/bin/env python
#SpaceHack! Game server main module
#York Hackspace January 2014
#This runs on a Raspberry Pi
import controls
import paho.mqtt.client as paho
import time
import random
import json
import logging as log
from GameStarter.gamestart import GameStarter
lifeDisplay = False
sound = True #Switch this off if you don't have pyGame
debugMode = False
#sleep times
blurbSleep = 4.5
if lifeDisplay:
import seven_segment_display as sev
import led_sign as led
def playSound(filename):
"""Play a sound, if enabled"""
if sound:
snd = pygame.mixer.Sound("sounds/48k/" + filename)
snd.play()
return snd
def resetSound():
if sound:
pygame.mixer.quit()
pygame.mixer.init(48000, -16, 2, 1024) #was 1024
if sound:
import pygame
#MQTT client to allow publishing
client = paho.Client('PiServer') # ID shown to the broker
server = "127.0.0.1" #Mosquitto MQTT broker running locally
# Clever start button handling
gs = None
def initGameStarter():
global gs
gs = GameStarter(4, 1.5, 5.0)
initGameStarter()
gsIDs = {}
nextID = 0
#Game variables
consoles = [] #all registered consoles
players = [] #all participating players
playerstats = {}
console = {}
currentsetup = {}
currenttimeout = 30.0 #TODO: Is this needed here? We set it in initGame()
lastgenerated = time.time()
numinstructions = 0
gamestate = 'initserver' #initserver, readytostart, waitingforplayers, initgame, setupround, playround, roundover, hyperspace, gameover
warningsound = None
game_state = {
"skip_intro": False
}
#Show when we've connected
def on_connect(mosq, obj, rc):
"""Receive MQTT connection notification"""
if rc == 0:
print("Connected to MQTT")
global gamestate
gamestate = 'readytostart'
else:
print("Failed - return code is " + rc)
def game_exit():
"""It's harsh right now but we should probably clean up a bit here."""
log.info("Quitting the Game at game exit")
sys.exit(0)
def skip_introduction():
log.info("Setting game state to be skipped")
game_state['skip_intro'] = True
def stop_game():
resetToWaiting()
resetSound()
def add_life():
playerstats['game']['lives'] += 1
showLives()
def process_command_message(msg):
log.info('Command Received. Topic: %s Message: %s', msg.topic, msg.payload)
try:
command = msg.topic.split('/')[1]
except IndexError:
log.error("Command: Failed to find command in %s", msg.topic)
else:
if command == "exit":
game_exit()
elif command == "skip_intro":
skip_introduction()
elif command == "stop_game":
stop_game()
elif command == "add_life":
add_life()
else:
log.warning("Bad command: %s", command)
def on_message(mosq, obj, msg):
"""Receive and process incoming MQTT published message"""
if msg.topic.startswith('command/'):
process_command_message(msg)
nodes = msg.topic.split('/')
print(gamestate + ' - ' + msg.topic + " - " + str(msg.payload))
if nodes[0]=='server':
if nodes[1]=='register':
config = json.loads(str(msg.payload))
consoleip = config['ip']
console[consoleip] = config
if not consoleip in consoles:
consoles.append(consoleip)
#set console up for game start
consolesetup = {}
if gamestate in ['readytostart', 'waitingforplayers']:
#Still waiting to start
consolesetup['instructions'] = controls.blurb['readytostart']
consolesetup['timeout'] = 0.0
consolesetup['controls'] = {}
print(config['controls'])
for control in config['controls']:
ctrlid = control['id']
consolesetup['controls'][ctrlid]={}
if 'gamestart' in control:
consolesetup['controls'][ctrlid]['type'] = 'button'
consolesetup['controls'][ctrlid]['enabled'] = 1
consolesetup['controls'][ctrlid]['name'] = controls.blurb['startbutton']
consolesetup['controls'][ctrlid]['gamestart'] = True
consolesetup['controls'][ctrlid]['definition'] = {}
else:
consolesetup['controls'][ctrlid]['type'] = 'inactive'
consolesetup['controls'][ctrlid]['enabled'] = 0
consolesetup['controls'][ctrlid]['name'] = ""
client.subscribe('clients/' + consoleip + '/' + ctrlid + '/value')
else:
#There's a game on, but this client's late for it
if consoleip in players:
players.remove(consoleip)
#Was this the last remaining player?
if len(players) == 0:
#back to start
resetToWaiting()
return
else:
#Game still active - sit the rest of it out
consolesetup['instructions'] = controls.blurb['gameinprogress']
consolesetup['controls'] = {}
for control in config['controls']:
ctrlid = control['id']
consolesetup['controls'][ctrlid]={}
consolesetup['controls'][ctrlid]['type'] = 'inactive'
consolesetup['controls'][ctrlid]['enabled'] = 0
consolesetup['controls'][ctrlid]['name'] = ""
client.subscribe('clients/' + consoleip + '/' + ctrlid + '/value')
if len(consolesetup) > 0:
currentsetup[consoleip] = consolesetup
client.publish('clients/' + consoleip + '/configure', json.dumps(consolesetup))
elif nodes[0] == 'clients':
consoleip = nodes[1]
ctrlid = nodes[2]
if nodes[3] == 'value':
value = str(msg.payload)
if consoleip in currentsetup:
if 'controls' in currentsetup[consoleip]:
if currentsetup[consoleip]['controls'][ctrlid]['type'] in ['button', 'toggle', 'selector']:
try:
value = int(value)
except ValueError:
return
receiveValue(consoleip, ctrlid, value)
def receiveValue(consoleip, ctrlid, value):
"""Process a received value for a control"""
global lastgenerated
global gamestate
global numinstructions
global gsIDs
global nextID
if gamestate == 'playround':
#Check posted value against current targets
matched = False
if 'definition' in currentsetup[consoleip]['controls'][ctrlid]:
currentsetup[consoleip]['controls'][ctrlid]['definition']['value'] = value
for targetip in players:
consoledef = console[targetip]
if ('target' in consoledef and consoledef['target']['console'] == consoleip
and consoledef['target']['control'] == ctrlid
and str(consoledef['target']['value']) == str(value)):
#Match
matched = True
clearCorruption(consoleip, ctrlid)
playSound(random.choice(controls.soundfiles['right']))
#update stats
playerstats[targetip]['instructions']['hit'] += 1
playerstats[consoleip]['targets']['hit'] += 1
numinstructions -= 1
if numinstructions <= 0:
#Round over
roundOver()
else:
#Pick a new target and carry on
pickNewTarget(targetip)
if not matched: #Need to also check if a game round has begun yet
#Suppress caring about button releases - only important in game starts
if not (currentsetup[consoleip]['controls'][ctrlid]['type'] == 'button' and str(value) == "0"):
playSound(random.choice(controls.soundfiles['wrong']))
elif gamestate == 'setupround':
if 'definition' in currentsetup[consoleip]['controls'][ctrlid]:
currentsetup[consoleip]['controls'][ctrlid]['definition']['value'] = value
elif gamestate in ['readytostart', 'waitingforplayers']:
#button push?
if 'gamestart' in currentsetup[consoleip]['controls'][ctrlid]:
if value:
#Add to list of players
if not consoleip in gsIDs:
gsIDs[consoleip] = nextID
nextID += 1
#players.append(consoleip)
gs.push(gsIDs[consoleip])
else:
#remove from list of players
if consoleip in gsIDs:
gs.release(gsIDs[consoleip])
#players.remove(consoleip)
#Either way, reset the clock for game start
gamestate = 'waitingforplayers'
lastgenerated = time.time()
#Define a new set of controls for each client for this game round and send it to them as JSON.
def defineControls():
"""Define a new set of controls for each client for this game round and send it to them as JSON."""
emergency = controls.getEmergency()
print(emergency)
for consoleip in players:
print("Defining console " + consoleip)
consolesetup={}
consolesetup['instructions']=emergency
consolesetup['timeout'] = currenttimeout
consolesetup['controls']={}
#Pay attention to 'enabled' for the control as a whole
for control in (x for x in console[consoleip]["controls"] if 'enabled' not in x or x['enabled'] == 1):
ctrlid = control['id']
consolesetup['controls'][ctrlid]={}
#Pay attention to 'enabled' attribute
if 'enabled' in control:
consolesetup['controls'][ctrlid]['enabled']=control['enabled']
else:
consolesetup['controls'][ctrlid]['enabled']=1
#In case LCDs fail - allow a 'fixed name' we can tape over the LCD
if 'fixedname' in control:
consolesetup['controls'][ctrlid]['name'] = str(control['fixedname'])
else: #Normal case - generate a new control name
consolesetup['controls'][ctrlid]['name']=controls.getControlName(control['width'], 2, 12)
#Pay attention to 'enabled' for particular supported mode
ctrldef = random.choice([x for x in control['supported'] if 'enabled' not in x or x['enabled'] == 1])
ctrltype = ctrldef['type']
if ctrltype in ['words', 'verbs']:
if ctrldef['fixed']:
targetrange = ctrldef['list']
elif 'safe' in ctrldef and ctrldef['safe']:
targetrange=controls.safewords
elif 'list' in ctrldef:
if ctrldef['list']=='allcontrolwords':
targetrange=controls.allcontrolwords
elif ctrldef['list']=='passwd':
targetrange=controls.passwd
elif ctrldef['list']=='verbs':
targetrange=controls.verbs
else:
targetrange = controls.allcontrolwords
elif ctrltype=='verbs':
targetrange = controls.verbs
else:
targetrange = controls.allcontrolwords
#Create a predetermined list?
if not ctrldef['fixed']:
reallyfinished = False
while not reallyfinished:
wordpool = []
finished=False
while not finished:
newword = random.choice(targetrange)
if not newword in wordpool:
wordpool.append(newword)
if len(wordpool) == ctrldef['quantity']:
finished=True
if ctrldef['quantity'] != 2 or len(wordpool[0]) + len(wordpool[1]) < 14:
reallyfinished = True
ctrldef['pool'] = sorted(wordpool)
else:
ctrldef['pool'] = ctrldef['list']
#Pick a starting value
if 'assignable' in ctrldef and ctrldef['assignable']:
if ctrltype in ['words', 'verbs']:
ctrldef['value']=random.choice(ctrldef['pool'])
elif ctrltype == 'selector':
ctrldef['value'] = random.choice(range(ctrldef['min'],ctrldef['max']+1))
elif ctrltype == 'colour':
ctrldef['value'] = random.choice(ctrldef['values'])
elif ctrltype == 'toggle':
ctrldef['value'] = random.choice(range(2))
elif ctrltype == 'button':
ctrldef['value'] = 0
elif ctrltype == 'pin':
ctrldef['value'] = ''
consolesetup['controls'][ctrlid]['type'] = ctrltype
consolesetup['controls'][ctrlid]['definition']=ctrldef
print("Control " + ctrlid + " is " + ctrldef['type'] + ": " + consolesetup['controls'][ctrlid]['name'])
currentsetup[consoleip]=consolesetup
client.publish('clients/' + consoleip + '/configure', json.dumps(consolesetup))
#Get a choice from a range that isn't the same as the old value
def getChoice(choicerange, oldval):
"""Get a choice from a range that isn't the same as the old value."""
finished=False
while not finished:
retval = random.choice(choicerange)
if retval != oldval:
finished=True
return retval
#Pick a new instruction to display on a given console
def pickNewTarget(consoleip):
"""Pick a new instruction to display on a given console."""
#pick a random console and random control from that console
targetconsole = random.choice(players)
targetsetup = currentsetup[targetconsole]
targetctrlid = random.choice(targetsetup['controls'].keys())
targetcontrol = targetsetup['controls'][targetctrlid]
targetname = targetcontrol['name']
targetdef = targetcontrol['definition']
targettimeout = currenttimeout
if 'scalefactor' in targetdef:
targettimeout *= targetdef['scalefactor']
targetinstruction = ''
#pick a new target based on the control type and current value
ctrltype = targetcontrol['type']
if 'value' in targetdef:
curval = targetdef['value']
else:
curval=''
if ctrltype == 'button':
targetval=1
targetinstruction = controls.getButtonAction(targetname)
elif ctrltype == 'toggle':
if curval == 0:
targetval=1
else:
targetval=0
targetinstruction = controls.getToggleAction(targetname, targetval)
elif ctrltype == 'selector':
targetrange = range(targetdef['min'],targetdef['max']+1)
targetval = getChoice(targetrange, curval)
targetinstruction = controls.getSelectorAction(targetname, targetrange, targetval, curval)
elif ctrltype == 'colour':
targetrange = targetdef['values']
targetval = getChoice(targetrange, curval)
targetinstruction = controls.getColourAction(targetname, targetval)
elif ctrltype in ['words', 'verbs']:
targetrange = targetdef['pool']
targetval=getChoice(targetrange, curval)
if 'list' in targetdef:
if targetdef['list']=='passwd':
targetinstruction = controls.getPasswdAction(targetname, targetval)
elif targetdef['list']=='verbs' or ctrltype == 'verbs':
targetinstruction = controls.getVerbListAction(targetname, targetval)
elif ctrltype == 'verbs':
targetinstruction = controls.getVerbListAction(targetname, targetval)
if targetinstruction=='':
targetinstruction = controls.getWordAction(targetname, targetval)
elif ctrltype == 'pin':
finished=False
while not finished:
newpin=''
for i in range(4):
newpin += str(random.choice(range(10)))
if newpin != curval:
finished=True
targetval=newpin
targetinstruction = controls.getPinAction(targetname, targetval)
else:
print("Unhandled type: " + ctrltype)
#Now we have targetval and targetinstruction for this consoleip, store and publish it
console[consoleip]['instructions']=targetinstruction
console[consoleip]['target']={"console": targetconsole, "control": targetctrlid, "value": targetval, "timestamp": time.time(), "timeout": targettimeout}
print("Instruction: " + consoleip + '/' + targetctrlid + ' - ' + ctrltype + ' (was ' + str(curval) + ') ' + str(targetinstruction))
#update game stats
playerstats[consoleip]['instructions']['total'] += 1
playerstats[targetconsole]['targets']['total'] += 1
#publish!
client.publish('clients/' + consoleip + '/timeout', str(targettimeout))
client.publish('clients/' + consoleip + '/instructions', str(targetinstruction))
def showRound():
client.publish('status/round', str(playerstats['game']['rounds']))
def showLives():
client.publish('status/lives', str(playerstats['game']['lives']))
if lifeDisplay:
lives = playerstats['game']['lives']
print "Lives remaining: " + str(lives)
if 0 <= lives <= 9:
sev.displayDigit(lives)
if lives == 0:
led.solid(led.CODE_Col_White)
elif lives == 1:
led.flash(led.CODE_Col_Red, led.CODE_Col_Red1, 120)
elif lives == 2:
led.solid(led.CODE_Col_Red1)
elif lives == 3:
led.solid(led.CODE_Col_Red2)
elif lives == 4:
led.solid(led.CODE_Col_Yellow)
elif lives == 5:
led.solid(led.CODE_Col_Green)
def clearLives():
client.publish('status/lives', '')
if lifeDisplay:
sev.clear()
led.solid(led.CODE_Display_Fade)
time.sleep(1.1)
led.solid(led.CODE_Display_On)
def checkTimeouts():
"""Check all targets for expired instructions"""
global numinstructions, warningsound
for consoleip in players:
consoledef = console[consoleip]
if 'target' in consoledef and consoledef['target']['timestamp'] + consoledef['target']['timeout'] < time.time():
#Expired instruction
playSound(random.choice(controls.soundfiles['wrong']))
playerstats[consoleip]['instructions']['missed'] += 1
playerstats[consoledef['target']['console']]['targets']['missed'] += 1
numinstructions -= 1
playerstats['game']['lives'] -= 1
showLives()
if playerstats['game']['lives'] <= 0:
#Game over!
gameOver()
elif numinstructions <= 0:
#Round over
roundOver()
else:
#Pick a new target and carry on
increaseCorruption(consoledef['target']['console'], consoledef['target']['control'])
pickNewTarget(consoleip)
#Start a warning sound if we're on our last life
if playerstats['game']['lives'] == 1 and sound:
warningsound = pygame.mixer.Sound("sounds/48k/" + random.choice(controls.soundfiles['warning']))
warningsound.play(-1)
def increaseCorruption(consoleip, ctrlid):
try:
"""Introduce text corruptions to control names as artificial 'malfunctions'"""
ctrldef = currentsetup[consoleip]['controls'][ctrlid]
if 'corruptedname' in ctrldef:
corruptednamelist = list(ctrldef['corruptedname'])
else:
corruptednamelist = list(ctrldef['name'])
count = 3
while count > 0:
#Try to get a printable character, this is different for HD44780 than Nokia but I just use the HD44780 here
ascii = random.choice(range(12 * 16))
ascii += 32
if ascii > 128:
ascii += 32
#Position to change - avoid spaces so corrupt name prints the same
pos = random.choice(range(len(corruptednamelist)))
if corruptednamelist[pos] != ' ':
corruptednamelist[pos] = chr(ascii)
count -= 1
corruptedname = ''.join(corruptednamelist)
ctrldef['corruptedname'] = corruptedname
client.publish("clients/" + consoleip + "/" + ctrlid + "/name", corruptedname)
except:
pass
def clearCorruption(consoleip, ctrlid):
"""Reset the corrupted control name when the player gets it right"""
ctrldef = currentsetup[consoleip]['controls'][ctrlid]
if 'corruptedname' in ctrldef:
del ctrldef['corruptedname']
client.publish("clients/" + consoleip + "/" + ctrlid + "/name", str(ctrldef['name']))
def tellAllPlayers(consolelist, message):
"""Simple routine to broadcast a message to a list or consoles"""
for consoleip in consolelist:
client.publish('clients/' + consoleip + '/instructions', str(message))
client.loop(0)
def send_intro_text():
if not debugMode:
for txt in controls.blurb['intro']:
client.loop(0)
if game_state['skip_intro']:
game_state['skip_intro'] = False
return 'skip'
tellAllPlayers(players, txt)
time.sleep(blurbSleep)
game_state['skip_intro'] = False
def initGame():
"""Kick off a new game"""
#Start game!
global gamestate
global currenttimeout
global nextID
gamestate = 'initgame'
game_state['skip_intro'] = False
clearLives()
# get game players from GameStarter
for key, value in gsIDs.iteritems():
if gs.isStartablePlayer(value):
print("Player %d (%s) startable" % (value, key))
players.append(key)
else:
print("Player %d (%s) not startable" % (value, key))
currenttimeout = 5.0 * ( 1 + len(players) )
initGameStarter()
print("Player IPs: %r, player IDs: %r" % (players, gsIDs))
for consoleip in players:
#Slight fudge in assuming control 5 is the big button
client.publish('clients/' + consoleip + '/5/name', "")
client.publish('clients/' + consoleip + '/5/name', "Get ready!")
tellAllPlayers(players, controls.blurb['logo'])
#Music
introSound=None
if sound:
#Pygame for sounds
resetSound()
introSound = playSound(controls.soundfiles['special']['fanfare'])
#cut off non-players from participating
for consoleip in list(set(consoles) - set(players)):
consolesetup = {}
consolesetup['instructions'] = controls.blurb['gameinprogress']
consolesetup['timeout'] = 0.0
consolesetup['controls'] = {}
for control in console[consoleip]['controls']:
ctrlid = control['id']
consolesetup['controls'][ctrlid]={}
consolesetup['controls'][ctrlid]['type'] = 'inactive'
consolesetup['controls'][ctrlid]['enabled'] = 0
consolesetup['controls'][ctrlid]['name'] = ""
client.subscribe('clients/' + consoleip + '/' + ctrlid + '/value')
client.publish('clients/' + consoleip + '/configure', json.dumps(consolesetup))
currentsetup[consoleip] = consolesetup
#Explanatory intro blurb
if send_intro_text() == 'skip':
if introSound:
introSound.stop()
#Setup initial game params
global playerstats
playerstats = {}
for consoleip in players:
playerstats[consoleip] = {}
playerstats[consoleip]['instructions'] = {} #stats on instructions you read out
playerstats[consoleip]['targets'] = {} #stats on instructions you should have implemented
playerstats[consoleip]['instructions']['total'] = 0
playerstats[consoleip]['instructions']['hit'] = 0
playerstats[consoleip]['instructions']['missed'] = 0
playerstats[consoleip]['targets']['total'] = 0
playerstats[consoleip]['targets']['hit'] = 0
playerstats[consoleip]['targets']['missed'] = 0
playerstats['game'] = {}
playerstats['game']['rounds'] = 0
showRound()
#continuous spaceship mix
if sound:
for fn in controls.soundfiles['continuous']:
snd = pygame.mixer.Sound("sounds/48k/" + fn)
snd.play(-1)
#start first round
initRound()
def initRound():
"""Kick off a new round"""
global numinstructions
global lastgenerated
global gamestate
gamestate = 'setupround'
playSound(random.choice(controls.soundfiles['atmosphere']))
#Dump another batch of random control names and action
defineControls()
playerstats['game']['rounds'] += 1
playerstats['game']['lives'] = 5
showLives()
showRound()
numinstructions = 10
lastgenerated = time.time()
def roundOver():
"""End the round and jump to Hyperspace"""
global gamestate
global currenttimeout
global lastgenerated
global warningsound
gamestate = 'roundover'
if sound and not warningsound is None:
warningsound.stop()
warningsound = None
#Zap all existing targets
for consoleip in players:
consoledef = console[consoleip]
if 'target' in consoledef:
del consoledef['target']
client.publish('clients/' + consoleip + '/timeout', "0.0")
#play sound?
tellAllPlayers(players, controls.blurb['hyperspace'])
playSound(controls.soundfiles['special']['hyperspace'])
lastgenerated = time.time()
currenttimeout *= 0.75
gamestate = 'hyperspace'
def gameOver():
"""End the current game and dole out the medals"""
global gamestate
#Check we're not already here (fixes issue #4)
if gamestate != 'playround':
return
gamestate = 'gameover'
for consoleip in players:
client.publish('clients/' + consoleip + '/timeout', "0.0")
tellAllPlayers(players, controls.blurb['ending']['splash'])
#play sound
if sound:
#Pygame for sounds
resetSound()
playSound(controls.soundfiles['special']['explosion'])
playSound(controls.soundfiles['special']['taps'])
for consoleip in players:
config = console[consoleip]
consolesetup = {}
consolesetup['instructions'] = str(controls.blurb['ending']['start'])
consolesetup['timeout'] = 0.0
consolesetup['controls'] = {}
for control in config['controls']:
ctrlid = control['id']
consolesetup['controls'][ctrlid]={}
consolesetup['controls'][ctrlid]['type'] = 'inactive'
consolesetup['controls'][ctrlid]['enabled'] = 0
consolesetup['controls'][ctrlid]['name'] = ""
client.publish("clients/" + consoleip + "/configure", json.dumps(consolesetup))
time.sleep(5.0)
instr = controls.blurb['ending']['you']
#stats for your instructions
for consoleip in players:
instryou = instr.replace("{1}", str(playerstats[consoleip]['instructions']['hit']))
instryou = instryou.replace("{2}", str(playerstats[consoleip]['instructions']['missed'] + playerstats[consoleip]['instructions']['hit']))
instryou = instryou.replace("{3}", str(playerstats[consoleip]['instructions']['missed']))
client.publish("clients/" + consoleip + "/instructions", str(instryou))
time.sleep(5.0)
#stats for your targets
instr = controls.blurb['ending']['them']
for consoleip in players:
instrthem = instr.replace("{1}", str(playerstats[consoleip]['targets']['hit']))
instrthem = instrthem.replace("{2}", str(playerstats[consoleip]['targets']['missed'] + playerstats[consoleip]['targets']['hit']))
client.publish("clients/" + consoleip + "/instructions", str(instrthem))
time.sleep(5.0)
tellAllPlayers(players, controls.blurb['ending']['end'])
time.sleep(5.0)
#medals!
for consoleip in players:
client.publish("clients/" + consoleip + "/instructions", str(controls.getMedal()))
time.sleep(15.0)
resetToWaiting()
def resetToWaiting():
"""Reset game back to waiting for new players"""
global gamestate
gamestate = 'readytostart'
clearLives()
for consoleip in consoles:
consolesetup = {}
consolesetup['instructions'] = controls.blurb['readytostart']
consolesetup['controls'] = {}
consolesetup['timeout'] = 0.0
config = console[consoleip]
for control in config['controls']:
ctrlid = control['id']
consolesetup['controls'][ctrlid]={}
if 'gamestart' in control:
consolesetup['controls'][ctrlid]['type'] = 'button'
consolesetup['controls'][ctrlid]['enabled'] = 1
consolesetup['controls'][ctrlid]['name'] = controls.blurb['startbutton']
consolesetup['controls'][ctrlid]['gamestart'] = True
consolesetup['controls'][ctrlid]['definition'] = {}
else:
consolesetup['controls'][ctrlid]['type'] = 'inactive'
consolesetup['controls'][ctrlid]['enabled'] = 0
consolesetup['controls'][ctrlid]['name'] = ""
currentsetup[consoleip] = consolesetup
client.publish('clients/' + consoleip + '/configure', json.dumps(consolesetup))
global lastgenerated
global numinstructions
global players
players = []
lastgenerated = time.time()
numinstructions = 0
#Main loop
#Connect to MQTT (final code should make this a retry loop)
client.on_connect = on_connect
client.on_message = on_message
client.connect(server)
#Main topic subscription point for clients to register their configurations to
client.subscribe('server/register')
client.subscribe('command/#')
client.publish('server/ready', 'started')
lastReady = time.time()
lastGSStep = time.time()
while(client.loop(0) == 0):
if time.time() - lastGSStep > 0.05:
lastGSStep = time.time()
gs.timeStep(0.05)
if time.time() - lastReady > 3.0:
lastReady = time.time()
client.publish('server/ready', 'ready')
#if gamestate == 'waitingforplayers' and len(players) >= 1 and time.time() - lastgenerated > 5.0:
if gamestate == 'waitingforplayers' and gs.shouldStart():
initGame()
elif gamestate == 'setupround' and time.time() - lastgenerated > 10.0:
gamestate = 'playround'
for consoleip in players:
pickNewTarget(consoleip)
elif gamestate == 'playround':
checkTimeouts()
elif gamestate == 'hyperspace' and time.time() - lastgenerated > 4.0:
initRound()
#If client.loop() returns non-zero, loop drops out to here.
#Final code should try to reconnect to MQTT and/or networking if so.