DEV Community

Cover image for Encryption Part 3: RSA Encryption
Aaron Wolf
Aaron Wolf

Posted on • Edited on

Encryption Part 3: RSA Encryption

RSA encryption

note: You can clone this repo and use it on your own machine.

RSA Encryption Overview

RSA was invented by NORAD in the early 1970's but was deemed top secret. A few years later in 1977, Ron Rivest, Adi Shamir and Leonard Adleman independantly invented the exact same process for encryption and called it RSA. RSA is the most used software in the world to date and every internet user has used RSA at some point whether they knew it or not.

It shares many mathematical similarities with Diffie-Hellman encryption, but it's conceptually a little different. Instead of thinking about public and private keys, RSA can be thought of as sending an unlocked padlock. It would be like someone who wants to receive a message sending an unlocked padlock to the sender who then locks his message and sends it back for decryption.

The problem RSA solves compared to Diffie-Hellman is that of key storage. When using Diffie-Hellman every key needs to be saved separately. This can be unwieldy for a large organizaion like a bank with many customers- the bank would need a different key for each customer. The solution to this is RSA! The bank can send the same lock, that they alone have keys for, out to many customers who can send encrypted messages back.

Common Uses of RSA encryption

Some common uses of RSA encryption are:

  • The Signal protocol: An open source protocol for sending secure messages. Used by programs like Signal and WhatsApp. Though the signal protocol also uses a special version of Diffie-Hellman on top of RSA.
  • HTTPS: The encrypted connection between a computer and a server from the World Wide Web uses SSL or TLS which is based on RSA.
  • VPN: Like HTTPS, connecting to a VPN uses the SSL or TLS protocols.
  • Encrypted File Systems: Many files and file systems are encrypted using RSA.

RSA Encryption Process

The entire process is based on Euler's theorem: mφ(n)1modnm^{φ(n)} \equiv 1 \mod n , where φ(n)φ(n) is the product of two large prime numbers.

Key Generation

  1. Select Two Prime Numbers: Choose two large prime numbers, p and q.
  2. Compute the Modulus n: Calculate n = p * q. The value of n is used as the modulus for both the public and private keys.
  3. Calculate the Totient (phi φφ ): Compute Euler's totient function, φ(n) = (p-1) * (q-1).
  4. Choose the Public Exponent e: Select an integer e such that 1 < e < φ(n) and e is coprime with φ(n). Often, e = 65537.
  5. Determine the Private Exponent d: Calculate d as the modular multiplicative inverse of e modulo φ(n).

Encryption Process

  1. Public Key: The public key consists of the modulus n and the public exponent e.
  2. Encrypting a Message: Convert the message into an integer m (e.g., via ASCII). Then, compute the ciphertext c using cme(modn)c ≡ m^e (mod n) .

Decryption Process

  1. Private Key: The private key is made of the modulus n and the private exponent d.
  2. Decrypting a Message: Decrypt the message by computing mcd(modn)m ≡ c^{d} (mod n) using the private key.

Security

  • RSA's security relies on the difficulty of factoring the product of two large prime numbers.
  • The security improves with larger key sizes but at the cost of slower encryption and decryption.

Let's get coding: Helper Functions

These functions are meant for demonstration, not performance. This is just to show how the process of RSA, but this would never be used in production. In modern applications this process is completely automated.

# Finds factors and determines if prime.

def factors(number):
    number = int(number)
    factors = set()

    # Check for divisibility by 2 first to handle even numbers quickly
    if number % 2 == 0:
        factors.update({2, number // 2})

    # Check for odd divisors only, up to the square root of the number
    for i in range(3, int(number**0.5) + 1, 2):
        if number % i == 0:
            factors.update({i, number // i})

    # Add 1 and the number itself to the factors
    factors.update({1, number})

    # A prime number has exactly two factors: 1 and itself
    return {'factors': factors, 'is_prime': len(factors) == 2}



# Determines greatest common divisor of 2 numbers

def gcd(a, b):
    while b:
        a, b = b, a % b
    return a



# encode a string into an array of each character's numerical value

def encode_string(string):
    char_array = []
    for char in string:
        char_array.append(ord(char))
    return char_array



# decode an array of numerical values to their corresponding character

def decode_string(arr):
    decoded_arr = []
    for i in arr:
        decoded_arr.append(chr(i))
    return "".join(decoded_arr)



# Calculate if numbers are co-prime (they have no common factors except 1)

def are_coprime(number1, number2):
    number1_factors = factors(number1)['factors']
    number2_factors = factors(number2)['factors']

    common_factors = number1_factors.intersection(number2_factors)

    # if the intersection of common factors is only 1 return True
    return common_factors == {1}

Enter fullscreen mode Exit fullscreen mode

Encryption Functions

Selecting the public key

ee should be an integer that is not only greater than 1 but also less than φ(n)φ(n) , where n=pqn = pq (the product of two distinct prime numbers pp and qq ).

ee and φ(n)φ(n) should be coprime, meaning they should have no common factors other than 1. This ensures that ee has a multiplicative inverse modφ(n)mod φ(n) .

The private exponent d is the modular multiplicative inverse of e modulo ϕ(n).
This means d is the number that satisfies the equation
d×e1 modϕ(n)d × e ≡ 1 \ modϕ(n) .

# calulate phi (φ) - the totient

def calculate_phi(p, q):
    if factors(p)['is_prime'] and factors(q)['is_prime']:
        return (p - 1) * (q - 1)

    raise ValueError('Not prime numbers')



# Calculate e - the public exponent

def find_suitable_e(totient):
    # Start with 65537 - A common value for `e` in RSA
    e = 65537

    # Check if e is less than totient and coprime with the totient
    # If not, decrement by 2 (to keep e as an odd number) and check again
    while e >= totient or not are_coprime(e, totient):
        e -= 2
        if e <= 2:
            raise ValueError("Could not find an appropriate 'e' value.")

    return e


# Encrypts each letter of the message with the exponent and modulus
def encrypt(message, e, n):
    if isinstance(message, list):
        encrypted_list = []
        for i in message:
            encrypted_list.append(pow(i, e, n))
        return encrypted_list
    else:
        return []



# Decrypts each letter of the message given d - the private key - and n - the modulus

def decrypt(message, d, n):
    decrypted = []
    for i in message:
        decrypted.append(pow(i, d, n))

    return decrypted



# Calculate d: the private key
def calculate_d(phi_n, e):
    return pow(e, -1, phi_n)

Enter fullscreen mode Exit fullscreen mode

Watch RSA in action

# Alice picks 2 prime numbers of similar size to compute `n`

while True:
    p1 = int(input('Pick your first prime: '))
    p2 = int(input('Pick your second prime: '))

    p1_is_prime = factors(p1)['is_prime']
    p2_is_prime = factors(p2)['is_prime']

    if p1_is_prime and p2_is_prime:
        print(f"\nAlice's primes are {p1} and {p2}")
        break
    elif not p1_is_prime and not p2_is_prime:
        print(f'{p1} and {p2} are both not prime')
    elif not p1_is_prime:
        print(f'{p1} is not prime.')
    elif not p2_is_prime:
        print(f'{p2} is not prime.')



# Calculate n as the product of p1 and p2

n = p1 * p2

print(f"Alice's n value is {p1} x {p2} = {n}")



# Calculate the totient - φ(n)

totient = calculate_phi(p1, p2)

print(f'φ(n) = ({p1} - 1) x ({p2} - 1) = {totient}')



# Calculate exponent e, such that it is odd and coprime with φ(n)

e = exponent(totient)

print(f'The exponent value is {e}')



# Calculate the private key `d`. Alice keeps this as her secret key.

d = calculate_d(totient, e)
print(f"Alice's private decryption key is {d}")
Enter fullscreen mode Exit fullscreen mode

Alice sends n and e. Bob uses these numbers to encrypt

Alice sends to Bob her values for n and e. This is like sending an open lock that Bob will use to lock up his message. He does this by calculating memod(n)m^{e} mod(n) . Where m is the numerical value of his message.

# Bob's message to Alice
message = input('Type a message to send: ')

encoded_message = encode_string(message)
print(f'Encoded but not encrypted message is:\n{encoded_message}\n')

encrypted_message = encrypt(encoded_message, e, n)
print(f'Encoded and encrypted message is:\n {encrypted_message}')
Enter fullscreen mode Exit fullscreen mode

Send the encrypted message

encrypted_message can now be sent from Bob to Alice without an eavesdropper (Eve) making any sense of the message. In practice this only works when large values of p and q are used, otherwise computers can still quite easily brute force the encryption.

Alice uses her private key d to decrypt the message

Since d is the multiplicative inverse of e in respect to φ(n), decrypting the message is easy. Each letter is decrypted with the function idmod(n)i^{d} mod (n) , where i is the encrypted numerical value of the letter, d is our private key, and n is the product of our two primes.

# Alice decrypts the message using `d`

decrypted_message = decrypt(encrypted_message, d, n)

print(f"The decrypted encoded message is the same as above\n{encoded_message} = {decrypted_message}\n")


decoded_message = decode_string(decrypted_message)
print(f"The plain text value is: {decoded_message}")
Enter fullscreen mode Exit fullscreen mode

Next topic to tackle: Hashing

Top comments (0)