sweetpea.constraints module

This module provides constraints for CNF generation.

sweetpea.constraints.validate_factor_and_level(block, factor, level)
Parameters
Return type

None

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

None

static apply(block, backend_request)
Parameters
Return type

None

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:

  1. Generate intermediate vars

    Using the fresh var counter, allocate numTrials * num_states new vars

  2. Entangle them with block vars

    Add to the CNF queue: toCNF(Iff(newVar, And(levels))), e.g., if the variable 1 indicates color:red, the var 3 indicates text:red, and the var 25 represents (color:red, text:red), do toCNF(Iff(25, And([1, 3])))

  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

None

static apply(block, backend_request)
Parameters
Return type

None

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

None

static apply(block, backend_request)
Parameters
Return type

None

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 from 0, but the boolean variables start from 1). 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 general Window rather than Transition or WithinTrial). 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
Return type

None

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

None

apply(block, backend_request)
Parameters
Return type

None

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 want AtMostKInARow 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
Return type

None

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
Return type

None

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
Return type

None

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
Return type

None

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

None

extract_simplelevel(block, level)

Recursively deciphers the excluded level to a list of combinations basic levels.

Parameters
Return type

List[Dict[sweetpea.primitives.Factor, sweetpea.primitives.SimpleLevel]]

apply(block, backend_request)
Parameters
Return type

None

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

None

apply(block, backend_request)
Parameters
Return type

None