forked from wkschwartz/pigosat
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathoptimize_test.go
112 lines (99 loc) · 2.85 KB
/
optimize_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
package pigosat
import "testing"
// TestMinimize will test optimal values from `from` to `to`.
const (
from = -32
to = -from
)
func init() {
if from >= to {
panic("from >= to")
}
}
type parameters struct {
lower, upper, optimal int
}
type arguments struct {
k int
status Status
solution Solution
}
type minimizer struct {
t *testing.T
args chan arguments
params parameters
}
func newMinimizer(lo, hi, opt int, t *testing.T) *minimizer {
m := &minimizer{params: parameters{lower: lo, upper: hi, optimal: opt}, t: t}
// A little testing by hand suggests 2 is faster than 0 or (to - from)
m.args = make(chan arguments, 2)
return m
}
func (m *minimizer) LowerBound() int { return m.params.lower }
func (m *minimizer) UpperBound() int { return m.params.upper }
func (m *minimizer) IsFeasible(k int) (status Status, solution Solution) {
if k < from {
m.t.Errorf("k too low: %d", k)
}
if k > to {
m.t.Errorf("k too hi: %d", k)
}
status = Satisfiable
if k < m.params.optimal {
status = Unsatisfiable
}
m.args <- arguments{k, status, solution}
return
}
func (m *minimizer) RecordSolution(k int, status Status, solution Solution) {
m.args <- arguments{k, status, solution}
}
// Check that RecordSolution is called with IsFeasible's output every time.
func checkFeasibleRecord(t *testing.T, v parameters, args <-chan arguments) {
var last arguments
count := 0
for arg := range args {
// Each call to IsFeasible is paried with a go RecordSolution. Thus
// we're looking for pairs of arguments.
if count%2 == 0 {
if arg.status == -1 { // sentinel
return
}
last = arg
continue
}
if arg.k != last.k || arg.status != last.status ||
!equal(arg.solution, last.solution) {
t.Errorf("%+v: feasible=%+v record=%+v", v, last, arg)
}
}
}
// TestMinimize tests that the bisection search that Minimize does correctly
// finds the optimal value within the lower and upper bounds, that optimal and
// feasible flags are set correctly, Minimizer.RecordSolution is always called
// after Minimizer.IsFeasible.
func TestMinimize(t *testing.T) {
for hi := from; hi <= to; hi++ {
for lo := from; lo <= hi; lo++ {
for opt := lo; opt <= hi+1; opt++ {
m := newMinimizer(lo, hi, opt, t)
go checkFeasibleRecord(t, m.params, m.args)
min, optimal, feasible := Minimize(m)
m.args <- arguments{status: -1} // sentinel
if opt <= hi && min != opt {
t.Errorf("%+v: min=%d", m.params, min)
}
if opt > lo && opt <= hi && !optimal {
t.Errorf("%+v: Should have been optimal", m.params)
} else if opt <= lo && optimal {
t.Errorf("%+v: Should not have been optimal", m.params)
}
if opt <= hi && !feasible {
t.Errorf("%+v: Should have been feasible", m.params)
} else if opt > hi && feasible {
t.Error("%+v: Should not have been feasible", m.params)
}
} // opt
} // lo
} // hi
} // func