Inheritance
Out of various types of ORM models inheritance ormar
currently supports two of them:
- Mixins
- Concrete table inheritance (with parents set to
abstract=True
)
Types of inheritance
The short summary of different types of inheritance is:
- Mixins [SUPPORTED] - don't subclass
ormar.Model
, just define fields that are later used on different models (likecreated_date
andupdated_date
on each model), only actual models create tables, but those fields from mixins are added - Concrete table inheritance [SUPPORTED] - means that parent is marked as abstract and each child has its own table with columns from a parent and own child columns, kind of similar to Mixins but parent also is a Model
- Single table inheritance [NOT SUPPORTED] - means that only one table is created with fields that are combination/sum of the parent and all children models but child models use only subset of column in db (all parent and own ones, skipping the other children ones)
- Multi/ Joined table inheritance [NOT SUPPORTED] - means that part of the columns is saved on parent model and part is saved on child model that are connected to each other by kind of one to one relation and under the hood you operate on two models at once
- Proxy models [NOT SUPPORTED] - means that only parent has an actual table, children just add methods, modify settings etc.
Mixins
To use Mixins just define a class that is not inheriting from an ormar.Model
but is
defining ormar.Fields
as class variables.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
|
Tip
Note that Mixins are not models, so you still need to inherit
from ormar.Model
as well as define Meta
class in the final model.
A Category class above will have four additional fields: created_date
, updated_date
,
created_by
and updated_by
.
There will be only one table created for model Category
(categories
), with Category
class fields
combined with all Mixins
fields.
Note that Mixin
in class name is optional but is a good python practice.
Concrete table inheritance
In concept concrete table inheritance is very similar to Mixins, but uses
actual ormar.Models
as base classes.
Warning
Note that base classes have abstract=True
set in Meta
class, if you try
to inherit from non abstract marked class ModelDefinitionError
will be raised.
Since this abstract Model will never be initialized you can skip metadata
and database
in it's Meta
definition.
But if you provide it - it will be inherited, that way you do not have to
provide metadata
and databases
in the final/concrete class
Note that you can always overwrite it in child/concrete class if you need to.
More over at least one of the classes in inheritance chain have to provide both database
and metadata
-
otherwise an error will be raised.
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 |
|
The list of inherited options/settings is as follows: metadata
, database
and constraints
.
Also methods decorated with @property_field
decorator will be inherited/recognized.
Of course apart from that all fields from base classes are combined and created in the concrete table of the final Model.
Tip
Note how you don't have to provide abstarct=False
in the final class - it's the default setting
that is not inherited.
Redefining fields in subclasses
Note that you can redefine previously created fields like in normal python class inheritance.
Whenever you define a field with same name and new definition it will completely replace the previously defined one.
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 |
|
Warning
If you declare UniqueColumns
constraint with column names, the final model has to have
a column with the same name declared. Otherwise, the ModelDefinitionError
will be raised.
So in example above if you do not provide name
for created_date
in RedefinedField
model
ormar will complain.
created_date: str = ormar.String(max_length=200) # exception
created_date: str = ormar.String(max_length=200, name="creation_date2") # exception
Relations in inheritance
You can declare relations in every step of inheritance, so both in parent and child classes.
When you define a relation on a child model level it's either overwriting the relation defined in parent model (if the same field name is used), or is accessible only to this child if you define a new relation.
When inheriting relations, you always need to be aware of related_name
parameter, that
has to be unique across a related model, when you define multiple child classes that inherit the
same relation.
If you do not provide related_name
parameter ormar calculates it for you. This works
with inheritance as all child models have to have different class names, which are used
to calculate the default related_name
(class.name.lower()+'s').
But, if you provide a related_name
this name cannot be reused in all child models as
they would overwrite each other on the related model side.
Therefore, you have two options:
- redefine relation field in child models and manually provide different
related_name
parameters - let this for
ormar
to handle -> auto adjusted related_name are: original related_name + "_" + child model table name
That might sound complicated but let's look at the following example:
ForeignKey relations
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 |
|
Now when you will inspect the fields on Person model you will get:
1 2 3 4 5 6 7 8 9 |
|
Note how you have trucks
and buss
fields that leads to Truck and Bus class that
this Person owns. There were no related_name
parameter so default names were used.
At the same time the co-owned cars need to be referenced by coowned_trucks
and coowned_buses
. Ormar appended _trucks
and _buses
suffixes taken from child
model table names.
Seems fine, but the default name for owned trucks is ok (trucks
) but the buss
is
ugly, so how can we change it?
The solution is pretty simple - just redefine the field in Bus class and provide
different related_name
parameter.
1 2 3 4 5 6 7 8 |
|
Now the columns looks much better.
1 2 3 4 5 6 7 8 9 |
|
Note
You could also provide related_name
for the owner
field, that way the proper suffixes
would be added.
owner: Person = ormar.ForeignKey(Person, related_name="owned")
and model fields for Person owned cars would become owned_trucks
and owned_buses
.
ManyToMany relations
Similarly, you can inherit from Models that have ManyToMany relations declared but there is one, but substantial difference - the Through model.
Since in the future the Through model will be able to hold additional fields and now it links only two Tables
(from
and to
ones), each child that inherits the m2m relation field has to have separate
Through model.
Of course, you can overwrite the relation in each Child model, but that requires additional
code and undermines the point of the whole inheritance. Ormar
will handle this for you if
you agree with default naming convention, which you can always manually overwrite in
children if needed.
Again, let's look at the example to easier grasp the concepts.
We will modify the previous example described above to use m2m relation for co_owners.
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 |
|
Ormar
automatically modifies related_name of the fields to include the table name
of the children models. The default name is original related_name + '_' + child table name.
That way for class Truck2 the relation defined in
owner: Person = ormar.ForeignKey(Person, related_name="owned")
becomes owned_trucks2
You can verify the names by inspecting the list of fields present on Person
model.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
|
But that's not all. It's kind of internal to ormar
but affects the data structure in the database,
so let's examine the through models for both Bus2
and Truck2
models.
1 2 3 4 5 6 7 8 9 |
|
As you can see above ormar
cloned the Through model for each of the Child classes and added
Child class name at the end, while changing the table names of the cloned fields
the name of the table from the child is used.
Note that original model is not only not used, the table for this model is removed from metadata:
1 2 3 |
|
So be aware that if you introduce inheritance along the way and convert a model into abstract parent model you may lose your data on through table if not careful.
Note
Note that original table name and model name of the Through model is never used. Only the cloned models tables are created and used.
Warning
Note that each subclass of the Model that has ManyToMany
relation defined generates
a new Through
model, meaning also new database table.
That means that each time you define a Child model you need to either manually create the table in the database, or run a migration (with alembic).
exclude_parent_fields
Ormar allows you to skip certain fields in inherited model that are coming from a parent model.
Note
Note that the same behaviour can be achieved by splitting the model into more abstract models and mixins - which is a preferred way in normal circumstances.
To skip certain fields from a child model, list all fields that you want to skip in model.Meta.exclude_parent_fields
parameter like follows:
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 |
|
Note how you simply need to provide field names and it will exclude the parent field regardless of from which parent model the field is coming from.
Note
Note that if you want to overwrite a field in child model you do not have to exclude it, simply overwrite the field declaration in child model with same field name.
Warning
Note that this kind of behavior can confuse mypy and static type checkers, yet accessing the non existing fields will fail at runtime. That's why splitting the base classes is preferred.
The same effect can be achieved by splitting base classes like:
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 |
|
That way you can inherit from both create and update classes if needed, and only one of them otherwise.