-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathminesweeperPerformanceTest.py
327 lines (262 loc) · 12.5 KB
/
minesweeperPerformanceTest.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
import numpy as np
import sys
np.set_printoptions(threshold=sys.maxsize)
from tkinter import *
import random, argparse
import minesweeperAI1
import minesweeperAI2
import time
import json
import os
# Here, we are creating our class, Window, and inheriting from the Frame
# class. Frame is a class from the tkinter module. (see Lib/tkinter/__init__)
class Window(Frame):
# Define settings upon initialization. Here you can specify
def __init__(self, master=None):
# parameters that you want to send through the Frame class.
Frame.__init__(self, master)
#reference to the master widget, which is the tk window
self.master = master
#with that, we want to then run init_window, which doesn't yet exist
self.outcome = 0
self.time = 0
def setupGenerate(self, numRows, numCols, numBombs, safeSquare, AIType):
# game variables that can be accessed in any method in the class. For example, to access the number of rows, use "self.numRows"
self.numRows = numRows
self.numCols = numCols
self.numBombs = numBombs
self.safeSquare = safeSquare
self.AIType = AIType
self.init_window(0)
def setupFile(self, testcase_filename, AIType):
# game variables that can be accessed in any method in the class. For example, to access the number of rows, use "self.numRows"
with open(testcase_filename) as fp:
data = json.load(fp)
boardSize = data['dim'].split(',')
safe = data['safe'].split(',')
# game variables that can be accessed in any method in the class. For example, to access the number of rows, use "self.numRows"
self.numRows = int(boardSize[0])
self.numCols = int(boardSize[1])
self.numBombs = int(data['bombs'])
self.safeSquare = (int(safe[0]), int(safe[1]))
self.AIType = AIType
self.gridInput = data['board']
self.init_window(1)
# Creation of init_window
def init_window(self, runMethod):
# changing the title of our master widget
self.master.title("Minesweeper")
# allowing the widget to take the full space of the root window
self.pack(fill=BOTH, expand=1)
self.button = []
for row in range(self.numRows):
curRow = []
for col in range(self.numCols):
curRow.append(Button(self, bg="gray", width=2, height=1, command=lambda rw=row, cl=col: self.open_button(rw, cl)))
curRow[col].grid(row=row, column=col)
self.button.append(curRow)
if runMethod == 0:
self.generate_board()
elif runMethod == 1:
self.create_board()
else:
print('Invalid format!')
exit()
AIAlgo1Button = Button(self, bg="blue", text="AI 1", width=6, height=5, command=self.AIAlgo)
AIAlgo1Button.place(x=600, y=150)
AIAlgo2Button = Button(self, bg="blue", text="AI 2", width=6, height=5, command=self.AIAlgo)
AIAlgo2Button.place(x=600, y=350)
self.AI = None
if self.AIType == 1:
self.AI = minesweeperAI1.AI1(self.numRows, self.numCols, self.numBombs, self.safeSquare)
else:
self.AI = minesweeperAI2.AI2(self.numRows, self.numCols, self.numBombs, self.safeSquare)
self.highlight_button(self.safeSquare)
self.AIAlgo()
# (DO NOT MODIFY): uncover a square and perform appriopriate action based on game rule
def open_button(self, r, c):
if not self.squareInBounds(r, c):
return
self.button[r][c].config(state="disabled")
if self.ans[r][c] != 9:
self.button[r][c]["text"] = self.ans[r][c]
self.button[r][c]["bg"] = "white"
else:
self.button[r][c]["text"] = self.ans[r][c]
self.button[r][c]["bg"] = "red"
# (helper function): return true if and only if all non-bomb squares have been uncovered (game is won)
def isGameWon(self):
cnt = 0
for row in range(self.numRows):
for col in range(self.numCols):
if self.button[row][col]['state'] != 'disabled': cnt += 1
return cnt == self.numBombs
# (helper function): return true if and only if a square (r, c) is within the game grid
def squareInBounds(self, r, c):
return r >= 0 and c >= 0 and r < self.numRows and c < self.numCols
# (helper function): return true if and only if the board has all covered (unopened) squares
def isNewBoard(self):
for row in range(self.numRows):
for col in range(self.numCols):
if self.button[row][col]['state'] == 'disabled':
return False
return True
# (helper function): change the color of specified square to blue to indicate this is the square you will select next
def highlight_button(self, square):
if self.squareInBounds(square[0], square[1]):
self.button[square[0]][square[1]]["bg"] = "blue"
# (helper function): get the current board state from player POV. Note -1 represents unknown square
def getBoardState(self):
board = np.full((self.numRows, self.numCols), -1)
for row in range(self.numRows):
for col in range(self.numCols):
value = 0
if self.button[row][col]['state'] == 'disabled':
if self.button[row][col]["text"] != "":
value = int(self.button[row][col]["text"])
board[row][col] = value
return board
# generate a random board based on inputs {self.numRows, self.numCols, self.numBombs, self.safeSquare}
def generate_board(self):
# Create a safe 3x3 grid around the safe starting square
bombsNotAllowed = []
for dx in range(-1, 2):
for dy in range(-1, 2):
r = self.safeSquare[0] + dx
c = self.safeSquare[1] + dy
if self.squareInBounds(r, c):
bombsNotAllowed.append((r, c))
self.ans = np.full((self.numRows, self.numCols), 0)
# Add self.numBombs number of bombs to the ans grid
self.bombLocations = []
while len(self.bombLocations) < self.numBombs:
tempX = random.randint(0, self.numRows - 1)
tempY = random.randint(0, self.numCols - 1)
if (tempX, tempY) not in self.bombLocations and (tempX, tempY) not in bombsNotAllowed:
self.bombLocations.append((tempX, tempY))
self.ans[tempX][tempY] = 9
# Add numbers 0-8 to the ans grid
for row in range(self.numRows):
for col in range(self.numCols):
if self.ans[row][col] == 0:
squareValue = 0
for dx in range(-1, 2):
for dy in range(-1, 2):
r = row + dx
c = col + dy
if self.squareInBounds(r, c) and self.ans[r][c] == 9:
squareValue += 1
self.ans[row][col] = squareValue
print(self.ans)
output = ""
for row in self.ans:
for val in row:
output = output + str(val)
print(output)
# generate a board based on test case input
def create_board(self):
self.ans = np.full((self.numRows, self.numCols), 0)
self.bombLocations = [(i % self.numCols, int(i/self.numCols)) for i, c in enumerate(self.gridInput) if c == 9 or c == '9']
# Add numbers 0-8 to the ans grid
for row in range(self.numRows):
for col in range(self.numCols):
self.ans[col][row] = self.gridInput[row * self.numCols + col]
print(f"starting board\n{self.ans}")
print(f"location of bombs: {self.bombLocations}")
# parse the user's command and perform the appropriate action.
def parseAIAlgo(self, userCommand):
if type(userCommand) is not tuple:
print("cannot parse command")
elif "open_square" in userCommand[0]:
self.nextSquareToOpen = userCommand[1]
self.highlight_button(self.nextSquareToOpen)
elif "final_answer" in userCommand[0]:
userAnswer = userCommand[1]
self.numDigs = np.count_nonzero(self.getBoardState() != -1)
if set(self.bombLocations) == set(userAnswer):
self.outcome = 1
print(f"CORRECT BOMB LIST! You performed {self.numDigs} digs.")
else:
self.outcome = -1
print(f"WRONG BOMB LIST. expected: {self.bombLocations}, received: {userAnswer}. You performed {self.numDigs} digs.")
root.destroy()
"""
Each time you click the button, We would recommend you to perform the following loop in your algorithm:
1) uncover the best square computed last iteration using `self.open_button(r, c)`. Note a safe square is given for the first iteration
2) perform your algorithm to find the best square using performAI1()
3) visually display the square you will select on the next iteration (blue square) using self.highlight_button(r, c)
"""
def AIAlgo(self):
if self.outcome != 0:
return # game is already over (won or loss)
if self.isNewBoard():
self.open_button(self.safeSquare[0], self.safeSquare[1])
else:
self.open_button(self.nextSquareToOpen[0], self.nextSquareToOpen[1])
if self.outcome != 0:
return # game is already over (won or loss)
boardState = self.getBoardState()
startTime = time.time()
userCommand = self.AI.performAI(boardState)
endTime = time.time()
self.time += (endTime - startTime)
self.parseAIAlgo(userCommand)
if self.outcome == 0:
self.AIAlgo()
if len(sys.argv) < 2:
print("usage: -f <file_name.json> <algo_type>, or -g <x_dim> <y_dim> <num_bombs> <safe_x> <safe_y> <algo_type> <num_games>")
elif sys.argv[1] in ["--generate", "-g"] and len(sys.argv) == 9:
sys.setrecursionlimit(100000)
numGames = int(sys.argv[8])
numWins = 0
numLosses = 0
totalDigs = 0
totalTime = 0
totalSatVars = 0
totalSatSteps = 0
totalSatQueries = 0
numRows, numCols, numBombs = int(sys.argv[2]), int(sys.argv[3]), int(sys.argv[4])
aitype = int(sys.argv[7])
for i in range(numGames):
# print(f"match={i+1}")
root = Tk()
root.geometry("800x800")
app = Window(master = root)
sys.stdout = open(os.devnull,"w")
app.setupGenerate(numRows, numCols, numBombs, safeSquare = (int(sys.argv[5]), int(sys.argv[6])), AIType = aitype)
root.mainloop()
sys.stdout = sys.__stdout__
outcome = "ERROR"
totalDigs += app.numDigs
totalTime += app.time
if app.outcome == -1:
outcome = "Incorrect Bomb List"
numLosses += 1
elif app.outcome == 1:
outcome = "Correct Bomb List"
numWins += 1
if app.AIType == 2:
totalSatVars += app.AI.total_sat_variables
totalSatSteps += app.AI.total_sat_solve_steps
totalSatQueries += app.AI.total_sat_solves
# print("\n************\n")
numCells = numRows*numCols
print(f'numRows={numRows}, numCols={numCols}, numBombs={numBombs}, numCells={numCells}, bombDensity={numBombs/numCells}')
if aitype == 2:
print(f'average SAT queries={totalSatQueries/numGames}, average SAT size={round(totalSatVars/totalSatQueries, 3)} variables, average backtracking steps={round(totalSatSteps/totalSatQueries, 3)}')
print(f"totalDigs={totalDigs}, averageDigs={totalDigs/numGames}, averagePerformance={round(totalDigs/numGames/numCells,3)}, totalTime={round(totalTime, 3)}, averageTime={round(totalTime/numGames, 3)}, numberOfTimeCorrectBombListReturned={numWins}, numberOfTimeIncorrectBombListReturned={numLosses}")
print('')
elif sys.argv[1] in ["--file", "-f"] and len(sys.argv) == 4:
root = Tk()
root.geometry("800x800")
app = Window(master = root)
app.setupFile(testcase_filename = sys.argv[2], AIType = int(sys.argv[3]))
root.mainloop()
outcome = "ERROR"
if app.outcome == -1:
outcome = "Incorrect Bomb List"
elif app.outcome == 1:
outcome = "Correct Bomb List"
print(f"totalDigs={app.numDigs}, totalTime={round(app.time, 3)}, outcome={outcome}")
else:
print("usage: -f <file_name.json> <algo_type>, or -g <x_dim> <y_dim> <num_bombs> <safe_x> <safe_y> <algo_type> <num_games>")