The betterangels_backend is built on Django, a Python web framework. It also utilizes Celery for distributed task processing, enabling the scheduling and execution of tasks.
Prerequisites:
- Visual Studio Code (VSCode)
- Docker
- Development containers
Setup:
-
Run Docker
-
Clone the monorepo repo from github
-
Open it in a VSCode workspace
-
Cmd + Shift + P and select "Dev Containers: Rebuild and Reopen in Container" (this command take a few minutes to complete). It will close and reopen a new VSCode workspace
-
Apply migrations:
yarn nx run betterangels-backend:migrate
-
Install dependencies (if you just built the container, this step might already be done)
poetry install
To start the Django backend server:
yarn nx start betterangels-backend
Once started, you can access the Django development server at the default address: http://localhost:8000/admin/ or the port you've configured.
Login using creds: [email protected]
/admin
-
Activate a poetry shell
poetry shell
-
Navigate to
betterangels-backend
app directorycd apps/betterangels-backend/
-
Start Django admin shell
django-admin shell --settings betterangels_backend.settings
To run the full test suite:
yarn nx test betterangels-backend
To run an individual test, add the full path of the test in dot notation. Example:
yarn nx test betterangels-backend accounts.tests.UsersManagersTests.test_create_user
To run tests with breakpoints via the terminal, you'll need to use a poetry shell
as described in the section above, then:
-
Add any breakpoint to your code/tests:
Example (using
IPython
which is already in the dev dependencies):from IPython import embed; embed()
-
Run tests using
python manage.py test
. Example:python manage.py test accounts/tests/test_user_manager.py::UserManagerTestCase::test_create_user
To use VSCode's debugger:
- Select the
Testing
(test tube) icon on the left - Add a breakpoint (red dot) on the line you would like to break at
- Navigate through the test directory to find the test you want to run and click the
Debug Test
(🐞▶️ ) button - Once the breakpoint is hit, you will be moved to the
Run and Debug
screen where you can step through your code - To access the interactive shell and run commands directly, go to the
Debug Console
tab of the terminal on the bottom
The scheduler, powered by Celery Beat, is responsible for triggering scheduled tasks. If you'd like to test the scheduled tasks, you will need to run the scheduler.
The scheduler is responsible for triggering scheduled tasks. If you'd like to test the scheduled tasks, you will need to run the scheduler.
To start the scheduler:
yarn nx run betterangels-backend:start-scheduler
Workers handle task execution. They can run without the scheduler if you're not testing scheduled tasks.
To start a worker:
yarn nx run betterangels-backend:start-worker
Note: While workers can run independently of the scheduler, the scheduler requires at least one worker to process the scheduled tasks.
Django provides a flexible way to handle email backends. By default, our configuration uses the file-based email backend to capture sent emails as files. This is helpful for local development and testing without actually sending real emails.
Configure the .env File: Set the email backend in your .env file to use the file-based backend:
POST_OFFICE_EMAIL_BACKEND=django.core.mail.backends.filebased.EmailBackend
Checking Sent Emails: After configuring the backend, any email sent from the application will be stored as a file under tmp/app-emails in the project's directory.
Reading Emails: Navigate to the tmp/app-emails directory and open the email files to view the rendered HTML content. Each file represents an individual email sent from the application.
Before switching to SES, ensure that you're authenticated to AWS using Single Sign-On (SSO).
aws sso login
Configure the .env File: Update the email backend in your .env file to use the django_ses backend:
POST_OFFICE_EMAIL_BACKEND=django_ses.SESBackend
Sending & Receiving: With the above configuration, any emails sent from the application will now be dispatched through Amazon SES.
Django PG History is used in this project to track changes to model instances over time. The django-pghistory
should already be added into the project's setting. If you're developing a new model or want to add historical tracking to an existing model, follow these steps:
Update your model to allow pghistory to track events. Edit the model file to include:
import pghistory
@pghistory.track()
class YourModel(models.Model):
# your fields here
Optionally, include specific events to track:
import pghistory
@pghistory.track(
pghistory.InsertEvent("your_model.add"),
pghistory.UpdateEvent("your_model.update"),
pghistory.DeleteEvent("your_model.remove"),
)
class YourModel(models.Model):
# your fields here
yarn nx run betterangels-backend:makemigrations
yarn nx run betterangels-backend:migrate
In order to keep track of the of an event, a custom context manager (pghistory.context
) needs to be added whenever the tracked model is being modified:
class Mutation:
def modify_tracked_model_mutation(self, info: Info, data: SomeMutationInput) -> TrackedModelType:
with transaction.atomic(), pghistory.context(
tracked_model_id=data.tracked_model_id, timestamp=timezone.now(), label=info.field_name
):
# your mutation logic here
To track the historical events for a certain model based on their id and timestamp, query the Context
table using the custom tracked_model_id
and timestamp
metadata fields added in Step 3.
Context.objects.filter(metadata__tracked_model_id=tracked_model_id).order_by("metadata__timestamp")