Ever wondered how two factor authentication apps works? I certainly did. One could just guess how SMS based tokens work, that's simple (although they shouldn't be used as per guidelines from NIST). But what about TOTP or Time based One Time Password, the ones in which you scan a QR code and the OTP generator app (like freeOTP and andOTP) gives you a new six digit token every 30 seconds or so?
I was, for quite some time, under the (very misinformed) impression that web servers which implement this method of 2FA expose an API and, by means of the QR code, give you the endpoint with some token and then the OTP generator app polls the server and gets a new ephemeral password which the user enters in the application. Straight forward, but plain wrong.
The belief was challenged when I noticed that the OTP generator app works irrespective of network connection, even if both devices are offline (that is, when working on localhost server). HOW? I dug further and I learned some very interesting things, some of which I wanted to write here.
The Name, Dude
Time based One Time Password, the name itself gives enough clue to guess that it uses time, and as such is independent of inter-communication as long as the two systems are in time-sync. But an adversary can be assumed to be in time-sync as well, right? Yes, that's where we bring in the secret (a randomly generated token) which is embedded in the QR code that you scan with your smartphone app. So we have the time which is in sync and we've established a way of transferring the secret from the server to the client. Turns out, that's all the data we need to keep generating secrets independently on the client and server side, completely offline once the initial secret sharing happens.
Basic OTP Algorithm
function generate_otp(secret, counter) { h = hmac(key=secret, message=counter, algorithm=sha1) offset = get_last_four_bits(h) pre_opt = get_32_bits_starting_from_offset(h) otp = get_desired_number_of_chars(pre_otp, N) } function get_totp(secret) { counter = epoch / 30 return generate_otp(secret, counter) } global counter; // get from database function get_hotp(secret) { return generate_otp(secret, counter) }
The basic OTP algorithm (both time and hash based) accept a secret and a counter value. Combining current time and the secret, a new 6 (or N) digit token is generated every 30 (or Ti) seconds. They differ in what the counter value supplied to the algorithm.
- TOTP: take the number of times the interval Ti can be fitted in the total number of seconds since epoch. Which is just a weird way of saying that the interval is the quotient when you divide the seconds_since_epoch number by the interval duration (Ti).
- HOTP: take the current counter stored persistently and use that. After use, increment the counter in the database.
- Compute HMAC value of message Ti and key secret, get the hex digest
- The last 4 bits (last digit in hex) is stored as offset
- Starting from offsetth bit, take 32 bits (8 hex digits) and discard the first bit (xor with 7ffffffff. This works because f = 1111 and 7 = 0111, so &'ing with 7 (0111) is equal to switching off the first bit)
- Convert the 32 bits hex to int, then take the least significant 6 bits (or N depending on requirement), and that is your OTP
- ?? Profit!
Security Consideration
The entire security of the OTP lies in the secrecy of the initial secret. If that's compromised, an attacker can easily generate as many OTPs as she wishes. Also, given the keyspace of the OTP, and also that many servers are designed to accept counter+1 and counter-1 OTPs, securing the system against bruteforce is a most.
One important aspect of these 2FA mechanisms is that losing your 2FA device means losing your account. This is especially the case with services like ProtonMail where the password is used to decrypt client data.
Naive Python Implementation
Given how simple this algorithm was, I tried to implement it. The core algorithm was literally less than five lines of python code. Here's a naive implementation of the same, and while it seems to work, I'd not use it anywhere.
Thank you for reading! PS Python noob, please don't judge! :P