core/homeassistant/auth/auth_store.py

289 lines
9.3 KiB
Python
Raw Normal View History

2018-07-13 09:43:08 +00:00
"""Storage for auth models."""
from collections import OrderedDict
2018-07-13 09:43:08 +00:00
from datetime import timedelta
from logging import getLogger
from typing import Any, Dict, List, Optional # noqa: F401
import hmac
2018-07-13 09:43:08 +00:00
from homeassistant.core import HomeAssistant, callback
2018-07-13 09:43:08 +00:00
from homeassistant.util import dt as dt_util
from . import models
STORAGE_VERSION = 1
STORAGE_KEY = 'auth'
class AuthStore:
"""Stores authentication info.
Any mutation to an object should happen inside the auth store.
The auth store is lazy. It won't load the data from disk until a method is
called that needs it.
"""
def __init__(self, hass: HomeAssistant) -> None:
2018-07-13 09:43:08 +00:00
"""Initialize the auth store."""
self.hass = hass
self._users = None # type: Optional[Dict[str, models.User]]
2018-07-13 09:43:08 +00:00
self._store = hass.helpers.storage.Store(STORAGE_VERSION, STORAGE_KEY)
async def async_get_users(self) -> List[models.User]:
2018-07-13 09:43:08 +00:00
"""Retrieve all users."""
if self._users is None:
await self._async_load()
assert self._users is not None
2018-07-13 09:43:08 +00:00
return list(self._users.values())
async def async_get_user(self, user_id: str) -> Optional[models.User]:
2018-07-13 09:43:08 +00:00
"""Retrieve a user by id."""
if self._users is None:
await self._async_load()
assert self._users is not None
2018-07-13 09:43:08 +00:00
return self._users.get(user_id)
async def async_create_user(
self, name: Optional[str], is_owner: Optional[bool] = None,
is_active: Optional[bool] = None,
system_generated: Optional[bool] = None,
credentials: Optional[models.Credentials] = None) -> models.User:
2018-07-13 09:43:08 +00:00
"""Create a new user."""
if self._users is None:
await self._async_load()
assert self._users is not None
2018-07-13 09:43:08 +00:00
kwargs = {
'name': name
} # type: Dict[str, Any]
2018-07-13 09:43:08 +00:00
if is_owner is not None:
kwargs['is_owner'] = is_owner
if is_active is not None:
kwargs['is_active'] = is_active
if system_generated is not None:
kwargs['system_generated'] = system_generated
new_user = models.User(**kwargs)
self._users[new_user.id] = new_user
if credentials is None:
self._async_schedule_save()
2018-07-13 09:43:08 +00:00
return new_user
# Saving is done inside the link.
await self.async_link_user(new_user, credentials)
return new_user
async def async_link_user(self, user: models.User,
credentials: models.Credentials) -> None:
2018-07-13 09:43:08 +00:00
"""Add credentials to an existing user."""
user.credentials.append(credentials)
self._async_schedule_save()
2018-07-13 09:43:08 +00:00
credentials.is_new = False
async def async_remove_user(self, user: models.User) -> None:
2018-07-13 09:43:08 +00:00
"""Remove a user."""
if self._users is None:
await self._async_load()
assert self._users is not None
2018-07-13 09:43:08 +00:00
self._users.pop(user.id)
self._async_schedule_save()
2018-07-13 09:43:08 +00:00
async def async_activate_user(self, user: models.User) -> None:
"""Activate a user."""
user.is_active = True
self._async_schedule_save()
async def async_deactivate_user(self, user: models.User) -> None:
"""Activate a user."""
user.is_active = False
self._async_schedule_save()
async def async_remove_credentials(
self, credentials: models.Credentials) -> None:
"""Remove credentials."""
if self._users is None:
await self._async_load()
assert self._users is not None
for user in self._users.values():
found = None
for index, cred in enumerate(user.credentials):
if cred is credentials:
found = index
break
if found is not None:
user.credentials.pop(found)
break
self._async_schedule_save()
async def async_create_refresh_token(
self, user: models.User, client_id: Optional[str] = None) \
-> models.RefreshToken:
2018-07-13 09:43:08 +00:00
"""Create a new token for a user."""
refresh_token = models.RefreshToken(user=user, client_id=client_id)
user.refresh_tokens[refresh_token.id] = refresh_token
self._async_schedule_save()
2018-07-13 09:43:08 +00:00
return refresh_token
async def async_remove_refresh_token(
self, refresh_token: models.RefreshToken) -> None:
"""Remove a refresh token."""
if self._users is None:
await self._async_load()
assert self._users is not None
for user in self._users.values():
if user.refresh_tokens.pop(refresh_token.id, None):
self._async_schedule_save()
break
async def async_get_refresh_token(
self, token_id: str) -> Optional[models.RefreshToken]:
"""Get refresh token by id."""
2018-07-13 09:43:08 +00:00
if self._users is None:
await self._async_load()
assert self._users is not None
2018-07-13 09:43:08 +00:00
for user in self._users.values():
refresh_token = user.refresh_tokens.get(token_id)
2018-07-13 09:43:08 +00:00
if refresh_token is not None:
return refresh_token
return None
async def async_get_refresh_token_by_token(
self, token: str) -> Optional[models.RefreshToken]:
"""Get refresh token by token."""
if self._users is None:
await self._async_load()
assert self._users is not None
found = None
for user in self._users.values():
for refresh_token in user.refresh_tokens.values():
if hmac.compare_digest(refresh_token.token, token):
found = refresh_token
return found
async def _async_load(self) -> None:
2018-07-13 09:43:08 +00:00
"""Load the users."""
data = await self._store.async_load()
# Make sure that we're not overriding data if 2 loads happened at the
# same time
if self._users is not None:
return
users = OrderedDict() # type: Dict[str, models.User]
2018-07-13 09:43:08 +00:00
if data is None:
self._users = users
2018-07-13 09:43:08 +00:00
return
for user_dict in data['users']:
users[user_dict['id']] = models.User(**user_dict)
2018-07-13 09:43:08 +00:00
for cred_dict in data['credentials']:
users[cred_dict['user_id']].credentials.append(models.Credentials(
id=cred_dict['id'],
is_new=False,
auth_provider_type=cred_dict['auth_provider_type'],
auth_provider_id=cred_dict['auth_provider_id'],
data=cred_dict['data'],
))
for rt_dict in data['refresh_tokens']:
# Filter out the old keys that don't have jwt_key (pre-0.76)
if 'jwt_key' not in rt_dict:
continue
created_at = dt_util.parse_datetime(rt_dict['created_at'])
if created_at is None:
getLogger(__name__).error(
'Ignoring refresh token %(id)s with invalid created_at '
'%(created_at)s for user_id %(user_id)s', rt_dict)
continue
2018-07-13 09:43:08 +00:00
token = models.RefreshToken(
id=rt_dict['id'],
user=users[rt_dict['user_id']],
client_id=rt_dict['client_id'],
created_at=created_at,
2018-07-13 09:43:08 +00:00
access_token_expiration=timedelta(
seconds=rt_dict['access_token_expiration']),
token=rt_dict['token'],
jwt_key=rt_dict['jwt_key']
2018-07-13 09:43:08 +00:00
)
users[rt_dict['user_id']].refresh_tokens[token.id] = token
2018-07-13 09:43:08 +00:00
self._users = users
@callback
def _async_schedule_save(self) -> None:
2018-07-13 09:43:08 +00:00
"""Save users."""
if self._users is None:
return
self._store.async_delay_save(self._data_to_save, 1)
@callback
def _data_to_save(self) -> Dict:
"""Return the data to store."""
assert self._users is not None
2018-07-13 09:43:08 +00:00
users = [
{
'id': user.id,
'is_owner': user.is_owner,
'is_active': user.is_active,
'name': user.name,
'system_generated': user.system_generated,
}
for user in self._users.values()
]
credentials = [
{
'id': credential.id,
'user_id': user.id,
'auth_provider_type': credential.auth_provider_type,
'auth_provider_id': credential.auth_provider_id,
'data': credential.data,
}
for user in self._users.values()
for credential in user.credentials
]
refresh_tokens = [
{
'id': refresh_token.id,
'user_id': user.id,
'client_id': refresh_token.client_id,
'created_at': refresh_token.created_at.isoformat(),
'access_token_expiration':
refresh_token.access_token_expiration.total_seconds(),
'token': refresh_token.token,
'jwt_key': refresh_token.jwt_key,
2018-07-13 09:43:08 +00:00
}
for user in self._users.values()
for refresh_token in user.refresh_tokens.values()
]
return {
2018-07-13 09:43:08 +00:00
'users': users,
'credentials': credentials,
'refresh_tokens': refresh_tokens,
}