-
Notifications
You must be signed in to change notification settings - Fork 13
/
pystockfish.py
297 lines (247 loc) · 9.4 KB
/
pystockfish.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
"""
pystockfish
~~~~~~~~~~~~~~~
Wraps the Stockfish chess engine. Assumes stockfish is
executable at the root level.
Built on Ubuntu 12.1 tested with Stockfish 120212.
:copyright: (c) 2013 by Jarret Petrillo.
:license: GNU General Public License, see LICENSE for more details.
"""
import re
import subprocess
from random import randint
MAX_MOVES = 200
UCI_MOVE_REGEX = "[a-h]\d[a-h]\d[qrnb]?"
PV_REGEX = " pv (?P<move_list>{0}( {0})*)".format(UCI_MOVE_REGEX)
class Match:
"""
The Match class setups a chess match between two specified engines. The white player
is randomly chosen.
deep_engine = Engine(depth=20)
shallow_engine = Engine(depth=10)
engines = {
'shallow': shallow_engine,
'deep': deep_engine,
}
m = Match(engines=engines)
m.move() advances the game by one move.
m.run() plays the game until completion or 200 moves have been played,
returning the winning engine name.
"""
def __init__(self, engines):
random_bin = randint(0, 1)
self.white = list(engines.keys())[random_bin]
self.black = list(engines.keys())[not random_bin]
self.white_engine = engines.get(self.white)
self.black_engine = engines.get(self.black)
self.moves = []
self.white_engine.newgame()
self.black_engine.newgame()
self.winner = None
self.winner_name = None
def move(self):
"""
Advance game by single move, if possible.
@return: logical indicator if move was performed.
"""
if len(self.moves) == MAX_MOVES:
return False
elif len(self.moves) % 2:
active_engine = self.black_engine
active_engine_name = self.black
inactive_engine = self.white_engine
inactive_engine_name = self.white
else:
active_engine = self.white_engine
active_engine_name = self.white
inactive_engine = self.black_engine
inactive_engine_name = self.black
active_engine.setposition(self.moves)
movedict = active_engine.bestmove()
bestmove = movedict.get('move')
info = movedict.get('info')
ponder = movedict.get('ponder')
self.moves.append(bestmove)
if info["score"]["eval"] == "mate":
matenum = info["score"]["value"]
if matenum > 0:
self.winner_engine = active_engine
self.winner = active_engine_name
elif matenum < 0:
self.winner_engine = inactive_engine
self.winner = inactive_engine_name
return False
if ponder != '(none)':
return True
def run(self):
"""
Returns the winning chess engine or "None" if there is a draw.
"""
while self.move():
pass
return self.winner
class Engine(subprocess.Popen):
"""
This initiates the Stockfish chess engine with Ponder set to False.
'param' allows parameters to be specified by a dictionary object with 'Name' and 'value'
with value as an integer.
i.e. the following explicitly sets the default parameters
{
"Contempt Factor": 0,
"Min Split Depth": 0,
"Threads": 1,
"Hash": 16,
"MultiPV": 1,
"Skill Level": 20,
"Move Overhead": 30,
"Minimum Thinking Time": 20,
"Slow Mover": 80,
}
If 'rand' is set to False, any options not explicitly set will be set to the default
value.
-----
USING RANDOM PARAMETERS
-----
If you set 'rand' to True, the 'Contempt' parameter will be set to a random value between
'rand_min' and 'rand_max' so that you may run automated matches against slightly different
engines.
"""
def __init__(self, depth=2, ponder=False, param={}, rand=False, rand_min=-10, rand_max=10):
subprocess.Popen.__init__(self,
'stockfish',
universal_newlines=True,
stdin=subprocess.PIPE,
stdout=subprocess.PIPE, )
self.depth = str(depth)
self.ponder = ponder
self.put('uci')
if not ponder:
self.setoption('Ponder', False)
base_param = {
"Write Debug Log": "false",
"Contempt Factor": 0, # There are some stockfish versions with Contempt Factor
"Contempt": 0, # and others with Contempt. Just try both.
"Min Split Depth": 0,
"Threads": 1,
"Hash": 16,
"MultiPV": 1,
"Skill Level": 20,
"Move Overhead": 30,
"Minimum Thinking Time": 20,
"Slow Mover": 80,
"UCI_Chess960": "false",
}
if rand:
base_param['Contempt'] = randint(rand_min, rand_max),
base_param['Contempt Factor'] = randint(rand_min, rand_max),
base_param.update(param)
self.param = base_param
for name, value in list(base_param.items()):
self.setoption(name, value)
def newgame(self):
"""
Calls 'ucinewgame' - this should be run before a new game
"""
self.put('ucinewgame')
self.isready()
def put(self, command):
self.stdin.write(command + '\n')
self.stdin.flush()
def flush(self):
self.stdout.flush()
def setoption(self, optionname, value):
self.put('setoption name %s value %s' % (optionname, str(value)))
stdout = self.isready()
if stdout.find('No such') >= 0:
print("stockfish was unable to set option %s" % optionname)
def setposition(self, moves=[]):
"""
Move list is a list of moves (i.e. ['e2e4', 'e7e5', ...]) each entry as a string. Moves must be in full algebraic notation.
"""
self.put('position startpos moves %s' % Engine._movelisttostr(moves))
self.isready()
def setfenposition(self, fen):
"""
set position in fen notation. Input is a FEN string i.e. "rnbqkbnr/pppppppp/8/8/4P3/8/PPPP1PPP/RNBQKBNR b KQkq e3 0 1"
"""
self.put('position fen %s' % fen)
self.isready()
def go(self):
self.put('go depth %s' % self.depth)
@staticmethod
def _movelisttostr(moves):
"""
Concatenates a list of strings.
This is format in which stockfish "setoption setposition" takes move input.
"""
return ' '.join(moves)
def bestmove(self):
"""
Get proposed best move for current position.
@return: dictionary with 'move', 'ponder', 'info' containing best move's UCI notation,
ponder value and info dictionary.
"""
self.go()
last_info = ""
while True:
text = self.stdout.readline().strip()
split_text = text.split(' ')
print(text)
if split_text[0] == "info":
last_info = Engine._bestmove_get_info(text)
if split_text[0] == "bestmove":
ponder = None if len(split_text[0]) < 3 else split_text[3]
return {'move': split_text[1],
'ponder': ponder,
'info': last_info}
@staticmethod
def _bestmove_get_info(text):
"""
Parse stockfish evaluation output as dictionary.
Examples of input:
"info depth 2 seldepth 3 multipv 1 score cp -656 nodes 43 nps 43000 tbhits 0 \
time 1 pv g7g6 h3g3 g6f7"
"info depth 10 seldepth 12 multipv 1 score mate 5 nodes 2378 nps 1189000 tbhits 0 \
time 2 pv h3g3 g6f7 g3c7 b5d7 d1d7 f7g6 c7g3 g6h5 e6f4"
"""
result_dict = Engine._get_info_pv(text)
result_dict.update(Engine._get_info_score(text))
single_value_fields = ['depth', 'seldepth', 'multipv', 'nodes', 'nps', 'tbhits', 'time']
for field in single_value_fields:
result_dict.update(Engine._get_info_singlevalue_subfield(text, field))
return result_dict
@staticmethod
def _get_info_singlevalue_subfield(info, field):
"""
Helper function for _bestmove_get_info.
Extracts (integer) values for single value fields.
"""
search = re.search(pattern=field + " (?P<value>\d+)", string=info)
return {field: int(search.group("value"))}
@staticmethod
def _get_info_score(info):
"""
Helper function for _bestmove_get_info.
Example inputs:
score cp -100 <- engine is behind 100 centipawns
score mate 3 <- engine has big lead or checkmated opponent
"""
search = re.search(pattern="score (?P<eval>\w+) (?P<value>-?\d+)", string=info)
return {"score": {"eval": search.group("eval"), "value": int(search.group("value"))}}
@staticmethod
def _get_info_pv(info):
"""
Helper function for _bestmove_get_info.
Extracts "pv" field from bestmove's info and returns move sequence in UCI notation.
"""
search = re.search(pattern=PV_REGEX, string=info)
return {"pv": search.group("move_list")}
def isready(self):
"""
Used to synchronize the python engine object with the back-end engine. Sends 'isready' and waits for 'readyok.'
"""
self.put('isready')
while True:
text = self.stdout.readline().strip()
if text == 'readyok':
return text