Source code for core.browser_session

from __future__ import annotations

from contextlib import suppress
from dogpile.cache.api import NO_VALUE
from hashlib import blake2b


from typing import overload, Any, TypeVar
from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Callable
    from typing import Never

    from .cache import RedisCacheRegion

[docs] OnDirtyCallback = Callable[['BrowserSession', str], None]
[docs] _T = TypeVar('_T')
[docs] class Prefixed: def __init__(self, prefix: str, cache: RedisCacheRegion): assert len(prefix) >= 24
[docs] self.prefix = prefix
[docs] self.cache = cache
[docs] def mangle(self, key: str) -> str: assert key return f'{self.prefix}:{key}'
[docs] def get(self, key: str) -> Any: return self.cache.get(self.mangle(key))
[docs] def set(self, key: str, value: Any) -> None: self.cache.set(self.mangle(key), value)
[docs] def delete(self, key: str) -> None: self.cache.delete(self.mangle(key))
[docs] def count(self) -> int: # note, this cannot be used in a Redis cluster - if we use that # we have to keep track of all keys separately return self.cache.backend.reader_client.eval(""" return #redis.pcall('keys', ARGV[1]) """, 0, f'{self.cache.namespace}:{self.prefix}:*')
[docs] def flush(self) -> int: # note, this cannot be used in a Redis cluster - if we use that # we have to keep track of all keys separately return self.cache.backend.reader_client.eval(""" if #redis.pcall('keys', ARGV[1]) > 0 then return redis.pcall('del', unpack(redis.call('keys', ARGV[1]))) end return 0 """, 0, f'{self.cache.namespace}:{self.prefix}:*')
[docs] class BrowserSession: """ A session bound to a token (session_id cookie). Used to store data about a client on the server. Instances should be called ``browser_session`` to make sure we don't confuse this with the orm sessions. Used by :class:`onegov.core.request.CoreRequest`. Example:: browser_session = request.browser_session browser_session.name = 'Foo' assert client.session.name == 'Foo' del client.session.name This class can act like an object, through attribute access, or like a dict, through square brackets. Whatever you prefer. """ def __init__( self, cache: RedisCacheRegion, token: str, on_dirty: OnDirtyCallback = lambda session, token: None ): # make it impossible to get the session token through key listing prefix = blake2b(token.encode('utf-8'), digest_size=24).hexdigest()
[docs] self._cache = Prefixed(prefix=prefix, cache=cache)
[docs] self._token = token
[docs] self._is_dirty = False
[docs] self._on_dirty = on_dirty
[docs] def has(self, name: str) -> bool: return self._cache.get(name) is not NO_VALUE
[docs] def flush(self) -> None: self._cache.flush() self.mark_as_dirty()
[docs] def count(self) -> int: return self._cache.count()
[docs] def pop(self, name: str, default: Any = None) -> Any: """ Returns the value for the given name, removing the value in the process. """ value = self.get(name, default=default) # we can run into a race condition here when two requests come in # simultaneously - one request will get the value and delete it, the # other will get the value and fail with an error when trying to # delete it # # we can be pragmatic - if it's gone, it doesn't need to be deleted with suppress(AttributeError): delattr(self, name) return value
[docs] def mark_as_dirty(self) -> None: if not self._is_dirty: self._on_dirty(self, self._token) self._is_dirty = True
@overload
[docs] def get(self, name: str) -> Any | None: ...
@overload def get(self, name: str, default: _T) -> Any | _T: ... def get(self, name: str, default: Any = None) -> Any: result = self._cache.get(name) if result is NO_VALUE: return default return result # FIXME: What is *args for, why do we allow this? For now we set # it to Never, so we get a mypy error, if there is any code # that uses this, if no code uses it then get rid of it!
[docs] def __getitem__(self, name: str, *args: Never) -> Any: result = self._cache.get(name) if result is NO_VALUE: raise KeyError return result
[docs] def __getattr__(self, name: str) -> Any: result = self._cache.get(name) if result is NO_VALUE: raise AttributeError return result
[docs] def __setattr__(self, name: str, value: Any) -> None: if name.startswith('_'): return super().__setattr__(name, value) self._cache.set(name, value) self.mark_as_dirty()
[docs] def __delattr__(self, name: str) -> None: if name.startswith('_'): return super().__delattr__(name) self._cache.delete(name) self.mark_as_dirty()
[docs] __setitem__ = __setattr__
[docs] __delitem__ = __delattr__
__delattr__ = __delattr__
[docs] __contains__ = has