-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathrsa.py
71 lines (58 loc) · 1.48 KB
/
rsa.py
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
from __future__ import unicode_literals
from math import sqrt
from Crypto.Random import random.randint as rand
import sympy
import time
import binascii
def gcd(a, b):
if b == 0:
return a
else:
return gcd(b, a % b)
def egcd(a, b):
x,y, u,v = 0,1, 1,0
while a != 0:
q, r = b//a, b%a
m, n = x-u*q, y-v*q
b,a, x,y, u,v = a,r, u,v, m,n
return b, x, y
def mod_inverse(a, m):
g, x, y = egcd(a, m)
if g != 1:
return None
else:
return x % m
def isprime(n):
if n < 2:
return False
elif n == 2:
return True
else:
for i in range(1, int(sqrt(n)) + 1):
if n % i == 0:
return False
return True
def generate_prime(bitlength):
a = '1'+'0'*(bitlength-1)
b = '1'*bitlength
p = sympy.randprime(int(a, 2), int(b, 2))
return p
def generate_keypair(keysize):
p = generate_prime(keysize)
q = generate_prime(keysize)
n = p * q
phi = (p-1)*(q-1)//gcd(p-1, q-1)
e = sympy.randprime(1,phi)
d = mod_inverse(e,phi)
if e != d:
return ((e, n), (d, n))
def encrypt(plain_text, package):
e, n = package
if plain_text > n:
print('Message is too large for key to handle')
msg_ciphertext = pow(plain_text, e, n)
return msg_ciphertext
def decrypt(msg_ciphertext, package):
d, n = package
msg_plaintext = pow(msg_ciphertext, d, n)
return binascii.unhexlify(hex(msg_plaintext)[2:]).decode()