{
  "query": "Tell me about the differences between DDD and MTV architecture in system architecture patterns.",
  "query-id": 24,
  "gold_response": "DDD vs. MTV in Django: core concepts, pros/cons, and concrete structures\n\nQuick take\n- Where business logic lives:\n  - MTV: often in views (and occasionally model methods).\n  - DDD: in the Domain layer (entities/value objects/services); views/controllers stay thin.\n- Model responsibility:\n  - MTV: Active Record (Django ORM model = data + persistence API).\n  - DDD: Rich, persistence-ignorant domain model; persistence handled by repositories in Infrastructure.\n\nCore concepts\n1) MTV (Model–Template–View)\n- Model: Django ORM classes map to tables; provide CRUD via .save(), .objects, etc.\n- View: Orchestrates request handling; often ends up containing business logic and calling ORM models.\n- Template: Presentation only.\n\n2) DDD in Django\n- Layers\n  - Presentation: Django views/DRF views/routers; HTTP in/out only.\n  - Application: Use-cases; coordinates repositories and domain objects; transaction boundaries.\n  - Domain: Entities, Value Objects, Aggregates, Domain Services (pure business logic, no framework imports).\n  - Infrastructure: Adapters to tech (Django ORM models, repository implementations, email/S3/redis gateways).\n- Key building blocks (tied to Django)\n  - Entity: Has identity and lifecycle (e.g., Order). Implement as pure Python class in domain; map to a Django ORM model in infrastructure.\n  - Value Object: Immutable concepts identified by value (e.g., Money, Email). Python dataclass; may be wrapped by custom Django model fields for persistence.\n  - Aggregate/Root: Consistency boundary (e.g., Order with OrderLines; Order is root). Save/load via the root’s repository.\n  - Domain Service: Operation that doesn’t fit an entity (e.g., pricing policy); pure Python in domain/services.py.\n  - Repository: Interface in domain; Django ORM-backed implementation in infrastructure (repositories.py) that maps to/from domain objects.\n  - Bounded Context: A cohesive subdomain (e.g., orders, billing). In Django, often one app per bounded context (don’t force it; split when language/meaning diverges).\n  - Ubiquitous Language: Use business terms consistently across code, tests, and docs.\n\nPros and cons\nMTV\n- Pros\n  - Fast to ship; minimal ceremony; excellent for CRUD.\n  - Leverages Django’s batteries (admin, ORM, forms, auth) directly.\n  - Lower cognitive load for small teams/features.\n- Cons\n  - Fat views and tight coupling to ORM over time.\n  - Business rules spread across views/signals/model methods; harder to test in isolation.\n  - Invariants and cross-entity consistency are harder to centralize.\n- Best fit\n  - Simple CRUD apps, prototypes, admin tools, short-lifecycle projects.\n\nDDD\n- Pros\n  - Business logic centralized and technology-agnostic; high testability.\n  - Clear boundaries/layers; easier to evolve complex rules.\n  - Aggregates protect invariants; repositories hide persistence details.\n- Cons\n  - More files/abstractions; higher initial cost.\n  - Mapping between domain objects and ORM adds boilerplate.\n  - Needs discipline on boundaries and language.\n- Best fit\n  - Complex domains, long-lived products, teams expecting evolving rules and integrations.\n\nDjango examples: the same use-case in MTV vs DDD\nUse-case: “Apply a 10% discount if user is VIP and order total > 50,000.”\n\nMTV (logic in the view)\n```python\n# orders/views.py\nfrom django.shortcuts import render, get_object_or_404\nfrom .models import Order\n\ndef apply_discount_view(request, order_id):\n    order = get_object_or_404(Order, id=order_id)\n    user = request.user\n\n    # Business logic is here in the view\n    if getattr(user, \"grade\", None) == \"VIP\" and order.total_price > 50000:\n        order.final_price = order.total_price * 0.9\n        order.save()\n\n    return render(request, \"orders/detail.html\", {\"order\": order})\n```\n\nDDD (logic in the domain; repository handles persistence)\nDomain\n```python\n# orders/domain/entities.py\nfrom dataclasses import dataclass\nfrom decimal import Decimal\n\n@dataclass\nclass Order:\n    id: int\n    total_price: Decimal\n    final_price: Decimal | None = None\n\n    def apply_vip_discount(self, threshold: Decimal = Decimal(\"50000\")) -> None:\n        if self.total_price > threshold:\n            self.final_price = (self.total_price * Decimal(\"0.9\")).quantize(Decimal(\"1.\"))\n```\n\n```python\n# orders/domain/repositories.py\nfrom typing import Protocol\nfrom .entities import Order\n\nclass OrderRepository(Protocol):\n    def get(self, order_id: int) -> Order: ...\n    def save(self, order: Order) -> None: ...\n```\n\nInfrastructure (Django ORM + repository impl)\n```python\n# orders/infrastructure/models.py (Django ORM)\nfrom django.db import models\n\nclass OrderModel(models.Model):\n    total_price = models.DecimalField(max_digits=12, decimal_places=0)\n    final_price = models.DecimalField(max_digits=12, decimal_places=0, null=True, blank=True)\n```\n\n```python\n# orders/infrastructure/repositories.py\nfrom .models import OrderModel\nfrom ..domain.entities import Order\nfrom ..domain.repositories import OrderRepository\n\nclass DjangoOrderRepository(OrderRepository):\n    def get(self, order_id: int) -> Order:\n        m = OrderModel.objects.get(id=order_id)\n        return Order(id=m.id, total_price=m.total_price, final_price=m.final_price)\n\n    def save(self, order: Order) -> None:\n        m = OrderModel.objects.get(id=order.id)\n        m.total_price = order.total_price\n        m.final_price = order.final_price\n        m.save()\n```\n\nApplication service (transaction + orchestration)\n```python\n# orders/application/services.py\nfrom django.db import transaction\nfrom ..domain.repositories import OrderRepository\n\n@transaction.atomic\ndef apply_discount_service(user_grade: str, order_id: int, repo: OrderRepository) -> None:\n    order = repo.get(order_id)\n    if user_grade == \"VIP\":\n        order.apply_vip_discount()\n    repo.save(order)\n```\n\nPresentation (thin view)\n```python\n# orders/presentation/views.py\nfrom django.http import HttpResponse\nfrom ..application.services import apply_discount_service\nfrom ..infrastructure.repositories import DjangoOrderRepository\n\nrepo = DjangoOrderRepository()\n\ndef apply_discount_view(request, order_id: int):\n    apply_discount_service(getattr(request.user, \"grade\", \"\"), order_id, repo)\n    return HttpResponse(\"Discount applied.\")\n```\n\nExample directory structures\nMTV (typical Django app)\n```\nmyproject/\n└── orders/\n    ├── migrations/\n    ├── templates/orders/detail.html\n    ├── admin.py\n    ├── models.py\n    ├── urls.py\n    └── views.py  # often contains business logic\n```\n\nDDD-flavored within Django (layered per bounded context)\n```\nmyproject/\n└── orders/\n    ├── presentation/\n    │   ├── urls.py\n    │   └── views.py           # thin; HTTP concerns only\n    ├── application/\n    │   └── services.py        # use-cases, transactions\n    ├── domain/\n    │   ├── entities.py        # Entities, Value Objects\n    │   ├── services.py        # Domain services (if needed)\n    │   └── repositories.py    # Repository interfaces (Protocols)\n    └── infrastructure/\n        ├── models.py          # Django ORM models\n        └── repositories.py    # Repository implementations (ORM mapping)\n```\n\nWhen to choose which\n- Choose MTV if your app is primarily CRUD, rules are simple, and you need speed. Keep views small, push reusable logic into model methods or service functions, and you’ll be fine.\n- Choose DDD if rules are complex, invariants matter, and you expect frequent changes. Start by extracting an Application layer and a Domain layer around your most complex use-cases.\n- Hybrid path (pragmatic): begin with MTV, then carve out domain entities/services and repositories for the hot spots as complexity emerges.\n\nFinal summary\n| Category | MTV (Model-Template-View) | DDD (Domain-Driven Design) |\n| --- | --- | --- |\n| Core philosophy | Rapid web dev via framework roles | Model the business; protect invariants |\n| Logic location | Mostly in views (sometimes models) | Domain layer (entities/VOs/services) |\n| Model style | Active Record (ORM knows DB) | Rich, persistence-ignorant domain model |\n| Persistence | Model.save(), QuerySet in app code | Repository hides ORM in Infrastructure |\n| Layers | View–Model–Template | Presentation–Application–Domain–Infrastructure |\n| Pros | Simple, productive, low ceremony | Testable, maintainable for complex rules |\n| Cons | Coupling, fat views, scattered rules | More files, mapping overhead |\n| Best fit | CRUD, prototypes, admin tools | Complex, long-lived, evolving domains |",
  "gold_information": [
    "In model–template–view architecture, business logic often resides in the view layer and sometimes in model methods.",
    "In domain-driven design, business logic is concentrated in a domain layer, keeping controllers and views thin.",
    "Model–template–view commonly uses an active record style where models couple data with persistence operations.",
    "Domain-driven design favors a rich, persistence-agnostic domain model with repositories handling persistence.",
    "Domain-driven design separates concerns into presentation, application, domain, and infrastructure layers.",
    "The application layer in domain-driven design coordinates use-cases and sets transaction boundaries.",
    "Domain-driven design models include entities, value objects, aggregates, and domain services.",
    "Repositories in domain-driven design are defined as domain interfaces with infrastructure implementations.",
    "Aggregates in domain-driven design define consistency boundaries and protect invariants.",
    "Model–template–view excels at simple CRUD and rapid development with low ceremony.",
    "Model–template–view tends to accumulate fat views and tight coupling to the ORM over time.",
    "Scattered business rules in model–template–view hinder isolated testing and centralized invariants.",
    "Domain-driven design improves testability and adaptability for complex and evolving rules.",
    "Domain-driven design introduces extra abstractions and mapping overhead that raise initial cost.",
    "Model–template–view is a good fit for prototypes, admin tools, and short-lifecycle applications.",
    "Domain-driven design suits complex domains and long-lived products with frequent rule changes and integrations.",
    "In domain-driven design, presentation handles I/O only while domain logic remains framework-independent.",
    "In model–template–view, persistence is invoked directly from application code via the ORM.",
    "In domain-driven design, persistence concerns are hidden behind repositories implemented in the infrastructure.",
    "Domain-driven design encourages bounded contexts and a ubiquitous language aligned with the business."
  ]
}