Ghost: Securing Webhooks
Life in security-first environments shifts your perspective on many things, even if you don't really want that. This story explains how to control the Ghost blog and webhooks message exchange.
Life in security-first environments shifts your perspective on many things, even if you don't want that. This story explains how to control the Ghost blog and webhooks message exchange.
You may have read my resentment post about moving from Zappier to Webmethods.io. In the middle of the implementation, I realized that Ghost custom integration protects only inbound API calls and does not offer documentation on outbound integrations.
As you can see, all you have is a small form with a name, blog event, webhook URI, and a mysterious secret. As soon as I enable integration flow endpoint protection, communication between my development site and the flow breaks.
The secret value has nothing to do with the Webmethod.io security key and does not impact the message's content. This makes me suspect my lack of knowledge in the area is much greater than I expected. It has taken some research on Standard Webhooks (I don't think the name makes them an industry standard) and a beneficial resource - Webhooks FYI.
At this moment, I realized that the only way to verify the message's origin and content authenticity was to use the message header X-Ghost-Signature. The platform's source code confirmed that the engine uses an HMAC signature to compute the header value from the body, nonce, and secret key (from the form).
Webmethods.io Integration doesn't have HMAC actions but offers a custom Node.js action with whitelisted packages and input/output interfaces. The code below receives an event body, the secret, and the HMAC sha256 header to confirm the source and integrity of the incoming event.
The code describes input (three properties), output (One boolean result), and action executor. This part uses input values to:
- Parse the X-Ghost-Signature header
- Compute the new SHA-256 hash using the body, webhook secret, and the incoming nonce t.
- Compare signature and computed has using function crypto.timeSafeEqual.
Now, my integration's first action uses a secret key to validate the message header and ensure that my social network accounts are not flooded with spam or worse.