-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathrun.py
85 lines (67 loc) · 2.89 KB
/
run.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
from bauhaus import Encoding, proposition, constraint
from bauhaus.utils import count_solutions, likelihood
# These two lines make sure a faster SAT solver is used.
from nnf import config
config.sat_backend = "kissat"
# Encoding that will store all of your constraints
E = Encoding()
# To create propositions, create classes for them first, annotated with "@proposition" and the Encoding
@proposition(E)
class BasicPropositions:
def __init__(self, data):
self.data = data
def _prop_name(self):
return f"A.{self.data}"
# Different classes for propositions are useful because this allows for more dynamic constraint creation
# for propositions within that class. For example, you can enforce that "at least one" of the propositions
# that are instances of this class must be true by using a @constraint decorator.
# other options include: at most one, exactly one, at most k, and implies all.
# For a complete module reference, see https://bauhaus.readthedocs.io/en/latest/bauhaus.html
@constraint.at_least_one(E)
@proposition(E)
class FancyPropositions:
def __init__(self, data):
self.data = data
def _prop_name(self):
return f"A.{self.data}"
# Call your variables whatever you want
a = BasicPropositions("a")
b = BasicPropositions("b")
c = BasicPropositions("c")
d = BasicPropositions("d")
e = BasicPropositions("e")
# At least one of these will be true
x = FancyPropositions("x")
y = FancyPropositions("y")
z = FancyPropositions("z")
# Build an example full theory for your setting and return it.
#
# There should be at least 10 variables, and a sufficiently large formula to describe it (>50 operators).
# This restriction is fairly minimal, and if there is any concern, reach out to the teaching staff to clarify
# what the expectations are.
def example_theory():
# Add custom constraints by creating formulas with the variables you created.
E.add_constraint((a | b) & ~x)
# Implication
E.add_constraint(y >> z)
# Negate a formula
E.add_constraint(~(x & y))
# You can also add more customized "fancy" constraints. Use case: you don't want to enforce "exactly one"
# for every instance of BasicPropositions, but you want to enforce it for a, b, and c.:
constraint.add_exactly_one(E, a, b, c)
return E
if __name__ == "__main__":
T = example_theory()
# Don't compile until you're finished adding all your constraints!
T = T.compile()
# After compilation (and only after), you can check some of the properties
# of your model:
print("\nSatisfiable: %s" % T.satisfiable())
print("# Solutions: %d" % count_solutions(T))
print(" Solution: %s" % T.solve())
print("\nVariable likelihoods:")
for v,vn in zip([a,b,c,x,y,z], 'abcxyz'):
# Ensure that you only send these functions NNF formulas
# Literals are compiled to NNF here
print(" %s: %.2f" % (vn, likelihood(T, v)))
print()