PKCS1_v1_5.py 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. # -*- coding: utf-8 -*-
  2. #
  3. # Cipher/PKCS1-v1_5.py : PKCS#1 v1.5
  4. #
  5. # ===================================================================
  6. # The contents of this file are dedicated to the public domain. To
  7. # the extent that dedication to the public domain is not available,
  8. # everyone is granted a worldwide, perpetual, royalty-free,
  9. # non-exclusive license to exercise all rights associated with the
  10. # contents of this file for any purpose whatsoever.
  11. # No rights are reserved.
  12. #
  13. # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  14. # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  15. # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  16. # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
  17. # BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
  18. # ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
  19. # CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  20. # SOFTWARE.
  21. # ===================================================================
  22. __all__ = ['new', 'PKCS115_Cipher']
  23. from Cryptodome import Random
  24. from Cryptodome.Util.number import bytes_to_long, long_to_bytes
  25. from Cryptodome.Util.py3compat import bord, is_bytes, _copy_bytes
  26. from Cryptodome.Util._raw_api import (load_pycryptodome_raw_lib, c_size_t,
  27. c_uint8_ptr)
  28. _raw_pkcs1_decode = load_pycryptodome_raw_lib("Cryptodome.Cipher._pkcs1_decode",
  29. """
  30. int pkcs1_decode(const uint8_t *em, size_t len_em,
  31. const uint8_t *sentinel, size_t len_sentinel,
  32. size_t expected_pt_len,
  33. uint8_t *output);
  34. """)
  35. def _pkcs1_decode(em, sentinel, expected_pt_len, output):
  36. if len(em) != len(output):
  37. raise ValueError("Incorrect output length")
  38. ret = _raw_pkcs1_decode.pkcs1_decode(c_uint8_ptr(em),
  39. c_size_t(len(em)),
  40. c_uint8_ptr(sentinel),
  41. c_size_t(len(sentinel)),
  42. c_size_t(expected_pt_len),
  43. c_uint8_ptr(output))
  44. return ret
  45. class PKCS115_Cipher:
  46. """This cipher can perform PKCS#1 v1.5 RSA encryption or decryption.
  47. Do not instantiate directly. Use :func:`Cryptodome.Cipher.PKCS1_v1_5.new` instead."""
  48. def __init__(self, key, randfunc):
  49. """Initialize this PKCS#1 v1.5 cipher object.
  50. :Parameters:
  51. key : an RSA key object
  52. If a private half is given, both encryption and decryption are possible.
  53. If a public half is given, only encryption is possible.
  54. randfunc : callable
  55. Function that returns random bytes.
  56. """
  57. self._key = key
  58. self._randfunc = randfunc
  59. def can_encrypt(self):
  60. """Return True if this cipher object can be used for encryption."""
  61. return self._key.can_encrypt()
  62. def can_decrypt(self):
  63. """Return True if this cipher object can be used for decryption."""
  64. return self._key.can_decrypt()
  65. def encrypt(self, message):
  66. """Produce the PKCS#1 v1.5 encryption of a message.
  67. This function is named ``RSAES-PKCS1-V1_5-ENCRYPT``, and it is specified in
  68. `section 7.2.1 of RFC8017
  69. <https://tools.ietf.org/html/rfc8017#page-28>`_.
  70. :param message:
  71. The message to encrypt, also known as plaintext. It can be of
  72. variable length, but not longer than the RSA modulus (in bytes) minus 11.
  73. :type message: bytes/bytearray/memoryview
  74. :Returns: A byte string, the ciphertext in which the message is encrypted.
  75. It is as long as the RSA modulus (in bytes).
  76. :Raises ValueError:
  77. If the RSA key length is not sufficiently long to deal with the given
  78. message.
  79. """
  80. # See 7.2.1 in RFC8017
  81. k = self._key.size_in_bytes()
  82. mLen = len(message)
  83. # Step 1
  84. if mLen > k - 11:
  85. raise ValueError("Plaintext is too long.")
  86. # Step 2a
  87. ps = []
  88. while len(ps) != k - mLen - 3:
  89. new_byte = self._randfunc(1)
  90. if bord(new_byte[0]) == 0x00:
  91. continue
  92. ps.append(new_byte)
  93. ps = b"".join(ps)
  94. assert(len(ps) == k - mLen - 3)
  95. # Step 2b
  96. em = b'\x00\x02' + ps + b'\x00' + _copy_bytes(None, None, message)
  97. # Step 3a (OS2IP)
  98. em_int = bytes_to_long(em)
  99. # Step 3b (RSAEP)
  100. m_int = self._key._encrypt(em_int)
  101. # Step 3c (I2OSP)
  102. c = long_to_bytes(m_int, k)
  103. return c
  104. def decrypt(self, ciphertext, sentinel, expected_pt_len=0):
  105. r"""Decrypt a PKCS#1 v1.5 ciphertext.
  106. This is the function ``RSAES-PKCS1-V1_5-DECRYPT`` specified in
  107. `section 7.2.2 of RFC8017
  108. <https://tools.ietf.org/html/rfc8017#page-29>`_.
  109. Args:
  110. ciphertext (bytes/bytearray/memoryview):
  111. The ciphertext that contains the message to recover.
  112. sentinel (any type):
  113. The object to return whenever an error is detected.
  114. expected_pt_len (integer):
  115. The length the plaintext is known to have, or 0 if unknown.
  116. Returns (byte string):
  117. It is either the original message or the ``sentinel`` (in case of an error).
  118. .. warning::
  119. PKCS#1 v1.5 decryption is intrinsically vulnerable to timing
  120. attacks (see `Bleichenbacher's`__ attack).
  121. **Use PKCS#1 OAEP instead**.
  122. This implementation attempts to mitigate the risk
  123. with some constant-time constructs.
  124. However, they are not sufficient by themselves: the type of protocol you
  125. implement and the way you handle errors make a big difference.
  126. Specifically, you should make it very hard for the (malicious)
  127. party that submitted the ciphertext to quickly understand if decryption
  128. succeeded or not.
  129. To this end, it is recommended that your protocol only encrypts
  130. plaintexts of fixed length (``expected_pt_len``),
  131. that ``sentinel`` is a random byte string of the same length,
  132. and that processing continues for as long
  133. as possible even if ``sentinel`` is returned (i.e. in case of
  134. incorrect decryption).
  135. .. __: https://dx.doi.org/10.1007/BFb0055716
  136. """
  137. # See 7.2.2 in RFC8017
  138. k = self._key.size_in_bytes()
  139. # Step 1
  140. if len(ciphertext) != k:
  141. raise ValueError("Ciphertext with incorrect length (not %d bytes)" % k)
  142. # Step 2a (O2SIP)
  143. ct_int = bytes_to_long(ciphertext)
  144. # Step 2b (RSADP)
  145. m_int = self._key._decrypt(ct_int)
  146. # Complete step 2c (I2OSP)
  147. em = long_to_bytes(m_int, k)
  148. # Step 3 (not constant time when the sentinel is not a byte string)
  149. output = bytes(bytearray(k))
  150. if not is_bytes(sentinel) or len(sentinel) > k:
  151. size = _pkcs1_decode(em, b'', expected_pt_len, output)
  152. if size < 0:
  153. return sentinel
  154. else:
  155. return output[size:]
  156. # Step 3 (somewhat constant time)
  157. size = _pkcs1_decode(em, sentinel, expected_pt_len, output)
  158. return output[size:]
  159. def new(key, randfunc=None):
  160. """Create a cipher for performing PKCS#1 v1.5 encryption or decryption.
  161. :param key:
  162. The key to use to encrypt or decrypt the message. This is a `Cryptodome.PublicKey.RSA` object.
  163. Decryption is only possible if *key* is a private RSA key.
  164. :type key: RSA key object
  165. :param randfunc:
  166. Function that return random bytes.
  167. The default is :func:`Cryptodome.Random.get_random_bytes`.
  168. :type randfunc: callable
  169. :returns: A cipher object `PKCS115_Cipher`.
  170. """
  171. if randfunc is None:
  172. randfunc = Random.get_random_bytes
  173. return PKCS115_Cipher(key, randfunc)