"""Tests for the Z-Wave init."""
import asyncio
from collections import OrderedDict
from datetime import datetime
from unittest.mock import MagicMock, patch

import pytest
import voluptuous as vol

from homeassistant.bootstrap import async_setup_component
from homeassistant.components import zwave
from homeassistant.components.zwave import (
    CONF_DEVICE_CONFIG_GLOB,
    CONFIG_SCHEMA,
    DATA_NETWORK,
    const,
)
from homeassistant.const import ATTR_NAME
from homeassistant.helpers import device_registry as dr, entity_registry as er
from homeassistant.util import dt as dt_util

from tests.common import async_fire_time_changed, mock_registry
from tests.mock.zwave import MockEntityValues, MockNetwork, MockNode, MockValue


@pytest.fixture(autouse=True)
def mock_storage(hass_storage):
    """Autouse hass_storage for the TestCase tests."""


@pytest.fixture
async def zwave_setup(hass):
    """Zwave setup."""
    await async_setup_component(hass, "zwave", {"zwave": {}})
    await hass.async_block_till_done()


@pytest.fixture
async def zwave_setup_ready(hass, zwave_setup):
    """Zwave setup and set network to ready."""
    zwave_network = hass.data[DATA_NETWORK]
    zwave_network.state = MockNetwork.STATE_READY

    await hass.async_start()


async def test_valid_device_config(hass, mock_openzwave):
    """Test valid device config."""
    device_config = {"light.kitchen": {"ignored": "true"}}
    result = await async_setup_component(
        hass, "zwave", {"zwave": {"device_config": device_config}}
    )
    await hass.async_block_till_done()

    assert result


async def test_invalid_device_config(hass, mock_openzwave):
    """Test invalid device config."""
    device_config = {"light.kitchen": {"some_ignored": "true"}}
    result = await async_setup_component(
        hass, "zwave", {"zwave": {"device_config": device_config}}
    )
    await hass.async_block_till_done()

    assert not result


def test_config_access_error():
    """Test threading error accessing config values."""
    node = MagicMock()

    def side_effect():
        raise RuntimeError

    node.values.values.side_effect = side_effect
    result = zwave.get_config_value(node, 1)
    assert result is None


async def test_network_options(hass, mock_openzwave):
    """Test network options."""
    result = await async_setup_component(
        hass,
        "zwave",
        {"zwave": {"usb_path": "mock_usb_path", "config_path": "mock_config_path"}},
    )
    await hass.async_block_till_done()

    assert result

    network = hass.data[zwave.DATA_NETWORK]
    assert network.options.device == "mock_usb_path"
    assert network.options.config_path == "mock_config_path"


async def test_network_key_validation(hass, mock_openzwave):
    """Test network key validation."""
    test_values = [
        (
            "0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, "
            "0x0C, 0x0D, 0x0E, 0x0F, 0x10"
        ),
        (
            "0x01,0x02,0x03,0x04,0x05,0x06,0x07,0x08,0x09,0x0A,0x0B,0x0C,0x0D,"
            "0x0E,0x0F,0x10"
        ),
    ]
    for value in test_values:
        result = zwave.CONFIG_SCHEMA({"zwave": {"network_key": value}})
        assert result["zwave"]["network_key"] == value


async def test_erronous_network_key_fails_validation(hass, mock_openzwave):
    """Test failing erroneous network key validation."""
    test_values = [
        (
            "0x 01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, "
            "0x0C, 0x0D, 0x0E, 0x0F, 0x10"
        ),
        (
            "0X01,0X02,0X03,0X04,0X05,0X06,0X07,0X08,0X09,0X0A,0X0B,0X0C,0X0D,"
            "0X0E,0X0F,0X10"
        ),
        "invalid",
        "1234567",
        1234567,
    ]
    for value in test_values:
        with pytest.raises(vol.Invalid):
            zwave.CONFIG_SCHEMA({"zwave": {"network_key": value}})


async def test_auto_heal_midnight(hass, mock_openzwave, legacy_patchable_time):
    """Test network auto-heal at midnight."""
    await async_setup_component(hass, "zwave", {"zwave": {"autoheal": True}})
    await hass.async_block_till_done()

    network = hass.data[zwave.DATA_NETWORK]
    assert not network.heal.called

    time = datetime(2017, 5, 6, 0, 0, 0, tzinfo=dt_util.UTC)
    async_fire_time_changed(hass, time)
    await hass.async_block_till_done()
    await hass.async_block_till_done()
    assert network.heal.called
    assert len(network.heal.mock_calls) == 1


async def test_auto_heal_disabled(hass, mock_openzwave):
    """Test network auto-heal disabled."""
    await async_setup_component(hass, "zwave", {"zwave": {"autoheal": False}})
    await hass.async_block_till_done()

    network = hass.data[zwave.DATA_NETWORK]
    assert not network.heal.called

    time = datetime(2017, 5, 6, 0, 0, 0, tzinfo=dt_util.UTC)
    async_fire_time_changed(hass, time)
    await hass.async_block_till_done()
    assert not network.heal.called


async def test_setup_platform(hass, mock_openzwave):
    """Test invalid device config."""
    mock_device = MagicMock()
    hass.data[DATA_NETWORK] = MagicMock()
    hass.data[zwave.DATA_DEVICES] = {456: mock_device}
    async_add_entities = MagicMock()

    result = await zwave.async_setup_platform(hass, None, async_add_entities, None)
    assert not result
    assert not async_add_entities.called

    result = await zwave.async_setup_platform(
        hass, None, async_add_entities, {const.DISCOVERY_DEVICE: 123}
    )
    assert not result
    assert not async_add_entities.called

    result = await zwave.async_setup_platform(
        hass, None, async_add_entities, {const.DISCOVERY_DEVICE: 456}
    )
    assert result
    assert async_add_entities.called
    assert len(async_add_entities.mock_calls) == 1
    assert async_add_entities.mock_calls[0][1][0] == [mock_device]


async def test_zwave_ready_wait(hass, mock_openzwave, zwave_setup):
    """Test that zwave continues after waiting for network ready."""
    sleeps = []

    def utcnow():
        return datetime.fromtimestamp(len(sleeps))

    asyncio_sleep = asyncio.sleep

    async def sleep(duration, loop=None):
        if duration > 0:
            sleeps.append(duration)
        await asyncio_sleep(0)

    with patch("homeassistant.components.zwave.dt_util.utcnow", new=utcnow), patch(
        "asyncio.sleep", new=sleep
    ), patch.object(zwave, "_LOGGER") as mock_logger:
        hass.data[DATA_NETWORK].state = MockNetwork.STATE_STARTED

        await hass.async_start()

        assert len(sleeps) == const.NETWORK_READY_WAIT_SECS
        assert mock_logger.warning.called
        assert len(mock_logger.warning.mock_calls) == 1
        assert mock_logger.warning.mock_calls[0][1][1] == const.NETWORK_READY_WAIT_SECS


async def test_device_entity(hass, mock_openzwave):
    """Test device entity base class."""
    node = MockNode(node_id="10", name="Mock Node")
    value = MockValue(
        data=False,
        node=node,
        instance=2,
        object_id="11",
        label="Sensor",
        command_class=const.COMMAND_CLASS_SENSOR_BINARY,
    )
    power_value = MockValue(
        data=50.123456, node=node, precision=3, command_class=const.COMMAND_CLASS_METER
    )
    values = MockEntityValues(primary=value, power=power_value)
    device = zwave.ZWaveDeviceEntity(values, "zwave")
    device.hass = hass
    device.value_added()
    device.update_properties()
    await hass.async_block_till_done()

    assert not device.should_poll
    assert device.unique_id == "10-11"
    assert device.name == "Mock Node Sensor"
    assert device.extra_state_attributes[zwave.ATTR_POWER] == 50.123


async def test_node_removed(hass, mock_openzwave):
    """Test node removed in base class."""
    # Create a mock node & node entity
    node = MockNode(node_id="10", name="Mock Node")
    value = MockValue(
        data=False,
        node=node,
        instance=2,
        object_id="11",
        label="Sensor",
        command_class=const.COMMAND_CLASS_SENSOR_BINARY,
    )
    power_value = MockValue(
        data=50.123456, node=node, precision=3, command_class=const.COMMAND_CLASS_METER
    )
    values = MockEntityValues(primary=value, power=power_value)
    device = zwave.ZWaveDeviceEntity(values, "zwave")
    device.hass = hass
    device.entity_id = "zwave.mock_node"
    device.value_added()
    device.update_properties()
    await hass.async_block_till_done()

    # Save it to the entity registry
    registry = mock_registry(hass)
    registry.async_get_or_create("zwave", "zwave", device.unique_id)
    device.entity_id = registry.async_get_entity_id("zwave", "zwave", device.unique_id)

    # Create dummy entity registry entries for other integrations
    hue_entity = registry.async_get_or_create("light", "hue", 1234)
    zha_entity = registry.async_get_or_create("sensor", "zha", 5678)

    # Verify our Z-Wave entity is registered
    assert registry.async_is_registered(device.entity_id)

    # Remove it
    entity_id = device.entity_id
    await device.node_removed()

    # Verify registry entry for our Z-Wave node is gone
    assert not registry.async_is_registered(entity_id)

    # Verify registry entries for our other entities remain
    assert registry.async_is_registered(hue_entity.entity_id)
    assert registry.async_is_registered(zha_entity.entity_id)


async def test_node_discovery(hass, mock_openzwave):
    """Test discovery of a node."""
    mock_receivers = []

    def mock_connect(receiver, signal, *args, **kwargs):
        if signal == MockNetwork.SIGNAL_NODE_ADDED:
            mock_receivers.append(receiver)

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(hass, "zwave", {"zwave": {}})
        await hass.async_block_till_done()

    assert len(mock_receivers) == 1

    node = MockNode(node_id=14)
    await hass.async_add_executor_job(mock_receivers[0], node)
    await hass.async_block_till_done()

    assert hass.states.get("zwave.mock_node").state == "unknown"


async def test_unparsed_node_discovery(hass, mock_openzwave):
    """Test discovery of a node."""
    mock_receivers = []

    def mock_connect(receiver, signal, *args, **kwargs):
        if signal == MockNetwork.SIGNAL_NODE_ADDED:
            mock_receivers.append(receiver)

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(hass, "zwave", {"zwave": {}})
        await hass.async_block_till_done()

    assert len(mock_receivers) == 1

    node = MockNode(node_id=14, manufacturer_name=None, name=None, is_ready=False)

    sleeps = []

    def utcnow():
        return datetime.fromtimestamp(len(sleeps))

    asyncio_sleep = asyncio.sleep

    async def sleep(duration, loop=None):
        if duration > 0:
            sleeps.append(duration)
        await asyncio_sleep(0)

    with patch("homeassistant.components.zwave.dt_util.utcnow", new=utcnow), patch(
        "asyncio.sleep", new=sleep
    ), patch.object(zwave, "_LOGGER") as mock_logger:
        await hass.async_add_executor_job(mock_receivers[0], node)
        await hass.async_block_till_done()

        assert len(sleeps) == const.NODE_READY_WAIT_SECS
        assert mock_logger.warning.called
        assert len(mock_logger.warning.mock_calls) == 1
        assert mock_logger.warning.mock_calls[0][1][1:] == (
            14,
            const.NODE_READY_WAIT_SECS,
        )
    assert hass.states.get("zwave.unknown_node_14").state == "unknown"


async def test_node_ignored(hass, mock_openzwave):
    """Test discovery of a node."""
    mock_receivers = []

    def mock_connect(receiver, signal, *args, **kwargs):
        if signal == MockNetwork.SIGNAL_NODE_ADDED:
            mock_receivers.append(receiver)

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(
            hass,
            "zwave",
            {"zwave": {"device_config": {"zwave.mock_node": {"ignored": True}}}},
        )
        await hass.async_block_till_done()

    assert len(mock_receivers) == 1

    node = MockNode(node_id=14)
    await hass.async_add_executor_job(mock_receivers[0], node)
    await hass.async_block_till_done()

    assert hass.states.get("zwave.mock_node") is None


async def test_value_discovery(hass, mock_openzwave):
    """Test discovery of a node."""
    mock_receivers = []

    def mock_connect(receiver, signal, *args, **kwargs):
        if signal == MockNetwork.SIGNAL_VALUE_ADDED:
            mock_receivers.append(receiver)

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(hass, "zwave", {"zwave": {}})
        await hass.async_block_till_done()

    assert len(mock_receivers) == 1

    node = MockNode(node_id=11, generic=const.GENERIC_TYPE_SENSOR_BINARY)
    value = MockValue(
        data=False,
        node=node,
        index=12,
        instance=13,
        command_class=const.COMMAND_CLASS_SENSOR_BINARY,
        type=const.TYPE_BOOL,
        genre=const.GENRE_USER,
    )
    await hass.async_add_executor_job(mock_receivers[0], node, value)
    await hass.async_block_till_done()

    assert hass.states.get("binary_sensor.mock_node_mock_value").state == "off"


async def test_value_entities(hass, mock_openzwave):
    """Test discovery of a node."""
    mock_receivers = {}

    def mock_connect(receiver, signal, *args, **kwargs):
        mock_receivers[signal] = receiver

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(hass, "zwave", {"zwave": {}})
        await hass.async_block_till_done()

    zwave_network = hass.data[DATA_NETWORK]
    zwave_network.state = MockNetwork.STATE_READY

    await hass.async_start()

    assert mock_receivers

    await hass.async_add_executor_job(
        mock_receivers[MockNetwork.SIGNAL_ALL_NODES_QUERIED]
    )
    node = MockNode(node_id=11, generic=const.GENERIC_TYPE_SENSOR_BINARY)
    zwave_network.nodes = {node.node_id: node}
    value = MockValue(
        data=False,
        node=node,
        index=12,
        instance=1,
        command_class=const.COMMAND_CLASS_SENSOR_BINARY,
        type=const.TYPE_BOOL,
        genre=const.GENRE_USER,
    )
    node.values = {"primary": value, value.value_id: value}
    value2 = MockValue(
        data=False,
        node=node,
        index=12,
        instance=2,
        label="Mock Value B",
        command_class=const.COMMAND_CLASS_SENSOR_BINARY,
        type=const.TYPE_BOOL,
        genre=const.GENRE_USER,
    )
    node.values[value2.value_id] = value2

    await hass.async_add_executor_job(
        mock_receivers[MockNetwork.SIGNAL_NODE_ADDED], node
    )
    await hass.async_add_executor_job(
        mock_receivers[MockNetwork.SIGNAL_VALUE_ADDED], node, value
    )
    await hass.async_add_executor_job(
        mock_receivers[MockNetwork.SIGNAL_VALUE_ADDED], node, value2
    )
    await hass.async_block_till_done()

    assert hass.states.get("binary_sensor.mock_node_mock_value").state == "off"
    assert hass.states.get("binary_sensor.mock_node_mock_value_b").state == "off"

    ent_reg = er.async_get(hass)
    dev_reg = dr.async_get(hass)

    entry = ent_reg.async_get("zwave.mock_node")
    assert entry is not None
    assert entry.unique_id == f"node-{node.node_id}"
    node_dev_id = entry.device_id

    entry = ent_reg.async_get("binary_sensor.mock_node_mock_value")
    assert entry is not None
    assert entry.unique_id == f"{node.node_id}-{value.object_id}"
    assert entry.name is None
    assert entry.device_id == node_dev_id

    entry = ent_reg.async_get("binary_sensor.mock_node_mock_value_b")
    assert entry is not None
    assert entry.unique_id == f"{node.node_id}-{value2.object_id}"
    assert entry.name is None
    assert entry.device_id != node_dev_id
    device_id_b = entry.device_id

    device = dev_reg.async_get(node_dev_id)
    assert device is not None
    assert device.name == node.name
    old_device = device

    device = dev_reg.async_get(device_id_b)
    assert device is not None
    assert device.name == f"{node.name} ({value2.instance})"

    # test renaming without updating
    await hass.services.async_call(
        "zwave",
        "rename_node",
        {const.ATTR_NODE_ID: node.node_id, ATTR_NAME: "Demo Node"},
    )
    await hass.async_block_till_done()

    assert node.name == "Demo Node"

    entry = ent_reg.async_get("zwave.mock_node")
    assert entry is not None

    entry = ent_reg.async_get("binary_sensor.mock_node_mock_value")
    assert entry is not None

    entry = ent_reg.async_get("binary_sensor.mock_node_mock_value_b")
    assert entry is not None

    device = dev_reg.async_get(node_dev_id)
    assert device is not None
    assert device.id == old_device.id
    assert device.name == node.name

    device = dev_reg.async_get(device_id_b)
    assert device is not None
    assert device.name == f"{node.name} ({value2.instance})"

    # test renaming
    await hass.services.async_call(
        "zwave",
        "rename_node",
        {
            const.ATTR_NODE_ID: node.node_id,
            const.ATTR_UPDATE_IDS: True,
            ATTR_NAME: "New Node",
        },
    )
    await hass.async_block_till_done()

    assert node.name == "New Node"

    entry = ent_reg.async_get("zwave.new_node")
    assert entry is not None
    assert entry.unique_id == f"node-{node.node_id}"

    entry = ent_reg.async_get("binary_sensor.new_node_mock_value")
    assert entry is not None
    assert entry.unique_id == f"{node.node_id}-{value.object_id}"

    device = dev_reg.async_get(node_dev_id)
    assert device is not None
    assert device.id == old_device.id
    assert device.name == node.name

    device = dev_reg.async_get(device_id_b)
    assert device is not None
    assert device.name == f"{node.name} ({value2.instance})"

    await hass.services.async_call(
        "zwave",
        "rename_value",
        {
            const.ATTR_NODE_ID: node.node_id,
            const.ATTR_VALUE_ID: value.object_id,
            const.ATTR_UPDATE_IDS: True,
            ATTR_NAME: "New Label",
        },
    )
    await hass.async_block_till_done()

    entry = ent_reg.async_get("binary_sensor.new_node_new_label")
    assert entry is not None
    assert entry.unique_id == f"{node.node_id}-{value.object_id}"


async def test_value_discovery_existing_entity(hass, mock_openzwave):
    """Test discovery of a node."""
    mock_receivers = []

    def mock_connect(receiver, signal, *args, **kwargs):
        if signal == MockNetwork.SIGNAL_VALUE_ADDED:
            mock_receivers.append(receiver)

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(hass, "zwave", {"zwave": {}})
        await hass.async_block_till_done()

    assert len(mock_receivers) == 1

    node = MockNode(
        node_id=11,
        generic=const.GENERIC_TYPE_THERMOSTAT,
        specific=const.SPECIFIC_TYPE_THERMOSTAT_GENERAL_V2,
    )
    thermostat_mode = MockValue(
        data="Heat",
        data_items=["Off", "Heat"],
        node=node,
        command_class=const.COMMAND_CLASS_THERMOSTAT_MODE,
        genre=const.GENRE_USER,
    )
    setpoint_heating = MockValue(
        data=22.0,
        node=node,
        command_class=const.COMMAND_CLASS_THERMOSTAT_SETPOINT,
        index=1,
        genre=const.GENRE_USER,
    )

    await hass.async_add_executor_job(mock_receivers[0], node, thermostat_mode)
    await hass.async_block_till_done()

    def mock_update(self):
        self.hass.add_job(self.async_update_ha_state)

    with patch.object(
        zwave.node_entity.ZWaveBaseEntity, "maybe_schedule_update", new=mock_update
    ):
        await hass.async_add_executor_job(mock_receivers[0], node, setpoint_heating)
        await hass.async_block_till_done()

    assert (
        hass.states.get("climate.mock_node_mock_value").attributes["temperature"]
        == 22.0
    )
    assert (
        hass.states.get("climate.mock_node_mock_value").attributes[
            "current_temperature"
        ]
        is None
    )

    with patch.object(
        zwave.node_entity.ZWaveBaseEntity, "maybe_schedule_update", new=mock_update
    ):
        temperature = MockValue(
            data=23.5,
            node=node,
            index=1,
            command_class=const.COMMAND_CLASS_SENSOR_MULTILEVEL,
            genre=const.GENRE_USER,
            units="C",
        )
        await hass.async_add_executor_job(mock_receivers[0], node, temperature)
        await hass.async_block_till_done()

    assert (
        hass.states.get("climate.mock_node_mock_value").attributes["temperature"]
        == 22.0
    )
    assert (
        hass.states.get("climate.mock_node_mock_value").attributes[
            "current_temperature"
        ]
        == 23.5
    )


async def test_value_discovery_legacy_thermostat(hass, mock_openzwave):
    """Test discovery of a node. Special case for legacy thermostats."""
    mock_receivers = []

    def mock_connect(receiver, signal, *args, **kwargs):
        if signal == MockNetwork.SIGNAL_VALUE_ADDED:
            mock_receivers.append(receiver)

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(hass, "zwave", {"zwave": {}})
        await hass.async_block_till_done()

    assert len(mock_receivers) == 1

    node = MockNode(
        node_id=11,
        generic=const.GENERIC_TYPE_THERMOSTAT,
        specific=const.SPECIFIC_TYPE_SETPOINT_THERMOSTAT,
    )
    setpoint_heating = MockValue(
        data=22.0,
        node=node,
        command_class=const.COMMAND_CLASS_THERMOSTAT_SETPOINT,
        index=1,
        genre=const.GENRE_USER,
    )

    await hass.async_add_executor_job(mock_receivers[0], node, setpoint_heating)
    await hass.async_block_till_done()

    assert (
        hass.states.get("climate.mock_node_mock_value").attributes["temperature"]
        == 22.0
    )


async def test_power_schemes(hass, mock_openzwave):
    """Test power attribute."""
    mock_receivers = []

    def mock_connect(receiver, signal, *args, **kwargs):
        if signal == MockNetwork.SIGNAL_VALUE_ADDED:
            mock_receivers.append(receiver)

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(hass, "zwave", {"zwave": {}})
        await hass.async_block_till_done()

    assert len(mock_receivers) == 1

    node = MockNode(node_id=11, generic=const.GENERIC_TYPE_SWITCH_BINARY)
    switch = MockValue(
        data=True,
        node=node,
        index=12,
        instance=13,
        command_class=const.COMMAND_CLASS_SWITCH_BINARY,
        genre=const.GENRE_USER,
        type=const.TYPE_BOOL,
    )
    await hass.async_add_executor_job(mock_receivers[0], node, switch)

    await hass.async_block_till_done()

    assert hass.states.get("switch.mock_node_mock_value").state == "on"
    assert (
        "power_consumption"
        not in hass.states.get("switch.mock_node_mock_value").attributes
    )

    def mock_update(self):
        self.hass.add_job(self.async_update_ha_state)

    with patch.object(
        zwave.node_entity.ZWaveBaseEntity, "maybe_schedule_update", new=mock_update
    ):
        power = MockValue(
            data=23.5,
            node=node,
            index=const.INDEX_SENSOR_MULTILEVEL_POWER,
            instance=13,
            command_class=const.COMMAND_CLASS_SENSOR_MULTILEVEL,
            genre=const.GENRE_USER,  # to avoid exception
        )
        await hass.async_add_executor_job(mock_receivers[0], node, power)
        await hass.async_block_till_done()

    assert (
        hass.states.get("switch.mock_node_mock_value").attributes["power_consumption"]
        == 23.5
    )


async def test_network_ready(hass, mock_openzwave):
    """Test Node network ready event."""
    mock_receivers = []

    def mock_connect(receiver, signal, *args, **kwargs):
        if signal == MockNetwork.SIGNAL_ALL_NODES_QUERIED:
            mock_receivers.append(receiver)

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(hass, "zwave", {"zwave": {}})
        await hass.async_block_till_done()

    assert len(mock_receivers) == 1

    events = []

    def listener(event):
        events.append(event)

    hass.bus.async_listen(const.EVENT_NETWORK_COMPLETE, listener)

    await hass.async_add_executor_job(mock_receivers[0])
    await hass.async_block_till_done()

    assert len(events) == 1


async def test_network_complete(hass, mock_openzwave):
    """Test Node network complete event."""
    mock_receivers = []

    def mock_connect(receiver, signal, *args, **kwargs):
        if signal == MockNetwork.SIGNAL_AWAKE_NODES_QUERIED:
            mock_receivers.append(receiver)

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(hass, "zwave", {"zwave": {}})
        await hass.async_block_till_done()

    assert len(mock_receivers) == 1

    events = []

    def listener(event):
        events.append(event)

    hass.bus.async_listen(const.EVENT_NETWORK_READY, listener)

    await hass.async_add_executor_job(mock_receivers[0])
    await hass.async_block_till_done()

    assert len(events) == 1


async def test_network_complete_some_dead(hass, mock_openzwave):
    """Test Node network complete some dead event."""
    mock_receivers = []

    def mock_connect(receiver, signal, *args, **kwargs):
        if signal == MockNetwork.SIGNAL_ALL_NODES_QUERIED_SOME_DEAD:
            mock_receivers.append(receiver)

    with patch("pydispatch.dispatcher.connect", new=mock_connect):
        await async_setup_component(hass, "zwave", {"zwave": {}})
        await hass.async_block_till_done()

    assert len(mock_receivers) == 1

    events = []

    def listener(event):
        events.append(event)

    hass.bus.async_listen(const.EVENT_NETWORK_COMPLETE_SOME_DEAD, listener)

    await hass.async_add_executor_job(mock_receivers[0])
    await hass.async_block_till_done()

    assert len(events) == 1


async def test_entity_discovery(
    hass, mock_discovery, mock_import_module, mock_values, mock_openzwave, zwave_setup
):
    """Test the creation of a new entity."""
    (node, value_class, mock_schema) = mock_values

    registry = mock_registry(hass)

    entity_id = "mock_component.mock_node_mock_value"
    zwave_config = {"zwave": {}}
    device_config = {entity_id: {}}

    with patch.object(zwave, "discovery", mock_discovery), patch.object(
        zwave, "import_module", mock_import_module
    ):
        values = zwave.ZWaveDeviceEntityValues(
            hass=hass,
            schema=mock_schema,
            primary_value=value_class.primary,
            zwave_config=zwave_config,
            device_config=device_config,
            registry=registry,
        )
        assert not mock_discovery.async_load_platform.called

    assert values.primary is value_class.primary
    assert len(list(values)) == 3
    assert sorted(values, key=lambda a: id(a)) == sorted(
        [value_class.primary, None, None], key=lambda a: id(a)
    )

    with patch.object(zwave, "discovery", mock_discovery), patch.object(
        zwave, "import_module", mock_import_module
    ):
        values.check_value(value_class.secondary)
        await hass.async_block_till_done()

        assert mock_discovery.async_load_platform.called
        assert len(mock_discovery.async_load_platform.mock_calls) == 1

        args = mock_discovery.async_load_platform.mock_calls[0][1]
        assert args[0] == hass
        assert args[1] == "mock_component"
        assert args[2] == "zwave"
        assert args[3] == {
            const.DISCOVERY_DEVICE: mock_import_module().get_device().unique_id
        }
        assert args[4] == zwave_config

    assert values.secondary is value_class.secondary
    assert len(list(values)) == 3
    assert sorted(values, key=lambda a: id(a)) == sorted(
        [value_class.primary, value_class.secondary, None], key=lambda a: id(a)
    )

    mock_discovery.async_load_platform.reset_mock()
    with patch.object(zwave, "discovery", mock_discovery), patch.object(
        zwave, "import_module", mock_import_module
    ):
        values.check_value(value_class.optional)
        values.check_value(value_class.duplicate_secondary)
        values.check_value(value_class.no_match_value)
        await hass.async_block_till_done()

        assert not mock_discovery.async_load_platform.called

    assert values.optional is value_class.optional
    assert len(list(values)) == 3
    assert sorted(values, key=lambda a: id(a)) == sorted(
        [value_class.primary, value_class.secondary, value_class.optional],
        key=lambda a: id(a),
    )

    assert values._entity.value_added.called
    assert len(values._entity.value_added.mock_calls) == 1
    assert values._entity.value_changed.called
    assert len(values._entity.value_changed.mock_calls) == 1


async def test_entity_existing_values(
    hass, mock_discovery, mock_import_module, mock_values, mock_openzwave, zwave_setup
):
    """Test the loading of already discovered values."""
    (node, value_class, mock_schema) = mock_values

    registry = mock_registry(hass)

    entity_id = "mock_component.mock_node_mock_value"
    zwave_config = {"zwave": {}}
    device_config = {entity_id: {}}

    node.values = {
        value_class.primary.value_id: value_class.primary,
        value_class.secondary.value_id: value_class.secondary,
        value_class.optional.value_id: value_class.optional,
        value_class.no_match_value.value_id: value_class.no_match_value,
    }

    with patch.object(zwave, "discovery", mock_discovery), patch.object(
        zwave, "import_module", mock_import_module
    ):
        values = zwave.ZWaveDeviceEntityValues(
            hass=hass,
            schema=mock_schema,
            primary_value=value_class.primary,
            zwave_config=zwave_config,
            device_config=device_config,
            registry=registry,
        )
        await hass.async_block_till_done()

        assert mock_discovery.async_load_platform.called
        assert len(mock_discovery.async_load_platform.mock_calls) == 1
        args = mock_discovery.async_load_platform.mock_calls[0][1]
        assert args[0] == hass
        assert args[1] == "mock_component"
        assert args[2] == "zwave"
        assert args[3] == {
            const.DISCOVERY_DEVICE: mock_import_module().get_device().unique_id
        }
        assert args[4] == zwave_config
        assert not value_class.primary.enable_poll.called

    assert values.primary is value_class.primary
    assert values.secondary is value_class.secondary
    assert values.optional is value_class.optional
    assert len(list(values)) == 3
    assert sorted(values, key=lambda a: id(a)) == sorted(
        [value_class.primary, value_class.secondary, value_class.optional],
        key=lambda a: id(a),
    )


async def test_node_schema_mismatch(
    hass, mock_discovery, mock_import_module, mock_values, mock_openzwave, zwave_setup
):
    """Test node schema mismatch."""
    (node, value_class, mock_schema) = mock_values

    registry = mock_registry(hass)

    entity_id = "mock_component.mock_node_mock_value"
    zwave_config = {"zwave": {}}
    device_config = {entity_id: {}}

    node.generic = "no_match"
    node.values = {
        value_class.primary.value_id: value_class.primary,
        value_class.secondary.value_id: value_class.secondary,
    }
    mock_schema[const.DISC_GENERIC_DEVICE_CLASS] = ["generic_match"]

    with patch.object(zwave, "discovery", mock_discovery), patch.object(
        zwave, "import_module", mock_import_module
    ):
        values = zwave.ZWaveDeviceEntityValues(
            hass=hass,
            schema=mock_schema,
            primary_value=value_class.primary,
            zwave_config=zwave_config,
            device_config=device_config,
            registry=registry,
        )
        values._check_entity_ready()
        await hass.async_block_till_done()

        assert not mock_discovery.async_load_platform.called


async def test_entity_workaround_component(
    hass, mock_discovery, mock_import_module, mock_values, mock_openzwave, zwave_setup
):
    """Test component workaround."""
    (node, value_class, mock_schema) = mock_values

    registry = mock_registry(hass)

    node.manufacturer_id = "010f"
    node.product_type = "0b00"
    value_class.primary.command_class = const.COMMAND_CLASS_SENSOR_ALARM

    entity_id = "binary_sensor.mock_node_mock_value"
    zwave_config = {"zwave": {}}
    device_config = {entity_id: {}}

    mock_schema = {
        const.DISC_COMPONENT: "mock_component",
        const.DISC_VALUES: {
            const.DISC_PRIMARY: {
                const.DISC_COMMAND_CLASS: [const.COMMAND_CLASS_SWITCH_BINARY]
            }
        },
    }

    with patch.object(
        zwave, "async_dispatcher_send"
    ) as mock_dispatch_send, patch.object(
        zwave, "discovery", mock_discovery
    ), patch.object(
        zwave, "import_module", mock_import_module
    ):

        values = zwave.ZWaveDeviceEntityValues(
            hass=hass,
            schema=mock_schema,
            primary_value=value_class.primary,
            zwave_config=zwave_config,
            device_config=device_config,
            registry=registry,
        )
        values._check_entity_ready()
        await hass.async_block_till_done()

        assert mock_dispatch_send.called
        assert len(mock_dispatch_send.mock_calls) == 1
        args = mock_dispatch_send.mock_calls[0][1]
        assert args[1] == "zwave_new_binary_sensor"


async def test_entity_workaround_ignore(
    hass, mock_discovery, mock_import_module, mock_values, mock_openzwave, zwave_setup
):
    """Test ignore workaround."""
    (node, value_class, mock_schema) = mock_values

    registry = mock_registry(hass)

    entity_id = "mock_component.mock_node_mock_value"
    zwave_config = {"zwave": {}}
    device_config = {entity_id: {}}

    node.manufacturer_id = "010f"
    node.product_type = "0301"
    value_class.primary.command_class = const.COMMAND_CLASS_SWITCH_BINARY

    mock_schema = {
        const.DISC_COMPONENT: "mock_component",
        const.DISC_VALUES: {
            const.DISC_PRIMARY: {
                const.DISC_COMMAND_CLASS: [const.COMMAND_CLASS_SWITCH_BINARY]
            }
        },
    }

    with patch.object(zwave, "discovery", mock_discovery), patch.object(
        zwave, "import_module", mock_import_module
    ):
        values = zwave.ZWaveDeviceEntityValues(
            hass=hass,
            schema=mock_schema,
            primary_value=value_class.primary,
            zwave_config=zwave_config,
            device_config=device_config,
            registry=registry,
        )
        values._check_entity_ready()
        await hass.async_block_till_done()

        assert not mock_discovery.async_load_platform.called


async def test_entity_config_ignore(
    hass, mock_discovery, mock_import_module, mock_values, mock_openzwave, zwave_setup
):
    """Test ignore config."""
    (node, value_class, mock_schema) = mock_values

    registry = mock_registry(hass)

    entity_id = "mock_component.mock_node_mock_value"
    zwave_config = {"zwave": {}}
    device_config = {entity_id: {}}

    node.values = {
        value_class.primary.value_id: value_class.primary,
        value_class.secondary.value_id: value_class.secondary,
    }
    device_config = {entity_id: {zwave.CONF_IGNORED: True}}

    with patch.object(zwave, "discovery", mock_discovery), patch.object(
        zwave, "import_module", mock_import_module
    ):
        values = zwave.ZWaveDeviceEntityValues(
            hass=hass,
            schema=mock_schema,
            primary_value=value_class.primary,
            zwave_config=zwave_config,
            device_config=device_config,
            registry=registry,
        )
        values._check_entity_ready()
        await hass.async_block_till_done()

        assert not mock_discovery.async_load_platform.called


async def test_entity_config_ignore_with_registry(
    hass, mock_discovery, mock_import_module, mock_values, mock_openzwave, zwave_setup
):
    """Test ignore config.

    The case when the device is in entity registry.
    """
    (node, value_class, mock_schema) = mock_values

    registry = mock_registry(hass)

    entity_id = "mock_component.mock_node_mock_value"
    zwave_config = {"zwave": {}}
    device_config = {entity_id: {}}

    node.values = {
        value_class.primary.value_id: value_class.primary,
        value_class.secondary.value_id: value_class.secondary,
    }
    device_config = {"mock_component.registry_id": {zwave.CONF_IGNORED: True}}
    with patch.object(registry, "async_schedule_save"):
        registry.async_get_or_create(
            "mock_component",
            zwave.DOMAIN,
            "567-1000",
            suggested_object_id="registry_id",
        )

    with patch.object(zwave, "discovery", mock_discovery), patch.object(
        zwave, "import_module", mock_import_module
    ):
        zwave.ZWaveDeviceEntityValues(
            hass=hass,
            schema=mock_schema,
            primary_value=value_class.primary,
            zwave_config=zwave_config,
            device_config=device_config,
            registry=registry,
        )
        await hass.async_block_till_done()

        assert not mock_discovery.async_load_platform.called


async def test_entity_platform_ignore(
    hass, mock_discovery, mock_import_module, mock_values, mock_openzwave, zwave_setup
):
    """Test platform ignore device."""
    (node, value_class, mock_schema) = mock_values

    registry = mock_registry(hass)

    entity_id = "mock_component.mock_node_mock_value"
    zwave_config = {"zwave": {}}
    device_config = {entity_id: {}}

    node.values = {
        value_class.primary.value_id: value_class.primary,
        value_class.secondary.value_id: value_class.secondary,
    }

    import_module = MagicMock()
    platform = MagicMock()
    import_module.return_value = platform
    platform.get_device.return_value = None

    with patch.object(zwave, "discovery", mock_discovery), patch.object(
        zwave, "import_module", import_module
    ):
        zwave.ZWaveDeviceEntityValues(
            hass=hass,
            schema=mock_schema,
            primary_value=value_class.primary,
            zwave_config=zwave_config,
            device_config=device_config,
            registry=registry,
        )
        await hass.async_block_till_done()

        assert not mock_discovery.async_load_platform.called


async def test_config_polling_intensity(
    hass, mock_discovery, mock_import_module, mock_values, mock_openzwave, zwave_setup
):
    """Test polling intensity."""
    (node, value_class, mock_schema) = mock_values

    registry = mock_registry(hass)

    entity_id = "mock_component.mock_node_mock_value"
    zwave_config = {"zwave": {}}
    device_config = {entity_id: {}}

    node.values = {
        value_class.primary.value_id: value_class.primary,
        value_class.secondary.value_id: value_class.secondary,
    }
    device_config = {entity_id: {zwave.CONF_POLLING_INTENSITY: 123}}

    with patch.object(zwave, "discovery", mock_discovery), patch.object(
        zwave, "import_module", mock_import_module
    ):
        values = zwave.ZWaveDeviceEntityValues(
            hass=hass,
            schema=mock_schema,
            primary_value=value_class.primary,
            zwave_config=zwave_config,
            device_config=device_config,
            registry=registry,
        )
        values._check_entity_ready()
        await hass.async_block_till_done()

        assert mock_discovery.async_load_platform.called

    assert value_class.primary.enable_poll.called
    assert len(value_class.primary.enable_poll.mock_calls) == 1
    assert value_class.primary.enable_poll.mock_calls[0][1][0] == 123


async def test_device_config_glob_is_ordered():
    """Test that device_config_glob preserves order."""
    conf = CONFIG_SCHEMA({"zwave": {CONF_DEVICE_CONFIG_GLOB: OrderedDict()}})
    assert isinstance(conf["zwave"][CONF_DEVICE_CONFIG_GLOB], OrderedDict)


async def test_add_node(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave add_node service."""
    zwave_network = hass.data[DATA_NETWORK]

    await hass.services.async_call("zwave", "add_node", {})
    await hass.async_block_till_done()

    assert zwave_network.controller.add_node.called
    assert len(zwave_network.controller.add_node.mock_calls) == 1
    assert len(zwave_network.controller.add_node.mock_calls[0][1]) == 0


async def test_add_node_secure(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave add_node_secure service."""
    zwave_network = hass.data[DATA_NETWORK]

    await hass.services.async_call("zwave", "add_node_secure", {})
    await hass.async_block_till_done()

    assert zwave_network.controller.add_node.called
    assert len(zwave_network.controller.add_node.mock_calls) == 1
    assert zwave_network.controller.add_node.mock_calls[0][1][0] is True


async def test_remove_node(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave remove_node service."""
    zwave_network = hass.data[DATA_NETWORK]

    await hass.services.async_call("zwave", "remove_node", {})
    await hass.async_block_till_done()

    assert zwave_network.controller.remove_node.called
    assert len(zwave_network.controller.remove_node.mock_calls) == 1


async def test_cancel_command(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave cancel_command service."""
    zwave_network = hass.data[DATA_NETWORK]

    await hass.services.async_call("zwave", "cancel_command", {})
    await hass.async_block_till_done()

    assert zwave_network.controller.cancel_command.called
    assert len(zwave_network.controller.cancel_command.mock_calls) == 1


async def test_heal_network(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave heal_network service."""
    zwave_network = hass.data[DATA_NETWORK]

    await hass.services.async_call("zwave", "heal_network", {})
    await hass.async_block_till_done()

    assert zwave_network.heal.called
    assert len(zwave_network.heal.mock_calls) == 1


async def test_soft_reset(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave soft_reset service."""
    zwave_network = hass.data[DATA_NETWORK]

    await hass.services.async_call("zwave", "soft_reset", {})
    await hass.async_block_till_done()

    assert zwave_network.controller.soft_reset.called
    assert len(zwave_network.controller.soft_reset.mock_calls) == 1


async def test_test_network(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave test_network service."""
    zwave_network = hass.data[DATA_NETWORK]

    await hass.services.async_call("zwave", "test_network", {})
    await hass.async_block_till_done()

    assert zwave_network.test.called
    assert len(zwave_network.test.mock_calls) == 1


async def test_stop_network(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave stop_network service."""
    zwave_network = hass.data[DATA_NETWORK]

    with patch.object(hass.bus, "fire") as mock_fire:
        await hass.services.async_call("zwave", "stop_network", {})
        await hass.async_block_till_done()

        assert zwave_network.stop.called
        assert len(zwave_network.stop.mock_calls) == 1
        assert mock_fire.called
        assert len(mock_fire.mock_calls) == 1
        assert mock_fire.mock_calls[0][1][0] == const.EVENT_NETWORK_STOP


async def test_rename_node(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave rename_node service."""
    zwave_network = hass.data[DATA_NETWORK]

    zwave_network.nodes = {11: MagicMock()}
    await hass.services.async_call(
        "zwave",
        "rename_node",
        {const.ATTR_NODE_ID: 11, ATTR_NAME: "test_name"},
    )
    await hass.async_block_till_done()

    assert zwave_network.nodes[11].name == "test_name"


async def test_rename_value(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave rename_value service."""
    zwave_network = hass.data[DATA_NETWORK]

    node = MockNode(node_id=14)
    value = MockValue(index=12, value_id=123456, label="Old Label")
    node.values = {123456: value}
    zwave_network.nodes = {11: node}

    assert value.label == "Old Label"
    await hass.services.async_call(
        "zwave",
        "rename_value",
        {
            const.ATTR_NODE_ID: 11,
            const.ATTR_VALUE_ID: 123456,
            ATTR_NAME: "New Label",
        },
    )
    await hass.async_block_till_done()

    assert value.label == "New Label"


async def test_set_poll_intensity_enable(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave set_poll_intensity service, successful set."""
    zwave_network = hass.data[DATA_NETWORK]

    node = MockNode(node_id=14)
    value = MockValue(index=12, value_id=123456, poll_intensity=0)
    node.values = {123456: value}
    zwave_network.nodes = {11: node}

    assert value.poll_intensity == 0
    await hass.services.async_call(
        "zwave",
        "set_poll_intensity",
        {
            const.ATTR_NODE_ID: 11,
            const.ATTR_VALUE_ID: 123456,
            const.ATTR_POLL_INTENSITY: 4,
        },
    )
    await hass.async_block_till_done()

    enable_poll = value.enable_poll
    assert value.enable_poll.called
    assert len(enable_poll.mock_calls) == 2
    assert enable_poll.mock_calls[0][1][0] == 4


async def test_set_poll_intensity_enable_failed(
    hass, mock_openzwave, zwave_setup_ready
):
    """Test zwave set_poll_intensity service, failed set."""
    zwave_network = hass.data[DATA_NETWORK]

    node = MockNode(node_id=14)
    value = MockValue(index=12, value_id=123456, poll_intensity=0)
    value.enable_poll.return_value = False
    node.values = {123456: value}
    zwave_network.nodes = {11: node}

    assert value.poll_intensity == 0
    await hass.services.async_call(
        "zwave",
        "set_poll_intensity",
        {
            const.ATTR_NODE_ID: 11,
            const.ATTR_VALUE_ID: 123456,
            const.ATTR_POLL_INTENSITY: 4,
        },
    )
    await hass.async_block_till_done()

    enable_poll = value.enable_poll
    assert value.enable_poll.called
    assert len(enable_poll.mock_calls) == 1


async def test_set_poll_intensity_disable(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave set_poll_intensity service, successful disable."""
    zwave_network = hass.data[DATA_NETWORK]

    node = MockNode(node_id=14)
    value = MockValue(index=12, value_id=123456, poll_intensity=4)
    node.values = {123456: value}
    zwave_network.nodes = {11: node}

    assert value.poll_intensity == 4
    await hass.services.async_call(
        "zwave",
        "set_poll_intensity",
        {
            const.ATTR_NODE_ID: 11,
            const.ATTR_VALUE_ID: 123456,
            const.ATTR_POLL_INTENSITY: 0,
        },
    )
    await hass.async_block_till_done()

    disable_poll = value.disable_poll
    assert value.disable_poll.called
    assert len(disable_poll.mock_calls) == 2


async def test_set_poll_intensity_disable_failed(
    hass, mock_openzwave, zwave_setup_ready
):
    """Test zwave set_poll_intensity service, failed disable."""
    zwave_network = hass.data[DATA_NETWORK]

    node = MockNode(node_id=14)
    value = MockValue(index=12, value_id=123456, poll_intensity=4)
    value.disable_poll.return_value = False
    node.values = {123456: value}
    zwave_network.nodes = {11: node}

    assert value.poll_intensity == 4
    await hass.services.async_call(
        "zwave",
        "set_poll_intensity",
        {
            const.ATTR_NODE_ID: 11,
            const.ATTR_VALUE_ID: 123456,
            const.ATTR_POLL_INTENSITY: 0,
        },
    )
    await hass.async_block_till_done()

    disable_poll = value.disable_poll
    assert value.disable_poll.called
    assert len(disable_poll.mock_calls) == 1


async def test_remove_failed_node(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave remove_failed_node service."""
    zwave_network = hass.data[DATA_NETWORK]

    await hass.services.async_call(
        "zwave", "remove_failed_node", {const.ATTR_NODE_ID: 12}
    )
    await hass.async_block_till_done()

    remove_failed_node = zwave_network.controller.remove_failed_node
    assert remove_failed_node.called
    assert len(remove_failed_node.mock_calls) == 1
    assert remove_failed_node.mock_calls[0][1][0] == 12


async def test_replace_failed_node(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave replace_failed_node service."""
    zwave_network = hass.data[DATA_NETWORK]

    await hass.services.async_call(
        "zwave", "replace_failed_node", {const.ATTR_NODE_ID: 13}
    )
    await hass.async_block_till_done()

    replace_failed_node = zwave_network.controller.replace_failed_node
    assert replace_failed_node.called
    assert len(replace_failed_node.mock_calls) == 1
    assert replace_failed_node.mock_calls[0][1][0] == 13


async def test_set_config_parameter(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave set_config_parameter service."""
    zwave_network = hass.data[DATA_NETWORK]

    value_byte = MockValue(
        index=12,
        command_class=const.COMMAND_CLASS_CONFIGURATION,
        type=const.TYPE_BYTE,
    )
    value_list = MockValue(
        index=13,
        command_class=const.COMMAND_CLASS_CONFIGURATION,
        type=const.TYPE_LIST,
        data_items=["item1", "item2", "item3"],
    )
    value_button = MockValue(
        index=14,
        command_class=const.COMMAND_CLASS_CONFIGURATION,
        type=const.TYPE_BUTTON,
    )
    value_list_int = MockValue(
        index=15,
        command_class=const.COMMAND_CLASS_CONFIGURATION,
        type=const.TYPE_LIST,
        data_items=["1", "2", "3"],
    )
    value_bool = MockValue(
        index=16,
        command_class=const.COMMAND_CLASS_CONFIGURATION,
        type=const.TYPE_BOOL,
    )
    node = MockNode(node_id=14)
    node.get_values.return_value = {
        12: value_byte,
        13: value_list,
        14: value_button,
        15: value_list_int,
        16: value_bool,
    }
    zwave_network.nodes = {14: node}

    # Byte
    await hass.services.async_call(
        "zwave",
        "set_config_parameter",
        {
            const.ATTR_NODE_ID: 14,
            const.ATTR_CONFIG_PARAMETER: 12,
            const.ATTR_CONFIG_VALUE: 7,
        },
    )
    await hass.async_block_till_done()

    assert value_byte.data == 7

    # List
    await hass.services.async_call(
        "zwave",
        "set_config_parameter",
        {
            const.ATTR_NODE_ID: 14,
            const.ATTR_CONFIG_PARAMETER: 13,
            const.ATTR_CONFIG_VALUE: "item3",
        },
    )
    await hass.async_block_till_done()

    assert value_list.data == "item3"

    # Button
    await hass.services.async_call(
        "zwave",
        "set_config_parameter",
        {
            const.ATTR_NODE_ID: 14,
            const.ATTR_CONFIG_PARAMETER: 14,
            const.ATTR_CONFIG_VALUE: True,
        },
    )
    await hass.async_block_till_done()

    assert zwave_network.manager.pressButton.called
    assert zwave_network.manager.releaseButton.called

    # List of Ints
    await hass.services.async_call(
        "zwave",
        "set_config_parameter",
        {
            const.ATTR_NODE_ID: 14,
            const.ATTR_CONFIG_PARAMETER: 15,
            const.ATTR_CONFIG_VALUE: 3,
        },
    )
    await hass.async_block_till_done()

    assert value_list_int.data == "3"

    # Boolean Truthy
    await hass.services.async_call(
        "zwave",
        "set_config_parameter",
        {
            const.ATTR_NODE_ID: 14,
            const.ATTR_CONFIG_PARAMETER: 16,
            const.ATTR_CONFIG_VALUE: "True",
        },
    )
    await hass.async_block_till_done()

    assert value_bool.data == 1

    # Boolean Falsy
    await hass.services.async_call(
        "zwave",
        "set_config_parameter",
        {
            const.ATTR_NODE_ID: 14,
            const.ATTR_CONFIG_PARAMETER: 16,
            const.ATTR_CONFIG_VALUE: "False",
        },
    )
    await hass.async_block_till_done()

    assert value_bool.data == 0

    # Different Parameter Size
    await hass.services.async_call(
        "zwave",
        "set_config_parameter",
        {
            const.ATTR_NODE_ID: 14,
            const.ATTR_CONFIG_PARAMETER: 19,
            const.ATTR_CONFIG_VALUE: 0x01020304,
            const.ATTR_CONFIG_SIZE: 4,
        },
    )
    await hass.async_block_till_done()

    assert node.set_config_param.called
    assert len(node.set_config_param.mock_calls) == 1
    assert node.set_config_param.mock_calls[0][1][0] == 19
    assert node.set_config_param.mock_calls[0][1][1] == 0x01020304
    assert node.set_config_param.mock_calls[0][1][2] == 4
    node.set_config_param.reset_mock()


async def test_print_config_parameter(hass, mock_openzwave, zwave_setup_ready, caplog):
    """Test zwave print_config_parameter service."""
    zwave_network = hass.data[DATA_NETWORK]

    value1 = MockValue(
        index=12, command_class=const.COMMAND_CLASS_CONFIGURATION, data=1234
    )
    value2 = MockValue(
        index=13, command_class=const.COMMAND_CLASS_CONFIGURATION, data=2345
    )
    node = MockNode(node_id=14)
    node.values = {12: value1, 13: value2}
    zwave_network.nodes = {14: node}

    caplog.clear()

    await hass.services.async_call(
        "zwave",
        "print_config_parameter",
        {const.ATTR_NODE_ID: 14, const.ATTR_CONFIG_PARAMETER: 13},
    )
    await hass.async_block_till_done()

    assert "Config parameter 13 on Node 14: 2345" in caplog.text


async def test_print_node(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave print_node_parameter service."""
    zwave_network = hass.data[DATA_NETWORK]

    node = MockNode(node_id=14)

    zwave_network.nodes = {14: node}

    with patch.object(zwave, "_LOGGER") as mock_logger:
        await hass.services.async_call("zwave", "print_node", {const.ATTR_NODE_ID: 14})
        await hass.async_block_till_done()

        assert "FOUND NODE " in mock_logger.info.mock_calls[0][1][0]


async def test_set_wakeup(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave set_wakeup service."""
    zwave_network = hass.data[DATA_NETWORK]

    value = MockValue(index=12, command_class=const.COMMAND_CLASS_WAKE_UP)
    node = MockNode(node_id=14)
    node.values = {12: value}
    node.get_values.return_value = node.values
    zwave_network.nodes = {14: node}

    await hass.services.async_call(
        "zwave", "set_wakeup", {const.ATTR_NODE_ID: 14, const.ATTR_CONFIG_VALUE: 15}
    )
    await hass.async_block_till_done()

    assert value.data == 15

    node.can_wake_up_value = False
    await hass.services.async_call(
        "zwave", "set_wakeup", {const.ATTR_NODE_ID: 14, const.ATTR_CONFIG_VALUE: 20}
    )
    await hass.async_block_till_done()

    assert value.data == 15


async def test_reset_node_meters(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave reset_node_meters service."""
    zwave_network = hass.data[DATA_NETWORK]

    value = MockValue(
        instance=1, index=8, data=99.5, command_class=const.COMMAND_CLASS_METER
    )
    reset_value = MockValue(
        instance=1, index=33, command_class=const.COMMAND_CLASS_METER
    )
    node = MockNode(node_id=14)
    node.values = {8: value, 33: reset_value}
    node.get_values.return_value = node.values
    zwave_network.nodes = {14: node}

    await hass.services.async_call(
        "zwave",
        "reset_node_meters",
        {const.ATTR_NODE_ID: 14, const.ATTR_INSTANCE: 2},
    )
    await hass.async_block_till_done()

    assert not zwave_network.manager.pressButton.called
    assert not zwave_network.manager.releaseButton.called

    await hass.services.async_call(
        "zwave", "reset_node_meters", {const.ATTR_NODE_ID: 14}
    )
    await hass.async_block_till_done()

    assert zwave_network.manager.pressButton.called
    (value_id,) = zwave_network.manager.pressButton.mock_calls.pop(0)[1]
    assert value_id == reset_value.value_id
    assert zwave_network.manager.releaseButton.called
    (value_id,) = zwave_network.manager.releaseButton.mock_calls.pop(0)[1]
    assert value_id == reset_value.value_id


async def test_add_association(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave change_association service."""
    zwave_network = hass.data[DATA_NETWORK]

    ZWaveGroup = mock_openzwave.group.ZWaveGroup
    group = MagicMock()
    ZWaveGroup.return_value = group

    value = MockValue(index=12, command_class=const.COMMAND_CLASS_WAKE_UP)
    node = MockNode(node_id=14)
    node.values = {12: value}
    node.get_values.return_value = node.values
    zwave_network.nodes = {14: node}

    await hass.services.async_call(
        "zwave",
        "change_association",
        {
            const.ATTR_ASSOCIATION: "add",
            const.ATTR_NODE_ID: 14,
            const.ATTR_TARGET_NODE_ID: 24,
            const.ATTR_GROUP: 3,
            const.ATTR_INSTANCE: 5,
        },
    )
    await hass.async_block_till_done()

    assert ZWaveGroup.called
    assert len(ZWaveGroup.mock_calls) == 2
    assert ZWaveGroup.mock_calls[0][1][0] == 3
    assert ZWaveGroup.mock_calls[0][1][2] == 14
    assert group.add_association.called
    assert len(group.add_association.mock_calls) == 1
    assert group.add_association.mock_calls[0][1][0] == 24
    assert group.add_association.mock_calls[0][1][1] == 5


async def test_remove_association(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave change_association service."""
    zwave_network = hass.data[DATA_NETWORK]

    ZWaveGroup = mock_openzwave.group.ZWaveGroup
    group = MagicMock()
    ZWaveGroup.return_value = group

    value = MockValue(index=12, command_class=const.COMMAND_CLASS_WAKE_UP)
    node = MockNode(node_id=14)
    node.values = {12: value}
    node.get_values.return_value = node.values
    zwave_network.nodes = {14: node}

    await hass.services.async_call(
        "zwave",
        "change_association",
        {
            const.ATTR_ASSOCIATION: "remove",
            const.ATTR_NODE_ID: 14,
            const.ATTR_TARGET_NODE_ID: 24,
            const.ATTR_GROUP: 3,
            const.ATTR_INSTANCE: 5,
        },
    )
    await hass.async_block_till_done()

    assert ZWaveGroup.called
    assert len(ZWaveGroup.mock_calls) == 2
    assert ZWaveGroup.mock_calls[0][1][0] == 3
    assert ZWaveGroup.mock_calls[0][1][2] == 14
    assert group.remove_association.called
    assert len(group.remove_association.mock_calls) == 1
    assert group.remove_association.mock_calls[0][1][0] == 24
    assert group.remove_association.mock_calls[0][1][1] == 5


async def test_refresh_node(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave refresh_node service."""
    zwave_network = hass.data[DATA_NETWORK]

    node = MockNode(node_id=14)
    zwave_network.nodes = {14: node}
    await hass.services.async_call("zwave", "refresh_node", {const.ATTR_NODE_ID: 14})
    await hass.async_block_till_done()

    assert node.refresh_info.called
    assert len(node.refresh_info.mock_calls) == 1


async def test_set_node_value(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave set_node_value service."""
    zwave_network = hass.data[DATA_NETWORK]

    value = MockValue(index=12, command_class=const.COMMAND_CLASS_INDICATOR, data=4)
    node = MockNode(node_id=14, command_classes=[const.COMMAND_CLASS_INDICATOR])
    node.values = {12: value}
    node.get_values.return_value = node.values
    zwave_network.nodes = {14: node}

    await hass.services.async_call(
        "zwave",
        "set_node_value",
        {
            const.ATTR_NODE_ID: 14,
            const.ATTR_VALUE_ID: 12,
            const.ATTR_CONFIG_VALUE: 2,
        },
    )
    await hass.async_block_till_done()

    assert zwave_network.nodes[14].values[12].data == 2


async def test_set_node_value_with_long_id_and_text_value(
    hass, mock_openzwave, zwave_setup_ready
):
    """Test zwave set_node_value service."""
    zwave_network = hass.data[DATA_NETWORK]

    value = MockValue(
        index=87512398541236578,
        command_class=const.COMMAND_CLASS_SWITCH_COLOR,
        data="#ff0000",
    )
    node = MockNode(node_id=14, command_classes=[const.COMMAND_CLASS_SWITCH_COLOR])
    node.values = {87512398541236578: value}
    node.get_values.return_value = node.values
    zwave_network.nodes = {14: node}

    await hass.services.async_call(
        "zwave",
        "set_node_value",
        {
            const.ATTR_NODE_ID: 14,
            const.ATTR_VALUE_ID: "87512398541236578",
            const.ATTR_CONFIG_VALUE: "#00ff00",
        },
    )
    await hass.async_block_till_done()

    assert zwave_network.nodes[14].values[87512398541236578].data == "#00ff00"


async def test_refresh_node_value(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave refresh_node_value service."""
    zwave_network = hass.data[DATA_NETWORK]

    node = MockNode(
        node_id=14,
        command_classes=[const.COMMAND_CLASS_INDICATOR],
        network=zwave_network,
    )
    value = MockValue(
        node=node, index=12, command_class=const.COMMAND_CLASS_INDICATOR, data=2
    )
    value.refresh = MagicMock()

    node.values = {12: value}
    node.get_values.return_value = node.values
    zwave_network.nodes = {14: node}

    await hass.services.async_call(
        "zwave",
        "refresh_node_value",
        {const.ATTR_NODE_ID: 14, const.ATTR_VALUE_ID: 12},
    )
    await hass.async_block_till_done()

    assert value.refresh.called


async def test_heal_node(hass, mock_openzwave, zwave_setup_ready):
    """Test zwave heal_node service."""
    zwave_network = hass.data[DATA_NETWORK]

    node = MockNode(node_id=19)
    zwave_network.nodes = {19: node}
    await hass.services.async_call("zwave", "heal_node", {const.ATTR_NODE_ID: 19})
    await hass.async_block_till_done()

    assert node.heal.called
    assert len(node.heal.mock_calls) == 1


async def test_test_node(hass, mock_openzwave, zwave_setup_ready):
    """Test the zwave test_node service."""
    zwave_network = hass.data[DATA_NETWORK]

    node = MockNode(node_id=19)
    zwave_network.nodes = {19: node}
    await hass.services.async_call("zwave", "test_node", {const.ATTR_NODE_ID: 19})
    await hass.async_block_till_done()

    assert node.test.called
    assert len(node.test.mock_calls) == 1