canaille-globuzma/canaille/backends/__init__.py

195 lines
5.7 KiB
Python

import importlib
import os
from contextlib import contextmanager
from flask import g
from canaille.app import classproperty
class BaseBackend:
instance = None
def __init__(self, config):
self.config = config
BaseBackend.instance = self
self.register_models()
@classproperty
def instance(cls):
return cls.instance
def init_app(self, app):
@app.before_request
def before_request():
return self.setup()
@app.after_request
def after_request(response):
self.teardown()
return response
@contextmanager
def session(self, *args, **kwargs):
yield self.setup(*args, **kwargs)
self.teardown()
@classmethod
def install(self, config):
"""Prepare the database to host canaille data."""
raise NotImplementedError()
def setup(self):
"""Is called before each http request, it should open the connection to
the backend."""
def teardown(self):
"""Is called after each http request, it should close the connections
to the backend."""
@classmethod
def validate(cls, config):
"""Validate the config part dedicated to the backend.
It should raise :class:`~canaille.configuration.ConfigurationError` when
errors are met.
"""
raise NotImplementedError()
def query(self, model, **kwargs):
"""
Perform a query on the database and return a collection of instances.
Parameters can be any valid attribute with the expected value:
>>> backend.query(User, first_name="George")
If several arguments are passed, the methods only returns the model
instances that return matches all the argument values:
>>> backend.query(User, first_name="George", last_name="Abitbol")
If the argument value is a collection, the methods will return the
models that matches any of the values:
>>> backend.query(User, first_name=["George", "Jane"])
"""
raise NotImplementedError()
def fuzzy(self, model, query, attributes=None, **kwargs):
"""Works like :meth:`~canaille.backends.BaseBackend.query` but
attribute values loosely be matched."""
raise NotImplementedError()
def get(self, model, identifier=None, **kwargs):
"""Works like :meth:`~canaille.backends.BaseBackend.query` but return
only one element or :py:data:`None` if no item is matching."""
raise NotImplementedError()
def save(self, instance):
"""Validate the current modifications in the database."""
raise NotImplementedError()
def delete(self, instance):
"""Remove the current instance from the database."""
raise NotImplementedError()
def reload(self, instance):
"""Cancel the unsaved modifications.
>>> user = User.get(user_name="george")
>>> user.display_name
George
>>> user.display_name = "Jane"
>>> user.display_name
Jane
>>> BaseBackend.instance.reload(user)
>>> user.display_name
George
"""
raise NotImplementedError()
def update(self, instance, **kwargs):
"""Assign a whole dict to the current instance. This is useful to
update models based on forms.
>>> user = User.get(user_name="george")
>>> user.first_name
George
>>> backend.update({
... client,
... first_name="Jane",
... last_name="Calamity",
... })
>>> user.first_name
Jane
"""
for attribute, value in kwargs.items():
setattr(instance, attribute, value)
def check_user_password(self, user, password: str) -> bool:
"""Check if the password matches the user password in the database."""
raise NotImplementedError()
def set_user_password(self, user, password: str):
"""Set a password for the user."""
raise NotImplementedError()
def has_account_lockability(self):
"""Indicate wether the backend supports locking user accounts."""
raise NotImplementedError()
def register_models(self):
from canaille.app import models
module = ".".join(self.__class__.__module__.split(".")[:-1] + ["models"])
try:
backend_models = importlib.import_module(module)
except ModuleNotFoundError:
return
model_names = [
"AuthorizationCode",
"Client",
"Consent",
"Group",
"Token",
"User",
]
for model_name in model_names:
models.register(getattr(backend_models, model_name))
def setup_backend(app, backend=None):
if not backend:
prefix = "CANAILLE_"
available_backends_names = [
f"{prefix}{name}".upper() for name in available_backends()
]
configured_backend_names = [
key[len(prefix) :]
for key in app.config.keys()
if key in available_backends_names
]
backend_name = (
configured_backend_names[0].lower()
if configured_backend_names
else "memory"
)
module = importlib.import_module(f"canaille.backends.{backend_name}.backend")
backend_class = getattr(module, "Backend")
backend = backend_class(app.config)
backend.init_app(app)
with app.app_context():
g.backend = backend
app.backend = backend
return backend
def available_backends():
return {
elt.name
for elt in os.scandir(os.path.dirname(__file__))
if elt.is_dir() and os.path.exists(os.path.join(elt, "backend.py"))
}