{
  "title": "Django Service Layer & Repository Pattern for Complex Apps",
  "turns": [
    {
      "turn": 1,
      "user": "My Django models are getting too complex. Business logic that involves multiple models is all crammed into a few model methods. It's becoming hard to test and maintain. What are some common patterns to handle this in Django?",
      "assistant": "That's a common challenge as Django projects grow. Moving business logic out of models and views improves separation of concerns. A popular pattern is using a \"Service Layer.\"\n\n### What is a Service Layer?\nA service layer is a set of functions or classes that orchestrate business logic. It sits between your views (the presentation layer) and your models (the data layer).\n\n**Key Responsibilities:**\n- Encapsulate a single business use case (e.g., `place_order`, `register_user`).\n- Coordinate interactions between multiple models.\n- Handle transactions to ensure data consistency.\n\n### Simple Example\nImagine you have `Order` and `OrderItem` models. Placing an order involves creating both.\n\n**Directory Structure:**\n```\nyour_app/\n├── models.py\n├── services.py <-- Business logic lives here\n└── views.py\n```\n\n**`services.py`:**\n```python\nfrom django.db import transaction\nfrom .models import Order, OrderItem, Product\n\n@transaction.atomic\ndef place_order(user, product_ids: list[int]):\n    \"\"\" Creates an order and its items for a given user and product list. \"\"\"\n    order = Order.objects.create(customer=user)\n    for product_id in product_ids:\n        product = Product.objects.get(id=product_id)\n        OrderItem.objects.create(\n            order=order,\n            product=product,\n            price=product.price\n        )\n    # ... any other logic like sending a confirmation email ...\n    return order\n```\n\n**`views.py`:**\n```python\nfrom django.http import HttpResponse\nfrom . import services\n\ndef create_order_view(request):\n    product_ids = request.POST.getlist('product_ids')\n    order = services.place_order(user=request.user, product_ids=product_ids)\n    return HttpResponse(f\"Order {order.id} created!\")\n```\n\n**Benefits:**\n- **Testable:** You can now test `place_order` in isolation without needing an HTTP request.\n- **Reusable:** Other parts of your application (e.g., a management command) can call `place_order`.\n- **Leaner Views/Models:** Your views handle HTTP concerns, and models handle data representation."
    },
    {
      "turn": 2,
      "user": "Following up on that, my biggest issue is testing. When I test a service function like `place_order`, I still have to set up the entire database state for it and its related models. Is there a way to test the business logic in isolation from the Django ORM?",
      "assistant": "Yes, you can achieve that by abstracting the data access layer using the **Repository Pattern**. This pattern allows you to test your business logic (services) without touching the database by using a \"fake\" repository during tests.\n\n### The Repository Pattern in Django\nThe goal is to separate *how* you get data (the Django ORM) from the business logic that *uses* it.\n\n1. **Define an Interface (Abstract Repository):** Use Python's `abc` module or `typing.Protocol` to define what methods your repository should have. This is a contract, not concrete code.\n2. **Create a Concrete Implementation:** Write a class that implements the interface using the Django ORM.\n3. **Use the Repository in Your Service:** Your service function will depend on the abstract interface, not the concrete ORM implementation.\n\n### Step-by-Step Example\nLet's refactor our `place_order` service.\n\n**1. Define the Interface (`your_app/repositories.py`)**\n```python\nfrom abc import ABC, abstractmethod\nfrom .models import Product\n\nclass ProductRepository(ABC):\n    @abstractmethod\n    def get_product_by_id(self, product_id: int) -> Product:\n        ...\n# You would also create repositories for Order, etc.\n```\n\n**2. Create the Concrete Django ORM Implementation (`your_app/repositories.py`)**\n```python\n# ... in the same file ...\nfrom .models import Product\n\nclass DjangoProductRepository(ProductRepository):\n    def get_product_by_id(self, product_id: int) -> Product:\n        return Product.objects.get(id=product_id)\n```\n\n**3. Refactor the Service to Use the Repository (`your_app/services.py`)**\nThe service now receives a repository object instead of calling `Product.objects` directly. This is called \"Dependency Injection.\"\n```python\nfrom django.db import transaction\nfrom .models import Order, OrderItem\nfrom .repositories import ProductRepository  # Depend on the abstract interface\n\n@transaction.atomic\ndef place_order(user, product_ids: list[int], product_repo: ProductRepository):\n    order = Order.objects.create(customer=user)\n    for product_id in product_ids:\n        # Use the repository, NOT the ORM directly\n        product = product_repo.get_product_by_id(product_id)\n        OrderItem.objects.create(order=order, product=product, price=product.price)\n    return order\n```\n\n**4. Testing in Isolation (`your_app/tests.py`)**\nNow you can create a \"fake\" repository for your tests that doesn't use the database.\n```python\nimport unittest\nfrom .services import place_order\nfrom .repositories import ProductRepository\nfrom .models import Product  # We still need the model class for type hints\n\nclass FakeProductRepository(ProductRepository):\n    \"\"\"A repository that returns dummy data from memory.\"\"\"\n    def get_product_by_id(self, product_id: int) -> Product:\n        # Return a fake Product object, no database hit!\n        return Product(id=product_id, name=\"Test Product\", price=100.0)\n\nclass ServiceTests(unittest.TestCase):\n    def test_place_order_logic(self):\n        # 1. Create an instance of our fake repository\n        fake_repo = FakeProductRepository()\n        # 2. Call the service with the fake repository\n        # (Assuming you also have fake repositories for Order, etc.)\n        order = place_order(user=..., product_ids=..., product_repo=fake_repo)\n        # 3. Assert your business logic, without any database interaction\n        self.assertIsNotNone(order)\n        # ... more assertions ...\n```\n\nThis approach makes your core business logic completely independent of Django's ORM, making it much easier and faster to test."
    }
  ]
}