2019-01-31 16:42:22 +01:00
|
|
|
"""
|
|
|
|
Test the alternatives for crypto primitives
|
|
|
|
"""
|
|
|
|
|
|
|
|
import hashlib
|
2022-05-11 01:01:57 +02:00
|
|
|
import ssl
|
2019-01-31 16:42:22 +01:00
|
|
|
import unittest
|
|
|
|
from abc import ABCMeta, abstractmethod
|
2021-07-21 17:55:52 +02:00
|
|
|
from binascii import hexlify
|
2021-01-29 16:02:55 +01:00
|
|
|
|
2024-04-14 02:12:34 +02:00
|
|
|
from pybitmessage import highlevelcrypto, fallback
|
2019-01-31 16:42:22 +01:00
|
|
|
|
2021-01-29 16:02:55 +01:00
|
|
|
|
2019-01-31 16:42:22 +01:00
|
|
|
try:
|
2022-05-11 01:01:57 +02:00
|
|
|
from Crypto.Hash import RIPEMD160
|
2019-01-31 16:42:22 +01:00
|
|
|
except ImportError:
|
2022-05-11 01:01:57 +02:00
|
|
|
RIPEMD160 = None
|
2019-01-31 16:42:22 +01:00
|
|
|
|
2021-07-21 17:55:52 +02:00
|
|
|
from .samples import (
|
2021-12-09 18:46:02 +01:00
|
|
|
sample_deterministic_ripe, sample_double_sha512, sample_hash_data,
|
2021-07-27 21:21:20 +02:00
|
|
|
sample_msg, sample_pubsigningkey, sample_pubencryptionkey,
|
|
|
|
sample_privsigningkey, sample_privencryptionkey, sample_ripe,
|
2021-12-09 18:46:02 +01:00
|
|
|
sample_seed, sample_sig, sample_sig_sha1
|
2021-02-02 19:17:11 +01:00
|
|
|
)
|
|
|
|
|
2021-07-21 17:55:52 +02:00
|
|
|
|
2019-01-31 16:42:22 +01:00
|
|
|
_sha = hashlib.new('sha512')
|
|
|
|
_sha.update(sample_pubsigningkey + sample_pubencryptionkey)
|
|
|
|
|
|
|
|
pubkey_sha = _sha.digest()
|
|
|
|
|
|
|
|
|
|
|
|
class RIPEMD160TestCase(object):
|
|
|
|
"""Base class for RIPEMD160 test case"""
|
2019-02-16 12:57:30 +01:00
|
|
|
# pylint: disable=too-few-public-methods,no-member
|
2019-01-31 16:42:22 +01:00
|
|
|
__metaclass__ = ABCMeta
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def _hashdigest(self, data):
|
|
|
|
"""RIPEMD160 digest implementation"""
|
|
|
|
pass
|
|
|
|
|
|
|
|
def test_hash_string(self):
|
|
|
|
"""Check RIPEMD160 hash function on string"""
|
|
|
|
self.assertEqual(hexlify(self._hashdigest(pubkey_sha)), sample_ripe)
|
|
|
|
|
|
|
|
|
2022-05-11 01:01:57 +02:00
|
|
|
@unittest.skipIf(
|
|
|
|
ssl.OPENSSL_VERSION.startswith('OpenSSL 3'), 'no ripemd160 in openssl 3')
|
2019-01-31 16:42:22 +01:00
|
|
|
class TestHashlib(RIPEMD160TestCase, unittest.TestCase):
|
|
|
|
"""RIPEMD160 test case for hashlib"""
|
|
|
|
@staticmethod
|
|
|
|
def _hashdigest(data):
|
|
|
|
hasher = hashlib.new('ripemd160')
|
|
|
|
hasher.update(data)
|
|
|
|
return hasher.digest()
|
|
|
|
|
|
|
|
|
2022-05-11 01:01:57 +02:00
|
|
|
@unittest.skipUnless(RIPEMD160, 'pycrypto package not found')
|
2019-01-31 16:42:22 +01:00
|
|
|
class TestCrypto(RIPEMD160TestCase, unittest.TestCase):
|
|
|
|
"""RIPEMD160 test case for Crypto"""
|
|
|
|
@staticmethod
|
|
|
|
def _hashdigest(data):
|
2022-05-11 01:01:57 +02:00
|
|
|
return RIPEMD160.new(data).digest()
|
2019-11-20 16:35:51 +01:00
|
|
|
|
|
|
|
|
2021-07-20 19:06:01 +02:00
|
|
|
class TestHighlevelcrypto(unittest.TestCase):
|
|
|
|
"""Test highlevelcrypto public functions"""
|
|
|
|
|
2021-07-29 21:18:16 +02:00
|
|
|
def test_double_sha512(self):
|
|
|
|
"""Reproduce the example on page 1 of the Specification"""
|
|
|
|
self.assertEqual(
|
|
|
|
highlevelcrypto.double_sha512(sample_hash_data),
|
|
|
|
sample_double_sha512)
|
|
|
|
|
2021-12-09 17:44:57 +01:00
|
|
|
def test_randomBytes(self):
|
|
|
|
"""Dummy checks for random bytes"""
|
|
|
|
for n in (8, 32, 64):
|
|
|
|
data = highlevelcrypto.randomBytes(n)
|
|
|
|
self.assertEqual(len(data), n)
|
|
|
|
self.assertNotEqual(len(set(data)), 1)
|
|
|
|
self.assertNotEqual(data, highlevelcrypto.randomBytes(n))
|
|
|
|
|
2021-12-09 18:46:02 +01:00
|
|
|
def test_random_keys(self):
|
|
|
|
"""Dummy checks for random keys"""
|
|
|
|
priv, pub = highlevelcrypto.random_keys()
|
|
|
|
self.assertEqual(len(priv), 32)
|
|
|
|
self.assertEqual(highlevelcrypto.pointMult(priv), pub)
|
|
|
|
|
|
|
|
def test_deterministic_keys(self):
|
|
|
|
"""Generate deterministic keys, make ripe and compare it to sample"""
|
|
|
|
# encodeVarint(42) = b'*'
|
|
|
|
sigkey = highlevelcrypto.deterministic_keys(sample_seed, b'*')[1]
|
|
|
|
enkey = highlevelcrypto.deterministic_keys(sample_seed, b'+')[1]
|
|
|
|
self.assertEqual(
|
|
|
|
sample_deterministic_ripe,
|
2024-04-14 02:12:34 +02:00
|
|
|
hexlify(fallback.RIPEMD160Hash(
|
|
|
|
hashlib.sha512(sigkey + enkey).digest()).digest()))
|
2021-12-09 18:46:02 +01:00
|
|
|
|
2021-07-27 21:21:20 +02:00
|
|
|
def test_signatures(self):
|
|
|
|
"""Verify sample signatures and newly generated ones"""
|
|
|
|
pubkey_hex = hexlify(sample_pubsigningkey)
|
|
|
|
# pregenerated signatures
|
2021-07-27 23:44:19 +02:00
|
|
|
self.assertTrue(highlevelcrypto.verify(
|
|
|
|
sample_msg, sample_sig, pubkey_hex, "sha256"))
|
|
|
|
self.assertFalse(highlevelcrypto.verify(
|
|
|
|
sample_msg, sample_sig, pubkey_hex, "sha1"))
|
|
|
|
self.assertTrue(highlevelcrypto.verify(
|
|
|
|
sample_msg, sample_sig_sha1, pubkey_hex, "sha1"))
|
|
|
|
self.assertTrue(highlevelcrypto.verify(
|
|
|
|
sample_msg, sample_sig_sha1, pubkey_hex))
|
|
|
|
# new signatures
|
|
|
|
sig256 = highlevelcrypto.sign(sample_msg, sample_privsigningkey)
|
|
|
|
sig1 = highlevelcrypto.sign(sample_msg, sample_privsigningkey, "sha1")
|
2021-07-27 21:21:20 +02:00
|
|
|
self.assertTrue(
|
2021-07-27 23:44:19 +02:00
|
|
|
highlevelcrypto.verify(sample_msg, sig256, pubkey_hex))
|
2021-07-27 21:21:20 +02:00
|
|
|
self.assertTrue(
|
2021-07-27 23:44:19 +02:00
|
|
|
highlevelcrypto.verify(sample_msg, sig256, pubkey_hex, "sha256"))
|
2021-07-27 21:21:20 +02:00
|
|
|
self.assertTrue(
|
|
|
|
highlevelcrypto.verify(sample_msg, sig1, pubkey_hex))
|
|
|
|
|
2019-11-20 16:35:51 +01:00
|
|
|
def test_privtopub(self):
|
|
|
|
"""Generate public keys and check the result"""
|
|
|
|
self.assertEqual(
|
2021-07-20 19:06:01 +02:00
|
|
|
highlevelcrypto.privToPub(sample_privsigningkey),
|
2019-11-20 16:35:51 +01:00
|
|
|
hexlify(sample_pubsigningkey)
|
|
|
|
)
|
|
|
|
self.assertEqual(
|
2021-07-20 19:06:01 +02:00
|
|
|
highlevelcrypto.privToPub(sample_privencryptionkey),
|
2019-11-20 16:35:51 +01:00
|
|
|
hexlify(sample_pubencryptionkey)
|
|
|
|
)
|