"""The tests for the counter component."""
# pylint: disable=protected-access
import asyncio
import logging

from homeassistant.components.counter import (CONF_ICON, CONF_INITIAL,
                                              CONF_NAME, CONF_RESTORE,
                                              CONF_STEP, DOMAIN)
from homeassistant.const import (ATTR_FRIENDLY_NAME, ATTR_ICON)
from homeassistant.core import Context, CoreState, State
from homeassistant.setup import async_setup_component
from tests.common import (mock_restore_cache)
from tests.components.counter.common import (
    async_decrement, async_increment, async_reset)

_LOGGER = logging.getLogger(__name__)


async def test_config(hass):
    """Test config."""
    invalid_configs = [
        None,
        1,
        {},
        {'name with space': None},
    ]

    for cfg in invalid_configs:
        assert not await async_setup_component(hass, DOMAIN, {DOMAIN: cfg})


async def test_config_options(hass):
    """Test configuration options."""
    count_start = len(hass.states.async_entity_ids())

    _LOGGER.debug('ENTITIES @ start: %s', hass.states.async_entity_ids())

    config = {
        DOMAIN: {
            'test_1': {},
            'test_2': {
                CONF_NAME: 'Hello World',
                CONF_ICON: 'mdi:work',
                CONF_INITIAL: 10,
                CONF_RESTORE: False,
                CONF_STEP: 5,
            }
        }
    }

    assert await async_setup_component(hass, 'counter', config)
    await hass.async_block_till_done()

    _LOGGER.debug('ENTITIES: %s', hass.states.async_entity_ids())

    assert count_start + 2 == len(hass.states.async_entity_ids())
    await hass.async_block_till_done()

    state_1 = hass.states.get('counter.test_1')
    state_2 = hass.states.get('counter.test_2')

    assert state_1 is not None
    assert state_2 is not None

    assert 0 == int(state_1.state)
    assert ATTR_ICON not in state_1.attributes
    assert ATTR_FRIENDLY_NAME not in state_1.attributes

    assert 10 == int(state_2.state)
    assert 'Hello World' == \
        state_2.attributes.get(ATTR_FRIENDLY_NAME)
    assert 'mdi:work' == state_2.attributes.get(ATTR_ICON)


async def test_methods(hass):
    """Test increment, decrement, and reset methods."""
    config = {
        DOMAIN: {
            'test_1': {},
        }
    }

    assert await async_setup_component(hass, 'counter', config)

    entity_id = 'counter.test_1'

    state = hass.states.get(entity_id)
    assert 0 == int(state.state)

    async_increment(hass, entity_id)
    await hass.async_block_till_done()

    state = hass.states.get(entity_id)
    assert 1 == int(state.state)

    async_increment(hass, entity_id)
    await hass.async_block_till_done()

    state = hass.states.get(entity_id)
    assert 2 == int(state.state)

    async_decrement(hass, entity_id)
    await hass.async_block_till_done()

    state = hass.states.get(entity_id)
    assert 1 == int(state.state)

    async_reset(hass, entity_id)
    await hass.async_block_till_done()

    state = hass.states.get(entity_id)
    assert 0 == int(state.state)


async def test_methods_with_config(hass):
    """Test increment, decrement, and reset methods with configuration."""
    config = {
        DOMAIN: {
            'test': {
                CONF_NAME: 'Hello World',
                CONF_INITIAL: 10,
                CONF_STEP: 5,
            }
        }
    }

    assert await async_setup_component(hass, 'counter', config)

    entity_id = 'counter.test'

    state = hass.states.get(entity_id)
    assert 10 == int(state.state)

    async_increment(hass, entity_id)
    await hass.async_block_till_done()

    state = hass.states.get(entity_id)
    assert 15 == int(state.state)

    async_increment(hass, entity_id)
    await hass.async_block_till_done()

    state = hass.states.get(entity_id)
    assert 20 == int(state.state)

    async_decrement(hass, entity_id)
    await hass.async_block_till_done()

    state = hass.states.get(entity_id)
    assert 15 == int(state.state)


@asyncio.coroutine
def test_initial_state_overrules_restore_state(hass):
    """Ensure states are restored on startup."""
    mock_restore_cache(hass, (
        State('counter.test1', '11'),
        State('counter.test2', '-22'),
    ))

    hass.state = CoreState.starting

    yield from async_setup_component(hass, DOMAIN, {
        DOMAIN: {
            'test1': {
                CONF_RESTORE: False,
            },
            'test2': {
                CONF_INITIAL: 10,
                CONF_RESTORE: False,
            },
        }})

    state = hass.states.get('counter.test1')
    assert state
    assert int(state.state) == 0

    state = hass.states.get('counter.test2')
    assert state
    assert int(state.state) == 10


@asyncio.coroutine
def test_restore_state_overrules_initial_state(hass):
    """Ensure states are restored on startup."""
    mock_restore_cache(hass, (
        State('counter.test1', '11'),
        State('counter.test2', '-22'),
    ))

    hass.state = CoreState.starting

    yield from async_setup_component(hass, DOMAIN, {
        DOMAIN: {
            'test1': {},
            'test2': {
                CONF_INITIAL: 10,
            },
        }})

    state = hass.states.get('counter.test1')
    assert state
    assert int(state.state) == 11

    state = hass.states.get('counter.test2')
    assert state
    assert int(state.state) == -22


@asyncio.coroutine
def test_no_initial_state_and_no_restore_state(hass):
    """Ensure that entity is create without initial and restore feature."""
    hass.state = CoreState.starting

    yield from async_setup_component(hass, DOMAIN, {
        DOMAIN: {
            'test1': {
                CONF_STEP: 5,
            }
        }})

    state = hass.states.get('counter.test1')
    assert state
    assert int(state.state) == 0


async def test_counter_context(hass, hass_admin_user):
    """Test that counter context works."""
    assert await async_setup_component(hass, 'counter', {
        'counter': {
            'test': {}
        }
    })

    state = hass.states.get('counter.test')
    assert state is not None

    await hass.services.async_call('counter', 'increment', {
        'entity_id': state.entity_id,
    }, True, Context(user_id=hass_admin_user.id))

    state2 = hass.states.get('counter.test')
    assert state2 is not None
    assert state.state != state2.state
    assert state2.context.user_id == hass_admin_user.id


async def test_counter_min(hass, hass_admin_user):
    """Test that min works."""
    assert await async_setup_component(hass, 'counter', {
        'counter': {
            'test': {
                'minimum': '0',
                'initial': '0'
            }
        }
    })

    state = hass.states.get('counter.test')
    assert state is not None
    assert state.state == '0'

    await hass.services.async_call('counter', 'decrement', {
        'entity_id': state.entity_id,
    }, True, Context(user_id=hass_admin_user.id))

    state2 = hass.states.get('counter.test')
    assert state2 is not None
    assert state2.state == '0'

    await hass.services.async_call('counter', 'increment', {
        'entity_id': state.entity_id,
    }, True, Context(user_id=hass_admin_user.id))

    state2 = hass.states.get('counter.test')
    assert state2 is not None
    assert state2.state == '1'


async def test_counter_max(hass, hass_admin_user):
    """Test that max works."""
    assert await async_setup_component(hass, 'counter', {
        'counter': {
            'test': {
                'maximum': '0',
                'initial': '0'
            }
        }
    })

    state = hass.states.get('counter.test')
    assert state is not None
    assert state.state == '0'

    await hass.services.async_call('counter', 'increment', {
        'entity_id': state.entity_id,
    }, True, Context(user_id=hass_admin_user.id))

    state2 = hass.states.get('counter.test')
    assert state2 is not None
    assert state2.state == '0'

    await hass.services.async_call('counter', 'decrement', {
        'entity_id': state.entity_id,
    }, True, Context(user_id=hass_admin_user.id))

    state2 = hass.states.get('counter.test')
    assert state2 is not None
    assert state2.state == '-1'


async def test_configure(hass, hass_admin_user):
    """Test that setting values through configure works."""
    assert await async_setup_component(hass, 'counter', {
        'counter': {
            'test': {
                'maximum': '10',
                'initial': '10'
            }
        }
    })

    state = hass.states.get('counter.test')
    assert state is not None
    assert state.state == '10'
    assert 10 == state.attributes.get('maximum')

    # update max
    await hass.services.async_call('counter', 'configure', {
        'entity_id': state.entity_id,
        'maximum': 0,
    }, True, Context(user_id=hass_admin_user.id))

    state = hass.states.get('counter.test')
    assert state is not None
    assert state.state == '0'
    assert 0 == state.attributes.get('maximum')

    # disable max
    await hass.services.async_call('counter', 'configure', {
        'entity_id': state.entity_id,
        'maximum': None,
    }, True, Context(user_id=hass_admin_user.id))

    state = hass.states.get('counter.test')
    assert state is not None
    assert state.state == '0'
    assert state.attributes.get('maximum') is None

    # update min
    assert state.attributes.get('minimum') is None
    await hass.services.async_call('counter', 'configure', {
        'entity_id': state.entity_id,
        'minimum': 5,
    }, True, Context(user_id=hass_admin_user.id))

    state = hass.states.get('counter.test')
    assert state is not None
    assert state.state == '5'
    assert 5 == state.attributes.get('minimum')

    # disable min
    await hass.services.async_call('counter', 'configure', {
        'entity_id': state.entity_id,
        'minimum': None,
    }, True, Context(user_id=hass_admin_user.id))

    state = hass.states.get('counter.test')
    assert state is not None
    assert state.state == '5'
    assert state.attributes.get('minimum') is None

    # update step
    assert 1 == state.attributes.get('step')
    await hass.services.async_call('counter', 'configure', {
        'entity_id': state.entity_id,
        'step': 3,
    }, True, Context(user_id=hass_admin_user.id))

    state = hass.states.get('counter.test')
    assert state is not None
    assert state.state == '5'
    assert 3 == state.attributes.get('step')

    # update all
    await hass.services.async_call('counter', 'configure', {
        'entity_id': state.entity_id,
        'step': 5,
        'minimum': 0,
        'maximum': 9,
    }, True, Context(user_id=hass_admin_user.id))

    state = hass.states.get('counter.test')
    assert state is not None
    assert state.state == '5'
    assert 5 == state.attributes.get('step')
    assert 0 == state.attributes.get('minimum')
    assert 9 == state.attributes.get('maximum')