-
Notifications
You must be signed in to change notification settings - Fork 6
/
rsa.lua
79 lines (67 loc) · 2.15 KB
/
rsa.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
-- Simple RSA cryptosystem example
-- See https://en.wikipedia.org/wiki/RSA_(cryptosystem)
local bint = require 'bint'(512)
local function gcd(a, b)
while not bint.iszero(b) do
a, b = b, bint.umod(a, b)
end
return a
end
local function lcm(a, b)
return bint.abs(a * b) // gcd(a, b)
end
-- Returns modular multiplicative inverse m s.t. (a * m) % m == 1
local function modinv(a, m)
local r, inv, s = bint(m), bint.one(), bint.zero()
while not r:iszero() do
local quo, rem = bint.idivmod(a, r)
s, r, inv, a = inv - quo * s, rem, s, r
end
assert(a:isone(), 'no inverse')
return inv % m
end
-- 1. Choose two distinct primes
local p = bint('5011956100955230700919988831')
local q = bint('5989833698158308593254005067')
print('p = ' .. tostring(p))
print('q = ' .. tostring(q))
-- 2. Compute n = p * q
local n = p * q
print('n = ' .. tostring(n))
assert(n == bint('30020783547191766561527759475184666413598963507657406677'))
-- 3. Compute the totient
local phi_n = lcm(p - 1, q - 1)
print('phi_n = ' .. tostring(phi_n))
assert(phi_n == bint('5003463924531961093587959910697146102414237368913902130'))
-- 4. Choose any number e that 1 < e < phi_n and is coprime to phi_n
local e = bint('65537')
print('e = ' .. tostring(e))
-- 5. Compute d, the modular multiplicative inverse
local d = modinv(e, phi_n)
print('d = ' .. tostring(d))
assert(d == bint('2768292749187922993934715143535384861582621221551460873'))
-- The public key is (n, e), implement the encrypt function
local function encrypt(msg)
return bint.upowmod(msg, e, n)
end
-- The private key is (n, d), implement the decrypt function
local function decrypt(msg)
return bint.upowmod(msg, d, n)
end
-- Test encrypt and decrypt
print('Message encryption test:')
local msg = 'Hello world!'
print('Message: '..msg)
local x = bint.frombe(msg)
assert(x < n)
print('x = 0x' .. bint.tobase(x, 16))
local c = encrypt(x)
print('c = 0x' .. bint.tobase(c, 16))
assert(c == bint.frombase('81fa941a0bf7a387f0ad060b90a5cd251be4031b4df39a', 16))
local m = decrypt(c)
print('m = 0x' .. bint.tobase(m, 16))
assert(m == x)
local decoded = bint.tobe(m, true)
print('Decoded: '..decoded)
assert(decoded == msg)
print('success!')