Documentation Index
Fetch the complete documentation index at: https://docs.linkrunner.io/llms.txt
Use this file to discover all available pages before exploring further.
Overview
Webhooks allow you to receive real-time HTTP notifications when attribution events occur in your app. When a user installs your app or signs up, Linkrunner sends a POST request to your configured endpoint with detailed attribution data. Common use cases:- Server-side analytics and reporting
- CRM integration for user onboarding
- Real-time Slack notifications
- Custom attribution pipelines
Configuration
Setting Up Your Webhook URL
- Go to Linkrunner Settings → Webhooks
- Enter your webhook endpoint URL
- Save the configuration
Your endpoint must be publicly accessible and respond with a 2xx status code to acknowledge receipt.
Testing Your Endpoint
Before going live, verify your endpoint can:- Accept POST requests with JSON body
- Respond within a reasonable time (recommended under 5 seconds)
- Return a 2xx status code on success
Webhook Events
Linkrunner sends webhooks for the following events:| Event | Description |
|---|---|
install | Triggered when an app install is attributed to a campaign or organic source |
signup | Triggered when a user signup event is recorded via the SDK |
Install Webhook
Theinstall webhook is triggered immediately when Linkrunner attributes an app installation. At this point, your app has just been opened for the first time and the user hasn’t had a chance to sign up or log in yet. Because of this, the user_id field will be null in install webhooks.
However, the install webhook does include device identifiers (gaid for Android, idfa for iOS) when available. You can use these to match the install with the user later when they sign up.
Use the install webhook for:
- Tracking install counts by campaign
- Analyzing attribution data (network, ad creative, etc.)
- Monitoring app store conversion rates
- Storing device IDs to link with user accounts later
Signup Webhook
Thesignup webhook is triggered when you call the .signup() method in your app via the Linkrunner SDK. This happens after the user has signed up or logged in, so the user_id field will contain the ID you passed to the SDK.
The signup webhook includes device identifiers (gaid/idfa) along with the user_id, giving you a complete picture of both the device and the user. It also includes user identity fields (name, phone, email) and any custom parameters you passed via additional_data, such as referral codes or other custom key-value pairs.
Use the signup webhook for:
- Linking attribution data to your user records
- CRM integration and user onboarding flows
- Calculating signup conversion rates from installs
- Forwarding custom parameters (e.g., referral codes) to your backend
To receive signup webhooks, you must call
.signup() in your app after the user signs up or logs in. See your SDK’s usage guide for implementation details.Payload Structure
All webhooks are sent as POST requests with a JSON body.Headers
| Header | Description |
|---|---|
Content-Type | application/json |
linkrunner-key | Your project’s private key for authentication |
Body Parameters
| Field | Type | Description |
|---|---|---|
event_type | "install" | "signup" | The type of attribution event |
user_id | string | null | Customer user ID (if provided via SDK) |
campaign_id | string | Unique identifier for the campaign |
campaign_name | string | null | Human-readable campaign name |
network_name | string | null | Attribution network: ORGANIC, META, GOOGLE, etc. |
ad_channel | string | null | Ad channel: META, GOOGLE, TIKTOK, APPLE_SEARCH_ADS, etc. |
attributed_on | ISO 8601 date | Timestamp when attribution occurred |
installed_at | ISO 8601 date | null | Timestamp of app installation |
store_click_at | ISO 8601 date | null | Timestamp of store redirect click |
link | string | The campaign link URL |
app_version | string | null | Installed app version |
gaid | string | null | Google Advertising ID (Android) |
idfa | string | null | Identifier for Advertisers (iOS) |
name | string | null | User’s name (from user_data.name passed to the SDK) |
phone | string | null | User’s phone number (from user_data.phone passed to the SDK) |
email | string | null | User’s email address (from user_data.email passed to the SDK) |
additional_data | object | null | Custom parameters and device data (from the data field passed to the SDK) |
meta_campaign_details | object | null | Meta Ads campaign details (see below) |
google_campaign_details | object | null | Google Ads campaign details (see below) |
The
The
For
name, phone, and email fields are populated from user_data passed to the SDK’s .signup() or .trigger() methods.The
additional_data field is populated from the SDK data object.For
install events, these fields are null since user-level signup data is not yet available.Meta Campaign Details
When the attribution is from Meta Ads,meta_campaign_details contains:
| Field | Type | Description |
|---|---|---|
ad_creative_id | string | null | Meta ad creative ID |
ad_creative_name | string | null | Ad creative name |
ad_set_id | string | null | Ad set ID |
ad_set_name | string | null | Ad set name |
campaign_group_id | string | null | Campaign group ID |
campaign_group_name | string | null | Campaign group name |
account_id | string | null | Meta Ads account ID |
ad_objective_name | string | null | Campaign objective (e.g., APP_INSTALLS) |
is_instagram | boolean | null | Whether the ad was on Instagram |
publisher_platform | string | null | Platform where ad was shown |
platform_position | string | null | Placement position |
Google Campaign Details
When the attribution is from Google Ads,google_campaign_details contains:
| Field | Type | Description |
|---|---|---|
gclid | string | null | Google Click ID |
gbraid | string | null | Google app campaign tracking parameter |
ga_source | string | null | Google Analytics source |
ad_group_id | string | null | Ad group ID |
ad_group_name | string | null | Ad group name |
Additional Data
Theadditional_data object contains custom parameters and device data passed through the SDK. This is useful for forwarding arbitrary key-value pairs like referral codes or custom identifiers through the attribution flow.
Example:
Example Payloads
Install Event
Signup Event
Authentication
Every webhook request includes alinkrunner-key header containing your project’s private key. Use this to verify that requests are genuinely from Linkrunner.
Slack Integration
Linkrunner automatically formats webhook payloads for Slack when your URL containshooks.slack.com. Instead of raw JSON, Slack receives a rich Block Kit formatted message displaying:
- Event type and user ID
- App version and network
- Campaign name
- Attribution timestamps
- Device identifiers (GAID/IDFA)
- User identity (name, phone, email)
- Meta campaign details (if applicable)
- Create an Incoming Webhook in your Slack workspace
- Copy the webhook URL (format:
https://hooks.slack.com/services/...) - Paste it as your webhook URL in Linkrunner settings
Retry Behavior
If your endpoint fails to respond with a 2xx status code, Linkrunner retries the webhook with exponential backoff:| Attempt | Delay |
|---|---|
| 1st retry | 1 second |
| 2nd retry | 2 seconds |
| 3rd retry | 4 seconds |
Best Practices
Respond quickly
Return a 2xx status code as fast as possible. Process the webhook data asynchronously to avoid timeouts.
Implement idempotency
Store processed
campaign_id + user_id combinations to handle potential duplicate deliveries gracefully.Validate authentication
Always verify the
linkrunner-key header matches your private key before processing.Troubleshooting
Webhooks not being received?- Verify your endpoint URL is correct and publicly accessible
- Check that your server accepts POST requests with JSON body
- Ensure your firewall allows incoming requests from Linkrunner
- Verify the
linkrunner-keyheader validation in your code - Check your private key matches the one in your dashboard settings
name,phone, andemailrequire you to passuser_datato the SDK’s.signup()or.trigger()methodsuser_idrequires SDK configuration to be passed- Device identifiers (
gaid/idfa) depend on user consent and SDK implementation additional_datarequires passing adataobject to the SDK and contains those custom parameters













