core/homeassistant/components/hassio/auth.py

75 lines
2.3 KiB
Python
Raw Normal View History

"""Implement the auth feature from Hass.io for Add-ons."""
import logging
from ipaddress import ip_address
import os
from aiohttp import web
from aiohttp.web_exceptions import HTTPForbidden, HTTPNotFound
import voluptuous as vol
from homeassistant.core import callback
import homeassistant.helpers.config_validation as cv
from homeassistant.exceptions import HomeAssistantError
from homeassistant.components.http import HomeAssistantView
from homeassistant.components.http.const import KEY_REAL_IP
from homeassistant.components.http.data_validator import RequestDataValidator
from .const import ATTR_USERNAME, ATTR_PASSWORD, ATTR_ADDON
_LOGGER = logging.getLogger(__name__)
SCHEMA_API_AUTH = vol.Schema({
vol.Required(ATTR_USERNAME): cv.string,
vol.Required(ATTR_PASSWORD): cv.string,
vol.Required(ATTR_ADDON): cv.string,
}, extra=vol.ALLOW_EXTRA)
@callback
def async_setup_auth(hass):
"""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."""
for prv in self.hass.auth.auth_providers:
if prv.type == 'homeassistant':
return prv
_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:
raise HTTPForbidden() from None