import pytest
import bitstring
from pyln.proto import zbase32

not_str_bytes = [1, dict(), None, 3.4, object()]
messages = [b'this', b'is', b'a', b'split', b'message:', b'lightning', b'rocks']
zbase32_messages = [b'qtwg1ha', b'pf3o', b'cr', b'qpaga4mw', b'pi1zgh5bc71uw', b'ptwsq4dwp3wsh3a', b'qjzsg45u']
not_zbase32_messages = ['00', '[]', 'vv', '1234']


def test_message_to_bitarray():
    # This is an internal function and the input is supposed to be bytes. Not testing unexpected inputs.
    dummy_messages = [b'a' * i for i in range(1, 6)]
    for message in dummy_messages:
        not_padded_barr = bitstring.Bits(message)
        barr = zbase32._message_to_bitarray(message)
        assert isinstance(barr, bitstring.ConstBitStream)

        for i in range(len(barr)):
            # Then first len(not_padded_barr) bits are equal
            if i < len(not_padded_barr):
                assert not_padded_barr.bin[i] == barr.bin[i]
            # The remaining are zeros
            else:
                assert barr.bin[i] == '0'


def test_bitarray_to_message():
    # This is an internal function and the input is supposed to be BitArray. Not testing unexpected inputs.
    dummy_messages = [b'b' * i for i in range(1, 6)]

    for message in dummy_messages:
        barr = bitstring.Bits(message)
        not_padded_message = zbase32._bitarray_to_message(bitstring.BitArray(message))
        assert isinstance(not_padded_message, bytes)

        not_padded_barr = bitstring.Bits(not_padded_message)
        for i in range(len(not_padded_barr)):
            # Then first len(pre_padded_barr) bits are equal
            if i < len(not_padded_barr):
                assert not_padded_barr.bin[i] == barr.bin[i]
            # The remaining are zeros
            else:
                assert barr.bin[i] == '0'


def test_bitarray_to_u5():
    # This is an internal function and the input is supposed to be ConstBitStream of length multiple of 5.
    # Not testing unexpected inputs.
    barrs = [zbase32._message_to_bitarray(message) for message in messages]

    for barr in barrs:
        u5 = zbase32._bitarray_to_u5(barr)
        assert isinstance(u5, list)
        assert all(x in range(0, 31) for x in u5)


def test_u5_to_bitarray():
    # This is an internal function and the input is supposed to be a list of ints 0-31. Not testing unexpected inputs.
    u5s = [[0, 1, 2, 3, 4, 5], [15, 30, 24, 17, 8, 3, 21], [0], [28, 11]]

    for u5 in u5s:
        bitarray = zbase32._u5_to_bitarray(u5)
        assert isinstance(bitarray, bitstring.BitArray)


def test_is_zbase32_encoded():
    for message in zbase32_messages:
        assert zbase32.is_zbase32_encoded(message)

    for message in not_zbase32_messages:
        assert not zbase32.is_zbase32_encoded(message)


def test_encode():
    message = '1f76e8acd54afbf23610b7166ba689afcc9e8ec3c44e442e765012dfc1d299958827d0205f7e4e1a12620e7fc8ce1c7d3651acefde899c33f12b6958d3304106a0'
    zbase32_message = b'd75qtmgijm79rpooshmgzjwji9gj7dsdat8remuskyjp9oq1ugkaoj6orbxzhuo4njtyh96e3aq84p1tiuz77nchgxa1s4ka4carnbiy'
    assert(zbase32.encode(bytes.fromhex(message)) == zbase32_message)

    for message, expected_zbase32_message in zip(messages, zbase32_messages):
        zbase32_message = zbase32.encode(message)
        assert isinstance(zbase32_message, bytes)
        assert zbase32_message == expected_zbase32_message


def test_encode_wrong_inputs():
    # Message must be either str or bytes, any other type will be rejected
    for m in not_str_bytes:
        with pytest.raises(TypeError, match='message must be string or bytes'):
            zbase32.encode(m)


def test_decode():
    zbase32_message = b'd75qtmgijm79rpooshmgzjwji9gj7dsdat8remuskyjp9oq1ugkaoj6orbxzhuo4njtyh96e3aq84p1tiuz77nchgxa1s4ka4carnbiy'
    message = '1f76e8acd54afbf23610b7166ba689afcc9e8ec3c44e442e765012dfc1d299958827d0205f7e4e1a12620e7fc8ce1c7d3651acefde899c33f12b6958d3304106a0'
    assert(zbase32.decode(zbase32_message) == bytes.fromhex(message))

    for expected_message, zbase32_message in zip(messages, zbase32_messages):
        message = zbase32.decode(zbase32_message)
        assert isinstance(message, bytes)
        assert message == expected_message


def test_decode_wrong_inputs():
    # Message must be either str or bytes, any other type will be rejected

    for m in not_str_bytes:
        with pytest.raises(TypeError, match='message must be string or bytes'):
            zbase32.decode(m)

    # Message must also be zbase32 encoded, otherwise it will be rejected
    for m in not_zbase32_messages:
        with pytest.raises(ValueError, match='message is not zbase32 encoded'):
            zbase32.decode(m)