core/tests/components/shell_command/test_init.py

181 lines
6.2 KiB
Python
Raw Normal View History

2016-03-09 09:25:50 +00:00
"""The tests for the Shell command component."""
import asyncio
2015-10-12 04:30:17 +00:00
import os
import tempfile
from typing import Tuple
import unittest
from unittest.mock import Mock, patch
2015-10-12 04:30:17 +00:00
from homeassistant.components import shell_command
from homeassistant.setup import setup_component
2015-10-12 04:30:17 +00:00
2016-02-14 23:08:23 +00:00
from tests.common import get_test_home_assistant
2015-10-12 04:30:17 +00:00
@asyncio.coroutine
def mock_process_creator(error: bool = False) -> asyncio.coroutine:
"""Mock a coroutine that creates a process when yielded."""
2019-07-31 19:25:30 +00:00
@asyncio.coroutine
def communicate() -> Tuple[bytes, bytes]:
"""Mock a coroutine that runs a process when yielded.
Returns a tuple of (stdout, stderr).
"""
return b"I am stdout", b"I am stderr"
mock_process = Mock()
mock_process.communicate = communicate
mock_process.returncode = int(error)
return mock_process
2015-10-12 04:30:17 +00:00
class TestShellCommand(unittest.TestCase):
"""Test the shell_command component."""
2015-10-12 04:30:17 +00:00
def setUp(self): # pylint: disable=invalid-name
2018-08-19 20:29:08 +00:00
"""Set up things to be run when tests are started.
Also seems to require a child watcher attached to the loop when run
from pytest.
"""
2016-02-14 23:08:23 +00:00
self.hass = get_test_home_assistant()
asyncio.get_child_watcher().attach_loop(self.hass.loop)
2015-10-12 04:30:17 +00:00
def tearDown(self): # pylint: disable=invalid-name
2016-03-09 09:25:50 +00:00
"""Stop everything that was started."""
2015-10-12 04:30:17 +00:00
self.hass.stop()
def test_executing_service(self):
2016-03-09 09:25:50 +00:00
"""Test if able to call a configured service."""
2015-10-12 04:30:17 +00:00
with tempfile.TemporaryDirectory() as tempdirname:
2019-07-31 19:25:30 +00:00
path = os.path.join(tempdirname, "called.txt")
assert setup_component(
2019-07-31 19:25:30 +00:00
self.hass,
shell_command.DOMAIN,
{shell_command.DOMAIN: {"test_service": "date > {}".format(path)}},
)
self.hass.services.call("shell_command", "test_service", blocking=True)
self.hass.block_till_done()
assert os.path.isfile(path)
2015-10-12 04:41:44 +00:00
def test_config_not_dict(self):
"""Test that setup fails if config is not a dict."""
2019-07-31 19:25:30 +00:00
assert not setup_component(
self.hass,
shell_command.DOMAIN,
{shell_command.DOMAIN: ["some", "weird", "list"]},
)
2015-10-12 04:41:44 +00:00
def test_config_not_valid_service_names(self):
"""Test that setup fails if config contains invalid service names."""
2019-07-31 19:25:30 +00:00
assert not setup_component(
self.hass,
shell_command.DOMAIN,
{shell_command.DOMAIN: {"this is invalid because space": "touch bla.txt"}},
)
@patch(
"homeassistant.components.shell_command.asyncio.subprocess"
".create_subprocess_shell"
)
def test_template_render_no_template(self, mock_call):
"""Ensure shell_commands without templates get rendered properly."""
mock_call.return_value = mock_process_creator(error=False)
assert setup_component(
2019-07-31 19:25:30 +00:00
self.hass,
shell_command.DOMAIN,
{shell_command.DOMAIN: {"test_service": "ls /bin"}},
)
2019-07-31 19:25:30 +00:00
self.hass.services.call("shell_command", "test_service", blocking=True)
self.hass.block_till_done()
cmd = mock_call.mock_calls[0][1][0]
assert 1 == mock_call.call_count
2019-07-31 19:25:30 +00:00
assert "ls /bin" == cmd
2019-07-31 19:25:30 +00:00
@patch(
"homeassistant.components.shell_command.asyncio.subprocess"
".create_subprocess_exec"
)
def test_template_render(self, mock_call):
"""Ensure shell_commands with templates get rendered properly."""
2019-07-31 19:25:30 +00:00
self.hass.states.set("sensor.test_state", "Works")
mock_call.return_value = mock_process_creator(error=False)
2019-07-31 19:25:30 +00:00
assert setup_component(
self.hass,
shell_command.DOMAIN,
{
shell_command.DOMAIN: {
"test_service": ("ls /bin {{ states.sensor.test_state.state }}")
2019-07-31 19:25:30 +00:00
}
},
)
2019-07-31 19:25:30 +00:00
self.hass.services.call("shell_command", "test_service", blocking=True)
self.hass.block_till_done()
cmd = mock_call.mock_calls[0][1]
assert 1 == mock_call.call_count
2019-07-31 19:25:30 +00:00
assert ("ls", "/bin", "Works") == cmd
2019-07-31 19:25:30 +00:00
@patch(
"homeassistant.components.shell_command.asyncio.subprocess"
".create_subprocess_shell"
)
@patch("homeassistant.components.shell_command._LOGGER.error")
def test_subprocess_error(self, mock_error, mock_call):
"""Test subprocess that returns an error."""
mock_call.return_value = mock_process_creator(error=True)
2015-10-12 04:41:44 +00:00
with tempfile.TemporaryDirectory() as tempdirname:
2019-07-31 19:25:30 +00:00
path = os.path.join(tempdirname, "called.txt")
assert setup_component(
self.hass,
shell_command.DOMAIN,
{shell_command.DOMAIN: {"test_service": "touch {}".format(path)}},
)
2015-10-12 04:41:44 +00:00
2019-07-31 19:25:30 +00:00
self.hass.services.call("shell_command", "test_service", blocking=True)
2015-10-12 04:41:44 +00:00
self.hass.block_till_done()
assert 1 == mock_call.call_count
assert 1 == mock_error.call_count
assert not os.path.isfile(path)
2019-07-31 19:25:30 +00:00
@patch("homeassistant.components.shell_command._LOGGER.debug")
def test_stdout_captured(self, mock_output):
"""Test subprocess that has stdout."""
test_phrase = "I have output"
2019-07-31 19:25:30 +00:00
assert setup_component(
self.hass,
shell_command.DOMAIN,
{shell_command.DOMAIN: {"test_service": "echo {}".format(test_phrase)}},
)
2019-07-31 19:25:30 +00:00
self.hass.services.call("shell_command", "test_service", blocking=True)
self.hass.block_till_done()
assert 1 == mock_output.call_count
2019-07-31 19:25:30 +00:00
assert test_phrase.encode() + b"\n" == mock_output.call_args_list[0][0][-1]
2019-07-31 19:25:30 +00:00
@patch("homeassistant.components.shell_command._LOGGER.debug")
def test_stderr_captured(self, mock_output):
"""Test subprocess that has stderr."""
test_phrase = "I have error"
2019-07-31 19:25:30 +00:00
assert setup_component(
self.hass,
shell_command.DOMAIN,
{shell_command.DOMAIN: {"test_service": ">&2 echo {}".format(test_phrase)}},
)
2019-07-31 19:25:30 +00:00
self.hass.services.call("shell_command", "test_service", blocking=True)
self.hass.block_till_done()
assert 1 == mock_output.call_count
2019-07-31 19:25:30 +00:00
assert test_phrase.encode() + b"\n" == mock_output.call_args_list[0][0][-1]