cassowary-ruby 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +15 -0
- data/LICENSE +20 -0
- data/README.md +18 -0
- data/lib/cassowary.rb +25 -0
- data/lib/constraint/edit_or_stay_constraint.rb +40 -0
- data/lib/constraint/linear_constraint.rb +23 -0
- data/lib/constraint.rb +30 -0
- data/lib/ext/float.rb +24 -0
- data/lib/ext/numeric.rb +11 -0
- data/lib/ext/object.rb +11 -0
- data/lib/linear_expression.rb +163 -0
- data/lib/simplex_solver.rb +704 -0
- data/lib/strength.rb +31 -0
- data/lib/symbolic_weight.rb +115 -0
- data/lib/utils/equalities.rb +27 -0
- data/lib/variables/abstract_variable.rb +35 -0
- data/lib/variables/dummy_variable.rb +21 -0
- data/lib/variables/objective_variable.rb +17 -0
- data/lib/variables/slack_variable.rb +17 -0
- data/lib/variables/variable.rb +56 -0
- data/lib/variables.rb +5 -0
- data/test/test_abstract_methods.rb +22 -0
- data/test/test_cassowary.rb +184 -0
- data/test/test_ext.rb +38 -0
- data/test/test_helper.rb +12 -0
- data/test/test_variables.rb +33 -0
- metadata +106 -0
data/lib/strength.rb
ADDED
@@ -0,0 +1,31 @@
|
|
1
|
+
# Copyright (C) 2012 by Tim Felgentreff
|
2
|
+
|
3
|
+
module Cassowary
|
4
|
+
class Strength
|
5
|
+
attr_accessor :name, :symbolic_weight
|
6
|
+
|
7
|
+
def initialize(name = nil, symbolic_weight = nil)
|
8
|
+
self.name = name
|
9
|
+
self.symbolic_weight = symbolic_weight
|
10
|
+
end
|
11
|
+
|
12
|
+
def required?
|
13
|
+
self == RequiredStrength
|
14
|
+
end
|
15
|
+
|
16
|
+
def inspect
|
17
|
+
"#{name}"
|
18
|
+
end
|
19
|
+
|
20
|
+
def each
|
21
|
+
[RequiredStrength, StrongStrength, MediumStrength, WeakStrength].each do |str|
|
22
|
+
yield str
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
RequiredStrength = new "required"
|
27
|
+
StrongStrength = new "strong", SymbolicWeight.new([1.0])
|
28
|
+
MediumStrength = new "medium", SymbolicWeight.new([0.0, 1.0])
|
29
|
+
WeakStrength = new "weak", SymbolicWeight.new([0.0, 0.0, 1.0])
|
30
|
+
end
|
31
|
+
end
|
@@ -0,0 +1,115 @@
|
|
1
|
+
# Copyright (C) 2012 by Tim Felgentreff
|
2
|
+
|
3
|
+
module Cassowary
|
4
|
+
class SymbolicWeight
|
5
|
+
include Enumerable
|
6
|
+
include Comparable
|
7
|
+
|
8
|
+
StrengthLevels = 3
|
9
|
+
|
10
|
+
def initialize(levels = {})
|
11
|
+
@levels = [0.0] * StrengthLevels
|
12
|
+
case levels
|
13
|
+
when Hash
|
14
|
+
levels.each_pair do |k, v|
|
15
|
+
self[k] = v
|
16
|
+
end
|
17
|
+
when Array
|
18
|
+
levels.each_with_index do |e, idx|
|
19
|
+
self[idx] = e
|
20
|
+
end
|
21
|
+
else
|
22
|
+
raise InternalError
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
def each(*args, &block)
|
27
|
+
@levels.each(*args, &block)
|
28
|
+
end
|
29
|
+
|
30
|
+
def [](idx)
|
31
|
+
@levels[idx]
|
32
|
+
end
|
33
|
+
|
34
|
+
def []=(idx, value)
|
35
|
+
@levels[idx] = value
|
36
|
+
end
|
37
|
+
|
38
|
+
def *(n)
|
39
|
+
raise InternalError unless n.is_a? Numeric
|
40
|
+
result = SymbolicWeight.new
|
41
|
+
each_with_index do |e, idx|
|
42
|
+
result[idx] = e * n
|
43
|
+
end
|
44
|
+
result
|
45
|
+
end
|
46
|
+
|
47
|
+
def /(n)
|
48
|
+
raise InternalError unless n.is_a? Numeric
|
49
|
+
result = SymbolicWeight.new
|
50
|
+
each_with_index do |e, idx|
|
51
|
+
result[idx] = e / n
|
52
|
+
end
|
53
|
+
result
|
54
|
+
end
|
55
|
+
|
56
|
+
def +(n)
|
57
|
+
raise InternalError unless n.is_a? SymbolicWeight
|
58
|
+
result = SymbolicWeight.new
|
59
|
+
each_with_index do |e, idx|
|
60
|
+
result[idx] = e + n[idx]
|
61
|
+
end
|
62
|
+
result
|
63
|
+
end
|
64
|
+
|
65
|
+
def -(n)
|
66
|
+
raise InternalError unless n.is_a? SymbolicWeight
|
67
|
+
result = SymbolicWeight.new
|
68
|
+
each_with_index do |e, idx|
|
69
|
+
result[idx] = e - n[idx]
|
70
|
+
end
|
71
|
+
result
|
72
|
+
end
|
73
|
+
|
74
|
+
def <=>(other)
|
75
|
+
raise InternalError unless other.is_a? SymbolicWeight
|
76
|
+
each_with_index do |e, idx|
|
77
|
+
return -1 if e < other[idx]
|
78
|
+
return 1 if e > other[idx]
|
79
|
+
end
|
80
|
+
0
|
81
|
+
end
|
82
|
+
|
83
|
+
def cl_approx(s)
|
84
|
+
raise InternalError unless s.is_a? SymbolicWeight
|
85
|
+
each_with_index do |e, idx|
|
86
|
+
return false unless e.cl_approx(s[idx])
|
87
|
+
end
|
88
|
+
true
|
89
|
+
end
|
90
|
+
|
91
|
+
def cl_approx_zero
|
92
|
+
cl_approx Zero
|
93
|
+
end
|
94
|
+
|
95
|
+
def definitely_negative
|
96
|
+
epsilon = SimplexSolver::Epsilon
|
97
|
+
nepsilon = 0.0 - epsilon
|
98
|
+
each do |e|
|
99
|
+
return true if e < nepsilon
|
100
|
+
return false if e > epsilon
|
101
|
+
end
|
102
|
+
false
|
103
|
+
end
|
104
|
+
|
105
|
+
def symbolic_weight?
|
106
|
+
true
|
107
|
+
end
|
108
|
+
|
109
|
+
def inspect
|
110
|
+
"[" + @levels.join(",") + "]"
|
111
|
+
end
|
112
|
+
|
113
|
+
Zero = new([0.0] * StrengthLevels)
|
114
|
+
end
|
115
|
+
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
# Copyright (C) 2012 by Tim Felgentreff
|
2
|
+
|
3
|
+
module Cassowary
|
4
|
+
module Equalities
|
5
|
+
def cn_equal(expr, strength = Strength::RequiredStrength, weight = 1.0)
|
6
|
+
cn_equality(LinearEquation, self - expr, strength, weight)
|
7
|
+
end
|
8
|
+
|
9
|
+
def cn_geq(expr, strength = Strength::RequiredStrength, weight = 1.0)
|
10
|
+
cn_equality(LinearInequality, self - expr, strength, weight)
|
11
|
+
end
|
12
|
+
|
13
|
+
def cn_leq(expr, strength = Strength::RequiredStrength, weight = 1.0)
|
14
|
+
expr = expr.as_linear_expression if expr.is_a?(Numeric)
|
15
|
+
cn_equality(LinearInequality, expr - self, strength, weight)
|
16
|
+
end
|
17
|
+
|
18
|
+
private
|
19
|
+
def cn_equality(klass, expr, strength, weight)
|
20
|
+
cn = klass.new
|
21
|
+
cn.expression = expr
|
22
|
+
cn.strength = strength
|
23
|
+
cn.weight = weight
|
24
|
+
cn
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
@@ -0,0 +1,35 @@
|
|
1
|
+
# Copyright (C) 2012 by Tim Felgentreff
|
2
|
+
|
3
|
+
module Cassowary
|
4
|
+
class AbstractVariable
|
5
|
+
attr_accessor :name
|
6
|
+
|
7
|
+
def initialize(hash = {})
|
8
|
+
self.name = hash[:name]
|
9
|
+
end
|
10
|
+
|
11
|
+
def dummy?
|
12
|
+
false
|
13
|
+
end
|
14
|
+
|
15
|
+
def external?
|
16
|
+
raise NotImplementedError, "my subclass should have implemented #external?"
|
17
|
+
end
|
18
|
+
|
19
|
+
def pivotable?
|
20
|
+
raise NotImplementedError, "my subclass should have implemented #pivotable?"
|
21
|
+
end
|
22
|
+
|
23
|
+
def restricted?
|
24
|
+
raise NotImplementedError, "my subclass should have implemented #restricted?"
|
25
|
+
end
|
26
|
+
|
27
|
+
def inspect
|
28
|
+
if name
|
29
|
+
"#{name}"
|
30
|
+
else
|
31
|
+
"<CV#0x#{object_id.to_s(16)}>"
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
# Copyright (C) 2012 by Tim Felgentreff
|
2
|
+
|
3
|
+
module Cassowary
|
4
|
+
class DummyVariable < AbstractVariable
|
5
|
+
def dummy?
|
6
|
+
true
|
7
|
+
end
|
8
|
+
|
9
|
+
def external?
|
10
|
+
false
|
11
|
+
end
|
12
|
+
|
13
|
+
def pivotable?
|
14
|
+
false
|
15
|
+
end
|
16
|
+
|
17
|
+
def restricted?
|
18
|
+
true
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,56 @@
|
|
1
|
+
# Copyright (C) 2012 by Tim Felgentreff
|
2
|
+
|
3
|
+
module Cassowary
|
4
|
+
class Variable < AbstractVariable
|
5
|
+
include Equalities
|
6
|
+
|
7
|
+
attr_accessor :value
|
8
|
+
|
9
|
+
def initialize(hash)
|
10
|
+
super
|
11
|
+
self.value = hash[:value]
|
12
|
+
end
|
13
|
+
|
14
|
+
def *(expr)
|
15
|
+
self.as_linear_expression * expr
|
16
|
+
end
|
17
|
+
|
18
|
+
def +(expr)
|
19
|
+
self.as_linear_expression + expr
|
20
|
+
end
|
21
|
+
|
22
|
+
def -(expr)
|
23
|
+
self.as_linear_expression - expr
|
24
|
+
end
|
25
|
+
|
26
|
+
def /(expr)
|
27
|
+
self.as_linear_expression / expr
|
28
|
+
end
|
29
|
+
|
30
|
+
def as_linear_expression
|
31
|
+
expr = LinearExpression.new
|
32
|
+
expr.terms[self] = 1.0
|
33
|
+
expr
|
34
|
+
end
|
35
|
+
|
36
|
+
def external?
|
37
|
+
true
|
38
|
+
end
|
39
|
+
|
40
|
+
def pivotable?
|
41
|
+
false
|
42
|
+
end
|
43
|
+
|
44
|
+
def restricted?
|
45
|
+
false
|
46
|
+
end
|
47
|
+
|
48
|
+
def -@
|
49
|
+
-1.0.as_linear_expression * self
|
50
|
+
end
|
51
|
+
|
52
|
+
def inspect
|
53
|
+
"#{super}[#{value.inspect}]"
|
54
|
+
end
|
55
|
+
end
|
56
|
+
end
|
data/lib/variables.rb
ADDED
@@ -0,0 +1,22 @@
|
|
1
|
+
require File.expand_path("../test_helper", __FILE__)
|
2
|
+
|
3
|
+
class AbstractMethodsTest < Test::Unit::TestCase
|
4
|
+
def test_constraint
|
5
|
+
assert_raise NotImplementedError do
|
6
|
+
Cassowary::Constraint.new.expression
|
7
|
+
end
|
8
|
+
end
|
9
|
+
|
10
|
+
def test_abstract_variable
|
11
|
+
var = Cassowary::AbstractVariable.new
|
12
|
+
assert_raise NotImplementedError do
|
13
|
+
var.external?
|
14
|
+
end
|
15
|
+
assert_raise NotImplementedError do
|
16
|
+
var.pivotable?
|
17
|
+
end
|
18
|
+
assert_raise NotImplementedError do
|
19
|
+
var.restricted?
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,184 @@
|
|
1
|
+
require File.expand_path("../test_helper", __FILE__)
|
2
|
+
|
3
|
+
class CassowaryTests < Test::Unit::TestCase
|
4
|
+
include Cassowary
|
5
|
+
|
6
|
+
def test_add_delete1
|
7
|
+
x = Variable.new(name: 'x')
|
8
|
+
solver = SimplexSolver.new
|
9
|
+
solver.add_constraint x.cn_equal(100.0, Strength::WeakStrength)
|
10
|
+
c10 = x.cn_leq 10.0
|
11
|
+
c20 = x.cn_leq 20.0
|
12
|
+
solver.add_constraint c10
|
13
|
+
solver.add_constraint c20
|
14
|
+
assert x.value.cl_approx(10.0)
|
15
|
+
|
16
|
+
solver.remove_constraint c10
|
17
|
+
assert x.value.cl_approx(20.0)
|
18
|
+
|
19
|
+
solver.remove_constraint c20
|
20
|
+
assert x.value.cl_approx(100.0)
|
21
|
+
|
22
|
+
c10again = x.cn_leq 10.0
|
23
|
+
solver.add_constraint c10
|
24
|
+
solver.add_constraint c10again
|
25
|
+
assert x.value.cl_approx(10.0)
|
26
|
+
|
27
|
+
solver.remove_constraint c10
|
28
|
+
assert x.value.cl_approx(10.0)
|
29
|
+
|
30
|
+
solver.remove_constraint c10again
|
31
|
+
assert x.value.cl_approx(100.0)
|
32
|
+
end
|
33
|
+
|
34
|
+
def test_add_delete2
|
35
|
+
x = Variable.new name: 'x'
|
36
|
+
y = Variable.new name: 'y'
|
37
|
+
|
38
|
+
solver = SimplexSolver.new
|
39
|
+
solver.add_constraint x.cn_equal(100.0, Strength::WeakStrength)
|
40
|
+
solver.add_constraint y.cn_equal(120.0, Strength::StrongStrength)
|
41
|
+
|
42
|
+
c10 = x.cn_leq(10.0)
|
43
|
+
c20 = x.cn_leq(20.0)
|
44
|
+
solver.add_constraint c10
|
45
|
+
solver.add_constraint c20
|
46
|
+
assert x.value.cl_approx(10.0)
|
47
|
+
assert y.value.cl_approx(120.0)
|
48
|
+
|
49
|
+
solver.remove_constraint c10
|
50
|
+
assert x.value.cl_approx 20.0
|
51
|
+
assert y.value.cl_approx 120.0
|
52
|
+
|
53
|
+
cxy = (x * 2).cn_equal y
|
54
|
+
solver.add_constraint cxy
|
55
|
+
assert x.value.cl_approx 20
|
56
|
+
assert y.value.cl_approx 40
|
57
|
+
|
58
|
+
solver.remove_constraint c20
|
59
|
+
assert x.value.cl_approx 60
|
60
|
+
assert y.value.cl_approx 120
|
61
|
+
|
62
|
+
solver.remove_constraint cxy
|
63
|
+
assert x.value.cl_approx 100
|
64
|
+
assert y.value.cl_approx 120
|
65
|
+
|
66
|
+
cxy2 = (x * 1.as_linear_expression).cn_equal(1000)
|
67
|
+
solver.add_constraint cxy2
|
68
|
+
assert x.value.cl_approx 1000
|
69
|
+
end
|
70
|
+
|
71
|
+
def test_add_delete3
|
72
|
+
x = Variable.new name: 'x'
|
73
|
+
solver = SimplexSolver.new
|
74
|
+
c1 = x.cn_equal 100, Strength::WeakStrength, 5
|
75
|
+
c2 = x.cn_equal 200, Strength::WeakStrength
|
76
|
+
|
77
|
+
solver.add_constraint c1
|
78
|
+
solver.add_constraint c2
|
79
|
+
assert x.value.cl_approx 100
|
80
|
+
|
81
|
+
solver.remove_constraint c1
|
82
|
+
assert x.value.cl_approx 200
|
83
|
+
end
|
84
|
+
|
85
|
+
def test_inconsistent1
|
86
|
+
x = Variable.new name: 'x'
|
87
|
+
solver = SimplexSolver.new
|
88
|
+
solver.add_constraint x.cn_equal 10
|
89
|
+
assert_raise RequiredFailure do
|
90
|
+
solver.add_constraint x.cn_equal 5
|
91
|
+
end
|
92
|
+
end
|
93
|
+
|
94
|
+
def test_inconsistent2
|
95
|
+
x = Variable.new name: 'x'
|
96
|
+
solver = SimplexSolver.new
|
97
|
+
solver.add_constraint x.cn_geq 10
|
98
|
+
assert_raise RequiredFailure do
|
99
|
+
solver.add_constraint x.cn_leq 5
|
100
|
+
end
|
101
|
+
end
|
102
|
+
|
103
|
+
def test_stay1
|
104
|
+
x = Variable.new name: 'x', value: 20
|
105
|
+
solver = SimplexSolver.new
|
106
|
+
|
107
|
+
solver.add_stay x, Strength::WeakStrength
|
108
|
+
assert x.value.cl_approx 20
|
109
|
+
end
|
110
|
+
|
111
|
+
def test_two_solutions
|
112
|
+
x = Variable.new name: 'x'
|
113
|
+
y = Variable.new name: 'y'
|
114
|
+
|
115
|
+
solver = SimplexSolver.new
|
116
|
+
solver.add_constraint x.cn_leq y
|
117
|
+
solver.add_constraint y.cn_equal x + 3
|
118
|
+
solver.add_constraint x.cn_equal 10, Strength::WeakStrength
|
119
|
+
solver.add_constraint y.cn_equal 10, Strength::WeakStrength
|
120
|
+
|
121
|
+
assert(x.value.cl_approx(10) && y.value.cl_approx(13) ||
|
122
|
+
x.value.cl_approx(7) && y.value.cl_approx(10))
|
123
|
+
end
|
124
|
+
|
125
|
+
def test_weighted1
|
126
|
+
x = Variable.new name: 'x'
|
127
|
+
solver = SimplexSolver.new
|
128
|
+
|
129
|
+
c15 = x.cn_equal 15, Strength::WeakStrength
|
130
|
+
c20 = x.cn_equal 20, Strength::WeakStrength, 2
|
131
|
+
|
132
|
+
solver.add_constraint c15
|
133
|
+
assert x.value.cl_approx 15
|
134
|
+
|
135
|
+
solver.add_constraint c20
|
136
|
+
assert x.value.cl_approx 20
|
137
|
+
|
138
|
+
solver.remove_constraint c20
|
139
|
+
assert x.value.cl_approx 15
|
140
|
+
end
|
141
|
+
|
142
|
+
def test_edit1
|
143
|
+
x = Variable.new name: 'x', value: 20
|
144
|
+
y = Variable.new name: 'y', value: 30
|
145
|
+
|
146
|
+
solver = SimplexSolver.new
|
147
|
+
solver.add_stay x, Strength::WeakStrength
|
148
|
+
solver.add_constraint x.cn_geq 10
|
149
|
+
solver.add_constraint x.cn_leq 100
|
150
|
+
solver.add_constraint x.cn_equal y * 2
|
151
|
+
assert x.value.cl_approx 20
|
152
|
+
assert y.value.cl_approx 10
|
153
|
+
|
154
|
+
solver.add_edit_var y, Strength::StrongStrength
|
155
|
+
solver.begin_edit
|
156
|
+
solver.suggest_value y, 35
|
157
|
+
solver.resolve
|
158
|
+
assert x.value.cl_approx 70
|
159
|
+
assert y.value.cl_approx 35
|
160
|
+
|
161
|
+
solver.suggest_value y, 80
|
162
|
+
solver.resolve
|
163
|
+
assert x.value.cl_approx 100
|
164
|
+
assert y.value.cl_approx 50
|
165
|
+
|
166
|
+
solver.suggest_value y, 25
|
167
|
+
solver.resolve
|
168
|
+
assert x.value.cl_approx 50
|
169
|
+
assert y.value.cl_approx 25
|
170
|
+
|
171
|
+
solver.end_edit
|
172
|
+
assert x.value.cl_approx 50
|
173
|
+
assert y.value.cl_approx 25
|
174
|
+
|
175
|
+
solver.add_edit_var x, Strength::StrongStrength
|
176
|
+
solver.begin_edit
|
177
|
+
solver.suggest_value x, 44.0
|
178
|
+
solver.resolve
|
179
|
+
assert x.value.cl_approx 44
|
180
|
+
assert y.value.cl_approx 22
|
181
|
+
|
182
|
+
solver.end_edit
|
183
|
+
end
|
184
|
+
end
|
data/test/test_ext.rb
ADDED
@@ -0,0 +1,38 @@
|
|
1
|
+
require File.expand_path("../test_helper", __FILE__)
|
2
|
+
|
3
|
+
class ExtTests < Test::Unit::TestCase
|
4
|
+
def test_float_approx_zero
|
5
|
+
assert !1.1.cl_approx_zero
|
6
|
+
assert 0.0.cl_approx_zero
|
7
|
+
assert 0.1e-8.cl_approx_zero
|
8
|
+
end
|
9
|
+
|
10
|
+
def test_float_approx
|
11
|
+
assert 1.1.cl_approx 1.1
|
12
|
+
end
|
13
|
+
|
14
|
+
def test_float_negative
|
15
|
+
assert -1.1.definitely_negative
|
16
|
+
assert -0.1e-5.definitely_negative
|
17
|
+
assert ! -0.1e-8.definitely_negative
|
18
|
+
end
|
19
|
+
|
20
|
+
def test_numeric_as_linear_expression
|
21
|
+
expr = 1.as_linear_expression
|
22
|
+
assert expr.terms.empty?
|
23
|
+
assert expr.constant == 1.to_f
|
24
|
+
|
25
|
+
expr = 1.1.as_linear_expression
|
26
|
+
assert expr.terms.empty?
|
27
|
+
assert expr.constant == 1.1
|
28
|
+
end
|
29
|
+
|
30
|
+
def test_object_approx
|
31
|
+
assert "foo".cl_approx "foo"
|
32
|
+
assert ! Object.new.cl_approx(Object.new)
|
33
|
+
end
|
34
|
+
|
35
|
+
def test_object_symbolic_weight
|
36
|
+
assert !Object.new.symbolic_weight?
|
37
|
+
end
|
38
|
+
end
|
data/test/test_helper.rb
ADDED
@@ -0,0 +1,33 @@
|
|
1
|
+
require File.expand_path("../test_helper", __FILE__)
|
2
|
+
|
3
|
+
class VariablesTests < Test::Unit::TestCase
|
4
|
+
def test_operations
|
5
|
+
x = Cassowary::Variable.new name: 'x', value: 20
|
6
|
+
expr = x / 10
|
7
|
+
assert expr.constant == 0
|
8
|
+
assert expr.terms[x] == 0.1
|
9
|
+
|
10
|
+
expr = x * 10
|
11
|
+
assert expr.terms[x] == 10
|
12
|
+
|
13
|
+
expr = x - 10
|
14
|
+
assert expr.constant == -10
|
15
|
+
|
16
|
+
expr = x + 10
|
17
|
+
assert expr.constant == 10
|
18
|
+
|
19
|
+
expr = -x
|
20
|
+
assert expr.terms[x] == -1
|
21
|
+
end
|
22
|
+
|
23
|
+
def test_inspect
|
24
|
+
x = Cassowary::Variable.new name: 'x', value: 21.1
|
25
|
+
assert_equal "x[21.1]", x.inspect
|
26
|
+
|
27
|
+
x = Cassowary::Variable.new name: 'x'
|
28
|
+
assert_equal "x[nil]", x.inspect
|
29
|
+
|
30
|
+
x = Cassowary::SlackVariable.new
|
31
|
+
assert_equal "<CV#0x" + x.object_id.to_s(16) + ">", x.inspect
|
32
|
+
end
|
33
|
+
end
|