Skip to content

Commit

Permalink
DBZ-8063 refactor webhook MAC initialization
Browse files Browse the repository at this point in the history
- Rename 'standard_webhooks' namespace to 'webhooks'
- Use Clock for testing webhook timestamp
- Refactor Authenticator config initialization
  • Loading branch information
TimoWilhelm committed Jul 17, 2024
1 parent deb3aae commit 1ad1032
Show file tree
Hide file tree
Showing 6 changed files with 122 additions and 111 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@
import io.debezium.engine.DebeziumEngine;
import io.debezium.server.BaseChangeConsumer;
import io.debezium.server.http.jwt.JWTAuthenticatorBuilder;
import io.debezium.server.http.standard_webhooks.StandardWebhooksAuthenticatorBuilder;
import io.debezium.server.http.webhooks.StandardWebhooksAuthenticatorBuilder;
import io.debezium.util.Clock;
import io.debezium.util.Metronome;

Expand Down Expand Up @@ -136,23 +136,7 @@ void initWithConfig(Config config) throws URISyntaxException {
contentType = "application/json";
}

// Need to be able to throw an exception
// so not using ifPresent() syntax
Optional<String> authenticationType = config.getOptionalValue(PROP_AUTHENTICATION_PREFIX + PROP_AUTHENTICATION_TYPE, String.class);
if (authenticationType.isPresent()) {
String t = authenticationType.get();
if (t.equalsIgnoreCase(JWT_AUTHENTICATION)) {
JWTAuthenticatorBuilder builder = JWTAuthenticatorBuilder.fromConfig(config, PROP_AUTHENTICATION_PREFIX);
authenticator = builder.build();
}
else if (t.equalsIgnoreCase(STANDARD_WEBHOOKS_AUTHENTICATION)) {
StandardWebhooksAuthenticatorBuilder builder = StandardWebhooksAuthenticatorBuilder.fromConfig(config, PROP_AUTHENTICATION_PREFIX);
authenticator = builder.build();
}
else {
throw new DebeziumException("Unknown value '" + t + "' encountered for property " + PROP_AUTHENTICATION_PREFIX + PROP_AUTHENTICATION_TYPE);
}
}
authenticator = buildAuthenticator(config);

LOGGER.info("Using http content-type type {}", contentType);
LOGGER.info("Using sink URL: {}", sinkUrl);
Expand Down Expand Up @@ -185,6 +169,28 @@ public void handleBatch(List<ChangeEvent<Object, Object>> records, DebeziumEngin
committer.markBatchFinished();
}

private Authenticator buildAuthenticator(Config config) {
// Need to be able to throw an exception
// so not using ifPresent() syntax
Optional<String> authenticationType = config.getOptionalValue(PROP_AUTHENTICATION_PREFIX + PROP_AUTHENTICATION_TYPE, String.class);
if (authenticationType.isPresent()) {
String t = authenticationType.get();
if (t.equalsIgnoreCase(JWT_AUTHENTICATION)) {
JWTAuthenticatorBuilder builder = JWTAuthenticatorBuilder.fromConfig(config, PROP_AUTHENTICATION_PREFIX);
return builder.build();
}
else if (t.equalsIgnoreCase(STANDARD_WEBHOOKS_AUTHENTICATION)) {
StandardWebhooksAuthenticatorBuilder builder = StandardWebhooksAuthenticatorBuilder.fromConfig(config, PROP_AUTHENTICATION_PREFIX);
return builder.build();
}
else {
throw new DebeziumException("Unknown value '" + t + "' encountered for property " + PROP_AUTHENTICATION_PREFIX + PROP_AUTHENTICATION_TYPE);
}
}

return null;
}

private boolean recordSent(ChangeEvent<Object, Object> record, UUID messageId) throws InterruptedException {
boolean sent = false;
HttpResponse<String> r;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,13 @@
*
* Licensed under the Apache Software License version 2.0, available at http://www.apache.org/licenses/LICENSE-2.0
*/
package io.debezium.server.http.standard_webhooks;
package io.debezium.server.http.webhooks;

import java.net.http.HttpRequest.Builder;
import java.nio.charset.StandardCharsets;
import java.security.InvalidKeyException;
import java.security.NoSuchAlgorithmException;
import java.time.Clock;
import java.time.Instant;
import java.util.Base64;
import java.util.UUID;
Expand All @@ -27,11 +28,19 @@ public class StandardWebhooksAuthenticator implements Authenticator {
static final String UNBRANDED_MSG_TIMESTAMP_KEY = "webhook-timestamp";
private static final String HMAC_SHA256 = "HmacSHA256";

private final byte[] key;
private final Clock clock;
private final Mac sha512Hmac;

public StandardWebhooksAuthenticator(final String secret) {
this(secret, Clock.systemUTC());
}

@VisibleForTesting
StandardWebhooksAuthenticator(final String secret, Clock clock) {
super();

this.clock = clock;

String sec = secret;
if (sec.startsWith(StandardWebhooksAuthenticator.SECRET_PREFIX)) {
sec = sec.substring(StandardWebhooksAuthenticator.SECRET_PREFIX.length());
Expand All @@ -44,12 +53,20 @@ public StandardWebhooksAuthenticator(final String secret) {
throw new DebeziumException("Webhook secret must be between 24 and 64 bytes");
}

this.key = key;
try {
this.sha512Hmac = Mac.getInstance(HMAC_SHA256);
SecretKeySpec keySpec = new SecretKeySpec(key, HMAC_SHA256);
sha512Hmac.init(keySpec);
}
catch (InvalidKeyException | NoSuchAlgorithmException e) {
throw new DebeziumException("Failed to initialize HMAC-SHA256 signing algorithm", e);
}

}

@Override
public void setAuthorizationHeader(Builder httpRequestBuilder, final String bodyContent, final UUID messageId) {
final long timestamp = Instant.now().getEpochSecond();
final long timestamp = Instant.now(this.clock).getEpochSecond();
final String msgId = "msg_" + messageId;
final String signature = sign(msgId, timestamp, bodyContent);
httpRequestBuilder.header(StandardWebhooksAuthenticator.UNBRANDED_MSG_ID_KEY, msgId);
Expand All @@ -64,17 +81,10 @@ public boolean authenticate() throws InterruptedException {

@VisibleForTesting
String sign(final String msgId, final long timestamp, final String payload) {
try {
String toSign = String.format("%s.%s.%s", msgId, timestamp, payload);
Mac sha512Hmac = Mac.getInstance(HMAC_SHA256);
SecretKeySpec keySpec = new SecretKeySpec(this.key, HMAC_SHA256);
sha512Hmac.init(keySpec);
byte[] macData = sha512Hmac.doFinal(toSign.getBytes(StandardCharsets.UTF_8));
String signature = Base64.getEncoder().encodeToString(macData);
return String.format("v1,%s", signature);
}
catch (InvalidKeyException | NoSuchAlgorithmException e) {
throw new DebeziumException(e.getMessage());
}
// https://github.com/standard-webhooks/standard-webhooks/blob/main/spec/standard-webhooks.md#signature-scheme
final String toSign = String.format("%s.%s.%s", msgId, timestamp, payload);
byte[] macData = sha512Hmac.doFinal(toSign.getBytes(StandardCharsets.UTF_8));
final String signature = Base64.getEncoder().encodeToString(macData);
return String.format("v1,%s", signature);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -3,14 +3,14 @@
*
* Licensed under the Apache Software License version 2.0, available at http://www.apache.org/licenses/LICENSE-2.0
*/
package io.debezium.server.http.standard_webhooks;

import java.util.NoSuchElementException;
package io.debezium.server.http.webhooks;

import org.eclipse.microprofile.config.Config;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import io.debezium.DebeziumException;

public class StandardWebhooksAuthenticatorBuilder {
private static final Logger LOGGER = LoggerFactory.getLogger(StandardWebhooksAuthenticatorBuilder.class);

Expand All @@ -34,7 +34,7 @@ public StandardWebhooksAuthenticator build() {
if (secret == null) {
String msg = "Cannot build StandardWebhooksAuthenticator. Secret must be set.";
LOGGER.error(msg);
throw new NoSuchElementException(msg);
throw new DebeziumException(msg);
}

return new StandardWebhooksAuthenticator(secret);
Expand Down

This file was deleted.

Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
*
* Licensed under the Apache Software License version 2.0, available at http://www.apache.org/licenses/LICENSE-2.0
*/
package io.debezium.server.http.standard_webhooks;
package io.debezium.server.http.webhooks;

import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.eq;
Expand All @@ -15,6 +15,7 @@
import java.util.Optional;

import org.eclipse.microprofile.config.Config;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Test;

public class StandardWebhooksAuthenticatorBuilderTest {
Expand All @@ -24,12 +25,14 @@ public void verifyBuild() throws URISyntaxException {
StandardWebhooksAuthenticatorBuilder builder = new StandardWebhooksAuthenticatorBuilder();
builder.setSecret("whsec_MfKQ9r8GKYqrTwjUPD8ILPZIo2LaLaSw");

StandardWebhooksAuthenticator authenticator = builder.build();
Assertions.assertDoesNotThrow(() -> {
builder.build();
});
}

@Test
public void verifyBuildFromConfig() throws URISyntaxException {
Map<String, Object> configValues = Map.of("debezium.sink.http.webhook.secret", "whsec_MfKQ9r8GKYqrTwjUPD8ILPZIo2LaLaSw");
Map<String, Object> configValues = Map.of("debezium.sink.http.authentication.webhook.secret", "whsec_MfKQ9r8GKYqrTwjUPD8ILPZIo2LaLaSw");

Config result = mock(Config.class);

Expand All @@ -39,9 +42,10 @@ public void verifyBuildFromConfig() throws URISyntaxException {
when(result.getOptionalValue(eq(entry.getKey()), any())).thenReturn(Optional.of(value));
}

StandardWebhooksAuthenticatorBuilder builder = StandardWebhooksAuthenticatorBuilder.fromConfig(result, "debezium.sink.http.");
builder.setSecret("whsec_MfKQ9r8GKYqrTwjUPD8ILPZIo2LaLaSw");
StandardWebhooksAuthenticatorBuilder builder = StandardWebhooksAuthenticatorBuilder.fromConfig(result, "debezium.sink.http.authentication.");

StandardWebhooksAuthenticator authenticator = builder.build();
Assertions.assertDoesNotThrow(() -> {
builder.build();
});
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
/*
* Copyright Debezium Authors.
*
* Licensed under the Apache Software License version 2.0, available at http://www.apache.org/licenses/LICENSE-2.0
*/
package io.debezium.server.http.webhooks;

import java.net.URI;
import java.net.URISyntaxException;
import java.net.http.HttpHeaders;
import java.net.http.HttpRequest;
import java.time.Clock;
import java.time.Instant;
import java.time.ZoneOffset;
import java.util.Optional;
import java.util.UUID;

import org.junit.Test;
import org.junit.jupiter.api.Assertions;

public class StandardWebhooksAuthenticatorTest {

@Test
public void addAuthorizationHeader() throws URISyntaxException {
Clock clock = Clock.fixed(Instant.ofEpochSecond(1234), ZoneOffset.UTC);
UUID messageId = UUID.fromString("22bd292a-71ab-46fe-a460-8632d6754ac6");
StandardWebhooksAuthenticator authenticator = new StandardWebhooksAuthenticator(
"whsec_MfKQ9r8GKYqrTwjUPD8ILPZIo2LaLaSw", clock);

URI testURI = new URI("http://example.com");
String testEventContent = "{\"hello\":\"world\"}";
HttpRequest.Builder builder = HttpRequest.newBuilder(testURI);
builder.POST(HttpRequest.BodyPublishers.ofString(testEventContent));
authenticator.setAuthorizationHeader(builder, testEventContent, messageId);
HttpRequest request = builder.build();

HttpHeaders headers = request.headers();

Optional<String> idHeader = headers.firstValue("webhook-id");
Assertions.assertTrue(idHeader.isPresent());
Assertions.assertEquals("msg_22bd292a-71ab-46fe-a460-8632d6754ac6", idHeader.get());

Optional<String> timestampHeader = headers.firstValue("webhook-timestamp");
Assertions.assertTrue(timestampHeader.isPresent());
Assertions.assertEquals("1234", timestampHeader.get());

Optional<String> signatureHeader = headers.firstValue("webhook-signature");
Assertions.assertTrue(signatureHeader.isPresent());
String[] sigParts = signatureHeader.get().split(",");
Assertions.assertEquals(2, sigParts.length);
Assertions.assertEquals("v1", sigParts[0]);

// https://www.standardwebhooks.com/verify
String expected = "v1,qCVBRIv6rKQVhSJBAmUSE9GkdCdPe2j6xzzkm89UcoA=";

Assertions.assertEquals(expected, signatureHeader.get());
}
}

0 comments on commit 1ad1032

Please sign in to comment.