2019-12-21 06:52:43 +08:00
|
|
|
import random
|
|
|
|
import sys
|
2016-08-07 23:48:46 +08:00
|
|
|
|
2023-10-09 20:19:12 +08:00
|
|
|
from maths.greatest_common_divisor import gcd_by_iterative
|
|
|
|
|
2020-09-29 05:41:04 +08:00
|
|
|
from . import cryptomath_module as cryptomath
|
2019-12-21 06:52:43 +08:00
|
|
|
|
2019-12-26 19:50:12 +08:00
|
|
|
SYMBOLS = (
|
|
|
|
r""" !"#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`"""
|
|
|
|
r"""abcdefghijklmnopqrstuvwxyz{|}~"""
|
|
|
|
)
|
2016-08-07 23:48:46 +08:00
|
|
|
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2022-10-13 06:54:20 +08:00
|
|
|
def check_keys(key_a: int, key_b: int, mode: str) -> None:
|
2019-12-21 06:52:43 +08:00
|
|
|
if mode == "encrypt":
|
2022-10-13 06:54:20 +08:00
|
|
|
if key_a == 1:
|
2019-12-21 06:52:43 +08:00
|
|
|
sys.exit(
|
|
|
|
"The affine cipher becomes weak when key "
|
|
|
|
"A is set to 1. Choose different key"
|
|
|
|
)
|
2022-10-13 06:54:20 +08:00
|
|
|
if key_b == 0:
|
2019-12-21 06:52:43 +08:00
|
|
|
sys.exit(
|
|
|
|
"The affine cipher becomes weak when key "
|
|
|
|
"B is set to 0. Choose different key"
|
|
|
|
)
|
2022-10-13 06:54:20 +08:00
|
|
|
if key_a < 0 or key_b < 0 or key_b > len(SYMBOLS) - 1:
|
2019-10-05 13:14:13 +08:00
|
|
|
sys.exit(
|
2019-12-21 06:52:43 +08:00
|
|
|
"Key A must be greater than 0 and key B must "
|
|
|
|
f"be between 0 and {len(SYMBOLS) - 1}."
|
2019-10-05 13:14:13 +08:00
|
|
|
)
|
2023-10-09 20:19:12 +08:00
|
|
|
if gcd_by_iterative(key_a, len(SYMBOLS)) != 1:
|
2019-10-05 13:14:13 +08:00
|
|
|
sys.exit(
|
2022-10-13 06:54:20 +08:00
|
|
|
f"Key A {key_a} and the symbol set size {len(SYMBOLS)} "
|
2019-12-21 06:52:43 +08:00
|
|
|
"are not relatively prime. Choose a different key."
|
2019-10-05 13:14:13 +08:00
|
|
|
)
|
|
|
|
|
2016-08-07 23:48:46 +08:00
|
|
|
|
2019-12-21 06:52:43 +08:00
|
|
|
def encrypt_message(key: int, message: str) -> str:
|
2019-10-05 13:14:13 +08:00
|
|
|
"""
|
2020-06-16 16:09:19 +08:00
|
|
|
>>> encrypt_message(4545, 'The affine cipher is a type of monoalphabetic '
|
|
|
|
... 'substitution cipher.')
|
2016-08-07 23:48:46 +08:00
|
|
|
'VL}p MM{I}p~{HL}Gp{vp pFsH}pxMpyxIx JHL O}F{~pvuOvF{FuF{xIp~{HL}Gi'
|
2019-10-05 13:14:13 +08:00
|
|
|
"""
|
2022-10-13 06:54:20 +08:00
|
|
|
key_a, key_b = divmod(key, len(SYMBOLS))
|
|
|
|
check_keys(key_a, key_b, "encrypt")
|
|
|
|
cipher_text = ""
|
2016-08-07 23:48:46 +08:00
|
|
|
for symbol in message:
|
|
|
|
if symbol in SYMBOLS:
|
2022-10-13 06:54:20 +08:00
|
|
|
sym_index = SYMBOLS.find(symbol)
|
|
|
|
cipher_text += SYMBOLS[(sym_index * key_a + key_b) % len(SYMBOLS)]
|
2016-08-07 23:48:46 +08:00
|
|
|
else:
|
2022-10-13 06:54:20 +08:00
|
|
|
cipher_text += symbol
|
|
|
|
return cipher_text
|
2016-08-07 23:48:46 +08:00
|
|
|
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2019-12-21 06:52:43 +08:00
|
|
|
def decrypt_message(key: int, message: str) -> str:
|
2019-10-05 13:14:13 +08:00
|
|
|
"""
|
2020-06-16 16:09:19 +08:00
|
|
|
>>> decrypt_message(4545, 'VL}p MM{I}p~{HL}Gp{vp pFsH}pxMpyxIx JHL O}F{~pvuOvF{FuF'
|
|
|
|
... '{xIp~{HL}Gi')
|
2016-08-07 23:48:46 +08:00
|
|
|
'The affine cipher is a type of monoalphabetic substitution cipher.'
|
2019-10-05 13:14:13 +08:00
|
|
|
"""
|
2022-10-13 06:54:20 +08:00
|
|
|
key_a, key_b = divmod(key, len(SYMBOLS))
|
|
|
|
check_keys(key_a, key_b, "decrypt")
|
|
|
|
plain_text = ""
|
|
|
|
mod_inverse_of_key_a = cryptomath.find_mod_inverse(key_a, len(SYMBOLS))
|
2016-08-07 23:48:46 +08:00
|
|
|
for symbol in message:
|
|
|
|
if symbol in SYMBOLS:
|
2022-10-13 06:54:20 +08:00
|
|
|
sym_index = SYMBOLS.find(symbol)
|
|
|
|
plain_text += SYMBOLS[
|
|
|
|
(sym_index - key_b) * mod_inverse_of_key_a % len(SYMBOLS)
|
|
|
|
]
|
2016-08-07 23:48:46 +08:00
|
|
|
else:
|
2022-10-13 06:54:20 +08:00
|
|
|
plain_text += symbol
|
|
|
|
return plain_text
|
2016-08-07 23:48:46 +08:00
|
|
|
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2020-10-16 14:11:52 +08:00
|
|
|
def get_random_key() -> int:
|
2016-08-07 23:48:46 +08:00
|
|
|
while True:
|
2022-10-13 06:54:20 +08:00
|
|
|
key_b = random.randint(2, len(SYMBOLS))
|
|
|
|
key_b = random.randint(2, len(SYMBOLS))
|
2023-10-09 20:19:12 +08:00
|
|
|
if gcd_by_iterative(key_b, len(SYMBOLS)) == 1 and key_b % len(SYMBOLS) != 0:
|
2022-10-13 06:54:20 +08:00
|
|
|
return key_b * len(SYMBOLS) + key_b
|
2016-08-07 23:48:46 +08:00
|
|
|
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2021-03-22 14:59:51 +08:00
|
|
|
def main() -> None:
|
|
|
|
"""
|
|
|
|
>>> key = get_random_key()
|
|
|
|
>>> msg = "This is a test!"
|
|
|
|
>>> decrypt_message(key, encrypt_message(key, msg)) == msg
|
|
|
|
True
|
|
|
|
"""
|
|
|
|
message = input("Enter message: ").strip()
|
|
|
|
key = int(input("Enter key [2000 - 9000]: ").strip())
|
|
|
|
mode = input("Encrypt/Decrypt [E/D]: ").strip().lower()
|
|
|
|
|
|
|
|
if mode.startswith("e"):
|
|
|
|
mode = "encrypt"
|
|
|
|
translated = encrypt_message(key, message)
|
|
|
|
elif mode.startswith("d"):
|
|
|
|
mode = "decrypt"
|
|
|
|
translated = decrypt_message(key, message)
|
|
|
|
print(f"\n{mode.title()}ed text: \n{translated}")
|
|
|
|
|
|
|
|
|
2019-10-05 13:14:13 +08:00
|
|
|
if __name__ == "__main__":
|
2016-08-07 23:48:46 +08:00
|
|
|
import doctest
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2016-08-07 23:48:46 +08:00
|
|
|
doctest.testmod()
|
2020-07-02 22:32:15 +08:00
|
|
|
# main()
|