File: customizing.rst

package info (click to toggle)
flask-sqlalchemy 3.1.1-4
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 832 kB
  • sloc: python: 2,909; makefile: 27; sh: 14
file content (178 lines) | stat: -rw-r--r-- 6,155 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
Advanced Customization
======================

The various objects managed by the extension can be customized by passing arguments to
the :class:`.SQLAlchemy` constructor.


Model Class
-----------

SQLAlchemy models all inherit from a declarative base class. This is exposed as
``db.Model`` in Flask-SQLAlchemy, which all models extend. This can be customized by
subclassing the default and passing the custom class to ``model_class``.

The following example gives every model an integer primary key, or a foreign key for
joined-table inheritance.

.. note::
    Integer primary keys for everything is not necessarily the best database design
    (that's up to your project's requirements), this is only an example.

.. code-block:: python

    from sqlalchemy import Integer, String, ForeignKey
    from sqlalchemy.orm import DeclarativeBase, Mapped, mapped_column, declared_attr

    class Base(DeclarativeBase):
        @declared_attr.cascading
        @classmethod
        def id(cls):
            for base in cls.__mro__[1:-1]:
                if getattr(base, "__table__", None) is not None:
                        return mapped_column(ForeignKey(base.id), primary_key=True)
                else:
                    return mapped_column(Integer, primary_key=True)

    db = SQLAlchemy(app, model_class=Base)

    class User(db.Model):
        name: Mapped[str] = mapped_column(String)

    class Employee(User):
        title: Mapped[str] = mapped_column(String)


Abstract Models and Mixins
--------------------------

If behavior is only needed on some models rather than all models, use an abstract model
base class to customize only those models. For example, if some models should track when
they are created or updated.

.. code-block:: python

    from datetime import datetime
    from sqlalchemy import DateTime, Integer, String
    from sqlalchemy.orm import DeclarativeBase, Mapped, mapped_column, declared_attr

    class TimestampModel(db.Model):
        __abstract__ = True
        created: Mapped[datetime] = mapped_column(DateTime, nullable=False, default=datetime.utcnow)
        updated: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow, onupdate=datetime.utcnow)

    class Author(db.Model):
        id: Mapped[int] = mapped_column(Integer, primary_key=True)
        username: Mapped[str] = mapped_column(String, unique=True, nullable=False)

    class Post(TimestampModel):
        id: Mapped[int] = mapped_column(Integer, primary_key=True)
        title: Mapped[str] = mapped_column(String, nullable=False)

This can also be done with a mixin class, inheriting from ``db.Model`` separately.

.. code-block:: python

    class TimestampMixin:
        created: Mapped[datetime] = mapped_column(DateTime, nullable=False, default=datetime.utcnow)
        updated: Mapped[datetime] = mapped_column(DateTime, default=datetime.utcnow, onupdate=datetime.utcnow)

    class Post(TimestampMixin, db.Model):
        id: Mapped[int] = mapped_column(Integer, primary_key=True)
        title: Mapped[str] = mapped_column(String, nullable=False)


Disabling Table Name Generation
-------------------------------

Some projects prefer to set each model's ``__tablename__`` manually rather than relying
on Flask-SQLAlchemy's detection and generation. The simple way to achieve that is to
set each ``__tablename__`` and not modify the base class. However, the table name
generation can be disabled by setting `disable_autonaming=True` in the `SQLAlchemy` constructor.

.. code-block:: python

    class Base(sa_orm.DeclarativeBase):
        pass

    db = SQLAlchemy(app, model_class=Base, disable_autonaming=True)


Session Class
-------------

Flask-SQLAlchemy's :class:`.Session` class chooses which engine to query based on the
bind key associated with the model or table. However, there are other strategies such as
horizontal sharding that can be implemented with a different session class. The
``class_`` key to the ``session_options`` argument to the extension to change the
session class.

Flask-SQLAlchemy will always pass the extension instance as the ``db`` argument to the
session, so it must accept that to continue working. That can be used to get access to
``db.engines``.

.. code-block:: python

    from sqlalchemy.ext.horizontal_shard import ShardedSession
    from flask_sqlalchemy.session import Session

    class CustomSession(ShardedSession, Session):
        ...

    db = SQLAlchemy(session_options={"class_": CustomSession})


Query Class
-----------

.. warning::
    The query interface is considered legacy in SQLAlchemy. This includes
    ``session.query``, ``Model.query``, ``db.Query``, and ``lazy="dynamic"``
    relationships. Prefer using ``session.execute(select(...))`` instead.

It is possible to customize the query interface used by the session, models, and
relationships. This can be used to add extra query methods. For example, you could add
a ``get_or`` method that gets a row or returns a default.

.. code-block:: python

    from flask_sqlalchemy.query import Query

    class GetOrQuery(Query):
        def get_or(self, ident, default=None):
            out = self.get(ident)

            if out is None:
                return default

            return out

    db = SQLAlchemy(query_class=GetOrQuery)

    user = User.query.get_or(user_id, anonymous_user)

Passing the ``query_class`` argument will customize ``db.Query``, ``db.session.query``,
``Model.query``, and ``db.relationship(lazy="dynamic")`` relationships. It's also
possible to customize these on a per-object basis.

To customize a specific model's ``query`` property, set the ``query_class`` attribute on
the model class.

.. code-block:: python

    class User(db.Model):
        query_class = GetOrQuery

To customize a specific dynamic relationship, pass the ``query_class`` argument to the
relationship.

.. code-block:: python

    db.relationship(User, lazy="dynamic", query_class=GetOrQuery)

To customize only ``session.query``, pass the ``query_cls`` key to the
``session_options`` argument to the constructor.

.. code-block:: python

    db = SQLAlchemy(session_options={"query_cls": GetOrQuery})