Today, in 2024, project is largely unmaintained. I'm using it as a playground for some things I want to try out, but I don't recommend anyone build off this.
Remember logic puzzles that looked like this? They had clues like:
- There is one house between Tushar and Maui.
- Ruby's favorite video game is Pokemon.
- The chai drinker does not like the color blue.
I loved these as a kid, and watching Raymond Hettinger's PyCon 2019 talk inspired me to revisit these problems. He showed us how to solve them; I wanted to learn how to generate them.
Using modern Python and constraint satisfaction (SAT) solvers, this project can be used to create random zebra puzzles.
This project uses Python 3.12 and Poetry.
The entrypoint to this project is src/main.py
; accordingly, run it with [poetry run] python -m src.main
.
We use questionary to create a CLI with simple interactivity.
Other important files are:
puzzle.py
contains the main Puzzle class, which is mostly glue around clues and puzzle elementsclues.py
defines a clue (a higher-level statement that we can convert to a CNF) and several factories, likefound_at
orsame_house
;generate.py
has utilities to create clueselements.py
andtraptor_elements.py
have various puzzle elements
Finally, sat_utils.py
contains the basic utilities for interacting with the SAT solver pycosat. This code was almost entirely written by Raymond Hettinger for his 2019 talk (thank you!).
โฏ poetry run python -m src.main
? How large is the puzzle? (Use arrow keys)
ยป 5
4
3
? Choose a Traptor type (Use arrow keys)
๐ด Tropical
ยป ๐ฐ Mythical
? ๐ฅค Should the puzzle include smoothies? (Use arrow keys)
ยป Yes
No
? ๐ต Should the puzzle include bottlecaps? (Use arrow keys)
ยป Yes
No
Clues
-----
# ... Some flavor text omitted ...
1. The Lesser Traptor and the Swamp Traptor are in the same house.
2. The Ancient Traptor is directly left of the Chocolate smoothie.
3. The Ancient Traptor and the Lemon smoothie are in the same house.
4. The Volcano Traptor is directly left of the Restless Traptor.
5. The Cave Traptor and Green bottlecaps are in the same house.
6. There are two houses between the Lake Traptor and the Snapper Traptor.
7. Green bottlecaps and the Fierce Traptor are in the same house.
8. There is one house between Black bottlecaps and the Coconut smoothie.
9. There is one house between the Lesser Traptor and the Kiwi smoothie.
10. The Dweller Traptor and the Chocolate smoothie are in the same house.
11. Red bottlecaps and the Dweller Traptor are in the same house.
12. Red bottlecaps is directly left of the Hoarder Traptor.
13. There are two houses between the Starfruit smoothie and Yellow bottlecaps.
14. The Lake Traptor and Black bottlecaps are in the same house.
15. The Stalker Traptor and the Fierce Traptor are in the same house.
Solution
โโโโโโโโณโโโโโโโโโโโโโโโโโโโโโโโโโณโโโโโโโโโโโโโโโโโโโโโโโโโโโณโโโโโโโโโโโโโโโโโโโโโโโโโโณโโโโโโโโโโโโโโโโโโโโโโโโโณโโโโโโโโโโโโโโโโโโโโ
โ Nest โ MythicalTraptorPrimary โ MythicalTraptorSecondary โ MythicalTraptorTertiary โ Smoothie โ Bottlecap โ
โกโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฉ
โ 1 โ the Fierce Traptor โ the Cave Traptor โ the Stalker Traptor โ the Starfruit smoothie โ Green bottlecaps โ
โ 2 โ the Ancient Traptor โ the Lake Traptor โ the Crawler Traptor โ the Lemon smoothie โ Black bottlecaps โ
โ 3 โ the Lesser Traptor โ the Swamp Traptor โ the Dweller Traptor โ the Chocolate smoothie โ Red bottlecaps โ
โ 4 โ the Greater Traptor โ the Volcano Traptor โ the Hoarder Traptor โ the Coconut smoothie โ Yellow bottlecaps โ
โ 5 โ the Restless Traptor โ the Mountain Traptor โ the Snapper Traptor โ the Kiwi smoothie โ Blue bottlecaps โ
โโโโโโโโดโโโโโโโโโโโโโโโโโโโโโโโโโดโโโโโโโโโโโโโโโโโโโโโโโโโโโดโโโโโโโโโโโโโโโโโโโโโโโโโโดโโโโโโโโโโโโโโโโโโโโโโโโโดโโโโโโโโโโโโโโโโโโโโ
In February 2024, what do I need to consider this "done"?
- Add HTML templating, because the site that I build these for requires that I include things like BR tags.
- Improve the grammar nits in the output, like capitalization in the flavor text.
- Add the header/footer resources that I copy/paste into the submission.
Maybe, someday, I will consider a web interface. Could I lift the project using pyodide, or would the C-based pycosat cause problems? What about another logic programming interface, like PySAT, answer set programming, or others from this HN thread?
These may happen. But the project is finally at a place that I'm happy with, and so I'm excited to share and write about it.
2/18: make outputs simpler, less verbose, prettier. Update this README with new formatting and changes. Use the puzzle generator to create a new puzzle, finally!
2/17: update puzzle signature to accept the solution only, inferring the elements and element classes from it. Add a nicer display for the solution.
11/26: Clean up the puzzle/solution generation logic. Let the user choose the puzzle size.
11/25: Name some common types as type aliases (thanks, Python 3.12). Use logging instead of prints.
Add tests for clue types within generate.py
.
Fix a bug that I introduced in clues.py
. That file is so hard to understand, but it's also hard to write unit tests for; converting boolean expressions to CNF by hand is no fun.
Tune the weights of clues in the solution. Prepare to integrate the puzzle and solution classes.
Remove the checkbox-esque Smoothie selection, because it just gets in the way when trying to make puzzles quickly.
Continue adding tests. Finish up the unit tests for sat_utils.py
and start a couple for clues.py
, though that file is very hard to test because of the difficulty of hand-computing DNF-to-CNF conversions.
Next steps are unifying the puzzle and solution, creating a better __repr__
, and simplifying how we represent puzzle size (attribute on Puzzle
, tuple of ints 1 to N, just the number n_houses
, etc.).
Remove black and use ruff for formatting. Update dependencies. Add new CLI with simpler, clearer usage; python -m src.main
.
Rename SATLiteral
-> PuzzleElement
(smoothie, cat, etc.); this clarifies that it's not a Literal in the boolean sense and is instead a name for e.g., characters in a puzzle.
Create a new type SATLiteral
(which is a str
in a trenchcoat); this represents the literal in the boolean-variable sense, like "Value el is at house loc" or "Value el is not at house loc." Internally, these are represented by some integer i and its negative counterpart -i.
Use (the new) SATLiteral
as the return type of comb(el: str, loc: int) -> SATLiteral
, mapping puzzle elements to literals, and neg(el: SATLiteral) -> SATLiteral
to negate a literal.
Add Clause: tuple[SATLiteral, ...]
representing a "โจ" (boolean OR; disjunction) of literals; e.g., (1, -5, 6)
is a disjunction stating that x_1
is true, x_5
is not true, or x_6
is true.
Add ClueCNF: list[Clause]
, representing a "โง" (boolean AND; conjunction) of Clause
s. A puzzle in CNF is an "AND of ORs" ("โง of โจs" or "โง of Clauses").
Print the puzzle more clearly. Reduce verbosity of the clue reduction.
Randomly generate the puzzle (with a seed) on each run.
Move examples into their own files. Update to Python 3.12 (beta). Add more lint rules. Clean up imports.
Add dev tools (black, ruff, pyright); run black & ruff in pre-commit. Update some types.
- "โง" is the boolean AND
- "โจ" is the boolean OR
- a Clause is an "โจ of Literals"
- a CNF is an "โง of Clauses," or equivalently an "โง of โจs" ("AND of ORs")
A DNF, in contrast, is a "โจ of โงs." The DNF is the answer to a SAT problem; a DNF of "A or B or C" reads that A, B, and C are all valid (satisfying) assignments. Converting a CNF to a DNF is therefore NP-hard, since from the DNF you can read off solutions to the CNF.
CNF is an โง of โจs, where โจ is over variables or their negations (literals); an โจ of literals is also called a clause. DNF is an โจ of โงs; an โง of literals is called a term.
The core of this project is in expressing the zebra puzzle as a Boolean satisfiability (SAT) problem (Wikipedia.
The 3-SAT problem is known to be NP-complete. That is, there is probably no polynomial-time algorithm to solve it. This only describes the general case, though.
Our problem is small and well-specified (after all, we are creating these specifically for humans!). This makes it an excellent fit for modern SAT solvers, and Hettinger's talk (and notes) demonstrates this beautifully.