Show your Heart application
- General info
- Technologies
- External services and resources
- Local environment setup for development
- Internationalization and localization
- Testing
- Linter and formatter
- Commands
TO DO: include general description of the project's scope.
Many other technologies are used in the form of Python packages. Check out
the list in project.toml
.
- TailwindCSS
- FlowBite (a controls and components library for Tailwind)
- AlpineJS
- HTMX
- Docker
- npm, for Tailwind compilation
- Poetry, a python package manager
This project (which is a rewrite of a previous version made in Java) was originally developed by a group of cooperatives and other entities, raising the need of combining different skill sets and different ways of working. It also aims to be an easy to contribute and maintain open source project.
Given the expertise of the contributing cooperatives, two possible paradigms were considered:
- A SPA with Vue and Nuxt, consuming JSON API provided by Django Rest Framework.
- A plain HTML front-end rendered by the Django templates engine, with HTMX and AlpineJS for advanced interactivity.
We opted for the option 2, meaning that by default each interaction comes with a full page reload. In places where we considered that a more advanced ajax or client-side interaction is valuable, we used HTMX and/or AlpineJS.
Some of the reasons to go with the paradigm 2 are:
- Reduce the work load and complexity of the front-end development.
- Simplify the deployment steps and local environment setup.
- Allows us to kick-start the development by using this django boilerplate.
- We consider that this stack makes it easier to make changes in the future if other companies take over the maintenance and for any new contributors.
- Not using a heavy front-end JS framework removes the need for the front-end to handle routing, state, internationalization, duplication of logic, extra tests, higher-level framework and packages maintenance, build processes, etc.
There are several different frameworks and technologies to develop custom web applications to choose from. These are the reasons for our particular selection.
- Django: is widely used and robust, with a big community and ecosystem of packages.
- HTMX: in the context of the Django scene, it's the third most popular framework after React and jQuery. Check out the "What JavaScript framework(s) do you use?" section in this survey.
- TailwindCSS: is very popular (check out the "What CSS framework(s) do you use?" section of the aforementioned survey). It removes the need to define a CSS structure and naming decisions (which typically is complex and difficult to understand and maintain in the long term), instead, each HTML element contains all the information about how it's displayed.
- AlpineJS: Is lightweight, increasingly popular, and lets you define interactions directly in the HTML elements. At the same time it seamlessly integrates with vanilla JS blocks if you need to. It's a great complement to HTMX because you can let HTMX handle the requests to the backend and AlpineJS handle the client-side behaviour.
HTMX, TailwindCSS and AlpineJS all three have in common that the code goes into the HTML elements' properties, simplifying the development and maintenance and allowing for a good implementation of the Locality of Behaviour principle.
Nowadays, there's a constant stream of videos and articles sharing experiences and analysis of the HTMX stack. For now, we consider this one to be the best resource to understand it in a real life case.
The application needs are:
- To send transactional emails. It comes with a package to use the Sendgrid API, and it can use any other provider by using Django's SMTP backend, or adding other libraries to use other provider's APIs.
- To store media files. It comes with the necessary libraries to use any S3 provider.
- Access a PostgreSQL database.
- Storing the Docker images.
- A remote logging platform.
- A VPS or serverless service to run the Docker container.
In your local environment the app will run in a Docker container, along with a Postgres and a Selenium container.
The app's container mounts the /src
folder, therefore every change done in
the files inside /src
will also be applied in the container's code, then
Gunicorn will detect the change and automatically reload the app.
Given this setup, you could already work by cloning the repository, starting the
containers, and just editing the code inside /src
.
The problem with that is that it will be more difficult to browse around the code, for example, your IDE will not be able to find the source of the imported modules.
We recommend you to set it up in a way that you have a local virtual environment which will allow your IDE to correctly check references, jump to code definitions, etc, and that's the reason the next steps require you to install a couple of tools in your system.
If you don't intend to make any changes to the Tailwind classes or any css, you could setup only the backend part.
- Clone the repository into a local folder.
- Install Python 3.12, we recommend pyenv.py
for it, by doing
pyenv install 3.12
and then in the repository folderpyenv local 3.12
. Finally, check the version by going to the local folder and runningpython -V
. - Install Poetry or update it (
poetry self update
) to the latest version. - At the project's root, run
poetry install
. - Copy the
docker/.env.example
file todocker/.env
and modify as needed, but the initial setup should let the project initialize already. - Install or update Docker and from the
docker/
folder rundocker compose up --build
. - In another terminal, access the docker's container bash (
docker exec -it showyourheart-app bash
) and runpython manage.py migrate
. - Open
http://localhost:1601
.
In the future, when you pull a new version of the app, repeat the last 3 steps to make sure that you create an updated version of the Docker image and database migrations are applied.
Tailwind needs to "compile" the css files by scanning the templates and creating a compact css that includes only the Tailwind classes used in the project.
If you intend to change any Tailwind classes from the html files, you need to access container's bash and run these commands:
docker exec -it showyourheart-app bash
cd /front
npx tailwindcss -i /srv/assets/styles/input.css -o /srv/assets/styles/output.css --watch
This will start a process that will recompile the css each time you make changes to any tailwind classes in the templates.
Important: if you modify the package.json
, package-lock.json
or
tailwind.config.js
files, you must rebuild the docker image. One way to do it is
by running docker compose up --build
in the /docker
directory.
- Make use of comments for translators to help clarify the intent of the string.
- Make sure to use placeholders when needed. I.e.:
_("Hi %s, what's up?") % name
. - All string literals in python files or html templates must be written in english.
We use the core Django translations system, so the process of generating the translations files and the reference for the functions and template tags must be checked there.
For convenience, we configured the Docker image so it can create the translations files, given that it some systems you might need to install some additional library.
Run this command in the docker container:
python manage.py makemessages --all
Tips:
- When editing a .po file, start by checking that all new strings are in english. It could happen that by mistake a developer leaves a string in some other language. In that case, first translate this strings to english directly in the source code and then generate the .po files again.
- Beware of the strings that look like URL paths: they are URL paths and their translations must follow the same format (lowercase, without spaces and no special characters other than dashes).
Tests are stored in a subfolder tests
, both in the Django's main project folder
/src/project
and for each app in src/apps
.
We'll decide how to test every section of the application depending on what we consider more useful for each case. The main testing tool will be the Selenium test, that will reproduce as many user workflows as possible, specially the more critical ones. We'll use unit testing for areas where reproducing the steps with Selenium is too repetitive or tedious.
To run the tests connect to the Docker's container bash terminal and run:
python manage.py test
Tests will only work if you are running the project using the docker-compose.yml
and therefore the showyourheart-selenium
container is up.
We use Ruff for checking and applying linting and formatting.
To run these commands connect to the Docker's container bash terminal.
Check format:
ruff format --check .
Perform code formatting:
ruff format .
Check linting:
ruff check .
Perform linting to the code:
ruff check --fix .
In some cases, you might need:
ruff check --fix --unsafe-fixes .
And (as always) check the diff of all changes before commiting.
Note that migrations are not excluded from the linter/formatter on purpose because some of them are data migrations that we manually code, and we don't see any problem with the automatically generated ones being formatted as well.
To run the linter connect to the Docker's container bash terminal and run:
dennis-cmd lint --errorsonly src
There's an extended developer's guide here that is important for everyone to read before contributing to the project.
We use the django-extra-settings
library
to be able to customize the project on run-time.
When the project's migrations run for the first time, all the dynamic settings
declared in the EXTRA_SETTINGS_DEFAULTS
setting will be created.
You can add or edit settings there as long as you are aware that these changes will only be applied when the project is deployed to an empty database. For that reason, this should only happen before we hit production or 1.0.0.
Once the project is in production, any new dynamic setting or change in existing ones must be introduced by creating data migrations manually.
To use an image or file setting to refer to the uploaded file in a template you
can follow the example of the LOGO
setting in base.html
:
{% get_setting "LOGO" as logo %}
<img src="{% if logo %}{{ logo.url }}{% endif %}" alt="{% get_setting 'PROJECT_NAME' %} logo" />
Remember to load the tag in the template as described in the library's documentation.
The project
app includes the loaddevdata
command. Its purpose is to
populate the database with initial data, either generated or loaded from
fixtures, so when the developers needs to reset the database they don't have to
repeat all the set up steps and manually create models entries to work with.
Initially it only includes the creation of a Superuser and an Admin user which will not be superuser, but will have access to the admin panel and have all the permission groups assigned.
This user is created because normally superuser access is not given to our customer, instead, we keep a superuser account for the developers and create different user roles using permission groups, and within those users, the one with a higher access level is this Admin.
For that reason, when developing features within the admin panel, it's necessary to try them using the different user roles and NOT the superuser account.
Extend this command whenever new models are created or updated to maintain a good set of initial fixtures to work with.
The users
app will create an initial superuser during the migration process
(at 0002_data_superuser.py
) if the SUPERUSER_EMAIL
and SUPERUSER_PASSWORD
environment variables are set.
If the superuser was not created or was deleted and you want to recreate it with the environment settings credentials, you can run the following command:
python manage.py loaddevdata
Which will create it as well as populate the database with initial data for development.
Alternatively, you can also use the Django built-in createsuperuser
command.
If you need to split an admin.py or models.py file into multiple files using the
python packages technique (which is, you create a folder called admin
with a
__init__.py
file, and in this file you import the admin classes), beware of
two problems:
a. When running tests it will probably raise an AlreadyRegistered
exception
because of the reasons explained here.
b. Ruff will not accept the imports in the __init__
file unless you make them
explicit re-imports, like this:
from .base_admin import ModelAdmin as ModelAdmin