gino.loader 源代码

import warnings

from sqlalchemy import select
from sqlalchemy.schema import Column
from sqlalchemy.sql.elements import Label

from .declarative import Model


[文档]class Loader:
[文档] @classmethod def get(cls, value): from .crud import Alias if isinstance(value, Loader): rv = value elif isinstance(value, type) and issubclass(value, Model): rv = ModelLoader(value) elif isinstance(value, Alias): rv = AliasLoader(value) elif isinstance(value, Column): rv = ColumnLoader(value) elif isinstance(value, Label): rv = ColumnLoader(value.name) elif isinstance(value, tuple): rv = TupleLoader(value) elif callable(value): rv = CallableLoader(value) else: rv = ValueLoader(value) return rv
@property def query(self): rv = select(self.get_columns()) from_clause = self.get_from() if from_clause is not None: rv = rv.select_from(from_clause) return rv.execution_options(loader=self)
[文档] def do_load(self, row, context): raise NotImplementedError
[文档] def get_columns(self): return []
[文档] def get_from(self): return None
def __getattr__(self, item): return getattr(self.query, item)
_none = object() def _get_column(model, column_or_name) -> Column: if isinstance(column_or_name, str): return getattr(model, column_or_name) if isinstance(column_or_name, Column): if column_or_name in model: return column_or_name raise AttributeError('Column {} does not belong to model {}'.format( column_or_name, model)) raise TypeError('Unknown column {} with type {}'. format(column_or_name, type(column_or_name)))
[文档]class ModelLoader(Loader): def __init__(self, model, *columns, **extras): self.model = model self._distinct = None if columns: self.columns = [_get_column(model, name) for name in columns] else: self.columns = model self.extras = dict((key, self.get(value)) for key, value in extras.items()) self.on_clause = None self._none_as_none = True def _do_load(self, row, *, none_as_none=None): if none_as_none is None: none_as_none = self._none_as_none values = dict((c.name, row[c]) for c in self.columns if c in row) if none_as_none and all((v is None) for v in values.values()): return None rv = self.model() for c in self.columns: if c in row: # noinspection PyProtectedMember instance_key = self.model._column_name_map.invert_get(c.name) rv.__values__[instance_key] = row[c] return rv
[文档] def do_load(self, row, context): distinct = True if self._distinct: if context is None: context = {} ctx = context.setdefault(self._distinct, {}) key = tuple(row[col] for col in self._distinct) rv = ctx.get(key, _none) if rv is _none: rv = self._do_load(row, none_as_none=True) ctx[key] = rv else: distinct = False else: rv = self._do_load(row) if rv is None: return None, None else: for key, value in self.extras.items(): value, distinct_ = value.do_load(row, context) if distinct_ is not None: setattr(rv, key, value) return rv, distinct
[文档] def get_columns(self): yield from self.columns for subloader in self.extras.values(): yield from subloader.get_columns()
[文档] def get_from(self): rv = self.model for key, subloader in self.extras.items(): from_clause = subloader.get_from() if from_clause is not None: rv = rv.outerjoin(from_clause, getattr(subloader, 'on_clause', None)) return rv
[文档] def load(self, *columns, **extras): if columns: self.columns = [_get_column(self.model, name) for name in columns] self.extras.update((key, self.get(value)) for key, value in extras.items()) return self
[文档] def on(self, on_clause): self.on_clause = on_clause return self
[文档] def distinct(self, *columns): self._distinct = columns return self
[文档] def none_as_none(self, enabled=True): if not enabled: warnings.warn( 'The none_as_none feature will be always enabled in 1.0', PendingDeprecationWarning) self._none_as_none = enabled return self
[文档]class AliasLoader(ModelLoader): def __init__(self, alias, *columns, **extras): super().__init__(alias, *columns, **extras)
[文档]class ColumnLoader(Loader): def __init__(self, column): self.column = column
[文档] def do_load(self, row, context): return row[self.column], True
[文档]class TupleLoader(Loader): def __init__(self, values): self.loaders = tuple(self.get(value) for value in values)
[文档] def do_load(self, row, context): return tuple(loader.do_load(row, context)[0] for loader in self.loaders), True
[文档]class CallableLoader(Loader): def __init__(self, func): self.func = func
[文档] def do_load(self, row, context): return self.func(row, context), True
[文档]class ValueLoader(Loader): def __init__(self, value): self.value = value
[文档] def do_load(self, row, context): return self.value, True