{
  "query": "What is an anti-pattern in software development?",
  "query-id": 90,
  "gold_response": "Definition: An anti-pattern is a recurring “solution” that seems to help but degrades correctness, resilience, or evolvability over time. It’s identifiable by consistent symptoms and has known, better alternatives.\n\nBelow are three architecture-level anti-patterns common in distributed systems, with structure, why they’re harmful, and concrete refactorings.\n\n1) Distributed Monolith\n- Structure\n  - Multiple “microservices” tightly coupled by synchronous request chains, shared data models, or cross-service transactions.\n  - Services require coordinated releases; domain invariants span service boundaries.\n- Symptoms (what you observe)\n  - One service change forces updates/deploys in others; end-to-end tests are the only reliable tests.\n  - Outages cascade; long, deep call graphs with high fan-out/fan-in.\n- Problems (why it’s harmful)\n  - Tight runtime coupling: partial failures propagate; timeouts compound; retry storms and circuit-breaker thrash.\n  - Violates bounded contexts: low cohesion; invariants aren’t owned in one place; hard to reason about correctness.\n  - Poor testability and operability: unit tests give little confidence; observability needed across many hops.\n- Alternatives / Refactoring (decisions & trade-offs)\n  - Re-draw service boundaries around cohesive domain invariants. Decision: if two capabilities must be atomically consistent, keep them in one service; otherwise accept eventual consistency.\n  - Database-per-service; publish domain events via Outbox/CDC. Decision: at-least-once delivery + idempotency vs higher complexity for exactly-once semantics.\n  - Replace long sync chains with Sagas (orchestration vs choreography). Trade-off: simpler centralized control vs higher local autonomy.\n  - Apply resilience patterns: timeouts, circuit breakers, bulkheads, backpressure; provide graceful degradation and fallbacks.\n  - Migrate incrementally with the Strangler pattern to reduce blast radius.\n\n2) Shared Database Across Services (Integration DB)\n- Structure\n  - Multiple services directly read/write the same database/schema; cross-service foreign keys, triggers, and stored procedures enforce business rules.\n- Symptoms\n  - Any schema change requires multi-team coordination; blue/green deploys are risky.\n  - Test environments need the full shared DB to run; data coupling breaks local development.\n- Problems (why it’s harmful)\n  - Data-level coupling: schema drift ripples across services; breaks autonomy and versioning.\n  - Violates encapsulation of domain rules; invariants depend on other teams’ tables.\n  - Failure modes: deadlocks/lock contention across domains; partial updates create inconsistencies.\n  - Hard to secure and audit; least-privilege boundaries are blurred.\n- Alternatives / Refactoring (decisions & trade-offs)\n  - Database-per-service with API/event contracts, not tables, as the integration boundary. Decision: duplicate read models to avoid cross-service joins.\n  - Use event-driven propagation (CDC, domain events) to maintain materialized views or caches. Trade-off: staleness and duplication vs decoupling and deploy independence.\n  - For cross-service reporting/analytics, use a warehouse/lake with ETL/ELT; avoid ad hoc cross-service joins in OLTP.\n  - If strong cross-entity ACID is truly required, consolidate into one bounded context/service instead of faking it with a shared DB.\n\n3) Chatty Synchronous Calls (N+1 Across Services)\n- Structure\n  - A request fans out to many small synchronous calls (often inside loops) across services; fine-grained endpoints cause high round-trip counts.\n- Symptoms\n  - High p95/p99 latency; connection pools saturate; spikes cause cascading timeouts.\n  - Scaling is non-linear with traffic because each request amplifies into N remote calls.\n- Problems (why it’s harmful)\n  - Tail latency amplification: network hops multiply delays and failure probability.\n  - Tight temporal coupling: callers block on many dependencies; no degraded mode.\n  - Test brittleness; hard to load test and reproduce issues due to combinatorial fan-out.\n- Alternatives / Refactoring (decisions & trade-offs)\n  - Coarse-grain APIs and batch endpoints to eliminate N calls (e.g., fetch in bulk). Decision: combine per-entity operations vs a single vectorized call.\n  - Move composition to the edge (API Gateway/BFF) or use GraphQL with server-side caching. Trade-off: more logic at the edge vs simpler downstream services.\n  - Precompute/materialize read models (CQRS) and cache hot paths (edge cache, request coalescing). Decision: accept staleness for lower latency and fewer hops.\n  - Apply resilience: timeouts, circuit breakers, hedging (careful), idempotent retries, and fallbacks to cached snapshots.\n\nKey takeaway\n- Prefer high cohesion within a service, explicit contracts between services, and asynchronous, observable workflows where strong cross-boundary invariants are not essential. Make consistency, latency, and operability trade-offs explicit at design time.",
  "gold_information": [
    "An anti-pattern is a recurring solution that appears helpful but undermines correctness, resilience, or evolvability.",
    "An anti-pattern is identifiable by consistent symptoms and the existence of better alternatives.",
    "A distributed monolith occurs when multiple services are tightly coupled by synchronous calls, shared models, or cross-service transactions.",
    "Tightly coupled services force coordinated releases and cause cascading outages with deep call chains.",
    "Such coupling violates clear domain boundaries, reduces cohesion, and makes correctness hard to reason about.",
    "Refactoring a distributed monolith involves redrawing service boundaries around domain invariants and choosing appropriate consistency models.",
    "Using a database per service with event publication reduces coupling compared to shared transactional models.",
    "Replacing long synchronous chains with workflow patterns and resilience techniques improves fault tolerance.",
    "An incremental migration approach that isolates and replaces components gradually reduces risk during change.",
    "A shared database across services creates data-level coupling through cross-service tables, keys, and triggers.",
    "Shared schemas hinder autonomy, complicate versioning, and introduce deadlocks and inconsistencies.",
    "Refactoring a shared database uses API or event contracts as the integration boundary and duplicates read models to avoid cross-service joins.",
    "Event-driven propagation can maintain materialized views at the cost of some staleness and duplication.",
    "Cross-service analytics should use a warehouse or lake rather than ad hoc joins in transactional stores.",
    "If strong cross-entity transactions are required, the domain should be consolidated into a single service.",
    "Chatty synchronous calls fan out a request into many small remote calls, often inside loops.",
    "Such fan-out amplifies tail latency, saturates connection pools, and scales non-linearly with traffic.",
    "Refactoring chatty calls uses coarse-grained or batch endpoints to reduce round trips.",
    "Moving composition to the edge with server-side caching reduces downstream coupling and latency.",
    "Precomputed read models and caching trade freshness for lower latency and fewer network hops.",
    "Resilience practices like timeouts, circuit breakers, hedging, idempotent retries, and fallbacks mitigate failures.",
    "Favor high cohesion within services, explicit contracts between services, and asynchronous, observable workflows when strict cross-boundary invariants are unnecessary.",
    "Designs should make trade-offs among consistency, latency, and operability explicit."
  ]
}