v0 integration

Stripe Webhook Signature Verification Failing

Your v0-generated Next.js application receives Stripe webhook events but fails to verify their signatures, returning 400 errors or throwing 'Webhook signature verification failed' exceptions. Payments are processed by Stripe but your application never confirms them, leaving orders in a pending state and customers confused.

Stripe signs every webhook event with a signature derived from the raw request body and your webhook signing secret. If the body is modified in any way before verification (such as being parsed as JSON), or if the signing secret is incorrect, verification fails. This is a critical security feature that prevents attackers from sending fake webhook events to your application.

This is one of the most common Stripe integration issues because Next.js API routes automatically parse the request body as JSON, which modifies the raw body that Stripe's verification expects.

Error Messages You Might See

Stripe webhook signature verification failed No signatures found matching the expected signature for payload Webhook Error: Unexpected token in JSON at position 0 stripe.webhooks.constructEvent is not a function 400 Bad Request on /api/webhook
Stripe webhook signature verification failedNo signatures found matching the expected signature for payloadWebhook Error: Unexpected token in JSON at position 0stripe.webhooks.constructEvent is not a function400 Bad Request on /api/webhook

Common Causes

  • Body pre-parsed by Next.js — Next.js API routes parse JSON body automatically, but Stripe verification requires the raw body bytes
  • Wrong webhook secret — using the API secret key (sk_...) instead of the webhook signing secret (whsec_...)
  • Local vs production secret mismatch — Stripe CLI uses a different signing secret than the dashboard webhook endpoint
  • App Router body handling — Next.js App Router route handlers consume the body stream, making it unavailable for Stripe verification
  • Middleware modifying request — authentication or logging middleware reading the request body before the webhook handler

How to Fix It

  1. Disable body parsing — for Pages Router: export const config = { api: { bodyParser: false } }
  2. Read raw body in App Router — use const body = await request.text() to get the raw body string before any JSON parsing
  3. Use correct signing secret — get the webhook signing secret (whsec_...) from Stripe Dashboard > Webhooks > your endpoint > Signing secret
  4. Verify with Stripe SDK — use stripe.webhooks.constructEvent(body, sig, webhookSecret) with the raw body string
  5. Test with Stripe CLI — run stripe listen --forward-to localhost:3000/api/webhook and use the provided signing secret for local testing
  6. Return 200 quickly — process webhook asynchronously and return 200 immediately to avoid Stripe retry storms

Real developers can help you.

Antriksh Narang Antriksh Narang 5 years+ Experienced Dev (Specially in Web Development), can help in python, javascript, react, next.js and full stack web dev technologies. Omar Faruk Omar Faruk As a Product Engineer at Klasio, I contributed to end-to-end product development, focusing on scalability, performance, and user experience. My work spanned building and refining core features, developing dynamic website templates, integrating secure and reliable payment gateways, and optimizing the overall system architecture. I played a key role in creating a scalable and maintainable platform to support educators and learners globally. I'm enthusiastic about embracing new challenges and making meaningful contributions. Matt Butler Matt Butler Software Engineer @ AWS prajwalfullstack prajwalfullstack Hi Im a full stack developer, a vibe coded MVP to Market ready product, I'm here to help Matthew Jordan Matthew Jordan I've been working at a large software company named Kainos for 2 years, and mainly specialise in Platform Engineering. I regularly enjoy working on software products outside of work, and I'm a huge fan of game development using Unity. I personally enjoy Python & C# in my spare time, but I also specialise in multiple different platform-related technologies from my day job. Krishna Sai Kuncha Krishna Sai Kuncha Experienced Professional Full stack Developer with 8+ years of experience across react, python, js, ts, golang and react-native. Developed inhouse websearch tooling for AI before websearch was solved : ) Meïr Ankri Meïr Ankri Full-stack developer specializing in React / Next.js / Node.js with 6+ years of experience. I've worked across various sectors including automotive (Reezocar/Société Générale), healthcare (Medical Link SaaS), and e-commerce (Glasman). I build web apps end-to-end, from architecture to production, with a focus on scalability, performance, and code quality. I also mentor junior developers and contribute to technical decisions and code reviews. PawelPloszaj PawelPloszaj I'm fronted developer with 10+ years of experience with big projects. I have small backend background too Jaime Orts-Caroff Jaime Orts-Caroff I'm a Senior Android developer, open to work in various fields Matthew Butler Matthew Butler Systems Development Engineer @ Amazon Web Services

You don't need to be technical. Just describe what's wrong and a verified developer will handle the rest.

Get Help

Frequently Asked Questions

Why does webhook verification fail even with the correct secret?

Most likely the request body was parsed as JSON before verification. Stripe needs the raw body bytes. Disable body parsing with config = { api: { bodyParser: false } }.

Where do I find my webhook signing secret?

Go to Stripe Dashboard > Developers > Webhooks > click your endpoint > Signing secret. It starts with whsec_. Do not use your API key (sk_...).

How do I test webhooks locally?

Install Stripe CLI, run 'stripe listen --forward-to localhost:3000/api/webhook', and use the signing secret it outputs (whsec_...) for local testing.

Related v0 Issues

Can't fix it yourself?
Real developers can help.

You don't need to be technical. Just describe what's wrong and a verified developer will handle the rest.

Get Help