File: group_address_dpt_test.py

package info (click to toggle)
python-xknx 3.14.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 4,064 kB
  • sloc: python: 40,895; javascript: 8,556; makefile: 32; sh: 12
file content (141 lines) | stat: -rw-r--r-- 5,423 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
"""Test for group address dpt."""

from unittest.mock import Mock, patch

from xknx import XKNX
from xknx.dpt import DPTArray, DPTHumidity, DPTScaling, DPTTemperature
from xknx.telegram import GroupAddress, Telegram, TelegramDirection, apci


async def test_group_address_dpt_in_telegram_queue(xknx_no_interface: XKNX) -> None:
    """Test group address dpt."""
    xknx = xknx_no_interface
    telegram = Telegram(
        destination_address=GroupAddress("1/2/3"),
        direction=TelegramDirection.INCOMING,
        payload=apci.GroupValueWrite(DPTArray((0x7F,))),
    )
    read_telegram = Telegram(
        destination_address=GroupAddress("1/2/3"),
        direction=TelegramDirection.INCOMING,
        payload=apci.GroupValueRead(),
    )
    telegram_callback = Mock()
    xknx.telegram_queue.register_telegram_received_cb(telegram_callback)
    async with xknx:
        xknx.telegrams.put_nowait(telegram)
        await xknx.telegrams.join()
        assert telegram_callback.call_count == 1
        test_telegram = telegram_callback.call_args[0][0]
        assert test_telegram.decoded_data is None

        xknx.group_address_dpt.set({GroupAddress("1/2/3"): {"main": 5, "sub": 1}})
        xknx.telegrams.put_nowait(telegram)
        await xknx.telegrams.join()
        assert telegram_callback.call_count == 2
        test_telegram = telegram_callback.call_args[0][0]
        assert test_telegram.decoded_data is not None
        assert test_telegram.decoded_data.transcoder is DPTScaling
        assert test_telegram.decoded_data.value == 50
        # do nothing for read-telegrams
        xknx.telegrams.put_nowait(read_telegram)
        await xknx.telegrams.join()
        assert telegram_callback.call_count == 3
        test_telegram = telegram_callback.call_args[0][0]
        assert test_telegram.decoded_data is None


@patch("logging.Logger.warning")
@patch("logging.Logger.debug")
async def test_get_invalid_payload(
    logger_debug_mock: Mock, logger_warning_mock: Mock, xknx_no_interface: XKNX
) -> None:
    """Test processing when DPT doesn't fit payload."""
    xknx = xknx_no_interface
    xknx.group_address_dpt.set({GroupAddress("1/2/3"): {"main": 5}})
    telegram_callback = Mock()
    xknx.telegram_queue.register_telegram_received_cb(telegram_callback)

    telegram = Telegram(
        destination_address=GroupAddress("1/2/3"),
        direction=TelegramDirection.INCOMING,
        payload=apci.GroupValueWrite(DPTArray((0x01, 0x02))),  # wrong payload size
    )
    async with xknx:
        logger_debug_mock.reset_mock()
        logger_warning_mock.reset_mock()
        assert len(xknx.group_address_dpt.ga_decoding_error) == 0

        xknx.telegrams.put_nowait(telegram)
        await xknx.telegrams.join()
        assert len(xknx.group_address_dpt.ga_decoding_error) == 1
        assert telegram_callback.call_count == 1
        test_telegram = telegram_callback.call_args[0][0]
        assert test_telegram.decoded_data is None
        assert logger_debug_mock.call_count == 1  # telegram log
        assert logger_warning_mock.call_count == 1  # first occurrence goes to warning
        assert "DPT decoding error" in logger_warning_mock.call_args_list[0][0][0]
        # second telegram with same GA should go to debug
        logger_debug_mock.reset_mock()
        logger_warning_mock.reset_mock()
        xknx.telegrams.put_nowait(telegram)
        await xknx.telegrams.join()
        assert len(xknx.group_address_dpt.ga_decoding_error) == 1  # still 1 - same GA
        assert logger_warning_mock.call_count == 0
        assert logger_debug_mock.call_count == 2  # second occurrence goes to debug
        assert "DPT decoding error" in logger_debug_mock.call_args_list[0][0][0]

        assert xknx.group_address_dpt.ga_decoding_error == {GroupAddress("1/2/3")}


def test_set(xknx_no_interface: XKNX) -> None:
    """Test set."""
    xknx = xknx_no_interface
    xknx.group_address_dpt.set(
        {
            GroupAddress("1/2/3"): {"main": 5, "sub": 1},
            1: "temperature",
            "i-internal": "9.007",
        }
    )
    assert xknx.group_address_dpt._ga_dpts == {
        2563: DPTScaling,
        1: DPTTemperature,
        "i-internal": DPTHumidity,
    }
    assert xknx.group_address_dpt.get(GroupAddress("0/0/1")) is DPTTemperature


@patch("logging.Logger.warning")
@patch("logging.Logger.debug")
def test_set_invalid(
    logger_debug_mock: Mock, logger_warning_mock: Mock, xknx_no_interface: XKNX
) -> None:
    """Test set invalid dpts."""
    xknx = xknx_no_interface
    xknx.group_address_dpt.set(
        {
            GroupAddress("0/0/1"): {"main": None},
            2: "invalid",
            0: "temperature",  # 0 is not a valid GA
        }
    )
    assert logger_warning_mock.call_count == 1
    assert "Invalid group address" in logger_warning_mock.call_args[0][0]
    assert logger_debug_mock.call_count == 1
    assert "No transcoder found for DPTs" in logger_debug_mock.call_args[0][0]
    assert not xknx.group_address_dpt._ga_dpts


def test_clear(xknx_no_interface: XKNX) -> None:
    """Test clear."""
    xknx = xknx_no_interface
    xknx.group_address_dpt.set(
        {
            GroupAddress("1/2/3"): {"main": 5, "sub": 1},
            1: "temperature",
        }
    )
    assert len(xknx.group_address_dpt._ga_dpts) == 2
    xknx.group_address_dpt.clear()
    assert len(xknx.group_address_dpt._ga_dpts) == 0