-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathaihash.py
121 lines (106 loc) · 4.42 KB
/
aihash.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
import chess
import random
from table import table
from item import item
# Same as ai, but make move within method
class aihash:
def __init__(self):
self.max_depth = 2
self.max_q = 7
self.rand = [round(random.random()*10000000000000000) for i in range(768)]
#self.table = table()
# Pawns 0 & 6
# Knight 1 & 7
# Bishop 2 & 8
# Rook 3 & 9
# Queen 4 & 10
# King 5 & 11
def make_move(self, game):
moves = list(game.legal_moves)
max = [10000,0]
hash = self.get_hashkey(game)
self.table = table()
self.count = game.fullmove_number
for i in range(len(moves)):
value = self.move_value(game, -10000, 10000, 1, moves[i], hash)
if value < max[0]:
max[0] = value
max[1] = moves[i]
return max[1]
def move_value(self, game, alpha, beta, depth, move, hash):
value = 0
hash = self.mod_hashkey(hash, move, game)
last = self.table.get(hash)
if last is not None:
if last.depth >= self.distance:
return last.eval
if depth > self.max_q:
value = self.eval(game)
elif depth > self.max_depth and not game.gives_check(move) and not game.is_capture(move) and not game.is_check():
value = self.eval(game)
else:
game.push(move)
if game.is_game_over():
if game.is_checkmate():
value = 9999 if game.turn else -9999
elif game.turn:
value = self.min_val(game, alpha, beta, depth, move, hash)
else:
value = self.max_val(game, alpha, beta, depth, move, hash)
game.pop()
board = item(value, self.distance, self.count)
self.table.add(hash, board)
return value
def min_val(self, game, alpha, beta, depth, move, hash):
min_val = 10000
moves = list(game.legal_moves)
for i in range(len(moves)):
val = self.move_value(game, alpha, beta, depth + 1, moves[i], hash)
min_val = min(val, min_val)
beta = min(val, beta)
if beta <= alpha:
break
return min_val
def max_val(self, game, alpha, beta, depth, move, hash):
max_val = -10000
moves = list(game.legal_moves)
for i in range(len(moves)):
val = self.move_value(game, alpha, beta, depth + 1, moves[i], hash)
max_val = max(max_val, val)
alpha = max(alpha, val)
if beta <= alpha:
break
return max_val
def eval(self, game):
# Evalutes the current game state, from -9999 to 9999
eval = random.random()
for (piece, value) in [(chess.PAWN, 1),
(chess.BISHOP, 3),
(chess.KING, 0),
(chess.QUEEN, 10),
(chess.KNIGHT, 5),
(chess.ROOK, 3)]:
eval += len(game.pieces(piece, False)) * value
eval -= len(game.pieces(piece, True)) * value
# can also check things about the pieces position here
return eval
def min(one, two):
return one if one < two else two
def max(one, two):
return one if one > two else two
def get_hashkey(self, board):
key = 0
for i in range(2):
for j in range(1,7):
pos = list(board.pieces(j, True if i == 0 else False))
for k in pos:
key ^= self.rand[k + 64 * ((j - 1) + (0 if i == 0 else 6))]
return key
def mod_hashkey(self, hash, move, board):
# Modifies the hash key to include the move that has not been made yet
#m = chess.Move.from_uci(move)
hash ^= self.rand[move.from_square + 64 * ((board.piece_type_at(move.from_square) - 1) + (0 if board.color_at(move.from_square) == 0 else 6))]
hash ^= self.rand[move.to_square + 64 * ((board.piece_type_at(move.from_square) - 1) + (0 if board.color_at(move.from_square) == 0 else 6))]
if board.piece_type_at(move.to_square) is not None:
hash ^= self.rand[move.to_square + 64 * ((board.piece_type_at(move.to_square) - 1) + (0 if board.color_at(move.to_square) == 0 else 6))]
return hash