Skip to content

Commit

Permalink
Test for class object Processor
Browse files Browse the repository at this point in the history
  • Loading branch information
anand-skss committed May 22, 2024
1 parent e571ba8 commit 3073ef1
Show file tree
Hide file tree
Showing 3 changed files with 237 additions and 7 deletions.
14 changes: 7 additions & 7 deletions src/class_objectProcessor.py
Original file line number Diff line number Diff line change
Expand Up @@ -315,7 +315,7 @@ def processpubkey(self, data):
dataToStore = data[20:readPosition]
sha = hashlib.new('sha512')
sha.update(
'\x04' + publicSigningKey + '\x04' + publicEncryptionKey)
b'\x04' + publicSigningKey + b'\x04' + publicEncryptionKey)
ripe = RIPEMD160Hash(sha.digest()).digest()

if logger.isEnabledFor(logging.DEBUG):
Expand Down Expand Up @@ -354,9 +354,9 @@ def processpubkey(self, data):
' Sanity check failed.')
return
readPosition += 4
publicSigningKey = '\x04' + data[readPosition:readPosition + 64]
publicSigningKey = b'\x04' + data[readPosition:readPosition + 64]
readPosition += 64
publicEncryptionKey = '\x04' + data[readPosition:readPosition + 64]
publicEncryptionKey = b'\x04' + data[readPosition:readPosition + 64]
readPosition += 64
specifiedNonceTrialsPerByteLength = decodeVarint(
data[readPosition:readPosition + 10])[1]
Expand Down Expand Up @@ -513,9 +513,9 @@ def processmsg(self, data):
return
readPosition += sendersStreamNumberLength
readPosition += 4
pubSigningKey = '\x04' + decryptedData[readPosition:readPosition + 64]
pubSigningKey = b'\x04' + decryptedData[readPosition:readPosition + 64]
readPosition += 64
pubEncryptionKey = '\x04' + decryptedData[readPosition:readPosition + 64]
pubEncryptionKey = b'\x04' + decryptedData[readPosition:readPosition + 64]
readPosition += 64
if sendersAddressVersionNumber >= 3:
requiredAverageProofOfWorkNonceTrialsPerByte, varintLength = \
Expand Down Expand Up @@ -862,10 +862,10 @@ def processbroadcast(self, data):
)
readPosition += sendersStreamLength
readPosition += 4
sendersPubSigningKey = '\x04' + \
sendersPubSigningKey = b'\x04' + \
decryptedData[readPosition:readPosition + 64]
readPosition += 64
sendersPubEncryptionKey = '\x04' + \
sendersPubEncryptionKey = b'\x04' + \
decryptedData[readPosition:readPosition + 64]
readPosition += 64
if sendersAddressVersion >= 3:
Expand Down
21 changes: 21 additions & 0 deletions src/mockbm/class_objectProcessor.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
"""Mock function/class used in objectProcessor"""


def mockFunctionToPass():

Check notice on line 4 in src/mockbm/class_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pylint

C0111 / missing-docstring

Missing function docstring
pass


def sqlQueryMock(*args):

Check warning on line 8 in src/mockbm/class_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pylint

W0613 / unused-argument

Unused argument 'args'

Check notice on line 8 in src/mockbm/class_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pylint

C0111 / missing-docstring

Missing function docstring
return [(None, "mockData"),]

Check failure on line 9 in src/mockbm/class_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pycodestyle

E231

missing whitespace after ','

Check failure on line 9 in src/mockbm/class_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - flake8

E231

missing whitespace after ','


def sqlExecuteMock(*args):

Check warning on line 12 in src/mockbm/class_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pylint

W0613 / unused-argument

Unused argument 'args'

Check notice on line 12 in src/mockbm/class_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pylint

C0111 / missing-docstring

Missing function docstring
return


class SqlReadyMock(object):

Check notice on line 16 in src/mockbm/class_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pylint

R0903 / too-few-public-methods

Too few public methods (1/2)
"""Mock helper_sql.sql_ready event with dummy class"""
@staticmethod
def wait(time):

Check warning on line 19 in src/mockbm/class_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pylint

W0613 / unused-argument

Unused argument 'time'
"""Don't wait, return immediately"""
return
209 changes: 209 additions & 0 deletions src/tests/test_objectProcessor.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,209 @@
"""Test cases for ObjectProcessor"""

from binascii import hexlify
import threading
import state
import pybitmessage.class_objectProcessor as op

try:
from unittest.mock import patch, MagicMock
except ImportError:
from mock import patch, MagicMock, Mock

Check notice on line 11 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - flake8

F401

'mock.Mock' imported but unused

Check notice on line 11 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - flake8

F401

'mock.MagicMock' imported but unused

from .partial import TestPartialRun
from mockbm.class_objectProcessor import (
mockFunctionToPass,
sqlExecuteMock,
sqlQueryMock,
SqlReadyMock,
)
from .samples import (
sample_pubsigningkey,
sample_pubencryptionkey,
sample_privencryptionkey,
sample_deterministic_addr4,
sample_deterministic_addr3,
sample_privsigningkey
)


class TestObjectProcessor(TestPartialRun):
"""Test class for ObjectProcessor"""

@classmethod
def setUpClass(cls):
super(TestObjectProcessor, cls).setUpClass()

op.sqlQuery = sqlQueryMock
op.sqlExecute = sqlExecuteMock
op.sql_ready = SqlReadyMock
op.shared.reloadBroadcastSendersForWhichImWatching = mockFunctionToPass
op.shared.reloadMyAddressHashes = mockFunctionToPass
cls.queues = op.queues
cls.processor = op.objectProcessor()
cls.processor.start()

@classmethod
def tearDownClass(cls):
super(TestObjectProcessor, cls).tearDownClass()
for thread in threading.enumerate():
if thread.name == "objectProcessor":
op.queues.objectProcessorQueue.put(('checkShutdownVariable', 'no data'))

@patch("pybitmessage.class_objectProcessor.logger.debug")
@patch("pybitmessage.class_objectProcessor.queues.UISignalQueue.put")
def test_checkackdata(self, mock_UISignalQueue_put, mock_logger):
"""test checkdata"""
data = b"\x00" * 8 # nonce
data += b"\x00" * 8 # expiresTime
data += b"\x00" * 4 # getpubkey object type
data += b"\x04" # addressVersion
data += b"\x01" # streamNumber
data += b"\00" * 4 # behaviour bitfield
data += sample_pubsigningkey
data += sample_pubencryptionkey

# ackdata not in state.ackdataForWhichImWatching
state.ackdataForWhichImWatching = {}
self.processor.checkackdata(data)
mock_logger.assert_called_with('This object is not an acknowledgement bound for me.')

# ackdata is in state.ackdataForWhichImWatching
state.ackdataForWhichImWatching = {data[16:]: data}
self.processor.checkackdata(data)
self.assertEqual(len(state.ackdataForWhichImWatching), 0)
mock_UISignalQueue_put.assert_called_once()


@patch("pybitmessage.class_objectProcessor.protocol.checkIPAddress")

Check failure on line 78 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pycodestyle

E303

too many blank lines (2)

Check failure on line 78 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - flake8

E303

too many blank lines (2)
@patch("pybitmessage.class_objectProcessor.decodeVarint")
@patch("pybitmessage.class_objectProcessor.knownnodes.addKnownNode")
def test_processonion(self, mock_addknownNode, mock_decodeVarint, mock_checkIPAddress):
"""Test processonion"""
data = b"\x00" * 100
# decode varient called 3 times returns -> addressversion & lenth
# stream & length, port & length
mock_decodeVarint.side_effect = [(3, 1), (2, 1), (8080, 4), ]
mock_checkIPAddress.return_value = "127.0.0.1"
self.processor.processonion(data)
mock_addknownNode.assert_called_once()

@patch("pybitmessage.class_objectProcessor.queues.workerQueue.put")
def test_processgetpubkey_V3(
self, mock_workerqueue_put
):
"""Test processgetpukey with address version 3"""
HASH = b"\x01" * 20
gpk = b"\x00" * 8 # nonce
gpk += b"\x00" * 8 # expiresTime
gpk += b"\x00" * 4 # getpubkey object type
gpk += b"\x03" # addressVersion 3
gpk += b"\x01" # streamNumber
gpk += HASH # hash

# set address corresponding to hash
op.shared.myAddressesByHash[HASH] = sample_deterministic_addr3

self.processor.processgetpubkey(gpk)
mock_workerqueue_put.assert_called_once()

@patch("pybitmessage.class_objectProcessor.queues.workerQueue.put")
def test_processgetpubkey_V4(
self, mock_workerqueue_put
):
"""Test processgetpukey with address version 4"""
TAG = b"\x01" * 32
gpk = b"\x00" * 8 # nonce
gpk += b"\x00" * 8 # expiresTime
gpk += b"\x00" * 4 # getpubkey object type
gpk += b"\x04" # addressVersion 4
gpk += b"\x01" # streamNumber
gpk += TAG # tag

# set address corresponding to tag
op.shared.myAddressesByTag[TAG] = sample_deterministic_addr4

self.processor.processgetpubkey(gpk)
mock_workerqueue_put.assert_called_once()

def test_processpubkey_version2(
self,
):
"""Test processpubkey with version 2"""
ppk = b"\x00" * 8 # nonce
ppk += b"\x00" * 8 # expiresTime
ppk += b"\x00\x00\x00\x01" # getpubkey object type
ppk += b"\x02" # addressVersion
ppk += b"\x01" # streamNumber
ppk += b"\00" * 4 # behaviour bitfield
ppk += sample_pubsigningkey
ppk += sample_pubencryptionkey
self.processor.processpubkey(ppk)

def test_processpubkey_version3(
self,
):
"""Test processpubkey with version 3"""
ppk = b"\x00" * 8 # nonce
ppk += b"\x00" * 8 # expiresTime
ppk += b"\x00\x00\x00\x01" # getpubkey object type
ppk += b"\x03" # addressVersion
ppk += b"\x01" # streamNumber
ppk += b"\00" * 4 # behaviour bitfield
ppk += sample_pubsigningkey
ppk += sample_pubencryptionkey
ppk += b"\x01" # nonce_trials_per_byte
ppk += b"\x01" # extra_bytes
signature = op.highlevelcrypto.sign(ppk[8:-2], hexPrivkey=hexlify(sample_privsigningkey)) # signature
ppk += hex(len(signature)).encode() # sig_length
ppk += signature
self.processor.processpubkey(ppk)

Check warning on line 161 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pycodestyle

W293

blank line contains whitespace

Check warning on line 161 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - flake8

W293

blank line contains whitespace
def test_processpubkey_version4(
self
):
"""Test processpubkey with version 4"""
Tag = b"\00" * 32
pk = b"\x00" * 8 # nonce
pk += b"\x00" * 8 # expiresTime
pk += b"\x00\x00\x00\x01" # getpubkey object type
pk += b"\x04" # addressVersion
pk += b"\x01" # streamNumber
pk += Tag # tag

data_to_encrypt = b"\00" * 4 # behaviour bitfield
data_to_encrypt += sample_pubsigningkey
data_to_encrypt += sample_pubencryptionkey
data_to_encrypt += b"\x01" # nonce_trials_per_byte
data_to_encrypt += b"\x01" # extra_bytes
signature = op.highlevelcrypto.sign(pk[8:], hexPrivkey=hexlify(sample_privsigningkey)) # signature
data_to_encrypt += hex(len(signature)).encode() # sig_length
data_to_encrypt += signature

Check warning on line 182 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pycodestyle

W293

blank line contains whitespace

Check warning on line 182 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - flake8

W293

blank line contains whitespace
cryptor = op.highlevelcrypto.makeCryptor(sample_privencryptionkey)
encrypted_data = cryptor.encrypt(data_to_encrypt, hexlify(sample_pubencryptionkey))
pk += encrypted_data

Check warning on line 186 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pycodestyle

W293

blank line contains whitespace

Check warning on line 186 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - flake8

W293

blank line contains whitespace
state.neededPubkeys[Tag] = (sample_deterministic_addr4, cryptor)
self.processor.processpubkey(pk)

@patch("pybitmessage.class_objectProcessor.objectProcessor.sendMessages")
def test_possibleNewPubkey_with_addressV3(self, mock_sendMessages):
"""Test possibleNewPubkey with version 3 address"""
state.neededPubkeys = {sample_deterministic_addr3: "pubkey2"}

self.processor.possibleNewPubkey(sample_deterministic_addr3)
mock_sendMessages.assert_called_with(sample_deterministic_addr3)

@patch("pybitmessage.class_objectProcessor.highlevelcrypto.double_sha512")
@patch("pybitmessage.class_objectProcessor.objectProcessor.sendMessages")
def test_possibleNewPubkey_with_addressV4(self, mock_sendMessages, mock_double_sha512):
"""Test possibleNewPubkey with version 4 address"""
fake_encrypted_data = "\x01" * 64
state.neededPubkeys = {fake_encrypted_data[32:]: "pubkey"}

mock_double_sha512.return_value = fake_encrypted_data
self.processor.possibleNewPubkey(sample_deterministic_addr4)
mock_sendMessages.assert_called_with(sample_deterministic_addr4)


Check warning on line 209 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - pycodestyle

W391

blank line at end of file

Check warning on line 209 in src/tests/test_objectProcessor.py

View check run for this annotation

PyBitmessage Code Quality Checks / Code Quality - flake8

W391

blank line at end of file

0 comments on commit 3073ef1

Please sign in to comment.