In order to level up our programming skills, and have something to do while learning software engineering workflows, let's solve some puzzles in Python.
In this repository, you'll find twenty-five folders, each containing one puzzle. Each folder contains a README.md
with a description of the problem, and a .py
file with a stub of a solution. Your job is to complete the implementation.
Here are the puzzles in very approximate order of difficulty. (As you're dividing and conquering, keep the difficulty level in mind relative to each of your Python proficiency.)
- leap
- darts
- collatz-conjecture
- scrabble-score
- anagram
- acronym
- isogram
- pangram
- word-count
- rotational-cipher
- atbash-cipher
- isbn-verifier
- pig-latin
- roman-numerals
- run-length-encoding
- say
- matrix
- phone-number
- luhn
- matching-brackets
- poker
- affine-cipher
- change
- bowling
- book-store
Your job, as a team, is to solve as many as possible.
-
You've all been assigned to a team that has write access to this repository.
-
Keep your command-line cheat sheet handy. The main things you'll need:
cd folder_name
to navigate into a directory.cd ..
to navigate out of a directory (back to the parent folder).- Remember to use the tab key often to autocomplete file and folder names. It saves a ton of typing, and confirms the existence of the file or folder you are targeting; and will alert you to when you're not in the folder you think you are, if the autocomplete fails.
-
Clone the repository to your own computer. You can paste the URL of the repo into GitKraken, or use command-line Git:
$ git clone https://github.com/boothresearch/puzzles
-
Alternatively, you can use Gitpod to work on the puzzles instead of your own computer; if, for example, you're using a loaner that doesn't have Python installed.
-
The first thing you each should do is create your own version, or branch to work on. You can use GitKraken, or command-line Git:
$ git checkout -b your-name-first-branch
Substitute your own name, GitHub username, nickname, or initials where you see
your-name
above. -
Then, in your VSCode terminal,
cd
into whichever puzzle you want to work on, e.g. the first one,leap
:$ cd leap
-
Within each folder, there is a
README
that describes the problem to be solved. Read theREADME
inleap/
. -
Start editing
leap.py
; let's replace thepass
withreturn("Party like it's " + str(year))
:def leap_year(year): return("Party like it's " + str(year))
-
To try out our code, we could add a print statement to bottom of the file:
def leap_year(year): return("Party like it's " + str(year)) print(leap_year(1999))
and then run it from the command line:
$ python3 leap.py Party like it's 1999
but, instead, I usually find it more helpful to leave the file untouched and run the program interactively. That way I can try it out with a few different inputs:
$ python3 -i leap.py >>> leap_year(1999) "Party like it's 1999" >>> leap_year(2000) "Party like it's 2000" >>>
Note that the
-i
flag must come before the filename. -
cd ..
to navigate back up to the parent folder when you're ready to work on a different puzzle, and thencd
into that one. Rinse and repeat. -
Each of you should choose a task to work on. Divide and conquer as you see fit. Before each task, first take a snapshot of the work that you did so far on your current branch:
git add -A git commit -m "A description of what you did"
and then switch back to the original version:
git checkout main
Get the freshest version, in case there have been any updates:
git pull
And then start a new version for your new task:
git checkout -b whatever-my-next-task-is
More on this below.
- The official Python tutorial
- DataCamp's Introduction to Python
- DataCamp's Intermediate Python for Data Science
Git is a very powerful tool with a lot of commands. Here is the small subset of commands that I use a million times a day. We'll add to our toolbelt as time goes on:
-
To take a snapshot of your work (all of the files and subfolders), use the following two commands from the root folder of the repository:
git add -A git commit -m "Title of your snapshot"
-
Try to make the title somewhat descriptive of what you did since the last snapshot, but it's more important that you just make lots of commits. So if you must just say
git commit -m "WIP"
(for "work in progress"), that's fine; we can clean it up later. -
Our golden rule: Always Be Committing. As long as you do that, everything will be okay; we can recover from anything with git as long as you commit early and often. You cannot _over_commit but you can most certainly _under_commit.
-
To see the current status of the repository (are there any changed files since the last commit? etc):
git status
-
To see the line-by-line changes since the last commit:
git diff
or to see the contents of the last commit:
git show
-
To start a new version, or branch:
git checkout -b fancy-new-version-name
-
Again, branch early and often. There's no cost to it and there's lots of benefits to it. In particular, it's good to start a new branch for each task or feature. Or just a different approach to the same task, if you want to start over.
-
To list all branches:
git branch
-
To switch to another existing branch:
git checkout branch-name
-
Sometimes, if you have made edits to files that you haven't committed yet, it won't let you switch to another branch. Which is good; you have to make a decision about those changes first — do you want to save them or not? Your options:
- Make a commit first, and then you can switch to the other branch.
- If you don't want to pollute your current branch, you can make a new branch, commit the changes, and then switch to the other branch.
- You can quickly "stash" the changes with
git add -A; git stash
. This puts the changes into a randomly named branch that will eventually be deleted after a few weeks, but until then you can get the changes back if you want them. This is the equivalent of the above, but saves you the trouble of having to think of a name for the new branch.
I usually just think of
git add -A; git stash
as a convenient way to discard all the changes since my most recent commit, so that I can start afresh; but, there is a way to get those changes back if I want them (this only happens about once a year). -
To see the history of your current branch:
git log
You will see the author, date, and title of each commit preceded by a long sequence of letters and numbers known as the "SHA-1 hash" of the commit. This is a unique fingerprint of the snapshot.
-
(Advanced) To jump back to a prior commit:
git checkout [SHA of commit you want to go back to]
Find the SHA by looking at the output if
git log
. Just the first 7 or so digits of the SHA are enough; you don't need the whole thing.If you jump to a commit like this, you'll be in a "detached" state — i.e., not on any branch. This is okay for browsing, but it's best not to make any changes.
If you want to start a new branch from this point, though, that's perfectly fine — I do that all the time when I decide I want to try a new approach. Just
git checkout -b new-branch-name
as usual. -
When you're ready to send the work you've done on a branch back to GitHub.com:
git push
-
The very first time you
push
to a branch, it may ask you to do something like:git push --set-upstream origin your-branch-name
-
(If you're using Gitpod: the very first time you
push
from Gitpod, it may ask you to give it permission to do so from GitHub. Go ahead and do so.)
-
-
To retrieve the freshest version of the branch you're on from GitHub.com, in case there have been any updates:
git pull
-
When you're ready to start on a new task, the best practice is to first switch back to the
main
branch, get the freshest version, and then to start a new branch from there:git checkout main git pull git checkout -b my-next-task
-
Never make commits directly to the
main
branch. Always work on your own, experimental branch. We'll talk about why this is soon.
I know this is a lot of command-line Git commands, but just try to experiment and see how it feels compared to GitKraken. Please wave over an instructor as you experiment with these commands and ask questions; remember:
Questions are places in your mind where answers fit. If you haven’t asked the question, the answer has nowhere to go. — Clay Christensen
When you're reading to bring the changes from one branch (usually from one of your experimental branches, e.g. your-name-first-branch
) into another branch (usually into the main
branch), we have a couple of options:
- Fundamentally, we use Git's
merge
command at the command-line. - In the early days, I recommend pushing your branch to GitHub and using GitHub's or GitKraken's interface for merging.
To merge changes from e.g. your-name-first-branch
into main
using GitHub's interface:
-
Push your branch to GitHub:
git checkout your-name-first-branch git push
-
Merge the PR. You're given a few strategies to choose from on how to do so; choose Rebase and merge.
-
Switch back to
main
on your machine and fetch the merged version:git checkout main git pull
To merge changes from e.g. your-name-first-branch
into main
using the command-line:
-
First switch to
main
and make sure it is up to date:git checkout main git pull
-
Switch back to your feature branch and "rebase interactively" onto
main
. This means, essentially, sync up the feature branch with any changes that may have occurred onmain
since they time you created the branch:git checkout your-name-first-branch git rebase -i
-
A text editor will pop up with a list of all of the commits you made along the way on your feature branch. Replace
pick
for all but one of them withs
(forsquash
). Save and close the file. -
Another text editor will open where you can craft a wonderful commit, single message to communicate the WHY of your changes (the WHAT is told by the diff).
-
Now that all your messy work-in-progress commits have been ironed out, we can merge:
git checkout main git merge your-name-first-branch
-
That's it — verify with
git log
. It's as if you made one, beautiful commit directly tomain
.
You may have noticed that it's tedious to know when you've solved a problem completely; we can test the function a few times interactively with various arguments, but we have to do that over and over again while we're working. That's a lot of typing, and error prone!
Instead, it's a good idea to automate testing, rather than doing it manually. For each puzzle, you'll see a file ending in _test.py
. These are examples of how software engineers automate testing. You can run each one like any other Python program to see what they do.
There's also a command available called pytest
that you can run that has slightly more output, and you don't need to specify the file containing the tests; it will assume that any files ending in _test.py
contain tests and run them all. Give it a try.
How close were your solutions to passing all the tests?