This guide will show you how to subscribe to BigCommerce webhooks and use them to trigger global actions in Gadget.
Set up a webhook subscription
After setting up a BigCommerce connection, you can add BigCommerce webhooks as triggers to global actions.
To add a BigCommerce webhook as a trigger:
Navigate to an existing global action, or create a new one in api/actions
Click the + button to add a new trigger and select BigCommerce
Choose the webhook topic you want to trigger the action
When the selected webhooks are fired by BigCommerce, the global action will run.
Webhook payload
Webhooks sent by BigCommerce only include the id of the resource that triggered the webhook. To fetch the full resource data, you can use the included BigCommerce API client.
sample BigCommerce store/product/created payload
json
{
"id":113,
"storeHash":"wvpfsac141",
"type":"product"
}
For more information on reading and writing to BigCommerce, see the BigCommerce data guide.
Managing webhook scopes
Gadget automatically registers the BigCommerce webhook scopes you select as triggers in global actions. This means you can access the webhook payload in the action.
To see all scopes registered for a store, you can view the registeredWebhooks field in api/models/bigcommerce/store/data.
Manually registering webhooks
You can view the status of webhook registration on the store Installs page of the BigCommerce connection.
Hover over the Webhook Status of the store to see the list of webhooks that are registered for the store.
If webhooks are not all registered, you can click the Register webhooks button to manually register webhooks for a store. The only time that manual webhook registration is necessary is if there was a network issue during the initial registration.
Webhooks that are no longer used as triggers will be automatically unregistered.
Persisting session data in webhook-triggered actions
If you call an action from a global action that is triggered by a BigCommerce webhook, session data will not persist in the called action. This means that you will not be able to use connections.bigcommerce.current to get an instance of the BigCommerce API client, or connections.bigcommerce.currentStoreHash to get the store hash.
If you want to use the BigCommerce API client in an action that is called from a webhook-triggered global action, you will need to use connections.bigcommerce.forStoreHash() to get the API client for the store.
This means you also need the storeHash. The best way to access the storeHash in model actions is to relate your model records to the bigcommerce/store model and then fetch the hash manually before initializing the API client. Read more about setting up models to store BigCommerce data in the BigCommerce data guide.
If you are calling another global action from your webhook-triggered action, the storeHash can be passed in the params object.
Here is an example of a model action that fetches the storeHash and uses it to initialize the BigCommerce API:
Fetch storeHash and use it to init the BigCommerce API client
Webhooks in Gadget run on the built-in background action system. This means that you can view the payload of past webhooks by clicking on Queues in the Gadget editor.
Filter the Queues list
You can enter bigcommerce-webhook into the filter input to only display BigCommerce webhooks in the queue.
Clicking on an individual webhook will show you the payload that was sent by BigCommerce, as well as any errors that occurred during webhook processing.
If there is an error in the global action triggered by the BigCommerce webhook, the background actions system will automatically retry the action with the same webhook payload. If the retry count is reached and the action is not successful, the queued action will be marked as failed and you can view the error message.
Once the error has been fixed, you can re-run the action with the same webhook payload by clicking ⋮ → Retry now on the failed action.
Avoiding webhook loops
When creating global actions triggered by BigCommerce webhooks, it is important to avoid creating loops where the action triggers the same webhook that triggered the action.
For example, if a store/order/updated webhook triggers an action that updates the same product in BigCommerce, the update will trigger the store/order/updated webhook again, creating an infinite loop:
Avoid doing this in a global action, webhook will loop infinitely!
9const order =await bigcommerce.v2.get("/orders/{order_id}",{
10path:{
11order_id: orderId,
12},
13});
14
15// do custom work with the order
16// in this case, a secret discount!
17if(order.customer_message.includes("Gadget")){
18// update the same order!
19await bigcommerce.v2.put("/orders/{order_id}",{
20path:{
21order_id: orderId,
22},
23body:{
24// fields to update on order
25discount_amount:10.0,
26},
27});
28}
29};
30
31exportconstoptions:ActionOptions={
32triggers:{
33api:false,
34bigcommerce:{
35// action is triggered by the store/order/updated webhook
36webhooks:["store/order/updated"],
37},
38},
39};
This will constantly update the order with the discount, creating an infinite loop!
Avoiding these webhook loops requires some data to be stored in Gadget. Once data is stored in a model, you can use Gadget's record API that contains built-in change detection to see if code needs to be run, including any writes back to BigCommerce.
More information on change detection can be found in your API reference.
Here is how the above example can be modified to take advantage of change detection. First, save the order data in a bigcommerce/order data model by calling api.bigcommerce.order.update instead of writing back to BigCommerce directly:
Call a model action from the global action to save data
26// action is triggered by the store/order/updated webhook
27webhooks:["store/order/updated"],
28},
29},
30};
The bigcommerce/order record is updated in the database, and record.changes() is used to check and see if a write back to BigCommerce is required, breaking any possible webhook loops:
write to BigCommerce in api/models/bigcommerce/order/actions/update.js