Skip to the content.

Introduction

This lab provides you an opportunity to test your mettle against some challenging computational puzzles. These puzzles were selected from Project Euler and can be solved using Cryptol’s automated theorem proving interface.

Prerequisites

Before working through this lab, you’ll need

You’ll also need experience with

Skills You’ll Learn

By the end of this lab you will be a puzzle master!

Load This Module

This lab is a literate Cryptol document — that is, it can be loaded directly into the Cryptol interpreter. Load this module from within the Cryptol interpreter running in the cryptol-course directory with:

Loading module Cryptol
Cryptol> :m labs::ProjectEuler::ProjectEuler
Loading module Cryptol
Loading module labs::ProjectEuler::cipher1
Loading module labs::ProjectEuler::keylog
Loading module labs::ProjectEuler::cipher2
Loading module labs::ProjectEuler::ProjectEuler

We start by defining a new module for this lab and importing some accessory modules that we will use:

module labs::ProjectEuler::ProjectEuler where

import labs::ProjectEuler::cipher1
import labs::ProjectEuler::keylog
import labs::ProjectEuler::cipher2

You do not need to enter the above into the interpreter; the previous :m ... command loaded this literate Cryptol file automatically. In general, you should run Xcryptol-session commands in the interpreter and leave cryptol code alone to be parsed by :m ....

Properties in Cryptol, or How I Learned to Stop Worrying and Love SAT Solvers

You have learned in the other labs that you can use Cryptol to find answers to problems you coded using the :sat command. As a refresher, let’s look at a simple example. Suppose we want to find the integer that is one less than 1001 (as we said, it’s very simple). We do this by creating a function and a property:

inc : Integer -> Integer
inc x = x + 1

inc1001 : Integer -> Bit
property inc1001 x = inc x == 1001

Now that we have our function and a property about our function, we can test it using the Cryptol interpreter.

labs::ProjectEuler::ProjectEuler> :sat inc1001
Satisfiable
inc1001 1000 = True
(Total Elapsed Time: 0.023s, using "Z3")

Let’s do a more complicated example: use Cryptol to factor 3,000,013.

factor3000013 : Integer -> Integer -> Bit
factor3000013 x y =
    x * y == 3000013 /\
    x > 1            /\
    y > 1            /\
    x <= y

Note that if we don’t include the x > 1 /\ y > 1 clauses we get a trivial factorization. Now we can use Cryptol to factor our number:

labs::ProjectEuler::ProjectEuler> :sat factor3000013
Satisfiable
factor3000013 773 3881 = True
(Total Elapsed Time: 0.498s, using "Z3")

Clearly we don’t need SAT solvers to figure out how to subtract one. And writing simple factorization code in Python factored 3,000,013 just as fast. But there is something really interesting here: to solve the problem, instead of figuring out how to write a search algorithm which would solve the problem for us, we just had to write a solution checker in Cryptol and let a SAT solver find a valid solution for us! Usually it’s much easier to write a solution checker than write a search algorithm. We can’t always expect the SAT solver to find an answer quickly (this is known as as the P vs. NP problem), but there are many times where it will come back with an answer quickly. We like to say that it doesn’t cost much to ask a SAT solver.

Project Euler Problems

Project Euler is a collection of mathematical problems that can be solved using relatively simple computer code (i.e. the solution shouldn’t require optimized code to run on a supercomputer). Some of the questions lend themselves to querying a SAT solver. We’ve collected a few here for you to try your hand at.

ONE IMPORTANT NOTE: We have reworded the questions to make them more Cryptol-friendly. The Project Euler questions often ask you to find all of the numbers which satisfy some property and then combine them in some way (e.g. take their collective sum). For most of the exercises here, it is enough to write a property that will allow the SAT solver to find at least one answer.

Problem 9

A Pythagorean triplet is a set of three natural numbers, a < b < c, for which a2 + b2 = c2.

For example, 32 + 42 = 9 + 16 = 25 = 52.

There exists exactly one Pythagorean triplet for which a + b + c = 1000. Find this triple.

Problem 34

145 is a curious number, as 1! + 4! + 5! = 1 + 24 + 120 = 145.

Find all numbers which are equal to the sum of the factorial of their digits. Note: as 1! = 1 and 2! = 2 are not sums they are not included.

(Aside: these numbers are called factorions)

Hints:

Problem 36

The decimal number, 585 = 1001001001 (binary), is palindromic in both bases. Find at least three numbers which are palindromic in base 10 and base 2. (Please note that the palindromic number, in either base, may not include leading zeros.)

Problem 43

The number, 1406357289, is a 0 to 9 pandigital number because it is made up of each of the digits 0 to 9 in some order, but it also has a rather interesting sub-string divisibility property.

Let d1 be the first digit, d2 be the second digit, and so on. In this way, we note the following:

Find at least two 0 to 9 pandigital numbers with this property.

Problem 52

It can be seen that the number, 125874, and its double, 251748, contain exactly the same digits, but in a different order.

Find the smallest positive integer, x, such that 2x, 3x, 4x, 5x, and 6x, all contain the same digits.

Problem 59

Each character on a computer is assigned a unique code and the preferred standard is ASCII. For example, uppercase A = 65, asterisk (*) = 42, and lowercase k = 107.

A modern encryption method is to take a text file, convert the bytes to ASCII, then XOR each byte with a given value, taken from a secret key. The advantage with the XOR function is that using the same encryption key on the cipher text, restores the plain text; for example, 65 ^ 42 = 107, then 107 ^ 42 = 65.

For unbreakable encryption, the key is the same length as the plaintext message, and the key is made up of random bytes. The user would keep the encrypted message and the encryption key in different locations, and without both “halves”, it is impossible to decrypt the message.

Unfortunately, this method is impractical for most users, so the modified method is to use a password as a key. If the password is shorter than the message, which is likely, the key is repeated cyclically throughout the message. The balance for this method is using a sufficiently long password key for security, but short enough to be memorable.

Your task has been made easy, as the encryption key consists of three lower case characters. Using cipher1.cry, a file containing the encrypted ASCII codes, and the knowledge that the plain text must contain common English words, decrypt the message and find the sum of the ASCII values in the original text.

Note: cipher1.cry contains a different cipher encrypted under a different key from the original. The original Project Euler problem can be found in cipher2.cry.

Problem 79

A common security method used for online banking is to ask the user for three random characters from a passcode. For example, if the passcode was 531278, they may asked for the 2nd, 3rd, and 5th characters; the expected reply would be: 317.

The text file, keylog.cry, contains fifty successful login attempts.

Given that the three characters are always asked for in order, analyze the file so as to determine the shortest possible secret passcode of unknown length.

Throwback

Find a four-digit number (greater than 999 and less than 10000) such that the least significant four digits of the square is that number.

EXTRA CHALLENGE: What about five-digit numbers? Other numbers of digits?

Solicitation

How was your experience with this lab? Suggestions are welcome in the form of a ticket on the course GitHub page: https://github.com/weaversa/cryptol-course/issues

From here, you can go somewhere!

     
  - Cryptographic Properties  
  ! Project Euler (Answers)