Skip to main content

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

  1. Go to Linkrunner Settings → Webhooks
  2. Enter your webhook endpoint URL
  3. 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:
  1. Accept POST requests with JSON body
  2. Respond within a reasonable time (recommended under 5 seconds)
  3. Return a 2xx status code on success

Webhook Events

Linkrunner sends webhooks for the following events:
EventDescription
installTriggered when an app install is attributed to a campaign or organic source
signupTriggered when a user signup event is recorded via the SDK

Install Webhook

The install 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

The signup 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

HeaderDescription
Content-Typeapplication/json
linkrunner-keyYour project’s private key for authentication

Body Parameters

FieldTypeDescription
event_type"install" | "signup"The type of attribution event
user_idstring | nullCustomer user ID (if provided via SDK)
campaign_idstringUnique identifier for the campaign
campaign_namestring | nullHuman-readable campaign name
network_namestring | nullAttribution network: ORGANIC, META, GOOGLE, etc.
ad_channelstring | nullAd channel: META, GOOGLE, TIKTOK, APPLE_SEARCH_ADS, etc.
attributed_onISO 8601 dateTimestamp when attribution occurred
installed_atISO 8601 date | nullTimestamp of app installation
store_click_atISO 8601 date | nullTimestamp of store redirect click
linkstringThe campaign link URL
app_versionstring | nullInstalled app version
gaidstring | nullGoogle Advertising ID (Android)
idfastring | nullIdentifier for Advertisers (iOS)
namestring | nullUser’s name (from user_data.name passed to the SDK)
phonestring | nullUser’s phone number (from user_data.phone passed to the SDK)
emailstring | nullUser’s email address (from user_data.email passed to the SDK)
additional_dataobject | nullCustom parameters and device data (from the data field passed to the SDK)
meta_campaign_detailsobject | nullMeta Ads campaign details (see below)
google_campaign_detailsobject | nullGoogle Ads campaign details (see below)
The 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:
FieldTypeDescription
ad_creative_idstring | nullMeta ad creative ID
ad_creative_namestring | nullAd creative name
ad_set_idstring | nullAd set ID
ad_set_namestring | nullAd set name
campaign_group_idstring | nullCampaign group ID
campaign_group_namestring | nullCampaign group name
account_idstring | nullMeta Ads account ID
ad_objective_namestring | nullCampaign objective (e.g., APP_INSTALLS)
is_instagramboolean | nullWhether the ad was on Instagram
publisher_platformstring | nullPlatform where ad was shown
platform_positionstring | nullPlacement position

Google Campaign Details

When the attribution is from Google Ads, google_campaign_details contains:
FieldTypeDescription
gclidstring | nullGoogle Click ID
gbraidstring | nullGoogle app campaign tracking parameter
ga_sourcestring | nullGoogle Analytics source
ad_group_idstring | nullAd group ID
ad_group_namestring | nullAd group name

Additional Data

The additional_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:
{
  "id": "user_123",
  "name": "Test User",
  "email": "test@example.com",
  "phone": "9876543210",
  "device_data": {},
  "referral_code": "ABC123",
  "custom_param_name": "custom_value"
}

Example Payloads

Install Event

{
  "event_type": "install",
  "user_id": null,
  "campaign_id": "camp_XYZ123",
  "ad_channel": "META",
  "network_name": "META",
  "app_version": "2.4.1",
  "campaign_name": "Summer Promotion 2023",
  "attributed_on": "2026-03-24T08:11:00.464Z",
  "installed_at": "2026-03-24T08:11:00.464Z",
  "store_click_at": "2026-03-24T08:11:00.464Z",
  "link": "https://dl.linkrunner.io/?c=camp_XYZ123",
  "meta_campaign_details": { ... },
  "google_campaign_details": null,
  "gaid": "bk9384xs-p449-96ds-r132",
  "idfa": null,
  "name": null,
  "phone": null,
  "email": null,
  "additional_data": null
}

Signup Event

{
  "event_type": "signup",
  "user_id": "test_user_webhook_002",
  "campaign_id": "OgWmhiSXhG",
  "ad_channel": "META",
  "network_name": "ORGANIC",
  "app_version": "1.0.0",
  "campaign_name": "TOF - Free trial - AAA - DSDT - 17/12",
  "attributed_on": "2026-03-25T15:52:00.007Z",
  "installed_at": "2025-12-30T12:34:29.742Z",
  "store_click_at": null,
  "link": "https://dl.linkrunner.io/?c=OgWmhiSXhG&utm_source=meta_ads",
  "meta_campaign_details": null,
  "google_campaign_details": null,
  "gaid": "5faa2433-d7e1-4a8e-9a1c-a5880c26ab5c",
  "idfa": null,
  "name": "Test User",
  "phone": "9876543210",
  "email": "test@example.com",
  "additional_data": {
    "id": "test_user_webhook_002",
    "name": "Test User",
    "email": "test@example.com",
    "phone": "9876543210",
    "device_data": {},
    "referral_code": "ABC123",
    "custom_param_name": "custom_value"
  }
}

Authentication

Every webhook request includes a linkrunner-key header containing your project’s private key. Use this to verify that requests are genuinely from Linkrunner.
// Example: Verifying the webhook signature
const PRIVATE_KEY = process.env.LINKRUNNER_PRIVATE_KEY;

app.post('/webhook', (req, res) => {
  const receivedKey = req.headers['linkrunner-key'];

  if (receivedKey !== PRIVATE_KEY) {
    return res.status(401).json({ error: 'Unauthorized' });
  }

  // Process the webhook...
  res.status(200).json({ received: true });
});
Never expose your private key in client-side code. Store it securely as an environment variable.

Slack Integration

Linkrunner automatically formats webhook payloads for Slack when your URL contains hooks.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)
To set up Slack notifications:
  1. Create an Incoming Webhook in your Slack workspace
  2. Copy the webhook URL (format: https://hooks.slack.com/services/...)
  3. 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:
AttemptDelay
1st retry1 second
2nd retry2 seconds
3rd retry4 seconds
After 3 failed attempts, the webhook is marked as failed. Ensure your endpoint is reliable to avoid missing events.

Best Practices

1

Respond quickly

Return a 2xx status code as fast as possible. Process the webhook data asynchronously to avoid timeouts.
2

Implement idempotency

Store processed campaign_id + user_id combinations to handle potential duplicate deliveries gracefully.
3

Validate authentication

Always verify the linkrunner-key header matches your private key before processing.
4

Handle failures gracefully

Log failed webhook processing for debugging and implement alerting for critical failures.

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
Getting 401 errors?
  • Verify the linkrunner-key header validation in your code
  • Check your private key matches the one in your dashboard settings
Missing data in payload?
  • name, phone, and email require you to pass user_data to the SDK’s .signup() or .trigger() methods
  • user_id requires SDK configuration to be passed
  • Device identifiers (gaid/idfa) depend on user consent and SDK implementation
  • additional_data requires passing a data object to the SDK and contains those custom parameters
Need help? Contact support@linkrunner.io