Coding Patterns
Coding patterns we follow on our back end
The 3 Layers
Each backend consists of 3 layers that build on top of each other. Starting from the bottom layer, the layers are:
Models: Defines the data tables (a.k.a. models) and columns (a.k.a. fields).
Serializers: Handles converting data to objects and vice versa.
Views: Defines our API endpoints.
Because [data] models are our bottom layer, it can be said that our backend is model-centric. This means (almost) all our serializers and views and are model-serializers and model-views.
Models
Below are the conventions we follow when modelling our data.
First, understand how to define a Django model.
File Structure
TL;DR
For each model, create a file named after the model in the directory models
.
All models should be imported into models/__init__.py
to support importing multiple models from models
.
Any custom logic defined in a model-file should be tested in the directory models
, where each model has its own test-file following the naming convention {model}_test.py
. Model-tests should inherit ModelTestCase
and set their type parameter to be the model they are testing. The name of the model-test-case should follow the convention Test{model}
.
Warehousing Data
TL;DR
Determine whether the data stored in a model needs to be warehoused. Data should be warehoused if it can contribute to analyzing users' behavior. By default, new models should be warehoused as it's a rare occurrence that meaningful insights cannot be gained from analyzing its data.
Defining a model as a warehouse-model allows us to sync data from our database to our data warehouse before it's deleted from our database. See our data deletion strategy.
Defining a warehouse-model:
Defining a non-warehouse-model:
Defining Fields
TL;DR
When defining fields of any type, always set the verbose name and help text. These arguments aid future developers and super-users on the Django admin to understand the purpose of these fields.
Defining Foreign Keys
TL;DR
When defining a foreign key between 2 models, a few steps need to be taken to inform our static type checker of backward relationships between objects.
Say we have the following models:
From this we understand that an instance of Car
has the attribute owner
of type Person
. Django will create an attribute for "backward relationships" on each related object with the naming convention {model}_set
at runtime. Therefore, Person
has the attribute car_set
which is a set of type Car
.
The problem is static type checkers do not know attributes for backward relationships are going to being auto-generated. Therefore, we need to add type hints.
We import the model we are type hinting inside of if t.TYPE_CHECKING
to avoid circular-imports. We also define the name of the backward relationships to be the plural of the model's name to improve readability.
Defining Meta Classes
Models' meta class should inherit TypedModelMeta
to support type hinted Meta classes.
Defining Managers
TL;DR
If you're defining a model with a custom manager:
If you're defining an abstract model with a custom manager:
A model's manager should be defined within the model's class as Manager
. If the model inherits WarehouseModel
, the manager should inherit WarehouseModel.Manager
, providing a string of the model's name as the type parameter to inform the manager of the type of model it will be managing. Below the manager's definition, create a class-level attribute on the model called objects
which has its type set to Manager
and its value set to an instance of the Manager
class.
Django requires the User
model's manager to be defined globally (not locally nested within the User
class). This is an exception and is required by Django so that it may auto-discover the location of the user-manager.
If you need to create an abstract model with a custom manager, add a generic type parameter to the manager which is bound to the manager's model. A type variable will need to be defined before the abstract model, following the naming convention Any{model}
. However, to bind the type variable to the model before the model is defined, use a lazy binding by referencing the model's name as a string. Below the abstract model's manager, set the type of objects
to be Manager[t.Self]
so that the manager is managing the type of the inherited model (not the abstract model).
Defining Constraints
TL;DR
When defining constraints, the naming convention is {field}__{condition}
.
Test your custom constraint in models/{model}_test.py
. Your test name should follow the naming convention test_constraint__{constraint_name}
. The unit test should leverage CFL's assertion helper assert_check_constraint
, which will check the constraint is enforced under the expected conditions.
Define Verbose Names
When defining a model, set its verbose names in its meta attributes.
QuerySets as Properties
TL;DR
If there are query-sets that are filtered by a model instance's attributes, create properties on the model's class that return the query-sets. This achieves a shorthand that makes the code base less repetitive and more readable. It's only worth doing this if the query-set is used in 2 or more locations in the code base.
We import Song
outside the top level to avoid circular-imports.
Defining Settings
If a model has business logic that impacts its functionality based on custom conditions, add class-level settings on the model in all caps to configure the conditions.
For example, if the Person
model has a property which checks if the person is too young:
This will also help to make tests more robust as values can be dynamically calculated.
Serializers
Below are the conventions we follow when serializing our data.
First, understand how to serialize a Django model.
Model Serializers
For each model, we create a file with the naming convention {model}.py
in the directory serializers
. Within a model's serializer-file, we create one or more serializers. It's advised to create one serializer per model-view-set action. Each model-serializer should inherit CFL's ModelSerializer
by default and set the type parameter to the model being serialized.
To avoid repetitively setting the model being serialized to Person
, a base serializer may be created with the naming convention Base{model}Serializer
.
All serializers should be imported into serializers/__init__.py
to support importing multiple serializers from serializers
.
Any custom logic defined in a model-serializer-file should be tested in the directory serializers
, where each model has its own serializer-test-file following the naming convention {model}_test.py
. Model-serializer-test-cases should inherit CFL's ModelSerializerTestCase
, set the type parameter to be the model being serialized and set model_serializer_class
to the model-serializer being tested. The name of the model-serializer-test-case should follow the convention Test{model}Serializer
.
Model List Serializers
When defining model-list-serializers, follow the naming convention {model}ListSerializer
. Model-list-serializers should inherit CFL's ModelListSerializer
by default to support bulk-creating and bulk-updating model instances.
When overriding validate(self, attrs)
, always call super().validate(attrs)
first before implementing any additional validations. This is necessary to ensure the data is valid before bulk-creating or bulk-updating.
When overriding update(self, instance, validated_data)
, both instance
and validated_data
are of equal length and sorted in the correct order. Therefore, to get the data per model, use zip
on instance
and validated_data
.
Validation Errors
When defining validation errors, always set the code
of the validation error so that it may be asserted in a test. Error codes must always be unique per validation function so that they may be individually asserted in tests.
When testing validation errors, the error_code
that is expected to be raised must be provided. When testing the validations of a field, the test must follow the naming convention test_validate_{field}__{validation_error_code}
and use CFL's assert_validate_field
helper. When testing general validations, the test must follow the naming convention test_validate__{validation_error_code}
and use CFL's assert_validate
helper.
Testing create() and update()
If create(self, validated_data)
was overridden in a model-serializer, a test named test_create
will need to be created and use CFL's assert_create
helper. Likewise, if update(self, instance, validated_data)
was overridden in a model-serializer, a test named test_update
will need to be created and use CFL's assert_update
helper.
If create(self, validated_data)
was overridden in a model-list-serializer, a test named test_create_many
will need to be created and use CFL's assert_create_many
helper. Likewise, if update(self, instance, validated_data)
was overridden in a model-list-serializer, a test named test_update_many
will need to be created and use CFL's assert_update_many
helper.
Testing to_representation()
If to_representation(self, instance)
was overridden in a model-serializer, a test named test_to_representation
will need to be created and use CFL's assert_to_representation
helper.
Views
Below are the conventions we follow when viewing our data.
First, understand how to view a Django model.
Model View Sets
For each model, we create a file with the naming convention {model}.py
in the directory views
. Within a model's view-file, we create one view-set following the naming convention {model}ViewSet
. Each model-view-set should inherit CFL's ModelViewSet
by default and set the type parameter to the model being viewed.
All views should be imported into views/__init__.py
to support importing multiple views from views
.
Any custom logic defined in a model-view-file should be tested in the directory views
, where each model has its own view-test-file following the naming convention {model}_test.py
. Model-view-set-test-cases should inherit CFL's ModelViewSetTestCase
, set the type parameter to be the model being viewed, set model_view_set_class
to the model-view-set being tested and set basename
to the basename used to register the model-view-set in the urls. The name of the model-view-set-test-case should follow the convention Test{model}ViewSet
.
Register URLs
Each model-view-set needs to be registered in urls.py
using DRF's DefaultRouter
. Each registration needs to set prefix
to be the plural name of the model with kebab-casing, viewset
to be the model-view-set being registered and basename
to be the singular of the singular name of the model with kebab-casing.
If a model has a foreign key to another model, it's also acceptable to set prefix
to be a sub-path of the related model, where each model's name is plural with kebab casing.
Permissions
Permissions must at the very least be set per action in get_permissions
, but further conditions can be specified. All permission must be imported from CFL's package to support unit testing.
If get_permissions
is overridden, a test will need to be created for each action the model-view-set provides where each test follows the naming convention test_get_permissions__{action}
. Each test should use CFL's assert_get_permissions
helper.
Limiting Default Actions
If any of a model's default actions should not be allowed, they can be disallowed in one of 2 ways.
The first is by excluding an action's HTTP method from http_method_names
. The is the preferred way if no action with that HTTP method should be allowed.
The second is to not permit some or all users to trigger the action. The is the preferred way if only some actions for a HTTP should be allowed.
In the above example, actions that do not use HTTP GET are not allowed (e.g. The destroy action which uses HTTP DELETE is not allowed). Furthermore, of the 2 actions which use HTTP GET: list and retrieve, list is not allowed and only authenticated users are allowed to retrieve.
Custom Actions
If a custom action needs to be created for a model-view-set, use CFL's action
decorator. A custom action should set detail
and methods
.
Testing Actions
Each action provided by a model-view-set should have one test case that calls the action as it is intended to be called. Normally, unit tests can be said to test the "unhappy scenarios" - where we are asserting that the bad scenarios we are expecting to encounter are handled as expected. However, we will also test the "happy scenario" of each action to assert that the action does indeed work.
CFL has provided client-helpers to call and assert each of the default actions. Each action-test should follow the naming convention test_{action}
.
Custom Update or Bulk-Update Actions
If a custom update or bulk-update action needs to be created for a model-view-set, use CFL's ModelViewSet.update_action
or ModelViewSet.bulk_update_action
helpers and pass the name of the action as the first argument. As these actions require a serializer to be provided, you'll also need to override get_serializer_class
.
NOTE: Update actions use HTTP PUT and so it needs to be added to http_method_names
.
To test the "happy scenarios" of each update action, use CFL's client helpers update
or bulk_update
.
Testing get_queryset()
If you are overriding a model-view-set's get_queryset
callback, a test will need to be created for each action at the very least where each test follows naming convention test_get_queryset__{action}
. Each test will need to use CFL's assert_get_queryset
helper. Additional test dimensions can be specified if other factors affect a queryset.
In the above example, the default queryset is all insured cars. For the drive action, only insured owners can drive their cars. For the list action, insured or uninsured cars can be listed.
Testing get_serializer_class()
If you are overriding a model-view-set's get_serializer_class
callback, a test will need to be created for each action at the very least where each test follows naming convention test_get_serializer_class__{action}
. Each test will need to use CFL's assert_get_serializer_class
helper.
Testing get_serializer_context()
If you are overriding a model-view-set's get_serializer_context
callback, only the actions that have additional context will need to have a test created. The test should follow the naming convention test_get_serializer_context__{action}
. Each test will need to use CFL's assert_get_serializer_context
helper.
Last updated