|
5 | 5 | from cryptography.hazmat.primitives.ciphers import Cipher, algorithms, modes
|
6 | 6 |
|
7 | 7 | from django.conf import settings
|
8 |
| -from django.utils.encoding import force_bytes, force_text |
| 8 | +from django.utils.encoding import force_bytes, force_str |
9 | 9 |
|
10 | 10 |
|
| 11 | +class BaseCipher: |
| 12 | + def __init__(self, key, iv) -> None: |
| 13 | + self.key = key |
| 14 | + self.iv = iv |
| 15 | + |
| 16 | + def encrypt(self, text) -> str: |
| 17 | + return text |
| 18 | + |
| 19 | + def decrypt(self, encrypted) -> str: |
| 20 | + return encrypted |
| 21 | + |
| 22 | + |
| 23 | +class ECBCipher(BaseCipher): |
| 24 | + |
| 25 | + def encrypt(self, text): |
| 26 | + encryptor = Cipher(algorithms.AES(self.key), modes.ECB(), default_backend()).encryptor() |
| 27 | + padder = padding.PKCS7(algorithms.AES(self.key).block_size).padder() |
| 28 | + padded_data = padder.update(force_bytes(text)) + padder.finalize() |
| 29 | + encrypted_text = encryptor.update(padded_data) + encryptor.finalize() |
| 30 | + return force_str(base64.urlsafe_b64encode(encrypted_text)) |
| 31 | + |
| 32 | + def decrypt(self, encrypted): |
| 33 | + decryptor = Cipher(algorithms.AES(self.key), modes.ECB(), default_backend()).decryptor() |
| 34 | + padder = padding.PKCS7(algorithms.AES(self.key).block_size).unpadder() |
| 35 | + decrypted_text = decryptor.update(base64.urlsafe_b64decode(encrypted)) |
| 36 | + unpadded_text = padder.update(decrypted_text) + padder.finalize() |
| 37 | + return force_str(unpadded_text) |
| 38 | + |
| 39 | +class CBCCipher(BaseCipher): |
| 40 | + |
| 41 | + def encrypt(self, text) -> str: |
| 42 | + encryptor = Cipher(algorithms.AES(self.key), modes.CBC(self.iv), default_backend()).encryptor() |
| 43 | + padder = padding.PKCS7(algorithms.AES(self.key).block_size).padder() |
| 44 | + padded_data = padder.update(force_bytes(text)) + padder.finalize() |
| 45 | + encrypted_text = encryptor.update(padded_data) + encryptor.finalize() |
| 46 | + return force_str(base64.urlsafe_b64encode(encrypted_text)) |
| 47 | + |
| 48 | + def decrypt(self, encrypted) -> str: |
| 49 | + decryptor = Cipher(algorithms.AES(self.key), modes.CBC(self.iv), default_backend()).decryptor() |
| 50 | + padder = padding.PKCS7(algorithms.AES(self.key).block_size).unpadder() |
| 51 | + decrypted_text = decryptor.update(base64.urlsafe_b64decode(encrypted)) |
| 52 | + unpadded_text = padder.update(decrypted_text) + padder.finalize() |
| 53 | + return force_str(unpadded_text) |
| 54 | + |
11 | 55 | class Crypto:
|
12 | 56 |
|
13 |
| - def __init__(self, key=None): |
| 57 | + def __init__(self, key=None, mode=None, iv=None): |
14 | 58 | if key is None:
|
15 | 59 | key = getattr(settings, "MIRAGE_SECRET_KEY", None) or getattr(settings, "SECRET_KEY")
|
16 | 60 | assert len(key) >= 32, "mirage key length must more than 32!"
|
17 |
| - self.key = base64.urlsafe_b64encode(force_bytes(key))[:32] |
| 61 | + key = base64.urlsafe_b64encode(force_bytes(key))[:32] |
| 62 | + if mode is None: |
| 63 | + mode = getattr(settings, "MIRAGE_CIPHER_MODE", "ECB") |
| 64 | + if iv is None: |
| 65 | + iv=getattr(settings, "MIRAGE_CIPHER_IV", "1234567890abcdef") |
| 66 | + self.cipher = eval(f"{mode}Cipher")(key=key, iv=force_bytes(iv)) |
18 | 67 |
|
19 | 68 | def encrypt(self, text):
|
20 | 69 | if text is None:
|
21 | 70 | return None
|
22 | 71 | try:
|
23 |
| - self.try_decrypt(text) |
| 72 | + self.cipher.decrypt(text) |
24 | 73 | return text
|
25 | 74 | except Exception:
|
26 |
| - return self.try_encrypt(text) |
27 |
| - |
28 |
| - def try_encrypt(self, text): |
29 |
| - encryptor = Cipher(algorithms.AES(self.key), modes.ECB(), default_backend()).encryptor() |
30 |
| - padder = padding.PKCS7(algorithms.AES(self.key).block_size).padder() |
31 |
| - padded_data = padder.update(force_bytes(text)) + padder.finalize() |
32 |
| - encrypted_text = encryptor.update(padded_data) + encryptor.finalize() |
33 |
| - return force_text(base64.urlsafe_b64encode(encrypted_text)) |
34 |
| - |
35 |
| - def try_decrypt(self, encrypted_text): |
36 |
| - decryptor = Cipher(algorithms.AES(self.key), modes.ECB(), default_backend()).decryptor() |
37 |
| - padder = padding.PKCS7(algorithms.AES(self.key).block_size).unpadder() |
38 |
| - decrypted_text = decryptor.update(base64.urlsafe_b64decode(encrypted_text)) |
39 |
| - unpadded_text = padder.update(decrypted_text) + padder.finalize() |
40 |
| - return force_text(unpadded_text) |
| 75 | + return self.cipher.encrypt(text) |
41 | 76 |
|
42 |
| - def decrypt(self, encrypted_text): |
43 |
| - if encrypted_text is None: |
| 77 | + def decrypt(self, encrypted): |
| 78 | + if encrypted is None: |
44 | 79 | return None
|
45 | 80 | try:
|
46 |
| - return self.try_decrypt(encrypted_text) |
| 81 | + return self.cipher.decrypt(encrypted) |
47 | 82 | except Exception:
|
48 |
| - return encrypted_text |
| 83 | + return encrypted |
0 commit comments