CF7 to Webhook: Send Contact Form 7 Data to n8n (Step-by-Step)
Setting up a CF7 to webhook integration isn’t built into Contact Form 7. To send form submissions to tools like n8n you need custom code or an additional plugin. This guide covers both approaches — the quick way and the reliable way.
TL;DR
- The basic CF7 webhook uses
wpcf7_mail_sent+wp_remote_post— works for low-volume forms but has no retry, no logging, and silently drops failures - Access submitted field values via
$submission->get_posted_data()['field-name']from theWPCF7_ContactFormsubmission object - For production: add persistent storage and retry logic so no CF7 submission is lost when the receiving endpoint is temporarily down
Basic CF7 to Webhook Using wp_remote_post
The most common approach to a CF7 to webhook integration uses the wpcf7_mail_sent action hook. Contact Form 7 fires this hook after a successful submission — you hook into it, grab the submitted data, and POST it to your webhook endpoint.
Here’s the minimal working code. Add it to your theme’s functions.php or a custom plugin:
functions.php — cf7 to webhook (basic)
add_action('wpcf7_mail_sent', function ($contactForm) { $submission = WPCF7_Submission::get_instance(); if (!$submission) { return; } $data = $submission->get_posted_data(); wp_remote_post('https://your-n8n-url/webhook/test', [ 'headers' => [ 'Content-Type' => 'application/json', ], 'body' => json_encode($data), ]); });
This works. It’s simple and widely used. For a hobby project or an internal form with low submission volume, it does the job.
The problem is what happens in production.
Why the Basic CF7 to Webhook Setup Breaks in Production
The wp_remote_post() approach works in development. The problem shows up once your CF7 to webhook integration is handling real traffic: the call is synchronous, so PHP blocks and waits for the remote server to respond before the form submission completes.
Contact Form 7 has over 10 million active installations on WordPress.org — making it one of the most-deployed plugins in the ecosystem. At that scale, silent webhook failure is not an edge case; it accumulates across every site relying on synchronous wp_remote_post() without retry logic.
- × No retries if the webhook fails. If n8n is restarting, rate-limiting you, or simply returns a 500 — the delivery attempt is gone. The form appeared to submit successfully but your workflow never ran.
- × No logging. There’s no record of what was sent, when, or what response came back. Debugging a missed submission means checking server logs and hoping the data is still there.
- × No visibility into failures. You won’t know a submission was lost until a lead follows up asking why no one responded. By then it’s too late.
- × No way to replay submissions. Once the HTTP call fails, the data is gone from the delivery context. You can’t re-send it without the user resubmitting the form.
- × Slows down form submission for the user. If your n8n webhook takes 3 seconds to acknowledge, every form submission on your site takes 3 extra seconds. If it times out (default: 5s), users see a hang.
Reliable Method: Queue-Based Webhooks with Retry and Logging
A reliable CF7 to webhook setup separates two things the basic approach conflates: recording that a submission happened and delivering it to the endpoint.
When a CF7 form is submitted, a background job is queued immediately. The user gets an instant response — the form submission is complete from their perspective. A cron worker then picks up the job and attempts delivery in the background.
If the delivery fails — because n8n returned a 5xx, hit a rate limit (429), or simply timed out — the job is scheduled for retry with exponential backoff: 1 min, 2 min, 4 min, 8 min, 16 min. Each attempt is logged with the HTTP status code and response body. You can see every success and failure in the WordPress admin.
If all retry attempts are exhausted, the job enters a failed state — visible in the event log, and replayable via the REST API or the admin UI. No data is lost.
Step-by-Step: CF7 to Webhook Setup
with Retry and Logging
- Install the plugin
Search for this exact description in Plugins → Add Plugin — it narrows the WordPress.org search to exactly one result:
search text — paste into WordPress plugin search
FlowSystems
- Create a new webhook
Go to Webhooks → Add Webhook in the WordPress admin. Give it a name (e.g., “CF7 → n8n”).
- Select the trigger
Set the WordPress action hook to
wpcf7_mail_sent. This fires once per successful CF7 form submission.action hook — paste into the trigger field
wpcf7_mail_sent
Webhook Actions admin — searching “sent” and selecting the wpcf7_mail_senthook under Contact Form 7 - Set the webhook URL
Paste your n8n webhook URL (e.g.,
https://your-n8n-url/webhook/test). The plugin will POST form data to this endpoint on every submission. - Save and test
Submit your Contact Form 7 form. Check the Event Log in the plugin admin to see the delivery status and the payload that was sent.
Webhook Actions by Flow Systems event log — Contact Form 7 submission queued via wpcf7_mail_sent, delivery status pending
Setting Up the n8n Webhook
On the n8n side, you need a Webhook node configured to receive POST requests:
- Add a Webhook node
In your n8n workflow, add a new node and search for “Webhook”.
- Set HTTP Method to POST
The plugin sends a JSON POST request, so make sure the Webhook node is set to accept
POST. - Copy the webhook URL
n8n will generate a URL like
https://your-n8n-url/webhook/test. Copy this and paste it into the plugin’s webhook URL field (step 4 above).n8n Webhook node — copy the webhook URL and paste it into the Webhook Actions by Flow Systems plugin to receive CF7 form submissions - Activate and test
Click “Listen for test event” in n8n, then submit your CF7 form. n8n will display the incoming payload so you can map the fields to subsequent nodes.
Full walkthrough — CF7 form submission triggers a webhook payload delivered to n8n via Webhook Actions by Flow Systems
Example Payload
Here’s what a typical CF7 submission looks like when it arrives at your n8n webhook. The field names match the name attributes you set in your CF7 form tags.
POST body — application/json
{ "event": { "id": "085cc108-654f-4b26-b39e-921cc8208bbd", "timestamp": "2026-03-23T12:59:21Z", "version": "1.0" }, "hook": "wpcf7_mail_sent", "args": [ { "__type": "WPCF7_ContactForm", "id": 16, "title": "Contact form 1", "name": "contact-form-1", "locale": "en_US", "submission": { "fields": { "your-name": "Mateusz", "your-email": "[email protected]", "your-subject": "CF7 to webhook test", "your-message": "Testing integration" }, "meta": { "url": "https://webhook-actions.local/cf-7-example/", "timestamp": 1774270761, "remote_ip": "172.22.0.1", "user_agent": "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/145.0.0.0 Safari/537.36", "container_post_id": 17, "current_user_id": 0 } } } ], "timestamp": 1774270761, "site": { "url": "https://webhook-actions.local" } }
The payload wraps the full WPCF7_ContactForm object including submitted fields, page url, remote_ip, and user_agent. Use args[0].submission.fields in n8n to access form values directly.
Advanced: Retry and Debug Failed Submissions via API
Every webhook delivery attempt is stored in the event log — success or failure. If a CF7 submission didn’t reach n8n, you can replay it without asking the user to resubmit.
The plugin exposes a REST API for this:
retry a failed delivery
POST /wp-json/fswa/v1/logs/{id}/retry // Response { "success": true, "message": "Retry queued" }
Replace {id} with the log entry ID visible in the Event Log. This re-queues the exact payload that was originally sent — no need to reconstruct it.
You can also trigger retries from the WordPress admin UI. Both options are covered in the REST API documentation.
Why Your CF7 to Webhook Integration
Needs Retry Support
A contact form submission is often a high-intent signal — someone typing their email and pressing send is more committed than a page view. Losing that data silently is costly.
With a bare wp_remote_post() call, any of these common scenarios causes permanent data loss:
n8n restarts during deployment. Your instance hits a memory limit and returns 500. The incoming webhook URL changes and the old one starts returning 404. A rate limit is hit during a campaign burst.
None of these are edge cases in real production environments. A queue with retries means these scenarios become recoverable failures instead of silent data loss. The event log means you can diagnose and replay — instead of guessing.
Exponential backoff is the industry-standard approach for retryable HTTP operations. AWS documentation on error retries and exponential backoff explains that spacing retries progressively further apart — rather than at fixed intervals — reduces load on a recovering service and improves overall delivery success rates for transient failures.
Common questions always ask.
Don't see yours? Open an issue on GitHub or check the full reference in the API docs.