-
Notifications
You must be signed in to change notification settings - Fork 419
/
Copy pathhill.py
76 lines (67 loc) · 2.18 KB
/
hill.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
71
72
73
74
75
# Encrypt and decrypt a clear/ciphertext with the Hill Cipher
# GPL C polym
# Uses: http://stackoverflow.com/questions/4287721/easiest-way-to-perform-modular-matrix-inversion-with-python#answer-4293123
import numpy as np
global debug
debug=0
def modMatInv(A,p): # Finds the inverse of matrix A mod p
n=len(A)
A=np.matrix(A)
adj=np.zeros(shape=(n,n))
for i in range(0,n):
for j in range(0,n):
adj[i][j]=((-1)**(i+j)*int(round(np.linalg.det(minor(A,j,i)))))%p
return (modInv(int(round(np.linalg.det(A))),p)*adj)%p
def modInv(a,p): # Finds the inverse of a mod p, if it exists
for i in range(1,p):
if (i*a)%p==1: return i
raise ValueError(str(a)+" has no inverse mod "+str(p))
def minor(A,i,j): # Return matrix A with the ith row and jth column deleted
A=np.array(A)
minor=np.zeros(shape=(len(A)-1,len(A)-1))
p=0
for s in range(0,len(minor)):
if p==i: p=p+1
q=0
for t in range(0,len(minor)):
if q==j: q=q+1
minor[s][t]=A[p][q]
q=q+1
p=p+1
return minor
def encrypt(msg, key, sz):
triple = [list(msg[i*sz:(i*sz)+sz]) for i in range(0, len (msg)/sz)]
if debug>0: print triple
mul = [i[:] for i in triple]
for x in range(len(triple)):
for i in range(len(triple[x])):
triple[x][i]=ord(triple[x][i])-65
if debug>0: print triple
for x in range(len(triple)):
mul[x] = np.dot(key,triple[x]) % 26
if debug>0: print mul
enc=""
for x in range(len(mul)):
for s in range(0,sz): enc+=chr(mul[x][s]+65)
return enc
def decrypt(msg, key, sz):
try: deckey = modMatInv(key,26)
except ValueError: return
triple = [list(msg[i*sz:(i*sz)+sz]) for i in range(0, len (msg)/sz)]
mul = [i[:] for i in triple]
for x in range(len(triple)):
for i in range(len(triple[x])):
triple[x][i]=ord(triple[x][i])-65
if debug>0: print triple
for x in range(len(triple)):
mul[x] = np.dot(deckey,triple[x]) % 26
if debug>0: print mul
dec=""
for x in range(len(mul)):
for s in range(0,sz): dec+=chr(int(mul[x][s])+65)
return dec
encr="DKBRHSBTJVPHFKYJASSTZBOCOXXMRCKTNRJGGRSCBQJBRZVKPGAJOVANNEIHIKITIHKHUOFYISZBOCSIGEXQZYWEFANOAVEWQBJNYVUUYEAOIRRSWY"
correctkey = [[3,2,7], [4,5,6], [1,9,8]]
correctkey= np.transpose(correctkey)
sz=len(correctkey)
print decrypt(encr,correctkey,sz)