-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathencrypt.py
379 lines (317 loc) · 10.9 KB
/
encrypt.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
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
#####################################################################
# FrenchMasterSword, Cryptix, 2018
#####################################################################
# Encrypting functions
# Each should take at least three first arguments :
# the window instance : self (mostly for error messages)
# a boolean : True => encrypt, False => decrypt
# the text to process
#
# depending of the cypher, one or two keys may also be necessary
# this is used to control how many line fields should be editable.
# they should then be called key and key2
#
# They should return a string, or an exception to be processed
import json
import re
from functools import wraps
from itertools import cycle
from string import ascii_letters
# from math import ceil
from PySide2.QtWidgets import QMessageBox
with open('settings.json', 'r') as file:
settingsDict = json.load(file)
UNUSED = settingsDict["removed letter"]
REPLACED = settingsDict["replace letter"]
# catched is a dict {"ExceptionName": "Warning text"}
def catch(catched={}):
def catch_decorator(func):
@wraps(func)
def func_wrapper(*args):
try:
return func(*args)
except Exception as e:
name = e.__class__.__name__
if name in catched:
return QMessageBox.warning(args[0], "Warning", catched[name].replace('ERROR', e.args[0]))
else:
return QMessageBox.critical(args[0], "Error", repr(e))
return func_wrapper
return catch_decorator
def _isascii(string: str):
return all((char in ascii_letters for char in string))
def _create_alphabet(key: str, remove=True) -> list:
"""
generate a transposition alphabet
"""
key = re.sub(r'[^a-zA-Z]', '', key).upper # remove non alphabetical
alphabet = ''
for char in key:
if _isascii(char) and not char in alphabet:
alphabet += char
for i in range(65, 91):
if not chr(i) in alphabet:
alphabet += chr(i)
if not remove:
return alphabet
return alphabet.replace(UNUSED, '')
@catch()
def simple(self, encrypt: bool, text: str, key: str):
"""
Simply replace letters in the alphabet with those in the key.
You don't have to enter a complete one, as it will be generated from it.
"""
key = _create_alphabet(key, False)
pass
@catch({'ValueError': 'You should enter a valid integer as the key.'})
def caesar(self, encrypt: bool, text: str, key: str):
"""
Shifts the text's letter in the alphabet of the number given as key.
"""
key = int(key)
if not encrypt:
key = - key
result = ''
for char in text:
if _isascii(char):
letter = char.upper()
letter = chr((ord(letter) - 65 + key) % 26 + 65)
if char.islower():
letter = letter.lower()
else:
letter = char
result += letter
return result
@catch({'KeyError': '<b>ERROR</b> is not recognized in standard morse code'})
def morse(self, encrypt: bool, text: str):
"""
Transpose in standard morse code.
"""
morseCode = {
"A": ".-",
"B": "-...",
"C": "-.-.",
"D": "-..",
"E": ".",
"F": "..-.",
"G": "--.",
"H": "....",
"I": "..",
"J": ".---",
"K": "-.-",
"L": ".-..",
"M": "--",
"N": "-.",
"O": "---",
"P": ".--.",
"Q": "--.-",
"R": ".-.",
"S": "...",
"T": "-",
"U": "..-",
"V": "...-",
"W": ".--",
"X": "-..-",
"Y": "-.--",
"Z": "--..",
"0": "-----",
"1": ".----",
"2": "..---",
"3": "...--",
"4": "....-",
"5": ".....",
"6": "-....",
"7": "--...",
"8": "---..",
"9": "----.",
".": ".-.-.-",
",": "--..--",
"?": "..--..",
"'": ".----.",
"!": "-.-.--", # May be ---. in North America: Ignored
"/": "-..-.",
"(": "-.--.",
")": "-.--.",
"&": ".-...",
":": "---...",
";": "-.-.-.",
"=": "-...-",
"+": ".-.-.",
"-": "-....-",
"_": "..--.-",
'"': ".-..-.",
"$": "...-..-",
"@": ".--.-.",
" ": "/" # For program design
}
inverseMorseCode = dict((v,k) for (k,v) in morseCode.items())
if encrypt:
result = ''
for char in text:
if char == '\n':
result = result[:-1] + char
else:
result += morseCode[char.upper()]
result += ' '
result = result[:-1]
else:
result = []
textLines = text.split('\n')
for line in textLines:
line = line.strip(' /')
resultLine = ''
textList = line.split('/')
for word in textList:
word = word.strip(' ')
for char in word.split(' '):
resultLine += inverseMorseCode[char]
resultLine += ' '
resultLine = resultLine[:-1] # remove superficial ending space
result.append(resultLine)
result = '\n'.join(result)
return result
@catch({
'ValueError': 'Use only standard alphabet letters ; <b>ERROR</b> is not valid.',
'IndexError': 'A digit is out of the grid.'
})
def polybius(self, encrypt: bool, text: str, key: str):
"""
Replace letters by their abscissa and ordinate in a grid.
If a key is given, it starts filling the grid, and finishes with the rest of the alphabet.
The second grid is an example with 'CRYPTIX' used as key.
As there are only 25 squares, one letter is removed and replaced.
"""
key = _create_alphabet(key, False)
result = ''
if encrypt:
i = 0 # for error handling
for char in text.upper():
if char == UNUSED:
char = REPLACED
if _isascii(char):
pos = key.index(char)
char = str((pos+1)//5 + 1 - ((pos+1)%5==0))+ str((pos+1)%5)
result += char
i += 1
result = re.sub('0', '5', result)
else:
text = re.sub('0', '5', text)
# List of 5 lists of 5
polybius = [[key[y] for y in range(5*i, 5*(i+1))] for i in range(5)]
i = 0
while i < len(text):
char = text[i]
if char.isdigit():
char = polybius[int(text[i])-1][int(text[i+1])-1]
i += 1
result += char
i += 1
return result
@catch()
def adfgvx(self, encrypt: bool, text: str, key: str):
"""
Same as Polybius, but grid is indexed with these 6 letters and also encrypt digits.
"""
key = _create_alphabet(key, False)
pass
@catch({'IndexError': 'You need to enter a valid key'})
def vigenere(self, encrypt: bool, text: str, key: str):
"""
Uses the letters in the key to shift (as in Caesar cipher) the letters in the text (A:0, B:1, Z:25).
If it's shorter than the text, the key is repeated.
"""
if not encrypt:
encrypt = -1
result = ''
key = cycle(key.upper())
for letter in text.upper():
if _isascii(letter):
letter = chr((ord(letter) + (encrypt * ord(next(key))) - 130) % 26 + 65)
result += letter
return result
@catch()
def wolseley(self, encrypt: bool, text: str, key: str):
"""
Replaces letters with a reversed alphabet, missing a letter.
"""
key = _create_alphabet(key, False)
keyReverse = key[::-1]
if not encrypt:
key, keyReverse = keyReverse, key
result = ''
for letter in text.upper():
if _isascii(letter):
letter = keyReverse[key.find(letter)]
result += letter
return result
@catch({'ValueError': 'You need to enter a valid positive integer as the key'})
def gronsfeld(self, encrypt: bool, text: str, key: str):
"""
Uses the digits in the key to shift (as in Caesar cipher) the letters in the text.
If it's shorter than the text, the key is repeated.
"""
if not encrypt:
encrypt = -1
result = ''
key = cycle((int(i) for i in key))
for letter in text.upper():
if _isascii(letter):
letter = chr((ord(letter) + (encrypt * next(key) - 65)) % 26 + 65)
result += letter
return result
@catch({'ValueError': 'You need to enter valid integers as keys'})
def affine(self, encrypt: bool, text: str, key: str, key2: str):
"""
Given a and b constants, x letter of the plain text and y letter of the encrypted one, : y = ax + b (modulo 26).
Note that if a = 0, it's equivalent to Caesar cipher, and if b = 0, 'A' is always ciphered 'A'.
"""
key, key2 = int(key), int(key2)
result = ''
for letter in text.upper():
if _isascii(letter):
letter = ord(letter) - 65
if encrypt:
letter = letter * key + key2
else:
letter = (letter - key2) // key
letter = chr(letter % 26 + 65)
result += letter
return result
@catch({'IndexError': 'You need to enter a valid key'})
def beaufort(self, encrypt: bool, text: str, key: str):
"""
A bit like the opposite of Vigenere cipher.
Instead of adding the key's letters to those of the plain text ;
we substract the plain text's letters to those of the key.
"""
result = ''
key = cycle((ord(k) for k in key.upper()))
for letter in text.upper():
if _isascii(letter):
letter = chr((next(key) - ord(letter) + 130) % 26 + 65)
result += letter
return result
@catch()
def collon(self, encrypt: bool, text: str, key: str, key2: str):
"""
With the help of the grid on the left (which you can generate with the key),
each letter is converted to a bigram (a group of two letters)
representing the abscissa and ordinate (or the ordinate and abscissa) in the grid.
For instance, R will become CS (or SC). The script will randomly alternate these two options to renforce the cipher.
Then, each bigram is entered under the letter in the two lines,
and following a given number, the first and second line are added to the ciphered text.
Here the number being 7, it will be ICQCKKK then QZSQZSS then KKCICEE etc. until the end.
Notice that the ciphered text will be twice longer than the plain one: ICQCKKKQZSQZSSKKCICEEZVQQVVQCS.
"""
key2 = int(key2)
key = _create_alphabet(key, False)
# 5x5 grid
# grid = [[key[y] for y in range(5*i, 5*(i+1))] for i in range(5)]
for letter in text.upper():
if _isascii(letter):
pos = key.index(letter)
location = [(pos)//5 - ((pos)%5==0), (pos)%5]
print(location)