{
  "query": "Tell me how to apply DDD Layered Architecture in Django.",
  "query-id": 25,
  "gold_response": "Applying DDD Layered Architecture in Django for your Schedule Management System\n\n1) Is DDD Layered appropriate for this project?\n- Yes. The domain has non-trivial logic (auto timetable placement, sharing/permissions, cross-client support). DDD’s separation of concerns will keep algorithmic code and collaboration rules clean and testable. \n- Caveat: Use it where it pays off (schedules, tasks, sharing). Keep simple subsystems (e.g., notifications) closer to plain MTV to avoid over-engineering.\n\n2) Map DDD layers to Django MTV (and what goes where)\n- Presentation (DDD) → Django views/routers; DRF APIViews/ViewSets or Django Ninja routers; URLConf; request auth (Firebase token verification), response mapping. No business rules.\n- Application (DDD) → services.py and selectors.py. Orchestrates use-cases (transactions, calling domain services/repositories). No direct ORM calls in services except through repositories/selectors.\n- Domain (DDD) → domain entities, value objects, domain services (pure Python). No Django/ORM imports.\n- Infrastructure (DDD) → Django ORM models (treated as infrastructure), repositories.py (implement repository interfaces against ORM), external gateways (Firebase Admin SDK), caching.\n- Django “Model” (MTV) belongs to Infrastructure. Don’t put domain rules in Django models; keep them in domain/ or application/.\n\n3) Recommended app decomposition (bounded contexts)\n- users (profile, onboarding; integrate Firebase auth mapping)\n- schedules (calendars, entries, auto-placement)\n- tasks (task CRUD, linking to schedule)\n- sharing (friendship, ACL/permissions to view/edit schedules)\n- notifications (optional, can be simpler MTV)\n\n4) Directory structure and naming conventions\nSmall-to-medium (per-domain app, pragmatic)\nproject/\n  apps/\n    schedules/\n      presentation/\n        api.py               # DRF/Ninja endpoints only\n        urls.py\n      application/\n        services.py          # write/flow logic: create_, update_, share_\n        selectors.py         # read/query logic: get_, list_, exists_\n        unit_of_work.py      # optional: transaction helpers\n      domain/\n        entities.py          # Schedule, ScheduleEntry\n        value_objects.py     # TimeSlot, RecurrenceRule, UserId\n        domain_services.py   # AutoPlacementService, conflict detection\n        rules.py             # invariants, policies\n      infrastructure/\n        models.py            # Django ORM models (infrastructure, not domain)\n        repositories.py      # ScheduleRepository, TaskRepository\n        external/\n          firebase_auth.py   # token verification, user mapping\n      validators.py          # Pydantic schemas: CreateEntryIn, EntryOut, etc.\n      __init__.py\n\nFor larger scale, keep the same shape in each domain app. Add shared libs under apps/common/ (clock, id generation, emailing) and keep them framework-agnostic where possible.\n\nNaming guidelines\n- services.py: imperative verbs (create_schedule_entry, auto_place_entries, share_schedule_with_friend)\n- selectors.py: get_, list_, count_, exists_ (get_schedule, list_user_entries)\n- validators.py (Pydantic): Input DTOs suffix In, Output DTOs suffix Out (CreateEntryIn, EntryOut, ShareScheduleIn)\n- repositories.py: <Entity>Repository with method names that express intent (save_entry, list_entries_for_range, lock_calendar)\n- domain/*.py: Nouns for entities, precise nouns for value objects, verbs for domain services (place_entry, detect_conflicts)\n- presentation: api.py classes/functions named by resource (ScheduleEntryAPI, ScheduleViewSet) and route names by action\n\n5) Step-by-step wiring (view → service → selector/validator → repository → domain)\nUse case: Create an entry with automatic placement\n- Presentation (api.py)\n  - Verify Firebase ID token (middleware/auth backend). Map to local user_id.\n  - Parse/validate request body via validators.CreateEntryIn.\n  - Call services.create_schedule_entry(actor_id, data).\n  - Return EntryOut as JSON; set HTTP status and errors here only.\n- Application (services.py)\n  - Wrap in transaction.atomic().\n  - Read necessary state via selectors (e.g., list existing entries for range; read user preferences).\n  - Call domain_services.auto_place(...) to compute candidate TimeSlot(s) and apply conflict rules.\n  - Persist via repositories (save_entry or bulk_save) to ORM.\n  - Return an EntryOut DTO (no ORM objects leaking out).\n- Selectors (selectors.py)\n  - read-only, optimized queries (prefetch_related/select_related, caching if needed) like list_user_entries_for_range(user_id, start, end).\n- Domain (domain_services.py, entities.py)\n  - AutoPlacementService: pure Python. Inputs: desired duration, constraints, existing TimeSlot[]. Output: chosen TimeSlot or error (e.g., NoAvailableSlot).\n  - Entities/VOs enforce invariants (TimeSlot(start < end), ScheduleEntry cannot overlap in the same calendar in memory).\n- Infrastructure (repositories.py, models.py)\n  - Map domain entities to Django ORM models, handle select_for_update where needed.\n  - Keep ORM-specific details here (query building, bulk operations).\n\nMinimal code sketch (abbreviated, illustrative)\n- presentation/api.py\n  - data = CreateEntryIn.model_validate(request.data)\n  - out = services.create_schedule_entry(actor_id=request.user.id, data=data)\n  - return Response(out.model_dump(), status=201)\n- application/services.py\n  - with transaction.atomic():\n      entries = selectors.list_user_entries_for_range(actor_id, data.window_start, data.window_end)\n      slot = domain_services.auto_place(data.requested_duration, data.constraints, entries)\n      entry = domain.entities.ScheduleEntry.new(actor_id, data.title, slot)\n      saved = repositories.ScheduleRepository.save_entry(entry)\n      return validators.EntryOut.from_domain(saved)\n- domain/value_objects.py\n  - class TimeSlot: validate start < end; overlaps(other) -> bool\n- infrastructure/repositories.py\n  - class ScheduleRepository: save_entry(domain_entry) -> domain_entry; uses models.ScheduleEntry.objects.create(...)\n\n6) Django-specific notes (how it plugs into MTV)\n- Views: Keep thin. They authenticate, validate (Pydantic), dispatch to services, and serialize output. No business rules.\n- Models: Treat Django models as infrastructure. Avoid business logic in model methods/signals. Keep them as persistence schemas.\n- Templates: Not needed for Flutter; your presentation is API-only. If you add server-rendered pages later, treat them as another presentation adapter.\n\n7) Firebase Auth integration\n- Infrastructure: apps/schedules/infrastructure/external/firebase_auth.py implements token verification using Firebase Admin SDK.\n- Presentation: a custom DRF/Ninja authentication class or Django middleware verifies the token, then attaches a local user identity to request.user (create a local User on first login if needed). Keep domain logic unaware of Firebase specifics; services operate on a stable UserId value object.\n\n8) ORM choice and trade-offs\n- Default Django ORM (recommended here):\n  - Pros: tight integration with Django admin/migrations, stable, good enough for most scheduling queries, easy onboarding.\n  - Cons: repository abstraction still needed to avoid leaking ORM to services.\n- SQLAlchemy (via repository):\n  - Pros: richer query composition in some cases, easier domain/ORM mapping isolation.\n  - Cons: more plumbing in a Django project, loses batteries-included admin, dual stack complexity.\n- Decision for this project: Start with Django ORM behind repositories. Revisit if query complexity or performance demands surpass what Django ORM handles comfortably.\n\n9) Practical guidelines for each layer\n- Presentation\n  - DRF or Django Ninja. If you prefer Pydantic-first validation, Django Ninja pairs naturally; with DRF, still keep Pydantic DTOs in validators.py to avoid serializer-logic bloat.\n  - Error mapping: translate domain/application exceptions to HTTP codes here (e.g., NoAvailableSlot → 409).\n- Application\n  - One function per use case (transaction script), named by intent. Keep orchestration and transactions here.\n- Domain\n  - Keep it pure. No imports from Django or requests. Encapsulate invariants in VOs and domain services.\n- Infrastructure\n  - Repositories shield services/selectors from ORM details. Co-locate caching and locking strategies here.\n\n10) Example endpoints and naming (for your features)\n- Automatic placement: services.auto_place_entries, domain_services.auto_place\n- Schedule CRUD: services.create_schedule, services.update_schedule, selectors.get_schedule\n- Task linking: services.link_task_to_entry, selectors.list_tasks_for_entry\n- Friends/visibility: services.share_schedule_with_friend, services.update_share_permission, selectors.list_shared_schedules\n- Validators: CreateScheduleIn/Out, CreateEntryIn/Out, ShareScheduleIn\n\n11) When to choose Layered vs plain MTV\n- Choose Layered (this project): multiple domains, algorithmic logic, permissions, long-term maintenance, multi-client API.\n- Choose plain MTV (or a thin variant): admin-only utilities, simple notification preferences, small reports with trivial queries.\n\n12) First implementation checklist (1–2 days)\n- Create apps: users, schedules, tasks, sharing.\n- Add the directory skeletons above; commit empty modules (api.py, services.py, selectors.py, validators.py, repositories.py, domain/*).\n- Implement Firebase auth adapter (infrastructure/external) and presentation auth class.\n- Define core domain VOs and entities: TimeSlot, Schedule, ScheduleEntry, Friendship.\n- Implement repositories around Django ORM models; wire migrations.\n- Build one vertical slice: POST /schedules/entries with auto placement (validators → service → domain → repository → out DTO).\n- Add 1–2 selectors with efficient prefetching and basic caching for list endpoints.\n\nSummary\n- DDD Layered fits your schedule app. Keep Django’s strengths (ORM, routing) but move rules into application/domain layers.\n- Use a domain-centric app layout with services/selectors/validators/repositories separated.\n- Start with Django ORM behind repositories; prefer Pydantic DTOs; pick DRF or Ninja based on your team’s comfort.\n- Implement one vertical slice to set conventions, then scale out by domain.",
  "gold_information": [
    "Layered architecture suits domains with non-trivial scheduling logic and collaboration rules.",
    "Apply the layered approach selectively and keep simple subsystems in a basic model–view style.",
    "The presentation layer resides in views and routers and contains no business rules.",
    "The presentation layer performs authentication, input validation, and response mapping.",
    "The application layer lives in services and orchestrates use cases and transactions.",
    "The application layer calls repositories and selectors instead of using the ORM directly.",
    "The domain layer contains entities, value objects, domain services, and policies as pure code.",
    "The domain layer must not import framework or ORM modules.",
    "The infrastructure layer holds ORM models, repository implementations, external gateways, and caching.",
    "Framework model classes belong to the infrastructure layer rather than the domain layer.",
    "Avoid business logic in ORM models and signals.",
    "Decompose the system into bounded contexts such as users, schedules, tasks, sharing, and notifications.",
    "Structure each domain app with presentation, application, domain, and infrastructure subpackages.",
    "Keep API endpoints in the presentation folder without business logic.",
    "Place write orchestration in services and read queries in selectors within the application layer.",
    "Optionally add a unit-of-work helper for transaction management in the application layer.",
    "Put entities, value objects, domain services, and rules under the domain folder.",
    "Place ORM models, repositories, and external adapters under the infrastructure folder.",
    "Use schema-based validators to define input and output DTOs.",
    "Name service functions with imperative verbs that reflect intent.",
    "Name selector functions with get_, list_, count_, or exists_ prefixes.",
    "Suffix input DTO names with In and output DTO names with Out.",
    "Name repository methods by intent and avoid leaking persistence concerns to callers.",
    "The presentation layer maps domain and application exceptions to HTTP status codes.",
    "The application layer gathers state via selectors, invokes domain services, and persists through repositories within a transaction.",
    "Selectors offer read-only, optimized queries and may apply caching.",
    "Domain services implement algorithms such as automatic placement as pure functions.",
    "Entities and value objects enforce invariants like non-overlapping time slots.",
    "Repositories map domain objects to persistence and apply locking where necessary.",
    "Keep views thin and templates optional for API-only clients.",
    "Treat server-rendered pages as an alternative presentation adapter if added later.",
    "Integrate an external identity provider via an infrastructure adapter and a presentation-layer authentication class.",
    "Keep domain code unaware of identity-provider details by operating on stable user identifiers.",
    "Prefer the framework’s built-in ORM behind repositories for most use cases.",
    "Consider an alternative ORM only if query complexity or isolation needs justify added plumbing.",
    "Choose layered architecture for multi-domain systems with algorithms, permissions, and multi-client APIs.",
    "Use a simpler model–view pattern for trivial utilities and small features.",
    "Start by creating separate domain apps for key bounded contexts.",
    "Scaffold the four layers in each app to establish consistent conventions.",
    "Implement repositories and migrations to back domain persistence.",
    "Build one vertical slice from endpoint through domain to database to validate the approach.",
    "Add selectors with efficient prefetching and basic caching for list endpoints.",
    "Keep business rules in the domain and application layers while using the framework for routing and persistence."
  ]
}