API Reference

Webhook Subscriptions

Webhook subscriptions are outbound delivery configurations. They define the URL Composio posts trigger events to, along with the signing secret and the set of event types you want to receive.

Reach for these endpoints to register where Composio should send events, to filter delivery to specific event types, and to manage the signing secret used to verify those deliveries. List the available event types with the /webhook_subscriptions/event_types endpoint, then create a subscription scoped to the ones you care about.

Each subscription is addressed by its id. You can update its URL and filters with PATCH, delete it, and rotate its signing secret with /webhook_subscriptions/{id}/rotate_secret if the secret leaks.

Every webhook request Composio sends includes webhook-id, webhook-timestamp, and webhook-signature headers. Store the secret as COMPOSIO_WEBHOOK_SECRET and verify each payload before trusting it. See Webhook verification for the SDK and manual verification flows.

Event types

A subscription's enabled_events controls which events get delivered to its URL. Two broad families exist:

  • Trigger events like composio.trigger.message — payloads emitted by triggers you've enabled (new email, new issue, etc.).
  • Lifecycle events like composio.connected_account.expired — emitted when a connected account changes state.

List everything you can subscribe to with the /webhook_subscriptions/event_types endpoint, then scope a subscription to the events you handle.

Detecting connection expiry

Composio automatically refreshes OAuth tokens before they expire. But when a refresh token is revoked or expires, the connection enters an EXPIRED state and the user must re-authenticate.

Subscribe to the composio.connected_account.expired event to detect this proactively, instead of waiting for a tool execution to fail.

This event is only available with V3 webhook payloads. New organizations use V3 by default.

Add composio.connected_account.expired to the subscription's enabled_events:

curl -X POST https://backend.composio.dev/api/v3.1/webhook_subscriptions \
  -H "X-API-KEY: <your-composio-api-key>" \
  -H "Content-Type: application/json" \
  -d '{
    "webhook_url": "https://example.com/webhook",
    "enabled_events": [
      "composio.trigger.message",
      "composio.connected_account.expired"
    ]
  }'

When a connection expires, Composio sends a webhook with the connected account details:

{
  "id": "evt_847cdfcd-d219-4f18-a6dd-91acd42ca94a",
  "type": "composio.connected_account.expired",
  "metadata": {
    "project_id": "pr_your-project-id",
    "org_id": "ok_your-org-id"
  },
  "data": {
    "id": "ca_your-connected-account-id",
    "toolkit": { "slug": "gmail" },
    "auth_config": {
      "id": "ac_your-auth-config-id",
      "auth_scheme": "OAUTH2"
    },
    "status": "EXPIRED",
    "status_reason": "OAuth refresh token expired"
  },
  "timestamp": "2026-02-06T12:00:00.000Z"
}

Route on type to handle expiry alongside trigger events:

from composio import Composio, WebhookEventType

composio = Composio()

@app.post("/webhook")
async def webhook_handler(request: Request):
    payload = await request.json()
    event_type = payload.get("type")

    if event_type == WebhookEventType.CONNECTION_EXPIRED:
        account_id = payload["data"]["id"]
        toolkit = payload["data"]["toolkit"]["slug"]

        # Look up the user and send them a re-auth link
        session = composio.create(user_id=lookup_user(account_id))
        connection_request = session.authorize(toolkit)
        notify_user(connection_request.redirect_url)

    elif event_type == WebhookEventType.TRIGGER_MESSAGE:
        # Handle trigger events
        pass

    return {"status": "ok"}
import { Composio } from '@composio/core';
const composio = new Composio();
type NextApiRequest = { body: any };
type NextApiResponse = { status: (code: number) => { json: (data: any) => void } };
declare function lookupUser(accountId: string): string;
declare function notifyUser(url: string): void;
// ---cut---
export default async function webhookHandler(req: NextApiRequest, res: NextApiResponse) {
  const payload = req.body;

  if (payload.type === 'composio.connected_account.expired') {
    const accountId = payload.data.id;
    const toolkit = payload.data.toolkit.slug;

    // Look up the user and send them a re-auth link
    const session = await composio.create(lookupUser(accountId));
    const connectionRequest = await session.authorize(toolkit);
    if (connectionRequest.redirectUrl) {
      notifyUser(connectionRequest.redirectUrl);
    }

  } else if (payload.type === 'composio.trigger.message') {
    // Handle trigger events
  }

  res.status(200).json({ status: 'ok' });
}

Always verify webhook signatures before processing events in production.

Endpoints

EndpointQuick Link
POST /api/v3.1/webhook_subscriptionsCreate webhook subscription
GET /api/v3.1/webhook_subscriptionsList webhook subscriptions
GET /api/v3.1/webhook_subscriptions/{id}Get webhook subscription
PATCH /api/v3.1/webhook_subscriptions/{id}Update webhook subscription
DELETE /api/v3.1/webhook_subscriptions/{id}Delete webhook subscription
POST /api/v3.1/webhook_subscriptions/{id}/rotate_secretRotate webhook secret
GET /api/v3.1/webhook_subscriptions/event_typesList available event types