-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcell_test.go
393 lines (314 loc) · 9.35 KB
/
cell_test.go
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
package sudoku
import (
"reflect"
"strconv"
"strings"
"testing"
)
const TOP_LEFT_CELL = `123|
456|
789|
---+`
const TOP_LEFT_CELL_NO_BLOCK_DIM = `12 |
456|
789|
---+`
const TOP_LEFT_CELL_FILLED = `•••|
•1•|
•••|
---+`
func TestCellCreation(t *testing.T) {
number := 1
data := strconv.Itoa(number)
cell := newCell(nil, 0, 0)
if cell.rank() != DIM {
t.Log("Cell's rank was not DIM when initalized")
t.Fail()
}
ref := cell.Reference()
if ref.Col != 0 || ref.Row != 0 {
t.Error("Cell ref came back wrong")
}
possibilities := cell.Possibilities()
if len(possibilities) != DIM {
t.Log("We got back the wrong number of possibilities")
t.Fail()
}
for i, possibility := range possibilities {
if possibility != i+1 {
t.Log("The possibilities list was not a monotonically increasing list: ", possibility, "/", i)
t.Fail()
}
}
for i := 1; i <= DIM; i++ {
if !cell.Possible(i) {
t.Log("Cell reported ", i, " was impossible even though it hadn't been touched")
t.Fail()
}
cell.setImpossible(i)
if cell.Possible(i) {
t.Log("Cell reported ", i, " was possible even though we explicitly set it as impossible")
t.Fail()
}
if cell.Invalid() {
t.Log("Cell reported it was invalid even though only one number is impossible.")
t.Fail()
}
if cell.rank() != DIM-1 {
t.Log("Cell reported an incorrect rank: ", cell.rank())
t.Fail()
}
cell.setImpossible(i)
cell.setPossible(i)
if cell.Possible(i) {
t.Log("Cell reported ", i, " was possible even though we'd only called Possible 1x and Impossible 2x")
t.Fail()
}
cell.setPossible(i)
if !cell.Possible(i) {
t.Log("Cell reported ", i, " was impossible even after matched calls to setPossible/setImpossible")
t.Fail()
}
cell.SetExcluded(i, true)
if cell.Possible(i) {
t.Log("A cell reported it was possible even though its number had been manually excluded")
t.Fail()
}
cell.ResetExcludes()
if !cell.Possible(i) {
t.Log("A cell thought it was not possible even after excludes were cleared")
t.Fail()
}
}
for i := 1; i <= DIM; i++ {
cell.setImpossible(i)
}
if !cell.Invalid() {
t.Log("Cell didn't realize it was invalid even though every number is impossible.")
t.Fail()
}
for i := 1; i <= DIM; i++ {
cell.setPossible(i)
if cell.implicitNumber() != i {
t.Log("Implicit number failed to notice that ", i, " should be implict number.")
t.Fail()
}
possibilities := cell.Possibilities()
if len(possibilities) != 1 {
t.Log("We got the wrong number of possibilities back")
t.Fail()
} else if possibilities[0] != i {
t.Log("We got the wrong possibility back")
t.Fail()
}
cell.setImpossible(i)
}
for i := 1; i <= DIM; i++ {
cell.setPossible(i)
}
if cell.Invalid() {
t.Log("Cell still thinks it's invalid even though we reset all possible counters.")
t.Fail()
}
cell.load(data)
if cell.Number() != number {
t.Log("Number came back wrong")
t.Fail()
}
if cell.Row() != 0 {
t.Log("Row came back wrong")
t.Fail()
}
if cell.Col() != 0 {
t.Log("Cell came back wrong")
t.Fail()
}
if cell.dataString() != data {
t.Log("Cell round-tripped out with different string than data in")
t.Fail()
}
//TODO: test failing for values that are too high.
for i := 1; i <= DIM; i++ {
if i == number {
if !cell.Possible(i) {
t.Log("We reported back a number we were explicitly set to was impossible")
t.Fail()
}
} else if cell.Possible(i) {
t.Log("We reported back that a number was possible when another number had been explicitly set")
t.Fail()
}
}
number = 2
cell.SetNumber(number)
if cell.Number() != number {
t.Log("Number came back wrong after being set with SetNumber")
t.Fail()
}
if cell.rank() != 0 {
t.Log("Cell with an explicit number came back with a non-0 Rank")
t.Fail()
}
for i := 1; i <= DIM; i++ {
if i == number {
if !cell.Possible(i) {
t.Log("We reported back a number we were explicitly set to was impossible (2nd set)")
t.Fail()
}
} else if cell.Possible(i) {
t.Log("We reported back that a number was possible when another number had been explicitly set (2nd set)")
t.Fail()
}
}
number = 0
cell.SetNumber(number)
if cell.Number() != number {
t.Log("Number came back wrong after being set with SetNumber to 0")
t.Fail()
}
for i := 1; i <= DIM; i++ {
if !cell.Possible(i) {
t.Log("We reported back a number was not possible when 0 had been explicitly set.")
t.Fail()
}
}
if cell.diagram() != TOP_LEFT_CELL {
t.Log("Diagram for the top left cell printed out incorrectly: \n", cell.diagram())
t.Fail()
}
cell.setImpossible(BLOCK_DIM)
if cell.diagram() != TOP_LEFT_CELL_NO_BLOCK_DIM {
t.Log("Diagram for the top left cell printed out incorrectly: \n", cell.diagram())
t.Fail()
}
cell.SetNumber(1)
if cell.diagram() != TOP_LEFT_CELL_FILLED {
t.Log("Diagram for the top left cell filled printed out incorrectly: \n", cell.diagram())
t.Fail()
}
}
func TestDiagramExtents(t *testing.T) {
//Testing this for real is hard, but what we can do is as at least make
//sure that the given indexes aren't running into any of the edges of the
//diagram.
//TODO: also test that the right cells are returned.
grid := NewGrid()
diagram := grid.Diagram(false)
diagramRows := strings.Split(diagram, "\n")
for i, cell := range grid.Cells() {
top, left, height, width := cell.DiagramExtents()
for r := top; r < top+height; r++ {
row := diagramRows[r]
for c := left; c < left+width; c++ {
char := strings.Split(row, "")[c]
if char == DIAGRAM_RIGHT || char == DIAGRAM_BOTTOM || char == DIAGRAM_CORNER {
t.Error("In cell", i, "invalid char at", r, c)
}
}
}
}
}
func TestMarks(t *testing.T) {
grid := NewGrid()
cell := grid.MutableCell(0, 0)
for i := 1; i < DIM+1; i++ {
if cell.Mark(i) {
t.Error("Zero cell had a mark:", i)
}
}
if cell.Mark(0) {
t.Error("Invalid index had true mark: 0")
}
if cell.Mark(DIM + 2) {
t.Error("Invalid index had true mark: ", DIM+2)
}
if len(cell.Marks()) != 0 {
t.Error("An empty cell already had marks")
}
cell.SetMark(1, true)
if !cell.Mark(1) {
t.Error("Cell with a mark on 1 did not read back")
}
cell.SetMark(2, true)
if !reflect.DeepEqual(cell.Marks(), IntSlice{1, 2}) {
t.Error("Cell with marks 1 and 2 set had wrong Marks List:", cell.Marks())
}
cell.ResetMarks()
for i := 1; i < DIM; i++ {
if cell.Mark(i) {
t.Error("Cell that had called ResetMarks still had a mark set at", i)
}
}
}
func TestCellLock(t *testing.T) {
grid := NewGrid()
cell := grid.MutableCell(3, 3)
if cell.Locked() {
t.Error("New cell was already locked")
}
cell.Lock()
if !cell.Locked() {
t.Error("Locked cell was not actually locked")
}
cell.Unlock()
if cell.Locked() {
t.Error("Unlocked cell was still locked")
}
cell.SetNumber(2)
if cell.Number() != 2 {
t.Error("Locking made it so SetNumber failed")
}
}
func TestSymmetry(t *testing.T) {
grid := NewGrid()
cell := grid.Cell(3, 3)
mutableCell := grid.MutableCell(3, 3)
partner := cell.SymmetricalPartner(SYMMETRY_BOTH)
if partner.Row() != 5 || partner.Col() != 5 {
t.Error("Got wrong symmetrical partner (both) for 3,3: ", partner)
}
mutablePartner := mutableCell.MutableSymmetricalPartner(SYMMETRY_BOTH)
if mutablePartner.Row() != 5 || mutablePartner.Col() != 5 {
t.Error("Got wrong symmetrical partner (both) for 3,3: ", mutablePartner)
}
partner = cell.SymmetricalPartner(SYMMETRY_HORIZONTAL)
if partner.Row() != 5 || partner.Col() != 3 {
t.Error("Got wrong symmetrical partner (horizontal) for 3,3: ", partner)
}
mutablePartner = mutableCell.MutableSymmetricalPartner(SYMMETRY_HORIZONTAL)
if mutablePartner.Row() != 5 || mutablePartner.Col() != 3 {
t.Error("Got wrong symmetrical partner (horizontal) for 3,3: ", mutablePartner)
}
partner = cell.SymmetricalPartner(SYMMETRY_VERTICAL)
if partner.Row() != 3 || partner.Col() != 5 {
t.Error("Got wrong symmetrical partner (vertical) for 3,3: ", partner)
}
mutablePartner = mutableCell.MutableSymmetricalPartner(SYMMETRY_VERTICAL)
if mutablePartner.Row() != 3 || mutablePartner.Col() != 5 {
t.Error("Got wrong symmetrical partner (vertical) for 3,3: ", mutablePartner)
}
partner = cell.SymmetricalPartner(SYMMETRY_ANY)
if partner == nil {
t.Error("Didn't get back a symmerical partner for (any) for 3,3")
}
mutablePartner = mutableCell.MutableSymmetricalPartner(SYMMETRY_ANY)
if mutablePartner == nil {
t.Error("Didn't get back a symmerical partner for (any) for 3,3")
}
partner = cell.SymmetricalPartner(SYMMETRY_NONE)
if partner != nil {
t.Error("Should have gotten back nil for SYMMETRY_NONE for 3,3, got: ", partner)
}
mutablePartner = mutableCell.MutableSymmetricalPartner(SYMMETRY_NONE)
if mutablePartner != nil {
t.Error("Should have gotten back nil for SYMMETRY_NONE for 3,3, got: ", mutablePartner)
}
cell = grid.Cell(4, 4)
if cell.SymmetricalPartner(SYMMETRY_BOTH) != nil || cell.SymmetricalPartner(SYMMETRY_HORIZONTAL) != nil || cell.SymmetricalPartner(SYMMETRY_VERTICAL) != nil {
t.Error("Middle cell got a symmetrical partner for some kind of symmetry.")
}
mutableCell = grid.MutableCell(4, 4)
if mutableCell.MutableSymmetricalPartner(SYMMETRY_BOTH) != nil || mutableCell.MutableSymmetricalPartner(SYMMETRY_HORIZONTAL) != nil || mutableCell.MutableSymmetricalPartner(SYMMETRY_VERTICAL) != nil {
t.Error("Middle cell got a mutable symmetrical partner for some kind of symmetry.")
}
}