| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323 |
- # sql/roles.py
- # Copyright (C) 2005-2025 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 Generic
- from typing import Optional
- from typing import TYPE_CHECKING
- from typing import TypeVar
- from .. import util
- from ..util.typing import Literal
- if TYPE_CHECKING:
- from ._typing import _PropagateAttrsType
- from .elements import Label
- from .selectable import _SelectIterable
- from .selectable import FromClause
- from .selectable import Subquery
- _T = TypeVar("_T", bound=Any)
- _T_co = TypeVar("_T_co", bound=Any, covariant=True)
- class SQLRole:
- """Define a "role" within a SQL statement structure.
- Classes within SQL Core participate within SQLRole hierarchies in order
- to more accurately indicate where they may be used within SQL statements
- of all types.
- .. versionadded:: 1.4
- """
- __slots__ = ()
- allows_lambda = False
- uses_inspection = False
- class UsesInspection:
- __slots__ = ()
- _post_inspect: Literal[None] = None
- uses_inspection = True
- class AllowsLambdaRole:
- __slots__ = ()
- allows_lambda = True
- class HasCacheKeyRole(SQLRole):
- __slots__ = ()
- _role_name = "Cacheable Core or ORM object"
- class ExecutableOptionRole(SQLRole):
- __slots__ = ()
- _role_name = "ExecutionOption Core or ORM object"
- class LiteralValueRole(SQLRole):
- __slots__ = ()
- _role_name = "Literal Python value"
- class ColumnArgumentRole(SQLRole):
- __slots__ = ()
- _role_name = "Column expression"
- class ColumnArgumentOrKeyRole(ColumnArgumentRole):
- __slots__ = ()
- _role_name = "Column expression or string key"
- class StrAsPlainColumnRole(ColumnArgumentRole):
- __slots__ = ()
- _role_name = "Column expression or string key"
- class ColumnListRole(SQLRole):
- """Elements suitable for forming comma separated lists of expressions."""
- __slots__ = ()
- class StringRole(SQLRole):
- """mixin indicating a role that results in strings"""
- __slots__ = ()
- class TruncatedLabelRole(StringRole, SQLRole):
- __slots__ = ()
- _role_name = "String SQL identifier"
- class ColumnsClauseRole(AllowsLambdaRole, UsesInspection, ColumnListRole):
- __slots__ = ()
- _role_name = (
- "Column expression, FROM clause, or other columns clause element"
- )
- @property
- def _select_iterable(self) -> _SelectIterable:
- raise NotImplementedError()
- class TypedColumnsClauseRole(Generic[_T_co], SQLRole):
- """element-typed form of ColumnsClauseRole"""
- __slots__ = ()
- class LimitOffsetRole(SQLRole):
- __slots__ = ()
- _role_name = "LIMIT / OFFSET expression"
- class ByOfRole(ColumnListRole):
- __slots__ = ()
- _role_name = "GROUP BY / OF / etc. expression"
- class GroupByRole(AllowsLambdaRole, UsesInspection, ByOfRole):
- __slots__ = ()
- # note there's a special case right now where you can pass a whole
- # ORM entity to group_by() and it splits out. we may not want to keep
- # this around
- _role_name = "GROUP BY expression"
- class OrderByRole(AllowsLambdaRole, ByOfRole):
- __slots__ = ()
- _role_name = "ORDER BY expression"
- class StructuralRole(SQLRole):
- __slots__ = ()
- class StatementOptionRole(StructuralRole):
- __slots__ = ()
- _role_name = "statement sub-expression element"
- class OnClauseRole(AllowsLambdaRole, StructuralRole):
- __slots__ = ()
- _role_name = (
- "ON clause, typically a SQL expression or "
- "ORM relationship attribute"
- )
- class WhereHavingRole(OnClauseRole):
- __slots__ = ()
- _role_name = "SQL expression for WHERE/HAVING role"
- class ExpressionElementRole(TypedColumnsClauseRole[_T_co]):
- # note when using generics for ExpressionElementRole,
- # the generic type needs to be in
- # sqlalchemy.sql.coercions._impl_lookup mapping also.
- # these are set up for basic types like int, bool, str, float
- # right now
- __slots__ = ()
- _role_name = "SQL expression element"
- def label(self, name: Optional[str]) -> Label[_T]:
- raise NotImplementedError()
- class ConstExprRole(ExpressionElementRole[_T]):
- __slots__ = ()
- _role_name = "Constant True/False/None expression"
- class LabeledColumnExprRole(ExpressionElementRole[_T]):
- __slots__ = ()
- class BinaryElementRole(ExpressionElementRole[_T]):
- __slots__ = ()
- _role_name = "SQL expression element or literal value"
- class InElementRole(SQLRole):
- __slots__ = ()
- _role_name = (
- "IN expression list, SELECT construct, or bound parameter object"
- )
- class JoinTargetRole(AllowsLambdaRole, UsesInspection, StructuralRole):
- __slots__ = ()
- _role_name = (
- "Join target, typically a FROM expression, or ORM "
- "relationship attribute"
- )
- class FromClauseRole(ColumnsClauseRole, JoinTargetRole):
- __slots__ = ()
- _role_name = "FROM expression, such as a Table or alias() object"
- _is_subquery = False
- named_with_column: bool
- class StrictFromClauseRole(FromClauseRole):
- __slots__ = ()
- # does not allow text() or select() objects
- class AnonymizedFromClauseRole(StrictFromClauseRole):
- __slots__ = ()
- if TYPE_CHECKING:
- def _anonymous_fromclause(
- self, *, name: Optional[str] = None, flat: bool = False
- ) -> FromClause: ...
- class ReturnsRowsRole(SQLRole):
- __slots__ = ()
- _role_name = (
- "Row returning expression such as a SELECT, a FROM clause, or an "
- "INSERT/UPDATE/DELETE with RETURNING"
- )
- class StatementRole(SQLRole):
- __slots__ = ()
- _role_name = "Executable SQL or text() construct"
- if TYPE_CHECKING:
- @util.memoized_property
- def _propagate_attrs(self) -> _PropagateAttrsType: ...
- else:
- _propagate_attrs = util.EMPTY_DICT
- class SelectStatementRole(StatementRole, ReturnsRowsRole):
- __slots__ = ()
- _role_name = "SELECT construct or equivalent text() construct"
- def subquery(self) -> Subquery:
- raise NotImplementedError(
- "All SelectStatementRole objects should implement a "
- ".subquery() method."
- )
- class HasCTERole(ReturnsRowsRole):
- __slots__ = ()
- class IsCTERole(SQLRole):
- __slots__ = ()
- _role_name = "CTE object"
- class CompoundElementRole(AllowsLambdaRole, SQLRole):
- """SELECT statements inside a CompoundSelect, e.g. UNION, EXTRACT, etc."""
- __slots__ = ()
- _role_name = (
- "SELECT construct for inclusion in a UNION or other set construct"
- )
- # TODO: are we using this?
- class DMLRole(StatementRole):
- __slots__ = ()
- class DMLTableRole(FromClauseRole):
- __slots__ = ()
- _role_name = "subject table for an INSERT, UPDATE or DELETE"
- class DMLColumnRole(SQLRole):
- __slots__ = ()
- _role_name = "SET/VALUES column expression or string key"
- class DMLSelectRole(SQLRole):
- """A SELECT statement embedded in DML, typically INSERT from SELECT"""
- __slots__ = ()
- _role_name = "SELECT statement or equivalent textual object"
- class DDLRole(StatementRole):
- __slots__ = ()
- class DDLExpressionRole(StructuralRole):
- __slots__ = ()
- _role_name = "SQL expression element for DDL constraint"
- class DDLConstraintColumnRole(SQLRole):
- __slots__ = ()
- _role_name = "String column name or column expression for DDL constraint"
- class DDLReferredColumnRole(DDLConstraintColumnRole):
- __slots__ = ()
- _role_name = (
- "String column name or Column object for DDL foreign key constraint"
- )
|