dsa.py 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. # This file is dual licensed under the terms of the Apache License, Version
  2. # 2.0, and the BSD License. See the LICENSE file in the root of this repository
  3. # for complete details.
  4. from __future__ import absolute_import, division, print_function
  5. from cryptography import utils
  6. from cryptography.exceptions import InvalidSignature
  7. from cryptography.hazmat.backends.openssl.utils import (
  8. _calculate_digest_and_algorithm,
  9. _check_not_prehashed,
  10. _warn_sign_verify_deprecated,
  11. )
  12. from cryptography.hazmat.primitives import hashes
  13. from cryptography.hazmat.primitives.asymmetric import (
  14. AsymmetricSignatureContext,
  15. AsymmetricVerificationContext,
  16. dsa,
  17. )
  18. def _dsa_sig_sign(backend, private_key, data):
  19. sig_buf_len = backend._lib.DSA_size(private_key._dsa_cdata)
  20. sig_buf = backend._ffi.new("unsigned char[]", sig_buf_len)
  21. buflen = backend._ffi.new("unsigned int *")
  22. # The first parameter passed to DSA_sign is unused by OpenSSL but
  23. # must be an integer.
  24. res = backend._lib.DSA_sign(
  25. 0, data, len(data), sig_buf, buflen, private_key._dsa_cdata
  26. )
  27. backend.openssl_assert(res == 1)
  28. backend.openssl_assert(buflen[0])
  29. return backend._ffi.buffer(sig_buf)[: buflen[0]]
  30. def _dsa_sig_verify(backend, public_key, signature, data):
  31. # The first parameter passed to DSA_verify is unused by OpenSSL but
  32. # must be an integer.
  33. res = backend._lib.DSA_verify(
  34. 0, data, len(data), signature, len(signature), public_key._dsa_cdata
  35. )
  36. if res != 1:
  37. backend._consume_errors()
  38. raise InvalidSignature
  39. @utils.register_interface(AsymmetricVerificationContext)
  40. class _DSAVerificationContext(object):
  41. def __init__(self, backend, public_key, signature, algorithm):
  42. self._backend = backend
  43. self._public_key = public_key
  44. self._signature = signature
  45. self._algorithm = algorithm
  46. self._hash_ctx = hashes.Hash(self._algorithm, self._backend)
  47. def update(self, data):
  48. self._hash_ctx.update(data)
  49. def verify(self):
  50. data_to_verify = self._hash_ctx.finalize()
  51. _dsa_sig_verify(
  52. self._backend, self._public_key, self._signature, data_to_verify
  53. )
  54. @utils.register_interface(AsymmetricSignatureContext)
  55. class _DSASignatureContext(object):
  56. def __init__(self, backend, private_key, algorithm):
  57. self._backend = backend
  58. self._private_key = private_key
  59. self._algorithm = algorithm
  60. self._hash_ctx = hashes.Hash(self._algorithm, self._backend)
  61. def update(self, data):
  62. self._hash_ctx.update(data)
  63. def finalize(self):
  64. data_to_sign = self._hash_ctx.finalize()
  65. return _dsa_sig_sign(self._backend, self._private_key, data_to_sign)
  66. @utils.register_interface(dsa.DSAParametersWithNumbers)
  67. class _DSAParameters(object):
  68. def __init__(self, backend, dsa_cdata):
  69. self._backend = backend
  70. self._dsa_cdata = dsa_cdata
  71. def parameter_numbers(self):
  72. p = self._backend._ffi.new("BIGNUM **")
  73. q = self._backend._ffi.new("BIGNUM **")
  74. g = self._backend._ffi.new("BIGNUM **")
  75. self._backend._lib.DSA_get0_pqg(self._dsa_cdata, p, q, g)
  76. self._backend.openssl_assert(p[0] != self._backend._ffi.NULL)
  77. self._backend.openssl_assert(q[0] != self._backend._ffi.NULL)
  78. self._backend.openssl_assert(g[0] != self._backend._ffi.NULL)
  79. return dsa.DSAParameterNumbers(
  80. p=self._backend._bn_to_int(p[0]),
  81. q=self._backend._bn_to_int(q[0]),
  82. g=self._backend._bn_to_int(g[0]),
  83. )
  84. def generate_private_key(self):
  85. return self._backend.generate_dsa_private_key(self)
  86. @utils.register_interface(dsa.DSAPrivateKeyWithSerialization)
  87. class _DSAPrivateKey(object):
  88. def __init__(self, backend, dsa_cdata, evp_pkey):
  89. self._backend = backend
  90. self._dsa_cdata = dsa_cdata
  91. self._evp_pkey = evp_pkey
  92. p = self._backend._ffi.new("BIGNUM **")
  93. self._backend._lib.DSA_get0_pqg(
  94. dsa_cdata, p, self._backend._ffi.NULL, self._backend._ffi.NULL
  95. )
  96. self._backend.openssl_assert(p[0] != backend._ffi.NULL)
  97. self._key_size = self._backend._lib.BN_num_bits(p[0])
  98. key_size = utils.read_only_property("_key_size")
  99. def signer(self, signature_algorithm):
  100. _warn_sign_verify_deprecated()
  101. _check_not_prehashed(signature_algorithm)
  102. return _DSASignatureContext(self._backend, self, signature_algorithm)
  103. def private_numbers(self):
  104. p = self._backend._ffi.new("BIGNUM **")
  105. q = self._backend._ffi.new("BIGNUM **")
  106. g = self._backend._ffi.new("BIGNUM **")
  107. pub_key = self._backend._ffi.new("BIGNUM **")
  108. priv_key = self._backend._ffi.new("BIGNUM **")
  109. self._backend._lib.DSA_get0_pqg(self._dsa_cdata, p, q, g)
  110. self._backend.openssl_assert(p[0] != self._backend._ffi.NULL)
  111. self._backend.openssl_assert(q[0] != self._backend._ffi.NULL)
  112. self._backend.openssl_assert(g[0] != self._backend._ffi.NULL)
  113. self._backend._lib.DSA_get0_key(self._dsa_cdata, pub_key, priv_key)
  114. self._backend.openssl_assert(pub_key[0] != self._backend._ffi.NULL)
  115. self._backend.openssl_assert(priv_key[0] != self._backend._ffi.NULL)
  116. return dsa.DSAPrivateNumbers(
  117. public_numbers=dsa.DSAPublicNumbers(
  118. parameter_numbers=dsa.DSAParameterNumbers(
  119. p=self._backend._bn_to_int(p[0]),
  120. q=self._backend._bn_to_int(q[0]),
  121. g=self._backend._bn_to_int(g[0]),
  122. ),
  123. y=self._backend._bn_to_int(pub_key[0]),
  124. ),
  125. x=self._backend._bn_to_int(priv_key[0]),
  126. )
  127. def public_key(self):
  128. dsa_cdata = self._backend._lib.DSAparams_dup(self._dsa_cdata)
  129. self._backend.openssl_assert(dsa_cdata != self._backend._ffi.NULL)
  130. dsa_cdata = self._backend._ffi.gc(
  131. dsa_cdata, self._backend._lib.DSA_free
  132. )
  133. pub_key = self._backend._ffi.new("BIGNUM **")
  134. self._backend._lib.DSA_get0_key(
  135. self._dsa_cdata, pub_key, self._backend._ffi.NULL
  136. )
  137. self._backend.openssl_assert(pub_key[0] != self._backend._ffi.NULL)
  138. pub_key_dup = self._backend._lib.BN_dup(pub_key[0])
  139. res = self._backend._lib.DSA_set0_key(
  140. dsa_cdata, pub_key_dup, self._backend._ffi.NULL
  141. )
  142. self._backend.openssl_assert(res == 1)
  143. evp_pkey = self._backend._dsa_cdata_to_evp_pkey(dsa_cdata)
  144. return _DSAPublicKey(self._backend, dsa_cdata, evp_pkey)
  145. def parameters(self):
  146. dsa_cdata = self._backend._lib.DSAparams_dup(self._dsa_cdata)
  147. self._backend.openssl_assert(dsa_cdata != self._backend._ffi.NULL)
  148. dsa_cdata = self._backend._ffi.gc(
  149. dsa_cdata, self._backend._lib.DSA_free
  150. )
  151. return _DSAParameters(self._backend, dsa_cdata)
  152. def private_bytes(self, encoding, format, encryption_algorithm):
  153. return self._backend._private_key_bytes(
  154. encoding,
  155. format,
  156. encryption_algorithm,
  157. self,
  158. self._evp_pkey,
  159. self._dsa_cdata,
  160. )
  161. def sign(self, data, algorithm):
  162. data, algorithm = _calculate_digest_and_algorithm(
  163. self._backend, data, algorithm
  164. )
  165. return _dsa_sig_sign(self._backend, self, data)
  166. @utils.register_interface(dsa.DSAPublicKeyWithSerialization)
  167. class _DSAPublicKey(object):
  168. def __init__(self, backend, dsa_cdata, evp_pkey):
  169. self._backend = backend
  170. self._dsa_cdata = dsa_cdata
  171. self._evp_pkey = evp_pkey
  172. p = self._backend._ffi.new("BIGNUM **")
  173. self._backend._lib.DSA_get0_pqg(
  174. dsa_cdata, p, self._backend._ffi.NULL, self._backend._ffi.NULL
  175. )
  176. self._backend.openssl_assert(p[0] != backend._ffi.NULL)
  177. self._key_size = self._backend._lib.BN_num_bits(p[0])
  178. key_size = utils.read_only_property("_key_size")
  179. def verifier(self, signature, signature_algorithm):
  180. _warn_sign_verify_deprecated()
  181. utils._check_bytes("signature", signature)
  182. _check_not_prehashed(signature_algorithm)
  183. return _DSAVerificationContext(
  184. self._backend, self, signature, signature_algorithm
  185. )
  186. def public_numbers(self):
  187. p = self._backend._ffi.new("BIGNUM **")
  188. q = self._backend._ffi.new("BIGNUM **")
  189. g = self._backend._ffi.new("BIGNUM **")
  190. pub_key = self._backend._ffi.new("BIGNUM **")
  191. self._backend._lib.DSA_get0_pqg(self._dsa_cdata, p, q, g)
  192. self._backend.openssl_assert(p[0] != self._backend._ffi.NULL)
  193. self._backend.openssl_assert(q[0] != self._backend._ffi.NULL)
  194. self._backend.openssl_assert(g[0] != self._backend._ffi.NULL)
  195. self._backend._lib.DSA_get0_key(
  196. self._dsa_cdata, pub_key, self._backend._ffi.NULL
  197. )
  198. self._backend.openssl_assert(pub_key[0] != self._backend._ffi.NULL)
  199. return dsa.DSAPublicNumbers(
  200. parameter_numbers=dsa.DSAParameterNumbers(
  201. p=self._backend._bn_to_int(p[0]),
  202. q=self._backend._bn_to_int(q[0]),
  203. g=self._backend._bn_to_int(g[0]),
  204. ),
  205. y=self._backend._bn_to_int(pub_key[0]),
  206. )
  207. def parameters(self):
  208. dsa_cdata = self._backend._lib.DSAparams_dup(self._dsa_cdata)
  209. dsa_cdata = self._backend._ffi.gc(
  210. dsa_cdata, self._backend._lib.DSA_free
  211. )
  212. return _DSAParameters(self._backend, dsa_cdata)
  213. def public_bytes(self, encoding, format):
  214. return self._backend._public_key_bytes(
  215. encoding, format, self, self._evp_pkey, None
  216. )
  217. def verify(self, signature, data, algorithm):
  218. data, algorithm = _calculate_digest_and_algorithm(
  219. self._backend, data, algorithm
  220. )
  221. return _dsa_sig_verify(self._backend, self, signature, data)