sweetpea.constraints module¶
This module provides constraints for CNF generation.
- sweetpea.constraints.validate_factor_and_level(block, factor, level)¶
- Parameters
block (sweetpea.blocks.Block) –
factor (sweetpea.primitives.Factor) –
level (Union[sweetpea.primitives.SimpleLevel, sweetpea.primitives.DerivedLevel]) –
- Return type
- class sweetpea.constraints.Consistency¶
Bases:
sweetpea.base_constraint.Constraint
This constraint ensures that only one level of each factor is ‘on’ at a time. So for instance in the experiment:
color = Factor("color", ["red", "blue"]) text = Factor("text", ["red", "blue"]) design = crossing = [color, text, conFactor] experiment = fully_cross_block(design, crossing, [])
The first trial is represented by the boolean vars
[1, 2, 3, 4]
:1 is true iff the trial is color:red
2 is true iff the trial is color:blue
3 is true iff the trial is text:red
4 is true iff the trial is text:blue
The second trial is represented by the boolean vars
[5-8]
, the third by[9-12]
, the fourth by[13-16]
. So this desugaring applies the following constraints:sum(1, 2) EQ 1 sum(3, 4) EQ 1 sum(5, 6) EQ 1 ... sum(15, 16) EQ 1
- validate(block)¶
Constraints can’t be completely validated in isolation. This function will be called on all constraints with the block during the block initialization to ensure all constraints are valid.
- Parameters
block (sweetpea.blocks.Block) –
- Return type
- static apply(block, backend_request)¶
- Parameters
block (sweetpea.blocks.Block) –
backend_request (sweetpea.backend.BackendRequest) –
- Return type
- class sweetpea.constraints.FullyCross¶
Bases:
sweetpea.base_constraint.Constraint
We represent the fully crossed constraint by allocating additional boolean variables to represent each unique state. Only factors in crossing will contribute to the number of states (there may be factors in the design that aren’t in the crossing).
Continuing with the example from
Consistency
, we will represent the states:(color:red, text:red) (color:red, text:blue) (color:blue, text:red) (color:blue, text:blue)
The steps taken are:
Generate intermediate vars
Using the fresh var counter, allocate
numTrials * num_states
new varsEntangle them with block vars
Add to the CNF queue:
toCNF(Iff(newVar, And(levels)))
, e.g., if the variable1
indicatescolor:red
, the var3
indicatestext:red
, and the var25
represents(color:red, text:red)
, dotoCNF(Iff(25, And([1, 3])))
1 hot the states e.g., 1 red circle, etc
Same as
Consistency
above, collect all the state vars that represent each state & enforce that only one of those states is true, e.g.,sum(25, 29, 33, 37) EQ 1
(and 3 more of these for each of the other states).
- validate(block)¶
Constraints can’t be completely validated in isolation. This function will be called on all constraints with the block during the block initialization to ensure all constraints are valid.
- Parameters
block (sweetpea.blocks.Block) –
- Return type
- static apply(block, backend_request)¶
- Parameters
block (sweetpea.blocks.FullyCrossBlock) –
backend_request (sweetpea.backend.BackendRequest) –
- Return type
- class sweetpea.constraints.MultipleCross¶
Bases:
sweetpea.base_constraint.Constraint
- validate(block)¶
Constraints can’t be completely validated in isolation. This function will be called on all constraints with the block during the block initialization to ensure all constraints are valid.
- Parameters
block (sweetpea.blocks.Block) –
- Return type
- static apply(block, backend_request)¶
- Parameters
block (sweetpea.blocks.MultipleCrossBlock) –
backend_request (sweetpea.backend.BackendRequest) –
- Return type
- class sweetpea.constraints.Derivation(derived_idx, dependent_idxs, factor)¶
Bases:
sweetpea.base_constraint.Constraint
A derivation such as:
Derivation(4, [[0, 2], [1, 3]])
where the index of the derived level is
4
, and[[0, 2], [1, 3]]
is the list of dependent indices, represents the logical formula:4 iff (0 and 2) or (1 and 3)
These indicies are used the get the corresponding trial variables. Continuing from the example in of processDerivations, the first trial is represented by variables
[1-6]
(notice this feels like an off-by-one: the indicies start from0
, but the boolean variables start from1
). So we would use the indices to map onto the vars as:5 iff (1 and 3) or (2 and 4)
Then we convert to CNF directly, i.e.:
toCNF(Iff(5, Or(And(1,3), And(2,4))))
This is then done for all window-sizes, taking into account strides (which are specified only in
DerivedLevels
specified with a generalWindow
rather thanTransition
orWithinTrial
). We grab window-sized chunks of the variables that represent the trials, map the variables using the indices, and then convert to CNF. These chunks look like:window1: 1 2 3 4 5 6 window2: 7 8 9 10 11 12
So, for the second trial (since the window size in this example is
1
) it would be:11 iff (7 and 9) or (8 and 10)
90% sure this is the correct way to generalize to derivations involving 2+ levels & various windowsizes. One test is the experiment:
color = ["r", "b", "g"]; text = ["r", "b"]; conFactor; fullycross(color, text) + AtMostKInARow 1 conLevel
- Parameters
derived_idx (int) –
dependent_idxs (List[List[int]]) –
factor (sweetpea.primitives.DerivedFactor) –
- Return type
- validate(block)¶
Constraints can’t be completely validated in isolation. This function will be called on all constraints with the block during the block initialization to ensure all constraints are valid.
- Parameters
block (sweetpea.blocks.Block) –
- Return type
- apply(block, backend_request)¶
- Parameters
block (sweetpea.blocks.Block) –
backend_request (sweetpea.backend.BackendRequest) –
- Return type
- is_complex(block)¶
- Parameters
block (sweetpea.blocks.Block) –
- sweetpea.constraints.at_most_k_in_a_row(k, levels)¶
This desugars pretty directly into the llrequests. The only thing to do here is to collect all the boolean vars that match the same level & pair them up according to k.
Continuing with the example from
Consistency
, say we wantAtMostKInARow 1 ("color", "red")
, then we need to grab all the vars which indicate color-red:[1, 7, 13, 19]
and then wrap them up so that we’re making requests like:
sum(1, 7) LT 2 sum(7, 13) LT 2 sum(13, 19) LT 2
If it had been
AtMostKInARow 2 ("color", "red")
, the reqs would have been:sum(1, 7, 13) LT 3 sum(7, 13, 19) LT 3
- class sweetpea.constraints.AtMostKInARow(k, level)¶
Bases:
sweetpea.constraints._KInARow
- Parameters
level (Tuple[sweetpea.primitives.Factor, Union[sweetpea.primitives.SimpleLevel, sweetpea.primitives.DerivedLevel]]) –
- apply_to_backend_request(block, level, backend_request)¶
- Parameters
block (sweetpea.blocks.Block) –
level (Tuple[sweetpea.primitives.Factor, Union[sweetpea.primitives.SimpleLevel, sweetpea.primitives.DerivedLevel]]) –
backend_request (sweetpea.backend.BackendRequest) –
- Return type
- sweetpea.constraints.at_least_k_in_a_row(k, levels)¶
This is more complicated that AtMostKInARow. We collect all the boolean vars that match the same level & pair them up according to k.
We want
AtLeastKInARow 2 ("color", "red")
, then we need to grab all the vars which indicate color-red:[1, 7, 13, 19]
and then wrap them up in CNF as follows:
If(1) Then (7) --------This is a corner case If(And(!1, 7)) Then (13) If(And(!7, 13)) Then (19) If(19) Then (13) --------This is a corner case
If it had been
AtLeastKInARow 3 ("color", "red")
, the CNF would have been:If(1) Then (7, 13) --------This is a corner case If(And(!1, 7)) Then (13, 19) If(19) Then (7, 13) --------This is a corner case
- class sweetpea.constraints.AtLeastKInARow(k, levels)¶
Bases:
sweetpea.constraints._KInARow
- apply_to_backend_request(block, level, backend_request)¶
- Parameters
block (sweetpea.blocks.Block) –
level (Tuple[sweetpea.primitives.Factor, Union[sweetpea.primitives.SimpleLevel, sweetpea.primitives.DerivedLevel]]) –
backend_request (sweetpea.backend.BackendRequest) –
- Return type
- sweetpea.constraints.exactly_k(k, levels)¶
Requires that if the given level exists at all, it must exist in a trial exactly
k
times.
- class sweetpea.constraints.ExactlyK(k, level)¶
Bases:
sweetpea.constraints._KInARow
- Parameters
level (Tuple[sweetpea.primitives.Factor, Union[sweetpea.primitives.SimpleLevel, sweetpea.primitives.DerivedLevel]]) –
- apply_to_backend_request(block, level, backend_request)¶
- Parameters
block (sweetpea.blocks.Block) –
level (Tuple[sweetpea.primitives.Factor, Union[sweetpea.primitives.SimpleLevel, sweetpea.primitives.DerivedLevel]]) –
backend_request (sweetpea.backend.BackendRequest) –
- Return type
- sweetpea.constraints.exactly_k_in_a_row(k, levels)¶
Requires that if the given level exists at all, it must exist in a sequence of exactly K.
- class sweetpea.constraints.ExactlyKInARow(k, level)¶
Bases:
sweetpea.constraints._KInARow
- Parameters
level (Tuple[sweetpea.primitives.Factor, Union[sweetpea.primitives.SimpleLevel, sweetpea.primitives.DerivedLevel]]) –
- apply_to_backend_request(block, level, backend_request)¶
- Parameters
block (sweetpea.blocks.Block) –
level (Tuple[sweetpea.primitives.Factor, Union[sweetpea.primitives.SimpleLevel, sweetpea.primitives.DerivedLevel]]) –
backend_request (sweetpea.backend.BackendRequest) –
- Return type
- sweetpea.constraints.exclude(factor, levels)¶
- class sweetpea.constraints.Exclude(factor, level)¶
Bases:
sweetpea.base_constraint.Constraint
- validate(block)¶
Constraints can’t be completely validated in isolation. This function will be called on all constraints with the block during the block initialization to ensure all constraints are valid.
- Parameters
block (sweetpea.blocks.Block) –
- Return type
- extract_simplelevel(block, level)¶
Recursively deciphers the excluded level to a list of combinations basic levels.
- Parameters
block (sweetpea.blocks.Block) –
level (sweetpea.primitives.DerivedLevel) –
- Return type
List[Dict[sweetpea.primitives.Factor, sweetpea.primitives.SimpleLevel]]
- apply(block, backend_request)¶
- Parameters
block (sweetpea.blocks.Block) –
backend_request (sweetpea.backend.BackendRequest) –
- Return type
- sweetpea.constraints.minimum_trials(trials)¶
- class sweetpea.constraints.MinimumTrials(trials)¶
Bases:
sweetpea.base_constraint.Constraint
- validate(block)¶
Constraints can’t be completely validated in isolation. This function will be called on all constraints with the block during the block initialization to ensure all constraints are valid.
- Parameters
block (sweetpea.blocks.Block) –
- Return type
- apply(block, backend_request)¶
- Parameters
block (sweetpea.blocks.Block) –
backend_request (sweetpea.backend.BackendRequest) –
- Return type