You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1513 lines
48 KiB
1513 lines
48 KiB
2 years ago
|
# ext/asyncio/scoping.py
|
||
|
# Copyright (C) 2005-2023 the SQLAlchemy authors and contributors
|
||
|
# <see AUTHORS file>
|
||
|
#
|
||
|
# This module is part of SQLAlchemy and is released under
|
||
|
# the MIT License: https://www.opensource.org/licenses/mit-license.php
|
||
|
|
||
|
from __future__ import annotations
|
||
|
|
||
|
from typing import Any
|
||
|
from typing import Callable
|
||
|
from typing import Generic
|
||
|
from typing import Iterable
|
||
|
from typing import Iterator
|
||
|
from typing import Optional
|
||
|
from typing import overload
|
||
|
from typing import Sequence
|
||
|
from typing import Tuple
|
||
|
from typing import Type
|
||
|
from typing import TYPE_CHECKING
|
||
|
from typing import TypeVar
|
||
|
from typing import Union
|
||
|
|
||
|
from .session import _AS
|
||
|
from .session import async_sessionmaker
|
||
|
from .session import AsyncSession
|
||
|
from ... import exc as sa_exc
|
||
|
from ... import util
|
||
|
from ...orm.session import Session
|
||
|
from ...util import create_proxy_methods
|
||
|
from ...util import ScopedRegistry
|
||
|
from ...util import warn
|
||
|
from ...util import warn_deprecated
|
||
|
|
||
|
if TYPE_CHECKING:
|
||
|
from .engine import AsyncConnection
|
||
|
from .result import AsyncResult
|
||
|
from .result import AsyncScalarResult
|
||
|
from .session import AsyncSessionTransaction
|
||
|
from ...engine import Connection
|
||
|
from ...engine import Engine
|
||
|
from ...engine import Result
|
||
|
from ...engine import Row
|
||
|
from ...engine import RowMapping
|
||
|
from ...engine.interfaces import _CoreAnyExecuteParams
|
||
|
from ...engine.interfaces import _CoreSingleExecuteParams
|
||
|
from ...engine.result import ScalarResult
|
||
|
from ...orm._typing import _IdentityKeyType
|
||
|
from ...orm._typing import _O
|
||
|
from ...orm._typing import OrmExecuteOptionsParameter
|
||
|
from ...orm.interfaces import ORMOption
|
||
|
from ...orm.session import _BindArguments
|
||
|
from ...orm.session import _EntityBindKey
|
||
|
from ...orm.session import _PKIdentityArgument
|
||
|
from ...orm.session import _SessionBind
|
||
|
from ...sql.base import Executable
|
||
|
from ...sql.elements import ClauseElement
|
||
|
from ...sql.selectable import ForUpdateArg
|
||
|
from ...sql.selectable import TypedReturnsRows
|
||
|
|
||
|
_T = TypeVar("_T", bound=Any)
|
||
|
|
||
|
|
||
|
@create_proxy_methods(
|
||
|
AsyncSession,
|
||
|
":class:`_asyncio.AsyncSession`",
|
||
|
":class:`_asyncio.scoping.async_scoped_session`",
|
||
|
classmethods=["close_all", "object_session", "identity_key"],
|
||
|
methods=[
|
||
|
"__contains__",
|
||
|
"__iter__",
|
||
|
"add",
|
||
|
"add_all",
|
||
|
"begin",
|
||
|
"begin_nested",
|
||
|
"close",
|
||
|
"commit",
|
||
|
"connection",
|
||
|
"delete",
|
||
|
"execute",
|
||
|
"expire",
|
||
|
"expire_all",
|
||
|
"expunge",
|
||
|
"expunge_all",
|
||
|
"flush",
|
||
|
"get_bind",
|
||
|
"is_modified",
|
||
|
"invalidate",
|
||
|
"merge",
|
||
|
"refresh",
|
||
|
"rollback",
|
||
|
"scalar",
|
||
|
"scalars",
|
||
|
"stream",
|
||
|
"stream_scalars",
|
||
|
],
|
||
|
attributes=[
|
||
|
"bind",
|
||
|
"dirty",
|
||
|
"deleted",
|
||
|
"new",
|
||
|
"identity_map",
|
||
|
"is_active",
|
||
|
"autoflush",
|
||
|
"no_autoflush",
|
||
|
"info",
|
||
|
],
|
||
|
)
|
||
|
class async_scoped_session(Generic[_AS]):
|
||
|
"""Provides scoped management of :class:`.AsyncSession` objects.
|
||
|
|
||
|
See the section :ref:`asyncio_scoped_session` for usage details.
|
||
|
|
||
|
.. versionadded:: 1.4.19
|
||
|
|
||
|
|
||
|
"""
|
||
|
|
||
|
_support_async = True
|
||
|
|
||
|
session_factory: async_sessionmaker[_AS]
|
||
|
"""The `session_factory` provided to `__init__` is stored in this
|
||
|
attribute and may be accessed at a later time. This can be useful when
|
||
|
a new non-scoped :class:`.AsyncSession` is needed."""
|
||
|
|
||
|
registry: ScopedRegistry[_AS]
|
||
|
|
||
|
def __init__(
|
||
|
self,
|
||
|
session_factory: async_sessionmaker[_AS],
|
||
|
scopefunc: Callable[[], Any],
|
||
|
):
|
||
|
"""Construct a new :class:`_asyncio.async_scoped_session`.
|
||
|
|
||
|
:param session_factory: a factory to create new :class:`_asyncio.AsyncSession`
|
||
|
instances. This is usually, but not necessarily, an instance
|
||
|
of :class:`_asyncio.async_sessionmaker`.
|
||
|
|
||
|
:param scopefunc: function which defines
|
||
|
the current scope. A function such as ``asyncio.current_task``
|
||
|
may be useful here.
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
self.session_factory = session_factory
|
||
|
self.registry = ScopedRegistry(session_factory, scopefunc)
|
||
|
|
||
|
@property
|
||
|
def _proxied(self) -> _AS:
|
||
|
return self.registry()
|
||
|
|
||
|
def __call__(self, **kw: Any) -> _AS:
|
||
|
r"""Return the current :class:`.AsyncSession`, creating it
|
||
|
using the :attr:`.scoped_session.session_factory` if not present.
|
||
|
|
||
|
:param \**kw: Keyword arguments will be passed to the
|
||
|
:attr:`.scoped_session.session_factory` callable, if an existing
|
||
|
:class:`.AsyncSession` is not present. If the
|
||
|
:class:`.AsyncSession` is present
|
||
|
and keyword arguments have been passed,
|
||
|
:exc:`~sqlalchemy.exc.InvalidRequestError` is raised.
|
||
|
|
||
|
"""
|
||
|
if kw:
|
||
|
if self.registry.has():
|
||
|
raise sa_exc.InvalidRequestError(
|
||
|
"Scoped session is already present; "
|
||
|
"no new arguments may be specified."
|
||
|
)
|
||
|
else:
|
||
|
sess = self.session_factory(**kw)
|
||
|
self.registry.set(sess)
|
||
|
else:
|
||
|
sess = self.registry()
|
||
|
if not self._support_async and sess._is_asyncio:
|
||
|
warn_deprecated(
|
||
|
"Using `scoped_session` with asyncio is deprecated and "
|
||
|
"will raise an error in a future version. "
|
||
|
"Please use `async_scoped_session` instead.",
|
||
|
"1.4.23",
|
||
|
)
|
||
|
return sess
|
||
|
|
||
|
def configure(self, **kwargs: Any) -> None:
|
||
|
"""reconfigure the :class:`.sessionmaker` used by this
|
||
|
:class:`.scoped_session`.
|
||
|
|
||
|
See :meth:`.sessionmaker.configure`.
|
||
|
|
||
|
"""
|
||
|
|
||
|
if self.registry.has():
|
||
|
warn(
|
||
|
"At least one scoped session is already present. "
|
||
|
" configure() can not affect sessions that have "
|
||
|
"already been created."
|
||
|
)
|
||
|
|
||
|
self.session_factory.configure(**kwargs)
|
||
|
|
||
|
async def remove(self) -> None:
|
||
|
"""Dispose of the current :class:`.AsyncSession`, if present.
|
||
|
|
||
|
Different from scoped_session's remove method, this method would use
|
||
|
await to wait for the close method of AsyncSession.
|
||
|
|
||
|
"""
|
||
|
|
||
|
if self.registry.has():
|
||
|
await self.registry().close()
|
||
|
self.registry.clear()
|
||
|
|
||
|
async def get(
|
||
|
self,
|
||
|
entity: _EntityBindKey[_O],
|
||
|
ident: _PKIdentityArgument,
|
||
|
*,
|
||
|
options: Optional[Sequence[ORMOption]] = None,
|
||
|
populate_existing: bool = False,
|
||
|
with_for_update: Optional[ForUpdateArg] = None,
|
||
|
identity_token: Optional[Any] = None,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
) -> Optional[_O]:
|
||
|
r"""Return an instance based on the given primary key identifier,
|
||
|
or ``None`` if not found.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.get` - main documentation for get
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
# this was proxied but Mypy is requiring the return type to be
|
||
|
# clarified
|
||
|
|
||
|
# work around:
|
||
|
# https://github.com/python/typing/discussions/1143
|
||
|
return_value = await self._proxied.get(
|
||
|
entity,
|
||
|
ident,
|
||
|
options=options,
|
||
|
populate_existing=populate_existing,
|
||
|
with_for_update=with_for_update,
|
||
|
identity_token=identity_token,
|
||
|
execution_options=execution_options,
|
||
|
)
|
||
|
return return_value
|
||
|
|
||
|
# START PROXY METHODS async_scoped_session
|
||
|
|
||
|
# code within this block is **programmatically,
|
||
|
# statically generated** by tools/generate_proxy_methods.py
|
||
|
|
||
|
def __contains__(self, instance: object) -> bool:
|
||
|
r"""Return True if the instance is associated with this session.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
The instance may be pending or persistent within the Session for a
|
||
|
result of True.
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.__contains__(instance)
|
||
|
|
||
|
def __iter__(self) -> Iterator[object]:
|
||
|
r"""Iterate over all pending or persistent instances within this
|
||
|
Session.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.__iter__()
|
||
|
|
||
|
def add(self, instance: object, _warn: bool = True) -> None:
|
||
|
r"""Place an object into this :class:`_orm.Session`.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
Objects that are in the :term:`transient` state when passed to the
|
||
|
:meth:`_orm.Session.add` method will move to the
|
||
|
:term:`pending` state, until the next flush, at which point they
|
||
|
will move to the :term:`persistent` state.
|
||
|
|
||
|
Objects that are in the :term:`detached` state when passed to the
|
||
|
:meth:`_orm.Session.add` method will move to the :term:`persistent`
|
||
|
state directly.
|
||
|
|
||
|
If the transaction used by the :class:`_orm.Session` is rolled back,
|
||
|
objects which were transient when they were passed to
|
||
|
:meth:`_orm.Session.add` will be moved back to the
|
||
|
:term:`transient` state, and will no longer be present within this
|
||
|
:class:`_orm.Session`.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.add_all`
|
||
|
|
||
|
:ref:`session_adding` - at :ref:`session_basics`
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.add(instance, _warn=_warn)
|
||
|
|
||
|
def add_all(self, instances: Iterable[object]) -> None:
|
||
|
r"""Add the given collection of instances to this :class:`_orm.Session`.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
See the documentation for :meth:`_orm.Session.add` for a general
|
||
|
behavioral description.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.add`
|
||
|
|
||
|
:ref:`session_adding` - at :ref:`session_basics`
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.add_all(instances)
|
||
|
|
||
|
def begin(self) -> AsyncSessionTransaction:
|
||
|
r"""Return an :class:`_asyncio.AsyncSessionTransaction` object.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
The underlying :class:`_orm.Session` will perform the
|
||
|
"begin" action when the :class:`_asyncio.AsyncSessionTransaction`
|
||
|
object is entered::
|
||
|
|
||
|
async with async_session.begin():
|
||
|
# .. ORM transaction is begun
|
||
|
|
||
|
Note that database IO will not normally occur when the session-level
|
||
|
transaction is begun, as database transactions begin on an
|
||
|
on-demand basis. However, the begin block is async to accommodate
|
||
|
for a :meth:`_orm.SessionEvents.after_transaction_create`
|
||
|
event hook that may perform IO.
|
||
|
|
||
|
For a general description of ORM begin, see
|
||
|
:meth:`_orm.Session.begin`.
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.begin()
|
||
|
|
||
|
def begin_nested(self) -> AsyncSessionTransaction:
|
||
|
r"""Return an :class:`_asyncio.AsyncSessionTransaction` object
|
||
|
which will begin a "nested" transaction, e.g. SAVEPOINT.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
Behavior is the same as that of :meth:`_asyncio.AsyncSession.begin`.
|
||
|
|
||
|
For a general description of ORM begin nested, see
|
||
|
:meth:`_orm.Session.begin_nested`.
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.begin_nested()
|
||
|
|
||
|
async def close(self) -> None:
|
||
|
r"""Close out the transactional resources and ORM objects used by this
|
||
|
:class:`_asyncio.AsyncSession`.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
This expunges all ORM objects associated with this
|
||
|
:class:`_asyncio.AsyncSession`, ends any transaction in progress and
|
||
|
:term:`releases` any :class:`_asyncio.AsyncConnection` objects which
|
||
|
this :class:`_asyncio.AsyncSession` itself has checked out from
|
||
|
associated :class:`_asyncio.AsyncEngine` objects. The operation then
|
||
|
leaves the :class:`_asyncio.AsyncSession` in a state which it may be
|
||
|
used again.
|
||
|
|
||
|
.. tip::
|
||
|
|
||
|
The :meth:`_asyncio.AsyncSession.close` method **does not prevent
|
||
|
the Session from being used again**. The
|
||
|
:class:`_asyncio.AsyncSession` itself does not actually have a
|
||
|
distinct "closed" state; it merely means the
|
||
|
:class:`_asyncio.AsyncSession` will release all database
|
||
|
connections and ORM objects.
|
||
|
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:ref:`session_closing` - detail on the semantics of
|
||
|
:meth:`_asyncio.AsyncSession.close`
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.close()
|
||
|
|
||
|
async def commit(self) -> None:
|
||
|
r"""Commit the current transaction in progress.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.commit()
|
||
|
|
||
|
async def connection(self, **kw: Any) -> AsyncConnection:
|
||
|
r"""Return a :class:`_asyncio.AsyncConnection` object corresponding to
|
||
|
this :class:`.Session` object's transactional state.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
This method may also be used to establish execution options for the
|
||
|
database connection used by the current transaction.
|
||
|
|
||
|
.. versionadded:: 1.4.24 Added \**kw arguments which are passed
|
||
|
through to the underlying :meth:`_orm.Session.connection` method.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.connection` - main documentation for
|
||
|
"connection"
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.connection(**kw)
|
||
|
|
||
|
async def delete(self, instance: object) -> None:
|
||
|
r"""Mark an instance as deleted.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
The database delete operation occurs upon ``flush()``.
|
||
|
|
||
|
As this operation may need to cascade along unloaded relationships,
|
||
|
it is awaitable to allow for those queries to take place.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.delete` - main documentation for delete
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.delete(instance)
|
||
|
|
||
|
@overload
|
||
|
async def execute(
|
||
|
self,
|
||
|
statement: TypedReturnsRows[_T],
|
||
|
params: Optional[_CoreAnyExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
_parent_execute_state: Optional[Any] = None,
|
||
|
_add_event: Optional[Any] = None,
|
||
|
) -> Result[_T]:
|
||
|
...
|
||
|
|
||
|
@overload
|
||
|
async def execute(
|
||
|
self,
|
||
|
statement: Executable,
|
||
|
params: Optional[_CoreAnyExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
_parent_execute_state: Optional[Any] = None,
|
||
|
_add_event: Optional[Any] = None,
|
||
|
) -> Result[Any]:
|
||
|
...
|
||
|
|
||
|
async def execute(
|
||
|
self,
|
||
|
statement: Executable,
|
||
|
params: Optional[_CoreAnyExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> Result[Any]:
|
||
|
r"""Execute a statement and return a buffered
|
||
|
:class:`_engine.Result` object.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.execute` - main documentation for execute
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.execute(
|
||
|
statement,
|
||
|
params=params,
|
||
|
execution_options=execution_options,
|
||
|
bind_arguments=bind_arguments,
|
||
|
**kw,
|
||
|
)
|
||
|
|
||
|
def expire(
|
||
|
self, instance: object, attribute_names: Optional[Iterable[str]] = None
|
||
|
) -> None:
|
||
|
r"""Expire the attributes on an instance.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
Marks the attributes of an instance as out of date. When an expired
|
||
|
attribute is next accessed, a query will be issued to the
|
||
|
:class:`.Session` object's current transactional context in order to
|
||
|
load all expired attributes for the given instance. Note that
|
||
|
a highly isolated transaction will return the same values as were
|
||
|
previously read in that same transaction, regardless of changes
|
||
|
in database state outside of that transaction.
|
||
|
|
||
|
To expire all objects in the :class:`.Session` simultaneously,
|
||
|
use :meth:`Session.expire_all`.
|
||
|
|
||
|
The :class:`.Session` object's default behavior is to
|
||
|
expire all state whenever the :meth:`Session.rollback`
|
||
|
or :meth:`Session.commit` methods are called, so that new
|
||
|
state can be loaded for the new transaction. For this reason,
|
||
|
calling :meth:`Session.expire` only makes sense for the specific
|
||
|
case that a non-ORM SQL statement was emitted in the current
|
||
|
transaction.
|
||
|
|
||
|
:param instance: The instance to be refreshed.
|
||
|
:param attribute_names: optional list of string attribute names
|
||
|
indicating a subset of attributes to be expired.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:ref:`session_expire` - introductory material
|
||
|
|
||
|
:meth:`.Session.expire`
|
||
|
|
||
|
:meth:`.Session.refresh`
|
||
|
|
||
|
:meth:`_orm.Query.populate_existing`
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.expire(instance, attribute_names=attribute_names)
|
||
|
|
||
|
def expire_all(self) -> None:
|
||
|
r"""Expires all persistent instances within this Session.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
When any attributes on a persistent instance is next accessed,
|
||
|
a query will be issued using the
|
||
|
:class:`.Session` object's current transactional context in order to
|
||
|
load all expired attributes for the given instance. Note that
|
||
|
a highly isolated transaction will return the same values as were
|
||
|
previously read in that same transaction, regardless of changes
|
||
|
in database state outside of that transaction.
|
||
|
|
||
|
To expire individual objects and individual attributes
|
||
|
on those objects, use :meth:`Session.expire`.
|
||
|
|
||
|
The :class:`.Session` object's default behavior is to
|
||
|
expire all state whenever the :meth:`Session.rollback`
|
||
|
or :meth:`Session.commit` methods are called, so that new
|
||
|
state can be loaded for the new transaction. For this reason,
|
||
|
calling :meth:`Session.expire_all` is not usually needed,
|
||
|
assuming the transaction is isolated.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:ref:`session_expire` - introductory material
|
||
|
|
||
|
:meth:`.Session.expire`
|
||
|
|
||
|
:meth:`.Session.refresh`
|
||
|
|
||
|
:meth:`_orm.Query.populate_existing`
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.expire_all()
|
||
|
|
||
|
def expunge(self, instance: object) -> None:
|
||
|
r"""Remove the `instance` from this ``Session``.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
This will free all internal references to the instance. Cascading
|
||
|
will be applied according to the *expunge* cascade rule.
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.expunge(instance)
|
||
|
|
||
|
def expunge_all(self) -> None:
|
||
|
r"""Remove all object instances from this ``Session``.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
This is equivalent to calling ``expunge(obj)`` on all objects in this
|
||
|
``Session``.
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.expunge_all()
|
||
|
|
||
|
async def flush(self, objects: Optional[Sequence[Any]] = None) -> None:
|
||
|
r"""Flush all the object changes to the database.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.flush` - main documentation for flush
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.flush(objects=objects)
|
||
|
|
||
|
def get_bind(
|
||
|
self,
|
||
|
mapper: Optional[_EntityBindKey[_O]] = None,
|
||
|
clause: Optional[ClauseElement] = None,
|
||
|
bind: Optional[_SessionBind] = None,
|
||
|
**kw: Any,
|
||
|
) -> Union[Engine, Connection]:
|
||
|
r"""Return a "bind" to which the synchronous proxied :class:`_orm.Session`
|
||
|
is bound.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
Unlike the :meth:`_orm.Session.get_bind` method, this method is
|
||
|
currently **not** used by this :class:`.AsyncSession` in any way
|
||
|
in order to resolve engines for requests.
|
||
|
|
||
|
.. note::
|
||
|
|
||
|
This method proxies directly to the :meth:`_orm.Session.get_bind`
|
||
|
method, however is currently **not** useful as an override target,
|
||
|
in contrast to that of the :meth:`_orm.Session.get_bind` method.
|
||
|
The example below illustrates how to implement custom
|
||
|
:meth:`_orm.Session.get_bind` schemes that work with
|
||
|
:class:`.AsyncSession` and :class:`.AsyncEngine`.
|
||
|
|
||
|
The pattern introduced at :ref:`session_custom_partitioning`
|
||
|
illustrates how to apply a custom bind-lookup scheme to a
|
||
|
:class:`_orm.Session` given a set of :class:`_engine.Engine` objects.
|
||
|
To apply a corresponding :meth:`_orm.Session.get_bind` implementation
|
||
|
for use with a :class:`.AsyncSession` and :class:`.AsyncEngine`
|
||
|
objects, continue to subclass :class:`_orm.Session` and apply it to
|
||
|
:class:`.AsyncSession` using
|
||
|
:paramref:`.AsyncSession.sync_session_class`. The inner method must
|
||
|
continue to return :class:`_engine.Engine` instances, which can be
|
||
|
acquired from a :class:`_asyncio.AsyncEngine` using the
|
||
|
:attr:`_asyncio.AsyncEngine.sync_engine` attribute::
|
||
|
|
||
|
# using example from "Custom Vertical Partitioning"
|
||
|
|
||
|
|
||
|
import random
|
||
|
|
||
|
from sqlalchemy.ext.asyncio import AsyncSession
|
||
|
from sqlalchemy.ext.asyncio import create_async_engine
|
||
|
from sqlalchemy.ext.asyncio import async_sessionmaker
|
||
|
from sqlalchemy.orm import Session
|
||
|
|
||
|
# construct async engines w/ async drivers
|
||
|
engines = {
|
||
|
'leader':create_async_engine("sqlite+aiosqlite:///leader.db"),
|
||
|
'other':create_async_engine("sqlite+aiosqlite:///other.db"),
|
||
|
'follower1':create_async_engine("sqlite+aiosqlite:///follower1.db"),
|
||
|
'follower2':create_async_engine("sqlite+aiosqlite:///follower2.db"),
|
||
|
}
|
||
|
|
||
|
class RoutingSession(Session):
|
||
|
def get_bind(self, mapper=None, clause=None, **kw):
|
||
|
# within get_bind(), return sync engines
|
||
|
if mapper and issubclass(mapper.class_, MyOtherClass):
|
||
|
return engines['other'].sync_engine
|
||
|
elif self._flushing or isinstance(clause, (Update, Delete)):
|
||
|
return engines['leader'].sync_engine
|
||
|
else:
|
||
|
return engines[
|
||
|
random.choice(['follower1','follower2'])
|
||
|
].sync_engine
|
||
|
|
||
|
# apply to AsyncSession using sync_session_class
|
||
|
AsyncSessionMaker = async_sessionmaker(
|
||
|
sync_session_class=RoutingSession
|
||
|
)
|
||
|
|
||
|
The :meth:`_orm.Session.get_bind` method is called in a non-asyncio,
|
||
|
implicitly non-blocking context in the same manner as ORM event hooks
|
||
|
and functions that are invoked via :meth:`.AsyncSession.run_sync`, so
|
||
|
routines that wish to run SQL commands inside of
|
||
|
:meth:`_orm.Session.get_bind` can continue to do so using
|
||
|
blocking-style code, which will be translated to implicitly async calls
|
||
|
at the point of invoking IO on the database drivers.
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.get_bind(
|
||
|
mapper=mapper, clause=clause, bind=bind, **kw
|
||
|
)
|
||
|
|
||
|
def is_modified(
|
||
|
self, instance: object, include_collections: bool = True
|
||
|
) -> bool:
|
||
|
r"""Return ``True`` if the given instance has locally
|
||
|
modified attributes.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
This method retrieves the history for each instrumented
|
||
|
attribute on the instance and performs a comparison of the current
|
||
|
value to its previously committed value, if any.
|
||
|
|
||
|
It is in effect a more expensive and accurate
|
||
|
version of checking for the given instance in the
|
||
|
:attr:`.Session.dirty` collection; a full test for
|
||
|
each attribute's net "dirty" status is performed.
|
||
|
|
||
|
E.g.::
|
||
|
|
||
|
return session.is_modified(someobject)
|
||
|
|
||
|
A few caveats to this method apply:
|
||
|
|
||
|
* Instances present in the :attr:`.Session.dirty` collection may
|
||
|
report ``False`` when tested with this method. This is because
|
||
|
the object may have received change events via attribute mutation,
|
||
|
thus placing it in :attr:`.Session.dirty`, but ultimately the state
|
||
|
is the same as that loaded from the database, resulting in no net
|
||
|
change here.
|
||
|
* Scalar attributes may not have recorded the previously set
|
||
|
value when a new value was applied, if the attribute was not loaded,
|
||
|
or was expired, at the time the new value was received - in these
|
||
|
cases, the attribute is assumed to have a change, even if there is
|
||
|
ultimately no net change against its database value. SQLAlchemy in
|
||
|
most cases does not need the "old" value when a set event occurs, so
|
||
|
it skips the expense of a SQL call if the old value isn't present,
|
||
|
based on the assumption that an UPDATE of the scalar value is
|
||
|
usually needed, and in those few cases where it isn't, is less
|
||
|
expensive on average than issuing a defensive SELECT.
|
||
|
|
||
|
The "old" value is fetched unconditionally upon set only if the
|
||
|
attribute container has the ``active_history`` flag set to ``True``.
|
||
|
This flag is set typically for primary key attributes and scalar
|
||
|
object references that are not a simple many-to-one. To set this
|
||
|
flag for any arbitrary mapped column, use the ``active_history``
|
||
|
argument with :func:`.column_property`.
|
||
|
|
||
|
:param instance: mapped instance to be tested for pending changes.
|
||
|
:param include_collections: Indicates if multivalued collections
|
||
|
should be included in the operation. Setting this to ``False`` is a
|
||
|
way to detect only local-column based properties (i.e. scalar columns
|
||
|
or many-to-one foreign keys) that would result in an UPDATE for this
|
||
|
instance upon flush.
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.is_modified(
|
||
|
instance, include_collections=include_collections
|
||
|
)
|
||
|
|
||
|
async def invalidate(self) -> None:
|
||
|
r"""Close this Session, using connection invalidation.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
For a complete description, see :meth:`_orm.Session.invalidate`.
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.invalidate()
|
||
|
|
||
|
async def merge(
|
||
|
self,
|
||
|
instance: _O,
|
||
|
*,
|
||
|
load: bool = True,
|
||
|
options: Optional[Sequence[ORMOption]] = None,
|
||
|
) -> _O:
|
||
|
r"""Copy the state of a given instance into a corresponding instance
|
||
|
within this :class:`_asyncio.AsyncSession`.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.merge` - main documentation for merge
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.merge(instance, load=load, options=options)
|
||
|
|
||
|
async def refresh(
|
||
|
self,
|
||
|
instance: object,
|
||
|
attribute_names: Optional[Iterable[str]] = None,
|
||
|
with_for_update: Optional[ForUpdateArg] = None,
|
||
|
) -> None:
|
||
|
r"""Expire and refresh the attributes on the given instance.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
A query will be issued to the database and all attributes will be
|
||
|
refreshed with their current database value.
|
||
|
|
||
|
This is the async version of the :meth:`_orm.Session.refresh` method.
|
||
|
See that method for a complete description of all options.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.refresh` - main documentation for refresh
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.refresh(
|
||
|
instance,
|
||
|
attribute_names=attribute_names,
|
||
|
with_for_update=with_for_update,
|
||
|
)
|
||
|
|
||
|
async def rollback(self) -> None:
|
||
|
r"""Rollback the current transaction in progress.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.rollback()
|
||
|
|
||
|
@overload
|
||
|
async def scalar(
|
||
|
self,
|
||
|
statement: TypedReturnsRows[Tuple[_T]],
|
||
|
params: Optional[_CoreSingleExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> Optional[_T]:
|
||
|
...
|
||
|
|
||
|
@overload
|
||
|
async def scalar(
|
||
|
self,
|
||
|
statement: Executable,
|
||
|
params: Optional[_CoreSingleExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> Any:
|
||
|
...
|
||
|
|
||
|
async def scalar(
|
||
|
self,
|
||
|
statement: Executable,
|
||
|
params: Optional[_CoreSingleExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> Any:
|
||
|
r"""Execute a statement and return a scalar result.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.scalar` - main documentation for scalar
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.scalar(
|
||
|
statement,
|
||
|
params=params,
|
||
|
execution_options=execution_options,
|
||
|
bind_arguments=bind_arguments,
|
||
|
**kw,
|
||
|
)
|
||
|
|
||
|
@overload
|
||
|
async def scalars(
|
||
|
self,
|
||
|
statement: TypedReturnsRows[Tuple[_T]],
|
||
|
params: Optional[_CoreSingleExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> ScalarResult[_T]:
|
||
|
...
|
||
|
|
||
|
@overload
|
||
|
async def scalars(
|
||
|
self,
|
||
|
statement: Executable,
|
||
|
params: Optional[_CoreSingleExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> ScalarResult[Any]:
|
||
|
...
|
||
|
|
||
|
async def scalars(
|
||
|
self,
|
||
|
statement: Executable,
|
||
|
params: Optional[_CoreSingleExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> ScalarResult[Any]:
|
||
|
r"""Execute a statement and return scalar results.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
:return: a :class:`_result.ScalarResult` object
|
||
|
|
||
|
.. versionadded:: 1.4.24 Added :meth:`_asyncio.AsyncSession.scalars`
|
||
|
|
||
|
.. versionadded:: 1.4.26 Added
|
||
|
:meth:`_asyncio.async_scoped_session.scalars`
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.scalars` - main documentation for scalars
|
||
|
|
||
|
:meth:`_asyncio.AsyncSession.stream_scalars` - streaming version
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.scalars(
|
||
|
statement,
|
||
|
params=params,
|
||
|
execution_options=execution_options,
|
||
|
bind_arguments=bind_arguments,
|
||
|
**kw,
|
||
|
)
|
||
|
|
||
|
@overload
|
||
|
async def stream(
|
||
|
self,
|
||
|
statement: TypedReturnsRows[_T],
|
||
|
params: Optional[_CoreAnyExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> AsyncResult[_T]:
|
||
|
...
|
||
|
|
||
|
@overload
|
||
|
async def stream(
|
||
|
self,
|
||
|
statement: Executable,
|
||
|
params: Optional[_CoreAnyExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> AsyncResult[Any]:
|
||
|
...
|
||
|
|
||
|
async def stream(
|
||
|
self,
|
||
|
statement: Executable,
|
||
|
params: Optional[_CoreAnyExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> AsyncResult[Any]:
|
||
|
r"""Execute a statement and return a streaming
|
||
|
:class:`_asyncio.AsyncResult` object.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.stream(
|
||
|
statement,
|
||
|
params=params,
|
||
|
execution_options=execution_options,
|
||
|
bind_arguments=bind_arguments,
|
||
|
**kw,
|
||
|
)
|
||
|
|
||
|
@overload
|
||
|
async def stream_scalars(
|
||
|
self,
|
||
|
statement: TypedReturnsRows[Tuple[_T]],
|
||
|
params: Optional[_CoreSingleExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> AsyncScalarResult[_T]:
|
||
|
...
|
||
|
|
||
|
@overload
|
||
|
async def stream_scalars(
|
||
|
self,
|
||
|
statement: Executable,
|
||
|
params: Optional[_CoreSingleExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> AsyncScalarResult[Any]:
|
||
|
...
|
||
|
|
||
|
async def stream_scalars(
|
||
|
self,
|
||
|
statement: Executable,
|
||
|
params: Optional[_CoreSingleExecuteParams] = None,
|
||
|
*,
|
||
|
execution_options: OrmExecuteOptionsParameter = util.EMPTY_DICT,
|
||
|
bind_arguments: Optional[_BindArguments] = None,
|
||
|
**kw: Any,
|
||
|
) -> AsyncScalarResult[Any]:
|
||
|
r"""Execute a statement and return a stream of scalar results.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
:return: an :class:`_asyncio.AsyncScalarResult` object
|
||
|
|
||
|
.. versionadded:: 1.4.24
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:meth:`_orm.Session.scalars` - main documentation for scalars
|
||
|
|
||
|
:meth:`_asyncio.AsyncSession.scalars` - non streaming version
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await self._proxied.stream_scalars(
|
||
|
statement,
|
||
|
params=params,
|
||
|
execution_options=execution_options,
|
||
|
bind_arguments=bind_arguments,
|
||
|
**kw,
|
||
|
)
|
||
|
|
||
|
@property
|
||
|
def bind(self) -> Any:
|
||
|
r"""Proxy for the :attr:`_asyncio.AsyncSession.bind` attribute
|
||
|
on behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.bind
|
||
|
|
||
|
@bind.setter
|
||
|
def bind(self, attr: Any) -> None:
|
||
|
self._proxied.bind = attr
|
||
|
|
||
|
@property
|
||
|
def dirty(self) -> Any:
|
||
|
r"""The set of all persistent instances considered dirty.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class
|
||
|
on behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class
|
||
|
on behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
E.g.::
|
||
|
|
||
|
some_mapped_object in session.dirty
|
||
|
|
||
|
Instances are considered dirty when they were modified but not
|
||
|
deleted.
|
||
|
|
||
|
Note that this 'dirty' calculation is 'optimistic'; most
|
||
|
attribute-setting or collection modification operations will
|
||
|
mark an instance as 'dirty' and place it in this set, even if
|
||
|
there is no net change to the attribute's value. At flush
|
||
|
time, the value of each attribute is compared to its
|
||
|
previously saved value, and if there's no net change, no SQL
|
||
|
operation will occur (this is a more expensive operation so
|
||
|
it's only done at flush time).
|
||
|
|
||
|
To check if an instance has actionable net changes to its
|
||
|
attributes, use the :meth:`.Session.is_modified` method.
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.dirty
|
||
|
|
||
|
@property
|
||
|
def deleted(self) -> Any:
|
||
|
r"""The set of all instances marked as 'deleted' within this ``Session``
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class
|
||
|
on behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class
|
||
|
on behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.deleted
|
||
|
|
||
|
@property
|
||
|
def new(self) -> Any:
|
||
|
r"""The set of all instances marked as 'new' within this ``Session``.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class
|
||
|
on behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class
|
||
|
on behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.new
|
||
|
|
||
|
@property
|
||
|
def identity_map(self) -> Any:
|
||
|
r"""Proxy for the :attr:`_orm.Session.identity_map` attribute
|
||
|
on behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class
|
||
|
on behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.identity_map
|
||
|
|
||
|
@identity_map.setter
|
||
|
def identity_map(self, attr: Any) -> None:
|
||
|
self._proxied.identity_map = attr
|
||
|
|
||
|
@property
|
||
|
def is_active(self) -> Any:
|
||
|
r"""True if this :class:`.Session` not in "partial rollback" state.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class
|
||
|
on behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class
|
||
|
on behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
.. versionchanged:: 1.4 The :class:`_orm.Session` no longer begins
|
||
|
a new transaction immediately, so this attribute will be False
|
||
|
when the :class:`_orm.Session` is first instantiated.
|
||
|
|
||
|
"partial rollback" state typically indicates that the flush process
|
||
|
of the :class:`_orm.Session` has failed, and that the
|
||
|
:meth:`_orm.Session.rollback` method must be emitted in order to
|
||
|
fully roll back the transaction.
|
||
|
|
||
|
If this :class:`_orm.Session` is not in a transaction at all, the
|
||
|
:class:`_orm.Session` will autobegin when it is first used, so in this
|
||
|
case :attr:`_orm.Session.is_active` will return True.
|
||
|
|
||
|
Otherwise, if this :class:`_orm.Session` is within a transaction,
|
||
|
and that transaction has not been rolled back internally, the
|
||
|
:attr:`_orm.Session.is_active` will also return True.
|
||
|
|
||
|
.. seealso::
|
||
|
|
||
|
:ref:`faq_session_rollback`
|
||
|
|
||
|
:meth:`_orm.Session.in_transaction`
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.is_active
|
||
|
|
||
|
@property
|
||
|
def autoflush(self) -> Any:
|
||
|
r"""Proxy for the :attr:`_orm.Session.autoflush` attribute
|
||
|
on behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class
|
||
|
on behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.autoflush
|
||
|
|
||
|
@autoflush.setter
|
||
|
def autoflush(self, attr: Any) -> None:
|
||
|
self._proxied.autoflush = attr
|
||
|
|
||
|
@property
|
||
|
def no_autoflush(self) -> Any:
|
||
|
r"""Return a context manager that disables autoflush.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class
|
||
|
on behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class
|
||
|
on behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
e.g.::
|
||
|
|
||
|
with session.no_autoflush:
|
||
|
|
||
|
some_object = SomeClass()
|
||
|
session.add(some_object)
|
||
|
# won't autoflush
|
||
|
some_object.related_thing = session.query(SomeRelated).first()
|
||
|
|
||
|
Operations that proceed within the ``with:`` block
|
||
|
will not be subject to flushes occurring upon query
|
||
|
access. This is useful when initializing a series
|
||
|
of objects which involve existing database queries,
|
||
|
where the uncompleted object should not yet be flushed.
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.no_autoflush
|
||
|
|
||
|
@property
|
||
|
def info(self) -> Any:
|
||
|
r"""A user-modifiable dictionary.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class
|
||
|
on behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class
|
||
|
on behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
The initial value of this dictionary can be populated using the
|
||
|
``info`` argument to the :class:`.Session` constructor or
|
||
|
:class:`.sessionmaker` constructor or factory methods. The dictionary
|
||
|
here is always local to this :class:`.Session` and can be modified
|
||
|
independently of all other :class:`.Session` objects.
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return self._proxied.info
|
||
|
|
||
|
@classmethod
|
||
|
async def close_all(self) -> None:
|
||
|
r"""Close all :class:`_asyncio.AsyncSession` sessions.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return await AsyncSession.close_all()
|
||
|
|
||
|
@classmethod
|
||
|
def object_session(cls, instance: object) -> Optional[Session]:
|
||
|
r"""Return the :class:`.Session` to which an object belongs.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
This is an alias of :func:`.object_session`.
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return AsyncSession.object_session(instance)
|
||
|
|
||
|
@classmethod
|
||
|
def identity_key(
|
||
|
cls,
|
||
|
class_: Optional[Type[Any]] = None,
|
||
|
ident: Union[Any, Tuple[Any, ...]] = None,
|
||
|
*,
|
||
|
instance: Optional[Any] = None,
|
||
|
row: Optional[Union[Row[Any], RowMapping]] = None,
|
||
|
identity_token: Optional[Any] = None,
|
||
|
) -> _IdentityKeyType[Any]:
|
||
|
r"""Return an identity key.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_asyncio.AsyncSession` class on
|
||
|
behalf of the :class:`_asyncio.scoping.async_scoped_session` class.
|
||
|
|
||
|
.. container:: class_bases
|
||
|
|
||
|
Proxied for the :class:`_orm.Session` class on
|
||
|
behalf of the :class:`_asyncio.AsyncSession` class.
|
||
|
|
||
|
This is an alias of :func:`.util.identity_key`.
|
||
|
|
||
|
|
||
|
|
||
|
""" # noqa: E501
|
||
|
|
||
|
return AsyncSession.identity_key(
|
||
|
class_=class_,
|
||
|
ident=ident,
|
||
|
instance=instance,
|
||
|
row=row,
|
||
|
identity_token=identity_token,
|
||
|
)
|
||
|
|
||
|
# END PROXY METHODS async_scoped_session
|