-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy path0-santini.py
141 lines (113 loc) · 4.41 KB
/
0-santini.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
alph = 'abcdefghijklmnopqrstuvwxyz '
#custom error
class CaesarCipherError(Exception):
'''Error executing Caesar cipher script'''
def read_file(name):
try:
# open in read mode a text file
with open(name, 'r') as in_file:
out_str = in_file.read()
except IOError as err:
raise CaesarCipherError('Error: Cannot read ' + name + ' file: ' + str(err))
# delete possible trailing newlines
return out_str.strip('\n')
# if n is positive, the array is shited (rotated) to the right
def shift_alph(arr, n):
# arr[n:] takes the characters from the index 'n' onwards
# arr[:n] takes the characters before the index 'n'
return arr[n:] + arr[:n]
def enc_dec(in_str, key_str, sign = 1):
# check sign
if (abs(sign) != 1):
err_msg = 'Error: sign parameter should be either 1 or -1, got: '
err_msg += sign
raise CaesarCipherError(err_msg)
# note that this error does not depend on user's input
# so it should not be raised if the script is correct
### compute resulting string
out_str = ''
shifted_alph = shift_alph(alph, key_str * sign)
# process character by character
for charIndex in range(len(in_str)):
# search for the 'in_str' character by character position in 'alph'
i = alph.find(in_str[charIndex])
if i < 0:
err_msg = '\nError: message contains invalid character: "'
err_msg += in_str[charIndex] + '"'
raise CaesarCipherError(err_msg)
# put that position 'i' in 'shifted_alph'.
# every loop cycle append the corresponding character to 'out_str'
out_str += shifted_alph[i]
return out_str
# function that performs Caesar cipher encryption
def encrypt(plain_text, shift_num):
# encrypt
cipher_text = enc_dec(plain_text, shift_num)
# write output to file
try:
with open('./ciphertext.txt', 'w+') as out_file:
out_file.write(cipher_text)
except IOError as err:
raise CaesarCipherError('\nError: cannot write ciphertext in ciphertext.txt: ' + str(err))
# note that the following message is not printed if the function
# raised some exception previously
print(f'\nEncrypted message correctly saved:\n"{cipher_text}"')
# function that performs Caesar cipher decryption
def decrypt(shift_num):
# read ciphertext from file
cipher_text = read_file('./ciphertext.txt')
print('\nThe ciphertext is:\n' + cipher_text)
# decrypt
plain_text = enc_dec(cipher_text, shift_num, -1)
# write result on the console
print('\nThe decrypted message is:\n' + plain_text)
def menu():
separator = '---------------'
# notice the "f" before the string
# which is used to allow for variable interpolation
print(f'\n{separator}\nEnter:\n 1 -> encrypt\n 2 -> decrypt\n 0 -> quit\n{separator}')
return input('> ')
#main
print('\nEncrypt and decrypt messages with the Caesar cipher method!')
while True:
choice = menu()
shift_num = 27
try:
#switch-case syntax in substitution to 'if elif elif elif ...'
match choice:
#* encrypt
case "1":
# ask user the key, check if input is valid integer
while abs(shift_num) > 26:
try:
shift_num = int(input('\nChoose a number between -26 and 26 to shift the alphabet\n> '))
except ValueError:
print('\nError: value entered is not a number, try again.')
# better try again... Return to the start of the loop
continue
if abs(shift_num) > 26:
print('\nError: expected value must be between -26 and 26')
# read message from console
plain_text = input('\nType message to encrypt, no special characters allowed\n> ').lower()
encrypt(plain_text, shift_num)
#* decrypt
case "2":
# ask user the key, check if input is valid integer
while abs(shift_num) > 26:
try:
shift_num = int(input('\nChoose a number between -26 and 26 to shift the alphabet\n> '))
except ValueError:
print('\nError: value entered is invalid, try again.')
# better try again... Return to the start of the loop
continue
if abs(shift_num) > 26:
print('\nError: value entered must be between -26 and 26')
decrypt(shift_num)
#* exit script
case "0":
exit()
#* default case
case _:
print('\nTry again, please choose a number from these options: 1, 2 or 0')
except CaesarCipherError as err:
print(err)