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.

prajwalfullstack prajwalfullstack Hi Im a full stack developer, a vibe coded MVP to Market ready product, I'm here to help Victor Denisov Victor Denisov Developer Caio Rodrigues Caio Rodrigues I'm a full-stack developer focused on building practical and scalable web applications. My main experience is with **React, TypeScript, and modern frontend architectures**, where I prioritize clean code, component reusability, and maintainable project structures. I have strong experience working with **dynamic forms, state management (Redux / React Hook Form), and complex data-driven interfaces**. I enjoy solving real-world problems by turning ideas into reliable software that companies can actually use in their daily operations. Beyond coding, I care about **software quality and architecture**, following best practices for componentization, code organization, and performance optimization. I'm also comfortable working across the stack when needed, integrating APIs, handling business logic, and helping transform prototypes into production-ready systems. My goal is always to deliver solutions that are **simple, efficient, and genuinely useful for the people using them.** Franck Plazanet Franck Plazanet I am a Strategic Engineering Leader with over 8 years of experience building high-availability enterprise systems and scaling high-performing technical teams. My focus is on bridging the gap between complex technology and business growth. Core Expertise: 🚀 Leadership: Managing and coaching teams of 15+ engineers, fostering a culture of accountability and continuous improvement. 🏗️ Architecture: Enterprise Core Systems, Multi-system Integration (ERP/API/ETL), and Core Database Structure. ☁️ Cloud & Scale: AWS Expert; architected systems handling 10B+ monthly requests and managing 100k+ SKUs. 📈 Business Impact: Aligning tech strategy with P&L goals to drive $70k+ in monthly recurring revenue. I thrive on "out-of-the-box" thinking to solve complex technical bottlenecks and am always looking for ways to use automation to improve business productivity. BurnHavoc BurnHavoc Been around fixing other peoples code for 20 years. Mehdi Ben Haddou Mehdi Ben Haddou - Founder of Chessigma (1M+ users) & many small projects - ex Founding Engineer @Uplane (YC F25) - ex Software Engineer @Amazon and @Booking.com Richard McSorley Richard McSorley Full-Stack Software Engineer with 8+ years building high-performance applications for enterprise clients. Shipped production systems at Walmart (4,000+ stores), Cigna (20M+ users), and Arkansas Blue Cross. 5 patents in retail/supply chain tech. Currently focused on AI integrations, automation tools, and TypeScript-first architectures. Luca Liberati Luca Liberati I work on monoliths and microservices, backends and frontends, manage K8s clusters and love to design apps architecture MFox MFox Full-stack professional senior engineer (15+years). Extensive experience in software development, qa, and IP networking. Costea Adrian Costea Adrian Embedded Engineer specilizing in perception systems. Latest project was a adas camera calibration system.

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