Plaid
- Required Backend Changes
- Add the webhook URL and secret into application config
- Update where you create your Plaid Link tokens to specify the webhook URL
- Notify WebhookDB about new Tokens/Items
- Update existing Items in Plaid itself to use the new webhook URL
- Notify WebhookDB about existing Tokens/Items
- Backfill transaction history
- Notify WebhookDB about updated Tokens/Items
- Processor Tokens
- 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:
- Add the webhook URL and secret (provided by WebhookDB) into your application’s config.
- Update where you create your Plaid Link tokens to specify the WebhookDB webhook URL.
- POST to the WebhookDB webhook URL from your backend when new Plaid Items are created (the public token is exchanged).
- Update existing “items” in Plaid itself to use the new WebhookDB webhook URL.
- POST to the WebhookDB webhook URL from your backend about existing Plaid Items/Tokens in your database.
- (Optional) Backfill Transaction history of already-fetched accounts.
- (Optional) POST to the WebhookDB webhook URL from your backend when Plaid Items are updated.
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.
Update where you create your Plaid Link tokens to specify the webhook URL
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’splaid_id
. You can use this to determine that a row in theplaid_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.
- The
- 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 toprocessor_account_id
).
- The
- So, for a Link Token-based row:
- Processor Token based rows (ie, any row with a non-null
processor_account_id
) will not have Item-specific columns set, like theirstatus
orerror
.- 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.
- So for example, a
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.