forked from Andarin/qtpy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbotmoves.py
335 lines (267 loc) · 10.8 KB
/
botmoves.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
#!/usr/bin/env python
#
# Copyright (C) 2010-2012 John Driscoll <[email protected]>
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
# QTPy Bot move database
import elixir
class BotMoves(elixir.Entity):
""" Interface for database of bot moves """
# State of the game before the bot's move is made in standard notation
state = elixir.Field(elixir.String(), primary_key=True)
# Bot's move in standard notation
move = elixir.Field(elixir.String())
# Number of games the move has been played in
played = elixir.Field(elixir.Integer(), default=1, index=True)
# Number of games won using this move
wins = elixir.Field(elixir.Integer(), default=0)
# Number of games lost using this move
losses = elixir.Field(elixir.Integer(), default=0)
# Number of games drawn using this move
draws = elixir.Field(elixir.Integer(), default=0)
# Number of times in-a-row this move has been played
played_in_sequence = elixir.Field(elixir.Integer(), default=1)
# Probability of win
winp = elixir.Field(elixir.Float(), default=0.0, index=True)
# Probabilty of loss
lossp = elixir.Field(elixir.Float(), default=0.0, index=True)
# Probability of draw
drawp = elixir.Field(elixir.Float(), default=0.0, index=True)
# Square number translation tables
_rot90 = [ 7, 4, 1, 8, 5, 2, 9, 6, 3 ]
_rot180 = [ 9, 8, 7, 6, 5, 4, 3, 2, 1 ]
_rot270 = [ 3, 6, 9, 2, 5, 8, 1, 4, 7 ]
_fliph = [ 3, 2, 1, 6, 5, 4, 9, 8, 7 ]
_flipv = [ 7, 8, 9, 4, 5, 6, 1, 2, 3 ]
_table = { '0': [ _rot90 ], '-0': [ _rot270 ],
'1': [ _rot180 ], '-1': [ _rot180 ],
'2': [ _rot270 ], '-2': [ _rot90 ],
'v': [ _flipv ], '-v': [ _flipv ],
'h': [ _fliph ], '-h': [ _fliph ],
'0h': [ _rot90, _fliph ], '-0h': [ _fliph, _rot270 ],
'0v': [ _rot90, _flipv ], '-0v': [ _flipv, _rot270 ] }
_valid_keys = [ None, '0', '1', '2', 'v', 'h', '0v', '0h' ]
@staticmethod
def update_state(state, move, outcome):
""" Update (or create) outcome probabilities for a game state
and move combo
Arguments:
state and move: standard notation game strings
outcome: array of player scores [ player, bot ]
"""
from bot import Bot
game = BotMoves.find_state(state, move)
outcome = BotMoves.outcome(outcome)
if game:
if game[0].count() > 1: # Dup!
# Use the most played move
dups = []
for game in game[0]: dups.append(game)
dups.sort(key=lambda g: g.played, reverse=True)
for i in range(1,len(dups)): dups[i].delete()
game = dups[0]
else:
game = game[0].one()
game.played += 1
if outcome == 1: game.wins += 1
elif outcome == 0: game.draws += 1
else: game.losses += 1
game.winp = float(game.wins) / float(game.played);
game.lossp = float(game.losses) / float(game.played);
game.drawp = float(game.draws) / float(game.played);
if game.played_in_sequence >= Bot.noise_factor():
game.played_in_sequence = 0
else:
game.played_in_sequence += 1
elixir.session.commit()
# Add mapped flag if all valid moves have been played
from botmovesmapped import BotMovesMapped
if not BotMovesMapped.has(game.state):
from state import State
state = State(game.state)
if not Bot.get_missing_move(state):
BotMovesMapped(state=game.state)
elixir.session.commit()
else:
# Create new record
w = d = l = 0
if outcome == 1: w = 1
elif outcome == 0: d = 1
else: l = 1
BotMoves(state=state,
move=move,
wins=w,
draws=d,
losses=l,
winp=float(w),
drawp=float(d),
lossp=float(l))
elixir.session.commit()
@staticmethod
def translate(moves, key):
""" Translates a pliable move list based on the key """
if not key: return moves
tables = BotMoves._table[key]
out = []
for move in moves: out.append(move)
for table in tables:
for i in range(len(out)):
sq1 = table[out[i][0]-1] if out[i][0] > 0 else 0
sq2 = table[out[i][1]-1] if out[i][1] > 0 else 0
if sq1 > sq2: sq1,sq2 = sq2, sq1
out[i] = (sq1,sq2)
return out
@staticmethod
def find_states(state):
""" Finds all game states of any orientation that equal state. """
states = [state]
my = BotMoves
def t(state, k): return my.translate(state, k)
a = BotMoves.pliable(state)
t = BotMoves.translate
s = BotMoves.searchable
# Look for game rotated 90 degrees
a90 = t(a, '0')
states.append(s(a90))
# Look for game rotated 180 degrees
a180 = t(a90, '0')
states.append(s(a180))
# Look for game rotated 270 degrees
states.append(s(t(a180, '0')))
# Look for game flipped vertically
states.append(s(t(a, 'v')))
# Look for game flipped horizontally
states.append(s(t(a, 'h')))
# Look for game flipped vertically and rotated 90 degrees
states.append(s(t(a90, 'v')))
# Look for game flipped horizontally and rotated 90 degrees
states.append(s(t(a90, 'h')))
return my.query.filter(my.state.in_(states))
@staticmethod
def find_state(state, move=None, outcome=None, key=None, my=None):
""" Finds a game state in the DB or returns None.
state is a game state in standard notation.
Looks for games of the supplied state in different variations
and optionally with the same move and outcome.
Arguments:
state: game string in standard notation
move: one or two game moves in standard notation (two moves in the
case that the the first move is a collapse)
outcome: an integer where -1 = bot loss, 0 = draw, 1 = bot win
Returns an array
[ 0: the game records in the DB,
1: the key used to transform the supplied state into the
variation found in the DB ]
If key is None, then the game was stored in the database in the same
format supplied Key is a string consisting of any of these (although
some combinations are not supported because they are redundant)
0: The state was rotated 90 degrees
1: The state was rotated 180 degrees
2: The state was rotated 270 degrees
v: The state was flipped vertically
h: The state was flipped horizontally
Supported keys: 0, 1, 2, h, v, 0h, 0v
"""
if not my: my = BotMoves
def get(state, move=None, outcome=None):
# Get helper: return query object for oriented state string
q = my.query.filter_by(state=BotMoves.searchable(state))
if move: q.filter_by(move=BotMoves.searchable(move))
if outcome:
if outcome == 1: q.filter_by(wins=1)
elif outcome == 0: q.filter_by(draws=1)
else: q.filter_by(losses=1)
return q
def t(state, k): return my.translate(state, k)
a = BotMoves.pliable(state)
m = BotMoves.pliable(move) if move else None
# key provided?
if key: return get(t(a,key),t(m,key),outcome)
t = BotMoves.translate
# Look for game
k = None
games = get(a, m, outcome)
if not games.count():
# Look for game rotated 90 degrees
k = '0'
a90 = t(a, k); m90 = t(m, k) if m else None
games = get(a90, m90, outcome)
if not games.count():
# Look for game rotated 180 degrees
k = '1'
a180 = t(a90, '0'); m180 = t(m90, '0') if m else None
games = get(a180, m180, outcome)
if not games.count():
# Look for game rotated 270 degrees
k = '2'
games = get(t(a180, '0'), t(m180, '0') if m else None, outcome)
if not games.count():
# Look for game flipped vertically
k = 'v'
games = get(t(a, k), t(m, k) if m else None, outcome)
if not games.count():
# Look for game flipped horizontally
k = 'h'
games = get(t(a, k), t(m, k) if m else None, outcome)
if not games.count():
# Look for game flipped vertically and rotated 90 degrees
k = '0v'
games = get(t(a90, 'v'), t(m90, 'v') if m else None, outcome)
if not games.count():
# Look for game flipped horizontally and rotated 90 degrees
k = '0h'
games = get(t(a90, 'h'), t(m90, 'h')
if m else None, outcome)
if not games.count(): return None
return [ games, k ]
@staticmethod
def transform(state, key):
""" Perform quick re-orientation on a movelist """
if key == None: return state
return BotMoves.searchable( \
BotMoves.translate( \
BotMoves.pliable(state), key))
@staticmethod
def fix(state, key):
""" Perform quick un-re-orientation on a movelist """
if key == None: return state
return BotMoves.searchable( \
BotMoves.translate( \
BotMoves.pliable(state), '-%s'%key if key[0] != '-' else key[1:]))
@staticmethod
def pliable(state):
""" Returns an array of tuples of the supplied game string
[ (sq1, sq2), ... ]
Opposite of searchable(moves)
state is a game state string in standard notation
"""
out = []
for move in state.split('/'): out.append((int(move[0]),int(move[1])));
return out
@staticmethod
def searchable(moves):
""" Returns a standard notation string based on the supplied array of
move tuples
Opposite of pliable
"""
out = []
for move in moves: out.append('%d%d'%(move[0],move[1]))
return '/'.join(out)
@staticmethod
def outcome(scores):
""" Returns integer describing the game outcome:
-1 for player win, 0 for draw, 1 for bot win """
if scores[0] == scores[1]: return 0
elif scores[0] > scores[1]: return -1
return 1