Your webhook endpoint will be accessible to the public internet, so it's important that you verify incoming webhooks to ensure they have actually come from LHV. Every incoming webhook will have an X-LHV-HMAC header, containing a hex-encoded SHA256 HMAC formed from the webhook body and your shared secret.
Managing Your Shared Secret
Every webhook configuration has a secret, shared between you and LHV, which is used for verifying webhook payloads. You can use a same one for different configurations, or have each one be different.
You can either set the secret yourself, or leave it absent and we will generate one automatically. To increase security, we will only show you the secret once - in the response from a create/edit request.
When you edit an existing webhook, you can overwrite the secret with a new one you choose, or leave it absent to generate a new secret (used with immediate effect). If you are editing other fields but want to keep the same secret, then you must provide it in the edit request body.
The secrets themselves are represented as strings, but HMACs work on raw bytes, so you will need to convert it to a byte array of UTF-8 characters (example code is shown later). You can use any valid character in your secret, but it's often wise to avoid special characters anyway.
Secrets should have high entropy to ensure security. For our automaticly-generated secrets, we use a 64-character alphabet ([a-zA-Z0-9_\-]) and generate a string of length 64. This works out to 384 bits of entropy (LENGTH * log2(ALPHABET_SIZE)).
If you're generating your own secret, remember to use a cryptographically secure PRNG. For Kotlin/Java/JVM, try SecureRandom.getInstanceStrong()
Verifying HMACs
To verify a webhook, you should generate the expected HMAC in the same way, and then confirm that it matches the value from the header. If the result is not an exact match (or header is missing), you should discard the webhook as illegitimate, and log this as an error.
For security, you should use a constant-time equals function (as shown in the code examples). If you compare hex strings rather than bytes, then do so case insensitively (e.g. "af12" == "AF12").
Here is rough pseudocode for how you should use the result of HMAC validation. Actual implementations of verifyHmac are given in the next section
Note that rawPayload must be the exact bytes of the HTTP request body. Whitespace trimming or reformatting will result in a failed HMAC comparison. If you can only extract the body as a string then you can still convert it to a UTF-8 byte array - the example code does this to turn secret into secretBytes, so you can copy that for the request body.
Make sure to always use the SHA256algorithm when processing HMACs.
Once you have verified the HMAC of the message, it's also worth making sure that the clientCode and clientCountry fields are correct, and that the eventTimestamp is within expected bounds.
require'openssl'require'rack/utils'# https://rubygems.org/gems/rack# For this implementation the payload and secret actually *should* just be stringsdefgenerate_hmac(payload,secret)OpenSSL::HMAC.digest('sha256', secret, payload)enddefverify_hmac(payload,secret,received_hmac_header) received_hmac = [received_hmac_header].pack('H*') expected_hmac = generate_hmac(payload, secret)Rack::Utils.secure_compare(received_hmac, expected_hmac)end