惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

Stack Overflow Blog
Stack Overflow Blog
酷 壳 – CoolShell
酷 壳 – CoolShell
P
Proofpoint News Feed
Apple Machine Learning Research
Apple Machine Learning Research
T
Tailwind CSS Blog
罗磊的独立博客
F
Future of Privacy Forum
The Register - Security
The Register - Security
MyScale Blog
MyScale Blog
P
Privacy & Cybersecurity Law Blog
V
Visual Studio Blog
T
Tenable Blog
F
Fortinet All Blogs
D
Docker
V
Vulnerabilities – Threatpost
Cyberwarzone
Cyberwarzone
A
Arctic Wolf
T
Threat Research - Cisco Blogs
I
Intezer
T
Tor Project blog
大猫的无限游戏
大猫的无限游戏
MongoDB | Blog
MongoDB | Blog
博客园 - 司徒正美
AWS News Blog
AWS News Blog
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
G
GRAHAM CLULEY
T
Threatpost
美团技术团队
K
Kaspersky official blog
F
Fox-IT International blog
Hugging Face - Blog
Hugging Face - Blog
Vercel News
Vercel News
P
Palo Alto Networks Blog
Google DeepMind News
Google DeepMind News
T
The Blog of Author Tim Ferriss
S
Schneier on Security
腾讯CDC
Cisco Talos Blog
Cisco Talos Blog
C
Check Point Blog
博客园 - 叶小钗
I
InfoQ
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
Blog — PlanetScale
Blog — PlanetScale
F
Full Disclosure
T
True Tiger Recordings
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
E
Exploit-DB.com RSS Feed
L
LINUX DO - 热门话题
J
Java Code Geeks
C
CERT Recently Published Vulnerability Notes

DEV Community

How I self-hosted my Flask app on an old laptop for almost free I built a free DSA interview prep site because I was tired of the existing options I built an AI agent that migrates Next.js Pages Router to App Router Prisma Query Logging and PostgreSQL: Where the ORM Ends and the Database Begins From Browser to Server : The Journey of an HTTP Request (Demystifying the Web’s Infrastructure) Santa Augmentcode Intent Ep.6 I Benchmarked 17 ESLint Security Plugins. Only One Found Every Vulnerability. How to Build a High-Performance Image Optimization Pipeline in 5 Minutes 50 Linux Commands Every DevOps Engineer Must Know Less Toil, More Flow - Automating the Path from Request to Implementation The Code Review Checklist I Actually Use How I run a small blog on Astro 5 + Content Collections Git: Best Practices for Professionals How IBM Bob Became My Everyday Coding Companion Solana Passkey Wallet: Replacing Seed Phrases with SIMD-0075 I built a small browser puzzle game about arrows I wrapped Claude Code in a zsh function. Here's every decision I almost got wrong. Mobile Game Optimization: A Unity Developer's Checklist Git: Best Practices for Beginners Three days I lost chasing a ghost that was already dead on disk Why Too Many Parts Hurt ClickHouse Performance Guardrails for Agent Output: Pluggable Validation Before and After LLM Calls Gemma Forge: Local AI Without the Setup Wall From Half‑dead Prototype to Local‑Only AI Medical Assistant: Rewiring MedClinic with GitHub Copilot Runninig a forkbomb in Jenkins What’s Actually Happening When You Use Git Preventing Recursive Tool Loops in LangChain Agents Building a Rock-Paper-Scissors CLI with TypeScript — Union Types, Conditionals, and Jest Your AI Coding Agent Wastes 80% of Its Context. Fixed That with Graph Theory. Why Flutter Has Become the Go-To Framework for Fintech App Development We built a scripting language just for AI agents. Here's why. Stop building AI inboxes. Build decision layers instead. Meme Monday Why I Built @editora/ui-react? Are AI tools the next level of abstraction in software development? Identity on Solana: Your Wallet Is Your Account One API Call Changed Everything The Internet Career Nobody Talks About Enough: What Is DevRel? Solar Panel Wiring Diagram: Series vs Parallel Hello everyone! Glad to join the dev.to community I Built an AI Agent That Tailors My Resume - Here's How Agents Actually Work I Built a WhatsApp OTP + AI Chatbot Platform for African Businesses MTP Explained — And Why It Matters for Android on Mac Most Beginners Learn Full-Stack Development Backwards GitHub Glow-Up: Open Source, READMEs, Badges, Streaks, Git and gh CLI System Design Cheat Sheet: Concepts Every Developer Should Know Are Junior Developer Roles Actually Dying? A Fresher's Honest Take Using DigitalOcean Droplets as Ephemeral Sandboxes for AI Agents I built a VSCode extension that visualises your code navigation as a call tree — made for legacy codebase pain Vite predev/prebuild: chaining scripts without losing your mind A website to save you from messy browser tabs Dear Web2 Developer... Solana is here calling Postgres JSONB indexes: GIN vs BTREE on the same column The $5 AI That Remembers Everything What are your goals for the week? #180 Zettelkasten for Developers: A Practical Method That Works OpenClaw vs Hermes Agent: Stars, Downloads & Usage 2026 `act` vs. `waitFor` Global Teams Don’t Struggle With Time Zones. They Struggle With Context Python as a JavaScript Dev $5.4 Billion in Damage. 8.5 Million Machines Down. Three YAML Controls Would Have Prevented It. Here's the Structural Analysis. 🚫 Stop Using PN532 V1 for Your NFC Projects (Real Debugging Experience) Probabilistic Graph Neural Inference for smart agriculture microgrid orchestration for extreme data sparsity scenarios Inference Is Becoming the New Steady-State Cost Center Why AI-Generated Code Is Always Good Enough — And Never Great I built a dark admin dashboard template in HTML — no React, no npm, just pure HTML What is the Difference Between Lattice-Based and Hash-Based Signatures? Next.js App Router caching: revalidate, dynamic, and no-store without the folklore Next.js App Router caching: revalidate, dynamic y no-store sin folklore I built Stashly — a full-stack content manager with a rich text editor published: false tags: react, node, mongodb, typescript Why I Started Building React Projects Instead of Just Watching Tutorials ? Every Tool Eventually Becomes Tuesday Nobody Warns You That Real Software Engineering Feels Chaotic Tích hợp VNPay, Stripe trong Odoo 19 BeautifulSoup and Requests for Web Scraping With Python: When Simple Still Works I Was Stuck Debugging React — Then Developer Tools Changed It Buck Converter Ripple: Sizing the Inductor and Capacitor With Confidence AWS Just Made Its MCP Server Generally Available. Here's What It Actually Gives AI Agents. RAMPART Tests Your AI Agents in Dev. What Catches Malicious Tool Calls in Production? Vibe Team Software Engineering: What a Real AI Human Dev Team Workflow Actually Looks Like An npm Package for AI Agent Orchestration Just Shipped With Its Front Door Unlocked. Here's What the CVE Actually Reveals. Microsoft Foundry Just Added CI/CD for AI Agents. Here's What That Actually Changes. The Best Career Insurance Is a Tech Event You Don't Want to Attend Your GitHub Profile Already Tells Recruiters More Than Your Resume. Most Devs Just Don't Surface It. How to Add Execution Budgets to OpenAI Agents SDK Binary Tree Interview Problems: 6 Traversal Patterns, 15 Problems We trained a personal voice DoRA on Qwen3-8B for $1.50 — beat stock model 100% in blind A/B Stop Leaking API Keys: Why I Built a Local-First Vault for Developers 🔐 RAG Explained: How Retrieval-Augmented Generation Actually Works I Built a Fast Async JioSaavn API Wrapper in Python 🎧 chown & chgrp Deploying Your First App on Kubernetes: A Beginner's Guide (Minikube & Kind) Logs in code It's called a PR "review" for a reason DePIN GPU Market: The Failed Job Receipt Developers Should Demand Why Your AI Agent Monitoring is Wrong (And How to Fix It) Lock Down Your Cloud Shares: A Beginner’s Guide to Azure Files Security. Building a Multi-Channel Content Syndication Pipeline with EmDash Plugins Turn Your Phone Into Voice Input for Any React Text Field Which package is bloating your Docker image?
Prisma query logging y PostgreSQL: dónde termina el ORM y empieza la base
Juan Torchia · 2026-05-25 · via DEV Community

Prisma query logging y PostgreSQL: dónde termina el ORM y empieza la base

Activé query logging en Prisma, vi los queries llegando a la consola, y asumí que tenía visibilidad completa sobre lo que pasaba en la base. Spoiler: no la tenía.

Los logs de Prisma muestran la query que el cliente envía y el tiempo que tardó desde la perspectiva del ORM — incluyendo serialización, red y el overhead del driver. Lo que no muestran es qué hace PostgreSQL con esa query adentro: si usó un índice, si hizo un sequential scan, si hubo lock wait, si el planner eligió mal el plan. Esa parte vive en Postgres, no en el ORM.

Mi tesis: los query logs de Prisma son una herramienta de debugging de patrones, no de diagnóstico de base de datos. Confundirlos lleva a buscar el problema en el lugar equivocado y a tomar decisiones de optimización sin evidencia real.


Qué dice la documentación oficial de Prisma — y qué no dice

La documentación oficial de Prisma logging es clara sobre lo que el sistema ofrece: tres niveles de log (INFO, WARN, ERROR) más el nivel especial query, que emite la query SQL, los parámetros, la duración y el target.

La configuración básica se ve así:

// Inicializamos el cliente con logging de queries habilitado
const prisma = new PrismaClient({
  log: [
    {
      emit: 'event',   // emitimos como evento para procesarlo nosotros
      level: 'query',
    },
    {
      emit: 'stdout',  // errores y warnings van directo a consola
      level: 'error',
    },
    {
      emit: 'stdout',
      level: 'warn',
    },
  ],
})

// Escuchamos el evento de query para loguear con estructura
prisma.$on('query', (e) => {
  console.log({
    query: e.query,       // SQL generado por Prisma
    params: e.params,     // parámetros bindeados
    duration: e.duration, // duración en ms desde el cliente Prisma
    target: e.target,     // nombre del datasource (ej: "db")
  })
})

Enter fullscreen mode Exit fullscreen mode

Lo que la doc no menciona explícitamente es que e.duration mide el tiempo desde que el cliente Prisma envía la query hasta que recibe la respuesta. Ese número incluye latencia de red, parsing del driver, serialización del resultado y eventual contención del connection pool. No es el tiempo que PostgreSQL tardó en ejecutar la query. Son cosas distintas y mezclarlas genera diagnósticos incorrectos.

Para capturar el tiempo real de ejecución en Postgres, necesitás pg_stat_statements o EXPLAIN ANALYZE directamente en la base. Esas herramientas viven del lado del motor, no del ORM.


El error más común: confundir duración de cliente con tiempo de ejecución en Postgres

Un patrón típico en equipos que empiezan a usar Prisma: ven una query con duration: 800 en los logs y concluyen que "la query es lenta". Puede ser cierto. Pero también puede ser que la query en Postgres tarde 20ms y los 780ms restantes sean contención en el pool, latencia de red o deserialization overhead de un resultado muy grande.

Sin distinción entre esos tiempos, cualquier optimización es especulativa.

Un escenario concreto donde esto pega: consultás una tabla con muchas columnas y seleccionás SELECT * porque Prisma, por defecto con findMany(), trae todos los campos. El tiempo de ejecución en Postgres puede ser razonable, pero el tiempo de transferencia y serialización del payload puede ser lo que infla la duración que ves en el log. La solución no es un índice — es un select explícito:

// En vez de traer todos los campos (comportamiento default de findMany)
const usuarios = await prisma.usuario.findMany()

// Seleccionamos solo lo que necesitamos
const usuarios = await prisma.usuario.findMany({
  select: {
    id: true,
    email: true,
    creadoEn: true,
    // excluimos columnas grandes como avatarBase64, metadataJson, etc.
  },
})

Enter fullscreen mode Exit fullscreen mode

Este cambio puede bajar la duración visible en logs sin tocar ningún índice. Si hubieras ido directo a Postgres a "optimizar la query", habrías perdido tiempo buscando un problema que no existía ahí.


Cuándo Prisma logging alcanza y cuándo necesitás mirar PostgreSQL

Esta es la decisión técnica que más importa. Armé una guía de criterios basada en lo que cada capa puede y no puede mostrarte:

Prisma query logging alcanza cuando:

  • Detectás un N+1: ves decenas de queries iguales en el log para una sola request. Este es el caso de uso donde Prisma logging brilla. Si querés profundizar en patrones de N+1 en Server Actions, hay más contexto en este post sobre Prisma y Next.js 16.
  • Buscás queries innecesarias: logs te muestran si una pantalla hace queries que no debería hacer.
  • Verificás que select explícito funciona: podés confirmar que Prisma genera el SQL correcto antes de llegar a la base.
  • Depurás filtros mal escritos: la query logueada te muestra si el where se traduce como esperás.
  • Mapeás frecuencia de queries por endpoint: con emit por evento podés contar y agrupar sin herramientas externas.

Necesitás mirar PostgreSQL directamente cuando:

  • La duración del cliente es alta pero el patrón de queries parece correcto: investigá pg_stat_statements para ver tiempo real en Postgres.
  • Sospechás un sequential scan: EXPLAIN ANALYZE en la misma query te dice si hay un índice que no se está usando.
  • Hay bloqueos o deadlocks: pg_locks y pg_stat_activity son las herramientas. Prisma no ve esto.
  • El problema aparece bajo carga pero no en local: puede ser contención del pool o autovacuum que se activa con volumen real. Ninguna de las dos cosas aparece en logs de ORM.
  • Querés entender el plan del query planner: el plan puede cambiar con los datos reales y con las estadísticas de la tabla. Solo EXPLAIN ANALYZE te lo muestra.
-- Corrés esto directamente en PostgreSQL para ver el plan real de ejecución
EXPLAIN (ANALYZE, BUFFERS, FORMAT TEXT)
SELECT u.id, u.email
FROM "Usuario" u
WHERE u.estado = 'activo'
ORDER BY u."creadoEn" DESC
LIMIT 50;

-- Buffers=true muestra cuántos bloques leyó de disco vs caché
-- Analyze=true ejecuta la query de verdad (cuidado en tablas con writes pesados)

Enter fullscreen mode Exit fullscreen mode


Checklist de diagnóstico: por dónde empezar

Antes de optimizar algo, respondé estas preguntas en orden:

1. ¿El log de Prisma muestra muchas queries para una sola operación?
   → Sí: revisá N+1, eager loading, relaciones mal cargadas
   → No: seguí

2. ¿El SQL generado tiene sentido? ¿Traemos columnas que no usamos?
   → Problema: agregá select explícito en Prisma
   → OK: seguí

3. ¿La duración en Prisma es alta de forma consistente o esporádica?
   → Esporádica: investigá pool contention, conexiones agotadas
   → Consistente: seguí

4. ¿Tenés pg_stat_statements habilitado en PostgreSQL?
   → No: habilitarlo es el próximo paso antes de seguir diagnosticando
   → Sí: buscá la query por query text y mirá mean_exec_time real

5. ¿El plan de ejecución usa índice o sequential scan?
   → EXPLAIN ANALYZE en la query real con datos reales
   → Si hay seq scan en tabla grande con filtros, ahí está el problema

Enter fullscreen mode Exit fullscreen mode


Límites claros: qué no podés concluir solo con Prisma logs

Esto importa y no lo suficiente gente lo dice:

  • No podés concluir que "la query es lenta" basándote solo en e.duration sin saber cuánto de ese tiempo es Postgres vs overhead del driver vs red.
  • No podés detectar lock waits ni deadlocks desde el cliente ORM. Un query que espera un lock va a aparecer con duración alta, pero el motivo es invisible desde Prisma.
  • No podés ver si autovacuum está compitiendo con tus writes. Ese ruido de fondo aparece como lentitud intermitente que no correlaciona con ningún patrón en el log del cliente.
  • No podés validar que un índice se está usando sin EXPLAIN. Que Prisma genere un WHERE correcto no garantiza que Postgres elija el índice que esperás.
  • No podés reproducir el comportamiento bajo carga real solo con logs locales. El pool tiene un tamaño máximo (configurable con connection_limit en el datasource), y la contención aparece cuando hay concurrencia real.

Si el diagnóstico requiere cualquiera de esos puntos, el log de Prisma es un punto de partida, no la respuesta.


FAQ: Prisma query logging y PostgreSQL

¿Cómo habilito el query logging en Prisma sin mandar todo a stdout?

Usá emit: 'event' en vez de emit: 'stdout' y manejás el evento prisma.$on('query', handler). Así podés filtrar, estructurar o mandarlo a tu sistema de logging sin contaminar la salida estándar en producción.

¿El duration del log de Prisma es el mismo que el tiempo de ejecución en PostgreSQL?

No. La duración del cliente Prisma incluye serialización, latencia de red y overhead del driver. El tiempo real de ejecución en Postgres lo obtenés con pg_stat_statements o EXPLAIN ANALYZE. Pueden diferir bastante dependiendo del tamaño del resultado y la latencia de red.

¿Cómo habilito pg_stat_statements en PostgreSQL?

Agregás pg_stat_statements a shared_preload_libraries en postgresql.conf, reiniciás el servidor y ejecutás CREATE EXTENSION IF NOT EXISTS pg_stat_statements; en la base. Desde ahí podés consultar pg_stat_statements para ver tiempos de ejecución reales por query.

¿Tiene sentido loguear queries en producción?

Depende del volumen. En producción con tráfico alto, loguear cada query puede generar overhead de I/O significativo. Una alternativa más prudente es loguear solo queries que superen un threshold de duración, o usar OpenTelemetry con sampling. El tema de observabilidad con trazas lo cubrí en el contexto de Spring Boot pero los principios son similares — más detalles en el post de OpenTelemetry.

¿Prisma tiene alguna forma de hacer EXPLAIN ANALYZE directamente?

No nativa. Podés usar prisma.$queryRaw para ejecutar EXPLAIN ANALYZE manualmente:

// Ejecutamos EXPLAIN ANALYZE via queryRaw para ver el plan real
const plan = await prisma.$queryRaw`
  EXPLAIN (ANALYZE, BUFFERS, FORMAT JSON)
  SELECT id, email FROM "Usuario" WHERE estado = 'activo'
`
console.log(JSON.stringify(plan, null, 2))

Enter fullscreen mode Exit fullscreen mode

Esto es útil en desarrollo para validar que el planner usa los índices que esperás.

¿Si no veo queries lentas en los logs de Prisma, puedo asumir que la base está bien?

No. La ausencia de queries lentas en el cliente no garantiza ausencia de problemas en Postgres. Puede haber table bloat, índices sin actualizar, autovacuum retrasado o queries que corren rápido individualmente pero generan presión acumulada. El diagnóstico de la base requiere sus propias herramientas.


Mi postura: son capas distintas, no alternativas

Lo incómodo de este tema es que la mayoría de la documentación de Prisma (incluyendo la oficial) muestra cómo configurar el logging sin aclarar explícitamente qué mide y qué no mide. Eso genera una suposición razonable pero incorrecta: que tener query logging activado equivale a tener visibilidad sobre el comportamiento de la base.

No es así. Prisma logging es debugging de capa ORM. PostgreSQL tiene su propia capa de observabilidad y necesita sus propias herramientas. Las dos son necesarias y se complementan, pero no se reemplazan.

Mi recomendación práctica: usá Prisma logging para detectar patrones de queries (N+1, selects innecesarios, queries duplicadas por request). Cuando el patrón parece correcto y el problema persiste, pasá a pg_stat_statements y EXPLAIN ANALYZE. No saltees el primer paso porque es más fácil de activar, pero tampoco te quedes ahí si la respuesta no aparece.

El próximo paso concreto: si tenés pg_stat_statements deshabilitado en tu base, eso es lo primero que habilitaría. Sin él, estás diagnosticando a ciegas en la capa que más importa.


Fuentes originales:


Este artículo fue publicado originalmente en juanchi.dev