2018-05-10 18:09:22 +00:00
|
|
|
"""Test the auth script to manage local users."""
|
2024-03-08 18:16:38 +00:00
|
|
|
|
2024-08-16 11:21:39 +00:00
|
|
|
import argparse
|
2024-05-28 16:37:38 +00:00
|
|
|
from asyncio import AbstractEventLoop
|
2024-07-01 09:51:51 +00:00
|
|
|
from collections.abc import Generator
|
2024-02-07 16:22:10 +00:00
|
|
|
import logging
|
2023-02-20 10:42:56 +00:00
|
|
|
from typing import Any
|
2021-01-01 21:31:56 +00:00
|
|
|
from unittest.mock import Mock, patch
|
|
|
|
|
2018-05-10 18:09:22 +00:00
|
|
|
import pytest
|
|
|
|
|
2018-07-13 09:43:08 +00:00
|
|
|
from homeassistant.auth.providers import homeassistant as hass_auth
|
2023-02-20 10:42:56 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
2019-12-09 15:52:24 +00:00
|
|
|
from homeassistant.scripts import auth as script_auth
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2018-07-13 13:31:20 +00:00
|
|
|
from tests.common import register_auth_provider
|
|
|
|
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2024-02-07 16:22:10 +00:00
|
|
|
@pytest.fixture(autouse=True)
|
2024-06-27 17:54:44 +00:00
|
|
|
def reset_log_level() -> Generator[None]:
|
2024-02-07 16:22:10 +00:00
|
|
|
"""Reset log level after each test case."""
|
|
|
|
logger = logging.getLogger("homeassistant.core")
|
|
|
|
orig_level = logger.level
|
|
|
|
yield
|
|
|
|
logger.setLevel(orig_level)
|
|
|
|
|
|
|
|
|
2018-06-29 04:02:45 +00:00
|
|
|
@pytest.fixture
|
2024-06-27 17:54:44 +00:00
|
|
|
def provider(hass: HomeAssistant) -> hass_auth.HassAuthProvider:
|
2018-07-13 13:31:20 +00:00
|
|
|
"""Home Assistant auth provider."""
|
2019-07-31 19:25:30 +00:00
|
|
|
provider = hass.loop.run_until_complete(
|
|
|
|
register_auth_provider(hass, {"type": "homeassistant"})
|
|
|
|
)
|
2018-07-13 13:31:20 +00:00
|
|
|
hass.loop.run_until_complete(provider.async_initialize())
|
|
|
|
return provider
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2018-06-29 04:02:45 +00:00
|
|
|
|
2024-06-27 17:54:44 +00:00
|
|
|
async def test_list_user(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
provider: hass_auth.HassAuthProvider,
|
|
|
|
capsys: pytest.CaptureFixture[str],
|
|
|
|
) -> None:
|
2018-05-10 18:09:22 +00:00
|
|
|
"""Test we can list users."""
|
2018-07-13 13:31:20 +00:00
|
|
|
data = provider.data
|
2019-07-31 19:25:30 +00:00
|
|
|
data.add_auth("test-user", "test-pass")
|
|
|
|
data.add_auth("second-user", "second-pass")
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2018-07-13 13:31:20 +00:00
|
|
|
await script_auth.list_users(hass, provider, None)
|
2018-05-10 18:09:22 +00:00
|
|
|
|
|
|
|
captured = capsys.readouterr()
|
|
|
|
|
2024-04-12 07:04:16 +00:00
|
|
|
assert captured.out == "test-user\nsecond-user\n\nTotal users: 2\n"
|
2018-05-10 18:09:22 +00:00
|
|
|
|
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_add_user(
|
2024-06-27 17:54:44 +00:00
|
|
|
hass: HomeAssistant,
|
|
|
|
provider: hass_auth.HassAuthProvider,
|
|
|
|
capsys: pytest.CaptureFixture[str],
|
|
|
|
hass_storage: dict[str, Any],
|
2023-02-20 10:42:56 +00:00
|
|
|
) -> None:
|
2018-05-10 18:09:22 +00:00
|
|
|
"""Test we can add a user."""
|
2018-07-13 13:31:20 +00:00
|
|
|
data = provider.data
|
2018-06-29 04:02:45 +00:00
|
|
|
await script_auth.add_user(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass, provider, Mock(username="paulus", password="test-pass")
|
|
|
|
)
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert len(hass_storage[hass_auth.STORAGE_KEY]["data"]["users"]) == 1
|
2018-05-10 18:09:22 +00:00
|
|
|
|
|
|
|
captured = capsys.readouterr()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert captured.out == "Auth created\n"
|
2018-05-10 18:09:22 +00:00
|
|
|
|
|
|
|
assert len(data.users) == 1
|
2019-07-31 19:25:30 +00:00
|
|
|
data.validate_login("paulus", "test-pass")
|
2018-05-10 18:09:22 +00:00
|
|
|
|
|
|
|
|
2024-06-27 17:54:44 +00:00
|
|
|
async def test_validate_login(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
provider: hass_auth.HassAuthProvider,
|
|
|
|
capsys: pytest.CaptureFixture[str],
|
|
|
|
) -> None:
|
2018-05-10 18:09:22 +00:00
|
|
|
"""Test we can validate a user login."""
|
2018-07-13 13:31:20 +00:00
|
|
|
data = provider.data
|
2019-07-31 19:25:30 +00:00
|
|
|
data.add_auth("test-user", "test-pass")
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2018-06-29 04:02:45 +00:00
|
|
|
await script_auth.validate_login(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass, provider, Mock(username="test-user", password="test-pass")
|
|
|
|
)
|
2018-05-10 18:09:22 +00:00
|
|
|
captured = capsys.readouterr()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert captured.out == "Auth valid\n"
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2018-06-29 04:02:45 +00:00
|
|
|
await script_auth.validate_login(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass, provider, Mock(username="test-user", password="invalid-pass")
|
|
|
|
)
|
2018-05-10 18:09:22 +00:00
|
|
|
captured = capsys.readouterr()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert captured.out == "Auth invalid\n"
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2018-06-29 04:02:45 +00:00
|
|
|
await script_auth.validate_login(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass, provider, Mock(username="invalid-user", password="test-pass")
|
|
|
|
)
|
2018-05-10 18:09:22 +00:00
|
|
|
captured = capsys.readouterr()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert captured.out == "Auth invalid\n"
|
2018-05-10 18:09:22 +00:00
|
|
|
|
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_change_password(
|
2024-06-27 17:54:44 +00:00
|
|
|
hass: HomeAssistant,
|
|
|
|
provider: hass_auth.HassAuthProvider,
|
|
|
|
capsys: pytest.CaptureFixture[str],
|
|
|
|
hass_storage: dict[str, Any],
|
2023-02-20 10:42:56 +00:00
|
|
|
) -> None:
|
2018-05-10 18:09:22 +00:00
|
|
|
"""Test we can change a password."""
|
2018-07-13 13:31:20 +00:00
|
|
|
data = provider.data
|
2019-07-31 19:25:30 +00:00
|
|
|
data.add_auth("test-user", "test-pass")
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2018-06-29 04:02:45 +00:00
|
|
|
await script_auth.change_password(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass, provider, Mock(username="test-user", new_password="new-pass")
|
|
|
|
)
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert len(hass_storage[hass_auth.STORAGE_KEY]["data"]["users"]) == 1
|
2018-05-10 18:09:22 +00:00
|
|
|
captured = capsys.readouterr()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert captured.out == "Password changed\n"
|
|
|
|
data.validate_login("test-user", "new-pass")
|
2018-05-10 18:09:22 +00:00
|
|
|
with pytest.raises(hass_auth.InvalidAuth):
|
2019-07-31 19:25:30 +00:00
|
|
|
data.validate_login("test-user", "test-pass")
|
2018-05-10 18:09:22 +00:00
|
|
|
|
|
|
|
|
2023-02-20 10:42:56 +00:00
|
|
|
async def test_change_password_invalid_user(
|
2024-06-27 17:54:44 +00:00
|
|
|
hass: HomeAssistant,
|
|
|
|
provider: hass_auth.HassAuthProvider,
|
|
|
|
capsys: pytest.CaptureFixture[str],
|
|
|
|
hass_storage: dict[str, Any],
|
2023-02-20 10:42:56 +00:00
|
|
|
) -> None:
|
2018-05-10 18:09:22 +00:00
|
|
|
"""Test changing password of non-existing user."""
|
2018-07-13 13:31:20 +00:00
|
|
|
data = provider.data
|
2019-07-31 19:25:30 +00:00
|
|
|
data.add_auth("test-user", "test-pass")
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2018-06-29 04:02:45 +00:00
|
|
|
await script_auth.change_password(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass, provider, Mock(username="invalid-user", new_password="new-pass")
|
|
|
|
)
|
2018-05-10 18:09:22 +00:00
|
|
|
|
2018-06-29 04:02:45 +00:00
|
|
|
assert hass_auth.STORAGE_KEY not in hass_storage
|
2018-05-10 18:09:22 +00:00
|
|
|
captured = capsys.readouterr()
|
2019-07-31 19:25:30 +00:00
|
|
|
assert captured.out == "User not found\n"
|
|
|
|
data.validate_login("test-user", "test-pass")
|
2018-05-10 18:09:22 +00:00
|
|
|
with pytest.raises(hass_auth.InvalidAuth):
|
2019-07-31 19:25:30 +00:00
|
|
|
data.validate_login("invalid-user", "new-pass")
|
2018-06-29 04:02:45 +00:00
|
|
|
|
|
|
|
|
2024-05-28 16:37:38 +00:00
|
|
|
def test_parsing_args(event_loop: AbstractEventLoop) -> None:
|
2018-06-29 04:02:45 +00:00
|
|
|
"""Test we parse args correctly."""
|
|
|
|
called = False
|
|
|
|
|
2024-08-16 11:21:39 +00:00
|
|
|
async def mock_func(
|
|
|
|
hass: HomeAssistant, provider: hass_auth.AuthProvider, args2: argparse.Namespace
|
|
|
|
) -> None:
|
2018-06-29 04:02:45 +00:00
|
|
|
"""Mock function to be called."""
|
|
|
|
nonlocal called
|
|
|
|
called = True
|
2019-07-31 19:25:30 +00:00
|
|
|
assert provider.hass.config.config_dir == "/somewhere/config"
|
2018-06-29 04:02:45 +00:00
|
|
|
assert args2 is args
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
args = Mock(config="/somewhere/config", func=mock_func)
|
2018-06-29 04:02:45 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
with patch("argparse.ArgumentParser.parse_args", return_value=args):
|
2018-06-29 04:02:45 +00:00
|
|
|
script_auth.run(None)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
assert called, "Mock function did not get called"
|