test_ElGamal.py 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. # -*- coding: utf-8 -*-
  2. #
  3. # SelfTest/PublicKey/test_ElGamal.py: Self-test for the ElGamal primitive
  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. """Self-test suite for Cryptodome.PublicKey.ElGamal"""
  23. __revision__ = "$Id$"
  24. import unittest
  25. from Cryptodome.SelfTest.st_common import list_test_cases, a2b_hex, b2a_hex
  26. from Cryptodome import Random
  27. from Cryptodome.PublicKey import ElGamal
  28. from Cryptodome.Util.number import bytes_to_long
  29. from Cryptodome.Util.py3compat import *
  30. class ElGamalTest(unittest.TestCase):
  31. #
  32. # Test vectors
  33. #
  34. # There seem to be no real ElGamal test vectors available in the
  35. # public domain. The following test vectors have been generated
  36. # with libgcrypt 1.5.0.
  37. #
  38. # Encryption
  39. tve=[
  40. {
  41. # 256 bits
  42. 'p' :'BA4CAEAAED8CBE952AFD2126C63EB3B345D65C2A0A73D2A3AD4138B6D09BD933',
  43. 'g' :'05',
  44. 'y' :'60D063600ECED7C7C55146020E7A31C4476E9793BEAED420FEC9E77604CAE4EF',
  45. 'x' :'1D391BA2EE3C37FE1BA175A69B2C73A11238AD77675932',
  46. 'k' :'F5893C5BAB4131264066F57AB3D8AD89E391A0B68A68A1',
  47. 'pt' :'48656C6C6F207468657265',
  48. 'ct1':'32BFD5F487966CEA9E9356715788C491EC515E4ED48B58F0F00971E93AAA5EC7',
  49. 'ct2':'7BE8FBFF317C93E82FCEF9BD515284BA506603FEA25D01C0CB874A31F315EE68'
  50. },
  51. {
  52. # 512 bits
  53. 'p' :'F1B18AE9F7B4E08FDA9A04832F4E919D89462FD31BF12F92791A93519F75076D6CE3942689CDFF2F344CAFF0F82D01864F69F3AECF566C774CBACF728B81A227',
  54. 'g' :'07',
  55. 'y' :'688628C676E4F05D630E1BE39D0066178CA7AA83836B645DE5ADD359B4825A12B02EF4252E4E6FA9BEC1DB0BE90F6D7C8629CABB6E531F472B2664868156E20C',
  56. 'x' :'14E60B1BDFD33436C0DA8A22FDC14A2CCDBBED0627CE68',
  57. 'k' :'38DBF14E1F319BDA9BAB33EEEADCAF6B2EA5250577ACE7',
  58. 'pt' :'48656C6C6F207468657265',
  59. 'ct1':'290F8530C2CC312EC46178724F196F308AD4C523CEABB001FACB0506BFED676083FE0F27AC688B5C749AB3CB8A80CD6F7094DBA421FB19442F5A413E06A9772B',
  60. 'ct2':'1D69AAAD1DC50493FB1B8E8721D621D683F3BF1321BE21BC4A43E11B40C9D4D9C80DE3AAC2AB60D31782B16B61112E68220889D53C4C3136EE6F6CE61F8A23A0'
  61. }
  62. ]
  63. # Signature
  64. tvs=[
  65. {
  66. # 256 bits
  67. 'p' :'D2F3C41EA66530838A704A48FFAC9334F4701ECE3A97CEE4C69DD01AE7129DD7',
  68. 'g' :'05',
  69. 'y' :'C3F9417DC0DAFEA6A05C1D2333B7A95E63B3F4F28CC962254B3256984D1012E7',
  70. 'x' :'165E4A39BE44D5A2D8B1332D416BC559616F536BC735BB',
  71. 'k' :'C7F0C794A7EAD726E25A47FF8928013680E73C51DD3D7D99BFDA8F492585928F',
  72. 'h' :'48656C6C6F207468657265',
  73. 'sig1':'35CA98133779E2073EF31165AFCDEB764DD54E96ADE851715495F9C635E1E7C2',
  74. 'sig2':'0135B88B1151279FE5D8078D4FC685EE81177EE9802AB123A73925FC1CB059A7',
  75. },
  76. {
  77. # 512 bits
  78. 'p' :'E24CF3A4B8A6AF749DCA6D714282FE4AABEEE44A53BB6ED15FBE32B5D3C3EF9CC4124A2ECA331F3C1C1B667ACA3766825217E7B5F9856648D95F05330C6A19CF',
  79. 'g' :'0B',
  80. 'y' :'2AD3A1049CA5D4ED207B2431C79A8719BB4073D4A94E450EA6CEE8A760EB07ADB67C0D52C275EE85D7B52789061EE45F2F37D9B2AE522A51C28329766BFE68AC',
  81. 'x' :'16CBB4F46D9ECCF24FF9F7E63CAA3BD8936341555062AB',
  82. 'k' :'8A3D89A4E429FD2476D7D717251FB79BF900FFE77444E6BB8299DC3F84D0DD57ABAB50732AE158EA52F5B9E7D8813E81FD9F79470AE22F8F1CF9AEC820A78C69',
  83. 'h' :'48656C6C6F207468657265',
  84. 'sig1':'BE001AABAFFF976EC9016198FBFEA14CBEF96B000CCC0063D3324016F9E91FE80D8F9325812ED24DDB2B4D4CF4430B169880B3CE88313B53255BD4EC0378586F',
  85. 'sig2':'5E266F3F837BA204E3BBB6DBECC0611429D96F8C7CE8F4EFDF9D4CB681C2A954468A357BF4242CEC7418B51DFC081BCD21299EF5B5A0DDEF3A139A1817503DDE',
  86. }
  87. ]
  88. def test_generate_180(self):
  89. self._test_random_key(180)
  90. def test_encryption(self):
  91. for tv in self.tve:
  92. d = self.convert_tv(tv, True)
  93. key = ElGamal.construct(d['key'])
  94. ct = key._encrypt(d['pt'], d['k'])
  95. self.assertEquals(ct[0], d['ct1'])
  96. self.assertEquals(ct[1], d['ct2'])
  97. def test_decryption(self):
  98. for tv in self.tve:
  99. d = self.convert_tv(tv, True)
  100. key = ElGamal.construct(d['key'])
  101. pt = key._decrypt((d['ct1'], d['ct2']))
  102. self.assertEquals(pt, d['pt'])
  103. def test_signing(self):
  104. for tv in self.tvs:
  105. d = self.convert_tv(tv, True)
  106. key = ElGamal.construct(d['key'])
  107. sig1, sig2 = key._sign(d['h'], d['k'])
  108. self.assertEquals(sig1, d['sig1'])
  109. self.assertEquals(sig2, d['sig2'])
  110. def test_verification(self):
  111. for tv in self.tvs:
  112. d = self.convert_tv(tv, True)
  113. key = ElGamal.construct(d['key'])
  114. # Positive test
  115. res = key._verify( d['h'], (d['sig1'],d['sig2']) )
  116. self.failUnless(res)
  117. # Negative test
  118. res = key._verify( d['h'], (d['sig1']+1,d['sig2']) )
  119. self.failIf(res)
  120. def test_bad_key3(self):
  121. tup = tup0 = list(self.convert_tv(self.tvs[0], 1)['key'])[:3]
  122. tup[0] += 1 # p += 1 (not prime)
  123. self.assertRaises(ValueError, ElGamal.construct, tup)
  124. tup = tup0
  125. tup[1] = 1 # g = 1
  126. self.assertRaises(ValueError, ElGamal.construct, tup)
  127. tup = tup0
  128. tup[2] = tup[0]*2 # y = 2*p
  129. self.assertRaises(ValueError, ElGamal.construct, tup)
  130. def test_bad_key4(self):
  131. tup = tup0 = list(self.convert_tv(self.tvs[0], 1)['key'])
  132. tup[3] += 1 # x += 1
  133. self.assertRaises(ValueError, ElGamal.construct, tup)
  134. def convert_tv(self, tv, as_longs=0):
  135. """Convert a test vector from textual form (hexadecimal ascii
  136. to either integers or byte strings."""
  137. key_comps = 'p','g','y','x'
  138. tv2 = {}
  139. for c in tv.keys():
  140. tv2[c] = a2b_hex(tv[c])
  141. if as_longs or c in key_comps or c in ('sig1','sig2'):
  142. tv2[c] = bytes_to_long(tv2[c])
  143. tv2['key']=[]
  144. for c in key_comps:
  145. tv2['key'] += [tv2[c]]
  146. del tv2[c]
  147. return tv2
  148. def _test_random_key(self, bits):
  149. elgObj = ElGamal.generate(bits, Random.new().read)
  150. self._check_private_key(elgObj)
  151. self._exercise_primitive(elgObj)
  152. pub = elgObj.publickey()
  153. self._check_public_key(pub)
  154. self._exercise_public_primitive(elgObj)
  155. def _check_private_key(self, elgObj):
  156. # Check capabilities
  157. self.failUnless(elgObj.has_private())
  158. # Sanity check key data
  159. self.failUnless(1<elgObj.g<(elgObj.p-1))
  160. self.assertEquals(pow(elgObj.g, elgObj.p-1, elgObj.p), 1)
  161. self.failUnless(1<elgObj.x<(elgObj.p-1))
  162. self.assertEquals(pow(elgObj.g, elgObj.x, elgObj.p), elgObj.y)
  163. def _check_public_key(self, elgObj):
  164. # Check capabilities
  165. self.failIf(elgObj.has_private())
  166. # Sanity check key data
  167. self.failUnless(1<elgObj.g<(elgObj.p-1))
  168. self.assertEquals(pow(elgObj.g, elgObj.p-1, elgObj.p), 1)
  169. def _exercise_primitive(self, elgObj):
  170. # Test encryption/decryption
  171. plaintext = 127218
  172. ciphertext = elgObj._encrypt(plaintext, 123456789)
  173. plaintextP = elgObj._decrypt(ciphertext)
  174. self.assertEquals(plaintext, plaintextP)
  175. # Test signature/verification
  176. signature = elgObj._sign(plaintext, 987654321)
  177. elgObj._verify(plaintext, signature)
  178. def _exercise_public_primitive(self, elgObj):
  179. plaintext = 92987276
  180. ciphertext = elgObj._encrypt(plaintext, 123456789)
  181. def get_tests(config={}):
  182. tests = []
  183. tests += list_test_cases(ElGamalTest)
  184. return tests
  185. if __name__ == '__main__':
  186. suite = lambda: unittest.TestSuite(get_tests())
  187. unittest.main(defaultTest='suite')