Plaid

  1. Required Backend Changes
    1. Add the webhook URL and secret into application config
    2. Update where you create your Plaid Link tokens to specify the webhook URL
    3. Notify WebhookDB about new Tokens/Items
    4. Update existing Items in Plaid itself to use the new webhook URL
    5. Notify WebhookDB about existing Tokens/Items
    6. Backfill transaction history
    7. Notify WebhookDB about updated Tokens/Items
  2. Processor Tokens
  3. Getting Help

The integrations here are part of WebhookDB Enterprise.

Some APIs we integrate with are extremely tricky, and we cannot create as seamless a user experience as we strive for. Plaid is one such API.

In order to integrate WebhookDB with Plaid, you will need to do some work on your backend in order to tell WebhookDB when new Items/Access Tokens are created. This is necessary since 1) Plaid does not let us know about Items, and 2) your backend telling WebhookDB about Access Tokens is more secure than having to open an endpoint to us.

A corollary of the above is that WebhookDB will need your Plaid account secret and item access tokens. While we store the access tokens in an encrypted column so they cannot be read by the DB, we still recommend you host your own data or run the self-hosted version of WebhookDB for particularly sensitive read/write credentials like this.

All that said- full integration with Plaid resources that allow syncs, backfills, webhooks, and incremental updates is extremely complex to set up, and can easily take several weeks to do right. The changes explained in this document should take less than a day, and will give you a fully-working Plaid sync with the resources you configure.

Required Backend Changes

There are, broadly, five (small) things that need to be done to integrate WebhookDB with your Plaid data:

We’ll walk through each of these steps exactly as we do them for clients.

Add the webhook URL and secret into application config

When you run webhookdb integrations create plaid_item_v1 from the WebhookDB CLI, it will prompt for a webhook signing secret; after you enter that, the CLI will print a URL.

Copy these values, and put them into configuration so your app can read them at runtime.

For example:

$ echo "WHDB_PLAID_WEBHOOK_URL=https://api.webhookdb.com/v1/integrations/svi_abc" >> ".env"
$ echo "WHDB_PLAID_WEBHOOK_SECRET=longstring" >> ".env"

In the below examples, we’ll assume environment variables named as above; replace them with however you handle config in your application.

When you create the Plaid Link token, you must specify the webhook that Plaid will use when items get updated. This will be the value of WHDB_PLAID_WEBHOOK_URL. Here is a cURL for it; this should be done from your backend.

curl -X POST https://sandbox.plaid.com/link/token/create -H 'Content-Type: application/json' -d '{
  "client_id": "${CLIENT_ID}",
  "secret": "${SECRET}",
  "user": { "client_user_id": "unique-per-user" },
  "client_name": "My App",
  "products": ["auth"],
  "country_codes": ["US"],
  "language": "en",
  "webhook": ${WHDB_PLAID_WEBHOOK_URL}",
  "redirect_uri": "https://domainname.com/oauth-page.html",
  "account_filters": {
      "depository": {
          "account_subtypes": ["checking"]
      }
  }
}'

This will ensure WebhookDB finds out about changes to your Plaid items. Note that you can still send notification webhooks using WebhookDB, in case you still need webhooks to hit your backend.

Notify WebhookDB about new Tokens/Items

After you exchange your Plaid public token for an access token, Plaid creates the ‘Item’ representing this link. You must tell WebhookDB about the new item, and provide the access token so we can fetch it (as mentioned, the token is stored encrypted in your organization’s WebhookDB database, only the WebhookDB server can decrypt it).

Here is an example of how you would do this in Ruby:

# These are part of application config, as per earlier steps.
whdb_plaid_webhook_url = ENV['WHDB_PLAID_WEBHOOK_URL']
whdb_plaid_webhook_secret = ENV['WHDB_PLAID_WEBHOOK_SECRET']
# Item ID and access token are returned by Plaid's token exchange call.
item_id = 'created-item-id' 
access_token = 'exchanged-access-token'
body = {
  webhook_type: "ITEM",
  webhook_code: "CREATED",
  item_id: item_id,
  access_token: access_token
}
# Notify WebhookDB about the new token.
resp = Net::HTTP.post(
  URI(whdb_plaid_webhook_url),
  body.to_json,
  {'Content-Type' => 'application/json', 'Whdb-Webhook-Secret' => whdb_plaid_webhook_secret}
)
raise "Bad response: #{resp.inspect}" unless resp.code == '200'

Note that both Plaid and your backend POST to the same WebhookDB URL.

Update existing Items in Plaid itself to use the new webhook URL

Once that change is deployed, and WebhookDB is notified when new Plaid Items are created, you must update your existing Plaid Items in Plaid so that Plaid sends updates to WebhookDB, rather than whatever was configured previously.

As per the Plaid docs, you must run something like this for each item (refer to Plaid docs for how to run this using their SDKs in various languages):

curl -X POST https://sandbox.plaid.com/item/webhook/update \
  -H 'Content-Type: application/json' \
  -d '{
    "client_id": ${CLIENT_ID},
    "secret": ${SECRET},
    "access_token": "access token for item",
    "webhook": "${WHDB_PLAID_WEBHOOK_URL}"
  }'

Notify WebhookDB about existing Tokens/Items

Once you are telling WebhookDB about new Plaid items, and Plaid is sending all notifications to WebhookDB, you must let us know about already-created items (so that we can act on those Plaid webhooks). Run the same code you have after the token exchange for each item/access token you have:

whdb_plaid_webhook_url = ENV['WHDB_PLAID_WEBHOOK_URL']
whdb_plaid_webhook_secret = ENV['WHDB_PLAID_WEBHOOK_SECRET']
MyApp::PlaidItem.each do |item|
    body = {
      webhook_type: "ITEM",
      webhook_code: "CREATED",
      item_id: item.plaid_item_id,
      access_token: item.plaid_access_token
    }
    resp = Net::HTTP.post(
      URI(whdb_plaid_webhook_url),
      body.to_json,
      {'Content-Type' => 'application/json', 'Whdb-Webhook-Secret' => whdb_plaid_webhook_secret}
    )
    raise "Bad response: #{resp.inspect}" unless resp.code == '200'
end

Backfill transaction history

Plaid sends Transaction webhook notifications for various reasons, as explained in their docs. As soon as WebhookDB sees one of these, we will backfill all the available transactions. So the history of items you add (and then request the history for) will always be present, as will the history of items added before your WebhookDB integration, once we see a webhook.

If, however, you must trigger a backfill of historical data separately from getting a Plaid webhook, you can fake one of Plaid’s webhooks. It’s recommended you use HISTORICAL_UPDATE so we fetch all data (otherwise we only fetch newer transactions).

Here again is some Ruby code:

whdb_plaid_webhook_url = ENV['WHDB_PLAID_WEBHOOK_URL']
whdb_plaid_webhook_secret = ENV['WHDB_PLAID_WEBHOOK_SECRET']
item_id = 'plaid-item-id' 
body = {
  webhook_type: "TRANSACTIONS",
  webhook_code: "HISTORICAL_UPDATE",
  item_id: item_id,
}
resp = Net::HTTP.post(
  URI(whdb_plaid_webhook_url),
  body.to_json,
  {'Content-Type' => 'application/json', 'Whdb-Webhook-Secret' => whdb_plaid_webhook_secret}
)
raise "Bad response: #{resp.inspect}" unless resp.code == '200'

Notify WebhookDB about updated Tokens/Items

Plaid sends webhooks for certain types of item changes, like when there is an error, or consent is going to expire.

Plaid does not, unfortunately, send a webhook when an error is cleared, like after a user goes through Update Mode and re-links their items. In these cases, you will need to send a notification to WebhookDB directly to tell it to sync down changes to the item. The webhook_type must be ITEM and the webhook_code must be UPDATED.

Here is an example of how you would do this in Ruby:

whdb_plaid_webhook_url = ENV['WHDB_PLAID_WEBHOOK_URL']
whdb_plaid_webhook_secret = ENV['WHDB_PLAID_WEBHOOK_SECRET']
body = {
  webhook_type: "ITEM",
  webhook_code: "UPDATED",
  item_id: item_id
}
resp = Net::HTTP.post(
  URI(whdb_plaid_webhook_url),
  body.to_json,
  {'Content-Type' => 'application/json', 'Whdb-Webhook-Secret' => whdb_plaid_webhook_secret}
)
raise "Bad response: #{resp.inspect}" unless resp.code == '200'

In the future, if Plaid starts sending webhooks on updates like this (which we think is pretty important!) you won’t have to do this from your backend.

Processor Tokens

WebhookDB also supports using Plaid Processor Tokens for syncing transactions.

To ease integration, all the instructions for Link Tokens also apply to Processor Tokens. Wherever you send "item_id" to WebhookDB, use the processor token’s account_id:

whdb_plaid_webhook_url = ENV['WHDB_PLAID_WEBHOOK_URL']
whdb_plaid_webhook_secret = ENV['WHDB_PLAID_WEBHOOK_SECRET']
# See https://plaid.com/docs/api/processors/#processor-token-create-request-account-id
account_id = "yzhf012345"
# See https://plaid.com/docs/api/processors/#processor-token-create-response-processor-token
processor_token = "processor-sandbox-abc123"
MyApp::PlaidItem.each do |item|
    body = {
      webhook_type: "ITEM",
      webhook_code: "CREATED",
      item_id: account_id,
      access_token: processor_token
    }
    resp = Net::HTTP.post(
      URI(whdb_plaid_webhook_url),
      body.to_json,
      {'Content-Type' => 'application/json', 'Whdb-Webhook-Secret' => whdb_plaid_webhook_secret}
    )
    raise "Bad response: #{resp.inspect}" unless resp.code == '200'
end

WebhookDB takes care of everything else for you, and you can treat these just like normal tokens.

The things to know are:

  • WebhookDB determines if an ‘Item’ call is a Link Token-based call or a Processor-Token based call by looking for the "processor-" prefix on the access token. You don’t have to worry about this- Plaid will always have the "processor-" prefix on its Processor Tokens.
  • When a new Plaid Item row is added, if the access token is a Processor Token (see previous point), the processor_account_id column will be set to the same value as the row’s plaid_id. You can use this to determine that a row in the plaid_item_v1 table is actually a row for a processor account.
    • So, for a Link Token-based row:
      • The processor_account_id column is null.
      • The plaid_id column stores the Item ID.
    • For a Processor Token-based row:
      • The processor_account_id column stores the Account ID.
      • The plaid_id column is also the Account ID (ie, identical to processor_account_id).
  • Processor Token based rows (ie, any row with a non-null processor_account_id) will not have Item-specific columns set, like their status or error.
    • So for example, a "WEBHOOK_CODE": "UPDATED" call would noop on Processor Token rows, whereas for Link Token rows, it would update the database row with the result of an API call to Plaid.

Getting Help

We know integrating with Plaid is pretty complex. But, having done the same thing without WebhookDB takes several times longer, in our experience- something like 1 to 2 hours (including backfilling) compared to days or weeks. In fact, when working with customers and clients, we have found it rare that someone gets all of the nuances of their Plaid integration set up correctly, if at all, such as webhooks.

If you need any help, we’re here to assist. Just email hello@webhookdb.com and we’ll get back to you right away.