HKCERT CTF 2022 – Base64 Encryption
Frequency analysis with a touch of heuristics.
The challenge looks deceptively simple. Chinese has over 50,000 characters. Base64 just has 64. So it should be easy right?
Haha nope. It's not as trivial as I thought.
Description
200 points. 3/5 ⭐️. 6/311 solves.
People said that base64 is an encoding, not an encryption. Did they have a misconception about that?
If you believe that base64 is just an encoding, then convince me that you are able to "decode" the article (which is in English).
Regardless, the challenge author is kind enough to provide a clue in the description: the plaintext is an article in English. We’ll see how this helps us later on.
Analysis
We’re provided with an encryption script chall.py
(written in Python), along with the generated ciphertext message.enc.txt
.
The script first encodes the plaintext in English:
...then "encrypts" it by mapping each Base64 character to another one:
The script uses
random.shuffle
without seeding. This means we can’t easily reproduce the character mapping (charmap
). We’ll need to try harder.Although the script reads the plaintext in binary format (
open('message.txt', 'rb')
), I’m banking on the clue that the plaintext is an English article—so hopefully there aren’t any weird characters.
So how do we go about cracking this? Brute-force will be undoubtedly inefficient as we have $64! \approx 1.27 \times 10^{89}$ mapping combinations to try. It would take years before we have any progress! Also we’d need to look at results to determine if the English looks right (or automate it by checking a word list)—this would take even more time! Regardless, we need to find some other way.
First Steps: Elimination by ASCII Range
Here’s one idea: since the plaintext is an English article, this means that most (if not all) characters are in the printable ASCII range (32-127). This means that the most significant bit (MSB) of each byte cannot be 1. We can use this to create a blacklist of mappings. For example, originally we have 64 possible mappings for the letter A
. After blacklisting, we may be left with, say, 16 possible mappings. This drastically reduces the search space.1
Since Base64 simply maps 8-bits to 6-bits, so 3 characters of ASCII would be translated to 4 characters of Base64.
We can check the mappings we’ve eliminated:
And that means the letter J
can only map to:
Neat! This will help us later on (when we resort to blatant educated guessing).
We can do a similar thing on the low end. Again, since the smallest printable ASCII character is 32, this means either the second or third MSBs would be set.2
Frequency Analysis with Known Text
Another idea comes to mind. Remember the plaintext is in English? Well, with English text, some letters appear more frequently than others. The same applies to words and sequences.
In the same vein, some letters and sequences in the Base64 encoding will also appear more frequently than others.
With this in mind, we can compare the ciphertext to the Base64 encoding of some random article (also in English, of course). For this, I copied some articles from CNN Lite (text-only, so it's easier to copy), encoded it, then analysed letter frequencies using dcode.fr. You could use this excellent article as well:
From this, we can deduce that 'w' was mapped from 'G' in the original encoding (due to the gap in frequency).
One useful option is the bigrams/n-grams option. We can tell dcode to analyse frequencies of groups of characters with a sliding window. This is useful to identify words and sequences.
Observe how "YoJP0H" occurs (relatively) frequently. This corresponds to "IHRoZS", which happens to be the Base64 encoding for " the".
More Heuristics
Frequency analysis is useful to group letters into buckets. But using frequency analysis alone is painful. Some guesswork is needed. Here's the complete process I went through:
Frequency Analysis: use dcode.fr to associate frequent characters.
We can make use of our earlier constraints to eliminate wrong guesses.3
Guesswork: guess English from the
nonsenseexisting characters.Google: after decoding a sizeable portion, let's pray and hope the plaintext is open-source. Then use the plaintext to derive the rest of the mapping.
It turns out the plaintext is—quite aptly—the Wikipedia summary of frequency analysis.
Finding the rest of the mappings was quite easy. After a bit more tuning, we get the flag.
Final Remarks
Usually I play reverse and don’t touch cryptography, but all I can say is: this was basically playing an English reverse challenge under the hood. Forget C, C++, Java, .Net, and Rust. Reversing English is the best. 😛
There are probably better ways to automatically perform frequency analysis and search for mappings. I went for a hybrid method of Python scripting + manually checking two dcode tabs. Perhaps a second monitor would’ve helped, but I have nowhere to place it. 😐
Flag
Footnotes
What about é and à, as in déjà vu? Well, although those are technically in the extended ASCII character set, they should be rare enough. (Also I think Python encodes them differently from regular ASCII.) ↩︎
But what about newline (
\n
, ASCII 10) and carriage return (\r
, ASCII 13)? These are also possible to have in plaintext messages. We shouldn’t entirely discount these, but as they’re relatively rare, we won’t consider them for now. ↩︎Later on, we removed the second/third-MSB constraint since it got in the way of decoding
\n
. ↩︎
Comments are back! Privacy-focused, without ads, bloatware 🤮, and trackers. Be one of the first to contribute to the discussion — I'd love to hear your thoughts.