Home >Backend Development >Python Tutorial >Creating a To-Do app with Django and HTMX - Part Adding the Todo model with TDD

Creating a To-Do app with Django and HTMX - Part Adding the Todo model with TDD

Mary-Kate Olsen
Mary-Kate OlsenOriginal
2025-01-04 07:05:34537browse

This is the second post of our series on how to build a Todo app using HTMX and Django. Click here for part 1.

In part 2, we will create the Todo model and implement its basic functionality with unit tests.

Creating the Todo model

In models.py let's create the Todo model, with the its basic attributes. We want a Todo item to be associated to a UserProfile, so that a user will only see their own items. A todo item will also have a title and a boolean attribute is_completed. We have a lot of future ideas for the Todo model, such as the ability to set a task as "in progress" besides being completed or not started, and due dates, but that's for later. Let's keep it simple by now to have something on the screen asap.

Note: In a real-world app we should probably consider using UUIDs as primary keys on the UserProfile and Todo models, but we'll keep it simple by now.

# core/models.py

from django.contrib.auth.models import AbstractUser
from django.db import models  # <-- NEW

class UserProfile(AbstractUser):
    pass

# NEW
class Todo(models.Model):
    title = models.CharField(max_length=255)
    is_completed = models.BooleanField(default=False)
    user = models.ForeignKey(
        UserProfile,
        related_name="todos",
        on_delete=models.CASCADE,
    )
    created_at = models.DateTimeField(auto_now_add=True)
    updated_at = models.DateTimeField(auto_now=True)

    def __str__(self):
        return self.title

Let's run the migrations for the new model:

❯ uv run python manage.py makemigrations
Migrations for 'core':
  core/migrations/0002_todo.py
    + Create model Todo
❯ uv run python manage.py migrate
Operations to perform:
  Apply all migrations: admin, auth, contenttypes, core, sessions
Running migrations:
  Applying core.0002_todo... OK

Writing our first tests

Let's write the first tests on our project. We want to ensure that a user will only see their own todo items, and not items from other users.

To help us writing the tests, we will add a new development dependency to our project, model-bakery, which simplifies the process of creating dummy Django model instances. We'll also add pytest-django.

❯ uv add model-bakery pytest-django --dev
Resolved 27 packages in 425ms
Installed 2 packagez in 12ms
 + model-bakery==1.20.0
 + pytest-django==4.9.0

In pyproject.toml we need to configure pytest, by adding some lines at the end of the file:

# pyproject.toml

# NEW
[tool.pytest.ini_options]
DJANGO_SETTINGS_MODULE = "todomx.settings"
python_files = ["test_*.py", "*_test.py", "testing/python/*.py"]

Now let's write our first test to ensure users only have access to their own todos.

# core/tests/test_todo_model.py

import pytest


@pytest.mark.django_db
class TestTodoModel:
    def test_todo_items_are_associated_to_users(self, make_todo, make_user):
        [user1, user2] = make_user(_quantity=2)

        for i in range(3):
            make_todo(user=user1, title=f"user1 todo {i}")

        make_todo(user=user2, title="user2 todo")

        assert {todo.title for todo in user1.todos.all()} == {
            "user1 todo 0",
            "user1 todo 1",
            "user1 todo 2",
        }

        assert {todo.title for todo in user2.todos.all()} == {"user2 todo"}

We're using a conftest.py file from pytest to have a place for all fixtures we plan to use in our tests. The model_bakery library makes it simple to create instances of UserProfile and Todo with minimal boilerplate.

#core/tests/conftest.py

import pytest
from model_bakery import baker


@pytest.fixture
def make_user(django_user_model):
    def _make_user(**kwargs):
        return baker.make("core.UserProfile", **kwargs)

    return _make_user


@pytest.fixture
def make_todo(make_user):
    def _make_todo(user=None, **kwargs):
        return baker.make("core.Todo", user=user or make_user(), **kwargs)

    return _make_todo

Let's run our tests!

❯ uv run pytest
Test session starts (platform: darwin, Python 3.12.8, pytest 8.3.4, pytest-sugar 1.0.0)
django: version: 5.1.4, settings: todomx.settings (from ini)
configfile: pyproject.toml
plugins: sugar-1.0.0, django-4.9.0
collected 1 item

 core/tests/test_todo_model.py ✓                                                                                                                                   100% ██████████

Results (0.25s):
       1 passed

Register an admin page for Todos

Finally, we can register an admin page for it:

# core/admin.py

from django.contrib import admin
from django.contrib.auth.admin import UserAdmin
from .models import Todo, UserProfile # <-- NEW

# .. previous code

# NEW
@admin.register(Todo)
class TodoAdmin(admin.ModelAdmin):
    model = Todo
    list_display = ["title", "is_completed", "user"]
    list_filter = ["is_completed"]
    search_fields = ["title"]
    list_per_page = 10
    ordering = ["title"]

We can now add some Todo's from admin!

Creating a To-Do app with Django and HTMX - Part Adding the Todo model with TDD

If you want to check the whole code until the end of part 2, you can check it on Github at the part 02 branch.

The above is the detailed content of Creating a To-Do app with Django and HTMX - Part Adding the Todo model with TDD. For more information, please follow other related articles on the PHP Chinese website!

Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn