2018-10-10 12:07:51 +00:00
|
|
|
"""Implement the auth feature from Hass.io for Add-ons."""
|
2019-02-14 15:01:46 +00:00
|
|
|
import logging
|
2018-10-10 12:07:51 +00:00
|
|
|
import os
|
2019-04-19 07:43:47 +00:00
|
|
|
from ipaddress import ip_address
|
2018-10-10 12:07:51 +00:00
|
|
|
|
2019-04-19 07:43:47 +00:00
|
|
|
import voluptuous as vol
|
2018-10-10 12:07:51 +00:00
|
|
|
from aiohttp import web
|
2018-10-10 23:02:00 +00:00
|
|
|
from aiohttp.web_exceptions import HTTPForbidden, HTTPNotFound
|
2018-10-10 12:07:51 +00:00
|
|
|
|
2019-04-19 07:43:47 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2018-10-10 12:07:51 +00:00
|
|
|
from homeassistant.components.http import HomeAssistantView
|
|
|
|
from homeassistant.components.http.const import KEY_REAL_IP
|
|
|
|
from homeassistant.components.http.data_validator import RequestDataValidator
|
2019-02-14 15:01:46 +00:00
|
|
|
from homeassistant.core import callback
|
|
|
|
from homeassistant.exceptions import HomeAssistantError
|
2019-04-19 07:43:47 +00:00
|
|
|
from homeassistant.helpers.typing import HomeAssistantType
|
2018-10-10 12:07:51 +00:00
|
|
|
|
2019-02-14 15:01:46 +00:00
|
|
|
from .const import ATTR_ADDON, ATTR_PASSWORD, ATTR_USERNAME
|
2018-10-10 12:07:51 +00:00
|
|
|
|
2018-10-11 08:55:38 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2018-10-10 12:07:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
SCHEMA_API_AUTH = vol.Schema({
|
|
|
|
vol.Required(ATTR_USERNAME): cv.string,
|
|
|
|
vol.Required(ATTR_PASSWORD): cv.string,
|
2018-10-11 08:55:38 +00:00
|
|
|
vol.Required(ATTR_ADDON): cv.string,
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
2018-10-10 12:07:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
@callback
|
2019-04-19 07:43:47 +00:00
|
|
|
def async_setup_auth_view(hass: HomeAssistantType):
|
2018-10-10 12:07:51 +00:00
|
|
|
"""Auth setup."""
|
|
|
|
hassio_auth = HassIOAuth(hass)
|
|
|
|
hass.http.register_view(hassio_auth)
|
|
|
|
|
|
|
|
|
|
|
|
class HassIOAuth(HomeAssistantView):
|
|
|
|
"""Hass.io view to handle base part."""
|
|
|
|
|
|
|
|
name = "api:hassio_auth"
|
|
|
|
url = "/api/hassio_auth"
|
|
|
|
|
|
|
|
def __init__(self, hass):
|
|
|
|
"""Initialize WebView."""
|
|
|
|
self.hass = hass
|
|
|
|
|
|
|
|
@RequestDataValidator(SCHEMA_API_AUTH)
|
|
|
|
async def post(self, request, data):
|
|
|
|
"""Handle new discovery requests."""
|
|
|
|
hassio_ip = os.environ['HASSIO'].split(':')[0]
|
|
|
|
if request[KEY_REAL_IP] != ip_address(hassio_ip):
|
|
|
|
_LOGGER.error(
|
|
|
|
"Invalid auth request from %s", request[KEY_REAL_IP])
|
|
|
|
raise HTTPForbidden()
|
|
|
|
|
|
|
|
await self._check_login(data[ATTR_USERNAME], data[ATTR_PASSWORD])
|
|
|
|
return web.Response(status=200)
|
|
|
|
|
|
|
|
def _get_provider(self):
|
|
|
|
"""Return Homeassistant auth provider."""
|
2019-03-11 02:55:36 +00:00
|
|
|
prv = self.hass.auth.get_auth_provider('homeassistant', None)
|
|
|
|
if prv is not None:
|
|
|
|
return prv
|
2018-10-10 12:07:51 +00:00
|
|
|
|
|
|
|
_LOGGER.error("Can't find Home Assistant auth.")
|
|
|
|
raise HTTPNotFound()
|
|
|
|
|
|
|
|
async def _check_login(self, username, password):
|
|
|
|
"""Check User credentials."""
|
|
|
|
provider = self._get_provider()
|
|
|
|
|
|
|
|
try:
|
|
|
|
await provider.async_validate_login(username, password)
|
|
|
|
except HomeAssistantError:
|
2018-10-10 23:02:00 +00:00
|
|
|
raise HTTPForbidden() from None
|