-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathluzhanqi.py
835 lines (632 loc) · 27.5 KB
/
luzhanqi.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
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
from collections import namedtuple, defaultdict
from itertools import product
from functools import reduce
import logging
from misc import (namedtuple_with_defaults, match_sequence,
find_connected_component, memoize_generator,
merge_dicts)
from coordinates import (CenteredOriginAxis, CoordinateSystem,
CoordinateSystemState)
# Represents spaces on the board.
Space = namedtuple_with_defaults('Space', 'name',
initial_placement=True, safe=False,
diagonals=False, quagmire=False)
# Represents the types of pieces that can go on the board.
Piece = namedtuple_with_defaults('Piece', 'name', 'symbol', 'initial_count',
order=None, sessile=False, bomb=False,
defeats_sessile_bombs=False,
railroad_corners=False,
reveal_flag_on_defeat=False,
initial_placement=None,
lose_on_defeat=False)
# Represents an attack - the attacked piece, whether we know the winner,
# and the winning piece (which will only be meaningful if we know it)
AttackInfo = namedtuple('AttackInfo', 'piece theoretical winner')
class Movement:
"""Represents a movement of a piece to a position on a board.
The following attributes are available (but shouldn't be manually
changed):
- board: the board that this movement is on
- piece: the piece being moved
- start: where the move is being made from
(can be None for an initial placement)
- end: the destination position of the move
- turn: the turn on which the move was made
(0 means initial placement)
- attack: an AttackInfo if this is an attack move
- move_types: a set containing the strings INITIAL, ROAD, RAILROAD,
or RAILROAD_CORNER, depending on how the move was made
"""
def __init__(self, board, piece, end, outcome=None, move_types=None):
"""Initialize the movement.
board is the LuzhanqiBoard on which the movement is to be
made, piece is the piece being moved, end is where the piece
is being moved to, and outcome is an attack outcome or None
if this is not an attack or if the outcome is unknown.
"""
self.board = board
self.piece = piece
self.start = piece.position
self.end = end
self.turn = board.turn
if (self.start is None) != (self.turn == 0):
raise ValueError()
if move_types:
self.move_types = move_types
else:
self.move_types = self.board.verify_move(self.piece, self.end)
if self.move_types is None:
raise ValueError()
end_piece = board.get(end)
if end_piece is not None:
theoretical = False
if outcome == 'win':
winner = piece
elif outcome == 'loss':
winner = end_piece
elif outcome == 'tie':
winner = None
elif outcome == None:
theoretical = True
winner = None
else:
raise ValueError('Invalid outcome value!')
self.attack = AttackInfo(end_piece, theoretical, winner)
else:
if outcome is not None:
raise ValueError('This is not an attack!')
self.attack = None
def is_type(self, move_type):
"""Check whether this Movement's type is only the given type."""
return len(self.move_types) == 1 and move_type in self.move_types
class BoardPiece:
"""Represents a piece on the board with a type and an event history.
This class encapsulates a piece type and a list of events that have
happened to the piece during the game. The piece type may be None
if we don't actually know what type of piece it is.
"""
def __init__(self, spec=None):
"""Initialize a BoardPiece with an optional piece type."""
self.events = []
self.movements = []
self.attacks = []
self.spec = spec
self.maybies = None
if not spec:
self.maybies = LuzhanqiBoard.pieces.copy()
def __str__(self):
if self.friendly:
return '+' + self.spec.symbol
else:
return '-' + ','.join(sorted(maybe.symbol
for maybe in self.maybies))
def _fatal_event(self, event):
if event.attack is None:
return None
return event.attack.winner is not self
def _event_possible_for_type(self, event, spec):
if event.start is None:
initial = spec.initial_placement
if initial and not LuzhanqiBoard.position_match(abs(event.end),
initial):
return False
elif spec.sessile and event.piece is self:
return False
elif (not spec.railroad_corners and event.piece is self and
event.is_type('RAILROAD_CORNER')):
return False
elif event.attack is not None:
attacker_type, attacked_type = None, None
if event.piece is self:
attacker_type = spec
else:
attacked_type = spec
simulated = LuzhanqiBoard.simulate_attack(event.piece,
attacker_type,
event.attack.piece,
attacked_type)
if simulated is not event.attack.winner:
return False
return True
def add_event(self, event):
"""Add the given event to this piece's event history."""
if self.dead:
raise RuntimeError('This piece is dead - nothing further '
'can happen to it')
if event.piece is self:
self.movements.append(event)
elif event.attack is not None and event.attack.piece is self:
if event.attack.theoretical:
raise RuntimeError('A theoretical attack cannot be an event!')
self.attacks.append(event)
else:
raise RuntimeError('This event is not relevant to this piece')
self.events.append(event)
if self.maybies is not None:
to_remove = set()
for piece in self.maybies:
if not self._event_possible_for_type(event, piece):
to_remove.add(piece)
self.maybies -= to_remove
def force_inference(self, spec):
"""Allows specifying that an opponent's piece is definitely a certain
type of piece.
This can be used if some external knowledge allows us to conclude
that a piece is a certain type of piece.
"""
if self.spec is not None:
raise RuntimeError('We already know what this piece is!')
if spec not in self.maybies:
raise RuntimeError('Attempt to infer that a piece is something '
'that it cannot be')
self.maybies = {spec}
@property
def initial(self):
"""Get this piece's initial position.
Returns None if the piece has not yet been placed.
"""
if len(self.movements) == 0:
return None
return self.movements[0].end
@property
def friendly(self):
"""Determine whether this is a friendly piece or not."""
return self.spec is not None
@property
def dead(self):
"""Determine whether this piece is dead or not."""
if len(self.events) == 0:
return False
return bool(self._fatal_event(self.events[-1]))
@property
def position(self):
"""Get this piece's current position.
Returns None if the piece is dead or has not yet been placed.
"""
if self.dead or not self.initial:
return None
return self.movements[-1].end
@property
def died_at(self):
"""Return the position where the piece died, if it is dead.
Returns None if the piece is not dead.
"""
if not self.dead:
return None
last_move = self.movements[-1]
if self._fatal_event(last_move):
return last_move.start
else:
return last_move.end
class LuzhanqiBoard:
"""A complete description of the Luzhanqi board and game.
This class contains a bunch of information about the game (as class
attributes) and instances of this class are capable of keeping
track of the game board and implementing the rules of the game.
"""
STATION = Space('Soldier Station')
CAMP = Space('Camp', safe=True, diagonals=True,
initial_placement=False)
HEADQUARTERS = Space('Headquarters', quagmire=True)
system = CoordinateSystem(CenteredOriginAxis('x', 5),
CenteredOriginAxis('y', 12))
Coord = system.Coord
board_spec = defaultdict(lambda: LuzhanqiBoard.STATION, {
Coord(1, 2): CAMP,
Coord(0, 3): CAMP,
Coord(1, 4): CAMP,
Coord(1, 6): HEADQUARTERS
})
# initial_counts should add up to 25
MARSHAL = Piece('Field Marshal', '9', 1, order=9,
reveal_flag_on_defeat=True)
GENERAL = Piece('General', '8', 1, order=8)
LIEUT_GENERAL = Piece('Lieutenant General', '7', 2, order=7)
BRIG_GENERAL = Piece('Brigadier General', '6', 2, order=6)
COLONEL = Piece('Colonel', '5', 2, order=5)
MAJOR = Piece('Major', '4', 2, order=4)
CAPTAIN = Piece('Captain', '3', 3, order=3)
COMMANDER = Piece('Commander', '2', 3, order=2)
ENGINEER = Piece('Engineer', '1', 3, order=1,
defeats_sessile_bombs=True,
railroad_corners=True)
BOMB = Piece('Bomb', 'B', 2, bomb=True,
initial_placement=('*', lambda y: y > 1))
LANDMINE = Piece('Landmine', 'L', 3, sessile=True, bomb=True,
initial_placement=('*', lambda y: y > 4))
FLAG = Piece('Flag', 'F', 1, sessile=True, lose_on_defeat=True,
initial_placement=HEADQUARTERS)
pieces = {
MARSHAL, GENERAL, LIEUT_GENERAL, BRIG_GENERAL,
COLONEL, MAJOR, CAPTAIN, COMMANDER, ENGINEER,
BOMB, LANDMINE, FLAG
}
railroads = {
# the railroad from (0, 5) to (2, 5)
((0, 1, 2), 5),
# the railroad from (0, 1) to (2, 1)
((0, 1, 2), 1),
# the railroad from (2, 0) to (2, 5)
(2, (0, 1, 2, 3, 4, 5)),
# the railroad from (0, 0) to (0, 1)
(0, (0, 1))
}
@classmethod
def simulate_attack(cls, attacker, attacker_type, attacked, attacked_type):
"""Determine whether the attacker or the attackee would win if
they had the given types.
The return value is either one of the piece objects passed or
None if neither piece would win (a tie).
"""
def get_type(piece, spec):
if spec:
if piece.spec:
raise ValueError('This piece already has a spec!')
return spec
else:
return piece.spec
attacker_type = get_type(attacker, attacker_type)
attacked_type = get_type(attacked, attacked_type)
if attacker_type.sessile:
raise RuntimeError('Sessile pieces cannot attack!')
if attacker_type.order and attacked_type.order:
if attacker_type.order > attacked_type.order:
return attacker
elif attacked_type.order > attacker_type.order:
return attacked
return None
if attacker_type.bomb:
return None
elif attacked_type.bomb:
if attacked_type.sessile and attacker_type.defeats_sessile_bombs:
return attacker
return None
# special case - anything wins against the flag
if attacked_type == cls.FLAG:
return attacker
raise RuntimeError('Cannot determine winner???')
@classmethod
def position_spec(cls, position):
"""Get the Space object that corresponds to the given position."""
return cls.board_spec[abs(position)]
@classmethod
def position_match(cls, position, matchval):
"""Check whether a position matches a given matchval.
This supports both match_sequence based matching and matching based
on the Space object at the given position.
"""
if isinstance(matchval, Space):
return cls.position_spec(position) == matchval
else:
return match_sequence(position, matchval)
@classmethod
def initial_positions(cls):
"""Return the initial placement positions for a player."""
nonneg = lambda i: i >= 0
absolutes = (position
for position in cls.system.coords_matching(nonneg, nonneg)
if cls.position_spec(position).initial_placement)
x_map = lambda axis, x: axis.original_and_reflection(x)
return cls.system.map_coord_components(absolutes, x=x_map)
@classmethod
def initial_enemy_positions(cls):
"""Return the initial placement positions for a player's opponent."""
initials = cls.initial_positions()
y_reflect = lambda axis, y: axis.reflection(y)
return cls.system.map_coord_components(initials, y=y_reflect)
@classmethod
def can_move(cls, piece):
"""Determine whether the given piece can move.
This is based both on the piece's type and on the space where it is
currently located.
"""
# can't move a sessile piece
if piece.spec is not None and piece.spec.sessile:
return False
# can't move off a quagmire space
if cls.position_spec(piece.position).quagmire:
return False
return True
@classmethod
@memoize_generator
def nonabsolute_railroad_lines(cls):
"""Return matchvals for the straight railroad lines on the board.
The naming of this function is based on the fact that it takes the
"absolute" railroad information in LuzhanqiBoard.railroads and
"de-absolutizes" it to describe the actual railroad lines on the
complete board.
"""
def nonabsolute_matchvals(matchval):
for component in matchval:
if not isinstance(component, tuple):
component = (component,)
reflection = tuple(-val for val in component if val != 0)
if 0 in component:
yield (component + reflection,)
else:
yield (component, reflection)
for line in cls.railroads:
for nonabsolute in product(*nonabsolute_matchvals(line)):
yield nonabsolute
@classmethod
def log_board_with_markers(cls, *marks_dicts, level=logging.DEBUG):
"""Log the board with the given marks and log level or DEBUG.
The arguments to this function should be dicts which map between
positions on the board and a string to put at that position. If
multiple dicts are passed, they are merged into one with values
with the same keys joined with commas. The level argument is
keyword-only.
"""
if not logging.getLogger().isEnabledFor(level):
return
all_marks = {}
for marks_dict in marks_dicts:
for position, mark in marks_dict.items():
if position in all_marks:
all_marks[position] += ';' + mark
else:
all_marks[position] = mark
col_widths = [reduce(max,
(len(mark) for position, mark in all_marks.items()
if position.x == col),
len(str(col)))
for col in cls.system.x]
horizontal_pad_size = 2
horizontal_pad = ' ' * horizontal_pad_size
vertical_pad_size = 1
def grid_line(edge='|', sep='|', row=None, dashes=False, marks={}):
if row:
line = str(row).rjust(2) + ' '
else:
line = ' '
line += edge
if dashes:
displays = ('-' * (horizontal_pad_size * 2 + width)
for width in col_widths)
else:
displays = (horizontal_pad +
str(marks.get(col, '')).center(width) +
horizontal_pad
for col, width in zip(cls.system.x, col_widths))
line += sep.join(displays)
line += edge
logging.log(level, line)
def pad_vertical():
for _ in range(vertical_pad_size):
grid_line()
def sep_line():
grid_line(edge='+', sep='+', dashes=True)
grid_line(edge=' ', sep=' ', marks={x: x for x in cls.system.x})
for row in cls.system.y:
sep_line()
pad_vertical()
pieces = {x: mark for (x, y), mark in all_marks.items()
if y == row}
grid_line(row=row, marks=pieces)
pad_vertical()
sep_line()
def __init__(self):
self.board = CoordinateSystemState(self.system)
self.friendly_pieces = set()
self.friendly_pieces_dead = set()
self.enemy_pieces = set()
self.enemy_pieces_dead = set()
self.turn = 0
def _verify_attack(self, piece, end):
if self.board[end] is not None:
# no friendly fire and no attacking a safe space
return (piece.friendly != self.board[end].friendly and
not self.position_spec(end).safe)
return True
@classmethod
@memoize_generator
def adjacent_railroad_moves(cls, spec, orig_position, position, corner):
"""Gets adjacent railroad moves and is used for traversing the
railroad.
To use this function correctly, you must know:
- the type of a piece
- which is at a specific position
- that it can travel to some other position on the railroad
- whether traveling to this other position consists of moving
around a railroad corner
These things correspond to the arguments to the function.
The return value is a dict mapping Coord objects to either
'RAILROAD' or 'RAILROAD_CORNER', depending on the move type.
"""
def component_values(line):
for position_component, line_component in zip(position, line):
values = (position_component - 1,
position_component,
position_component + 1)
yield tuple(val for val in values
if val in line_component)
for line in cls.nonabsolute_railroad_lines():
if (match_sequence(position, line) and
(not spec or
spec.railroad_corners or
match_sequence(orig_position, line))):
new_corner = corner
if new_corner is False:
new_corner = not match_sequence(orig_position, line)
for components in product(*component_values(line)):
if components != position:
yield components, new_corner
def _railroad_moves(self, piece):
def adjacent(position, corner):
if (position in self.system and
self.board[position] is not None and
self.board[position] != piece):
return
for adj, adj_corner in self.adjacent_railroad_moves(piece.spec,
piece.position,
position,
corner):
yield adj, adj_corner
moves = find_connected_component(piece.position, False, adjacent)
def to_result(moves):
for move, corner in moves.items():
if move != piece.position:
if corner:
move_type = 'RAILROAD_CORNER'
else:
move_type = 'RAILROAD'
try:
yield self.Coord(*move), move_type
except ValueError:
pass
return dict(to_result(moves))
@classmethod
@memoize_generator
def road_moves(cls, position):
"""Gets the road moves that a piece at the given position could make.
This a generator which produces 2-tuples of Coord objects
and the string 'ROAD'.
"""
move_type = 'ROAD'
either_side = lambda axis, i: (i - 1, i + 1)
for end in cls.system.map_coord_components_separately([position],
x=either_side,
y=either_side):
yield end, move_type
all_diagonals = cls.position_spec(position).diagonals
for end in cls.system.map_coord_components([position],
x=either_side,
y=either_side):
if all_diagonals or cls.position_spec(end).diagonals:
yield end, move_type
def _all_moves(self, piece):
return merge_dicts(dict(self.road_moves(piece.position)),
self._railroad_moves(piece))
def _valid_moves_for_piece(self, piece):
if not self.can_move(piece):
return set()
valid_moves = self._all_moves(piece)
valid_moves = {Movement(self, piece, move, move_types=move_types)
for move, move_types in valid_moves.items()
if self._verify_attack(piece, move)}
return valid_moves
def verify_move(self, piece, end):
"""Verify that the given piece can move to the given end position.
Returns the strings INITIAL, ROAD, or RAILROAD if the move is a
valid move. This string indicates how the move was made. Returns
None if the move is invalid.
"""
start = piece.position
if start is None:
if self.board[end] is not None:
return None
if not self.position_spec(end).initial_placement:
return None
if (piece.spec and piece.spec.initial_placement and
not self.position_match(end, piece.spec.initial_placement)):
return None
return {'INITIAL'}
if (start == end or
not self.can_move(piece) or
not self._verify_attack(piece, end)):
return None
all_moves = self._all_moves(piece)
try:
return all_moves[end]
except KeyError:
pass
return None
def _placement_order(self, order):
for piece in order:
yield piece
for piece in self.pieces:
if piece not in order:
yield piece
def _do_initial_placement(self, placement_order, get_placement):
positions = set(self.initial_positions())
for piece in self._placement_order(placement_order):
placement = piece.initial_placement
choices = positions
if placement is not None:
choices = (position
for position in positions
if self.position_match(position, placement))
choices = set(choices)
if len(choices) < piece.initial_count:
raise RuntimeError("Not enough choices to place piece!")
for _ in range(piece.initial_count):
choice = get_placement(piece, choices)
choices.remove(choice)
positions.remove(choice)
new_piece = BoardPiece(piece)
new_piece.add_event(Movement(self, new_piece, choice))
self.friendly_pieces.add(new_piece)
self.board[choice] = new_piece
def setup(self, placement_order, get_placements):
"""Set up the board, placing our own pieces in the specified way.
placement_order should be a list of Piece objects. This method
will guarantee that it will start placing types of pieces in that
order - after the pieces in that list have been placed, no
guarantees are made.
get_placements should be a function which takes a piece and a list
of potential placement positions and returns an iterable of chosen
placement positions. It should return piece.initial_count chosen
positions.
"""
if self.turn != 0:
raise RuntimeError('Cannot setup while a game is in progress!')
self._do_initial_placement(placement_order, get_placements)
for position in self.initial_enemy_positions():
new_piece = BoardPiece()
new_piece.add_event(Movement(self, new_piece, position))
self.enemy_pieces.add(new_piece)
self.board[position] = new_piece
self.turn = 1
def valid_moves(self):
"""Returns an iterable of valid moves as Movement objects."""
for piece in self.friendly_pieces:
for move in self._valid_moves_for_piece(piece):
yield move
def get_living_pieces(self):
"""Returns the set of our pieces that are still alive."""
return self.friendly_pieces
def get_living_enemy_pieces(self):
"""Returns the set of enemy pieces that are still alive."""
return self.enemy_pieces
def get(self, position):
"""Get a piece on the board at the given position."""
return self.board[position]
def _check_pulse(self, piece):
if not piece.dead:
return
self.board[piece.died_at] = None
if piece.friendly:
living_set = self.friendly_pieces
dead_set = self.friendly_pieces_dead
else:
living_set = self.enemy_pieces
dead_set = self.enemy_pieces_dead
living_set.remove(piece)
dead_set.add(piece)
def _move_on_board(self, movement):
if self.board[movement.end] is not None:
raise RuntimeError('Cannot move onto an occupied space')
self.board[movement.start] = None
self.board[movement.end] = movement.piece
def add_move(self, movement):
"""Add a move (a Movement object) to the board."""
moved = movement.piece
attacked = movement.attack.piece if movement.attack else None
moved.add_event(movement)
if attacked is not None:
attacked.add_event(movement)
self._check_pulse(moved)
self._check_pulse(attacked)
if not moved.dead:
self._move_on_board(movement)
self.turn += 1
def _layout_markers(self):
return {piece.position: str(piece)
for piece in self.friendly_pieces | self.enemy_pieces}
def log_board_layout(self, level=logging.DEBUG):
"""Log the board layout with the given log level or DEBUG."""
self.log_board_with_markers(self._layout_markers(), level=level)
def log_defeated_pieces(self):
for piece in self.enemy_pieces_dead:
logging.debug('{} ({}): {}'.format(piece.initial, piece.died_at,
piece))