name: dust-webhook-source
description: Implement a new built-in webhook source provider in front. Use when adding a webhook provider such as Linear, GitHub, or Fathom, including provider research, OAuth prerequisites, provider-specific types and client code, preset registration, UI components, and end-to-end testing.
Front Webhook Sources
Implement built-in webhook providers across two trees:
front/lib/triggers/built-in-webhooks/<provider>/— UI-safe: presets, schemas, types, React componentsfront/lib/api/triggers/built-in-webhooks/<provider>/— server-only: service + provider HTTP client
The boundary is load-bearing. Anything reached from the UI must never transitively import
@app/lib/api/config, OAuthAPI, or other backend modules — putting server code under lib/api/
is how we enforce that. Do not re-introduce a webhookService field on the preset or instantiate
the service from preset.ts: the leak that motivated the split will come right back.
The work spans provider research, OAuth prerequisites, typed remote metadata, a provider client, the remote webhook service, preset registration, services-map registration, and the setup/details UI.
Hard Prerequisites
Do not start implementation until both of these already exist:
front/lib/api/oauth/providers/<provider>.tscore/src/oauth/providers/<provider>.rs
The provider must support the webhooks use case and have scopes that can manage webhooks.
If the provider only allows webhooks to be created manually in its UI, stop. It is not a fit for a built-in webhook source.
Research First
Before writing code, document these answers from the provider docs:
- how to create a webhook programmatically
- required auth method and OAuth scopes
- request and response payloads for create and delete
- available event types and sample payloads
- how the incoming event type is identified: header or body field
- what user-configurable metadata is required, such as team or repository selection
- whether webhook signatures exist and how to verify them
This research determines the metadata types, client methods, event schemas, and UI.
Files To Touch
Minimal implementation:
front/types/triggers/webhooks.tsfront/types/triggers/webhooks_client_side.tsfront/lib/api/triggers/built-in-webhooks/services.ts(register the service instance)front/lib/api/triggers/built-in-webhooks/<provider>/<provider>_client.ts(server-only)front/lib/api/triggers/built-in-webhooks/<provider>/service.ts(server-only)front/lib/triggers/built-in-webhooks/<provider>/types.tsfront/lib/triggers/built-in-webhooks/<provider>/preset.tsfront/lib/triggers/built-in-webhooks/<provider>/client_side.tsfront/lib/triggers/built-in-webhooks/<provider>/schemas/*front/lib/triggers/built-in-webhooks/<provider>/components/*
Use existing providers as templates:
fathomfor the simplest shapelinearfor team-scoped webhook creationgithubfor repository-driven configuration
Workflow
1. Register the provider
In front/types/triggers/webhooks.ts:
- add the provider to
WEBHOOK_PROVIDERS - extend
WebhookProviderServiceDataMapif the provider needs extra OAuth-fetched data - register the preset in
WEBHOOK_PRESETS
In front/types/triggers/webhooks_client_side.ts:
- import the provider client-side preset
- register it in
CLIENT_SIDE_WEBHOOK_PRESETS
In front/lib/api/triggers/built-in-webhooks/services.ts:
- import the provider service class (from
lib/api/..., notlib/...) - register an instance in
WEBHOOK_SERVICES. This map is the server-only dispatch used by webhook create/delete/service-data endpoints. It must never be imported from UI code.
2. Define metadata and type guards
In types.ts, define:
- create-time metadata entered by the user
- persisted metadata that also includes provider webhook ids
- runtime type guards for both
- optional additional OAuth data such as teams or repositories
Always keep remote metadata serializable. No functions or class instances.
3. Implement the provider client
<provider>_client.ts lives under front/lib/api/triggers/built-in-webhooks/<provider>/ and
should be a thin wrapper over the provider API. It is server-only; do not import it from UI code.
Typical methods:
createWebhook(...)deleteWebhook(...)getTeams()/getRepositories()/ similar when setup requires selection data
Return typed results and map the provider response into the local metadata shape.
4. Implement the remote webhook service
In front/lib/api/triggers/built-in-webhooks/<provider>/service.ts, implement
RemoteWebhookService<"<provider>">. This file is server-only — it will import OAuthAPI and
@app/lib/api/config, which must never reach the client bundle.
Core responsibilities:
getServiceData(oauthToken)fetches setup data for the formcreateWebhooks(...)validates metadata, creates provider-side webhooks, and returnsupdatedRemoteMetadataincluding webhook idsdeleteWebhooks(...)removes provider-side webhooks using the persisted ids
When multiple webhooks may be created, handle partial success explicitly and return user-facing errors without losing the successfully created ids.
5. Define event schemas
Create one schema file per event, or one per coherent event family, under schemas/.
Each event entry should provide:
- the event name/value exposed in the UI
- a JSON schema for the payload
- an optional sample payload when it clarifies the shape
6. Build the presets
Split preset code correctly:
preset.tsexports aBaseWebhookPreset— data-only, UI-safe, worker-safe. Do not import React, icons, or the service here. The service is registered separately inWEBHOOK_SERVICES(see step 1).client_side.tsexports aClientSideWebhookPreset— it spreads the base preset and adds the provider icon plus React components.
Set on the base preset (preset.ts):
eventCheckto match how the provider exposes event type- the full list of supported events
- optional
webhookPageUrl
Set on the client-side preset (client_side.ts):
icon- the React components (
detailsComponent,createFormComponent, and optionallyoauthExtraConfigInput)
7. Build the UI components
Create:
CreateWebhook<Provider>Connection.tsxWebhookSource<Provider>Details.tsx
The create form should:
- rely on
front/components/triggers/CreateWebhookSourceWithProviderForm.tsxfor the OAuth connection flow viasetupOAuthConnection(...) - treat
connectionIdas an input prop; do not recreate the OAuth flow inside the provider-specific form - fetch provider-specific setup data with
useWebhookServiceData({ owner, connectionId, provider }) - call
onDataToCreateWebhookChange({ connectionId, remoteMetadata }) - call
onReadyToSubmitChange(true)only when configuration is complete
The details component should stay read-only and summarize the persisted remote metadata.
Common Gotchas
- forgetting to register the data preset in
WEBHOOK_PRESETSor the client-side preset inCLIENT_SIDE_WEBHOOK_PRESETS - forgetting to register the service in
WEBHOOK_SERVICES - putting the service or provider client under
lib/triggers/...instead oflib/api/triggers/...(leaksOAuthAPIand backend config into the client bundle) - importing React, icons, or the service from
preset.ts(any of those turnBaseWebhookPresetinto a client-only module and reintroduces the bundle leak) - putting the icon on the base preset rather than the client-side preset
- storing non-serializable remote metadata
- skipping type guards before using
remoteMetadata - assuming a single webhook when the provider needs one per team or repository
- ignoring partial success during multi-webhook creation
- forgetting to make local webhook URLs public during manual testing
Testing
For local testing, expose the local app publicly and set DUST_WEBHOOKS_PUBLIC_URL.
Manual checklist:
- complete the OAuth flow successfully
- create the webhook source in the UI without errors
- verify the webhook exists in the provider dashboard
- trigger a real provider event and inspect local logs
- confirm the incoming payload matches the registered schema and event detection logic
- delete the source and verify the provider-side webhook is removed
References
front/lib/resources/webhook_source_resource.tsfront/types/triggers/remote_webhook_service.tsfront/types/triggers/webhooks_source_preset.ts