From de6b08c0b97f4cea818e551fc96fbfcad58c0d70 Mon Sep 17 00:00:00 2001 From: TayGov Date: Mon, 24 Aug 2020 08:28:18 -0700 Subject: [PATCH 1/5] Adding bambora payment service --- src/bambora-payment-starter/README.md | 14 ++++ src/bambora-payment-starter/pom.xml | 79 ++++++++++++++++++ .../payment/starter/AutoConfiguration.java | 21 +++++ .../payment/starter/BamboraConstants.java | 4 + .../payment/starter/BamboraException.java | 9 ++ .../payment/starter/BamboraProperties.java | 31 +++++++ .../starter/managment/BamboraCardService.java | 9 ++ .../managment/BamboraCardServiceImpl.java | 82 +++++++++++++++++++ .../models/RecurringPaymentDetails.java | 64 +++++++++++++++ .../main/resources/META-INF/spring.factories | 2 + .../starter/AutoConfigurationTest.java | 26 ++++++ .../managment/BamboraCardServiceTest.java | 4 + src/pom.xml | 8 ++ 13 files changed, 353 insertions(+) create mode 100644 src/bambora-payment-starter/README.md create mode 100644 src/bambora-payment-starter/pom.xml create mode 100644 src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/AutoConfiguration.java create mode 100644 src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraConstants.java create mode 100644 src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraException.java create mode 100644 src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java create mode 100644 src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardService.java create mode 100644 src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java create mode 100644 src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/models/RecurringPaymentDetails.java create mode 100644 src/bambora-payment-starter/src/main/resources/META-INF/spring.factories create mode 100644 src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/AutoConfigurationTest.java create mode 100644 src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceTest.java diff --git a/src/bambora-payment-starter/README.md b/src/bambora-payment-starter/README.md new file mode 100644 index 0000000..536a938 --- /dev/null +++ b/src/bambora-payment-starter/README.md @@ -0,0 +1,14 @@ +# Read Me First +The following was discovered as part of building this project: + +* The original package name 'ca.bc.gov.open.bambora-payment-starter' is invalid and this project uses 'ca.bc.gov.open.bamborapaymentstarter' instead. + +# Getting Started + +### Reference Documentation +For further reference, please consider the following sections: + +* [Official Apache Maven documentation](https://maven.apache.org/guides/index.html) +* [Spring Boot Maven Plugin Reference Guide](https://docs.spring.io/spring-boot/docs/2.3.3.RELEASE/maven-plugin/reference/html/) +* [Create an OCI image](https://docs.spring.io/spring-boot/docs/2.3.3.RELEASE/maven-plugin/reference/html/#build-image) + diff --git a/src/bambora-payment-starter/pom.xml b/src/bambora-payment-starter/pom.xml new file mode 100644 index 0000000..6fbb06c --- /dev/null +++ b/src/bambora-payment-starter/pom.xml @@ -0,0 +1,79 @@ + + + 4.0.0 + + ca.bc.gov.open + bambora-payment-starter + 0.0.1-SNAPSHOT + + + 1.8 + 2.2.4.RELEASE + + + + + org.springframework.boot + spring-boot-starter-web-services + + + org.springframework.boot + spring-boot-starter-tomcat + + + + + + org.springframework.boot + spring-boot-starter-test + test + + + org.junit.vintage + junit-vintage-engine + + + + + + org.apache.commons + commons-lang3 + + + + + + + + org.springframework.boot + spring-boot-dependencies + ${spring-boot.version} + pom + import + + + ca.bc.gov.open + spring-starters-bom + 0.1.5 + pom + import + + + + + + + + org.apache.maven.plugins + maven-compiler-plugin + 3.5.1 + + ${java.version} + ${java.version} + + + + + + diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/AutoConfiguration.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/AutoConfiguration.java new file mode 100644 index 0000000..f411366 --- /dev/null +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/AutoConfiguration.java @@ -0,0 +1,21 @@ +package ca.bc.gov.open.bambora.payment.starter; + +import ca.bc.gov.open.bambora.payment.starter.managment.BamboraCardService; +import ca.bc.gov.open.bambora.payment.starter.managment.BamboraCardServiceImpl; +import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.context.annotation.Bean; + +public class AutoConfiguration { + + private final BamboraProperties bamboraProperties; + + public AutoConfiguration(BamboraProperties bamboraProperties) { + this.bamboraProperties = bamboraProperties; + } + + @Bean + @ConditionalOnMissingBean(BamboraCardService.class) + public BamboraCardService bamboraCardService() { + return new BamboraCardServiceImpl(bamboraProperties); + } +} diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraConstants.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraConstants.java new file mode 100644 index 0000000..6b67d1d --- /dev/null +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraConstants.java @@ -0,0 +1,4 @@ +package ca.bc.gov.open.bambora.payment.starter; + +public class BamboraConstants { +} diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraException.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraException.java new file mode 100644 index 0000000..3204e1d --- /dev/null +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraException.java @@ -0,0 +1,9 @@ +package ca.bc.gov.open.bambora.payment.starter; + +public class BamboraException extends RuntimeException { + + public BamboraException(String message, Throwable cause) { + super(message, cause); + } + +} diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java new file mode 100644 index 0000000..53f6685 --- /dev/null +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java @@ -0,0 +1,31 @@ +package ca.bc.gov.open.bambora.payment.starter; + +import org.springframework.boot.context.properties.ConfigurationProperties; +import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.context.annotation.Configuration; + +@Configuration +@EnableConfigurationProperties(BamboraProperties.class) +@ConfigurationProperties(prefix = "bambora") +public class BamboraProperties { + private String merchantId; + private String hostedProfileUrl; + private String hostedProfileServiceVersion; + private String key; + + public String getMerchantId() { + return merchantId; + } + + public void setMerchantId(String merchantId) { + this.merchantId = merchantId; + } + + public String getHostedProfileUrl() { + return hostedProfileUrl; + } + + public void setHostedProfileUrl(String hostedProfileUrl) { + this.hostedProfileUrl = hostedProfileUrl; + } +} diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardService.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardService.java new file mode 100644 index 0000000..75f03e9 --- /dev/null +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardService.java @@ -0,0 +1,9 @@ +package ca.bc.gov.open.bambora.payment.starter.managment; + +import ca.bc.gov.open.bambora.payment.starter.managment.models.RecurringPaymentDetails; +import com.sun.jndi.toolkit.url.Uri; + +public interface BamboraCardService { + Uri setupRecurringPayment(RecurringPaymentDetails recurringPaymentDetails); + +} diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java new file mode 100644 index 0000000..75abf78 --- /dev/null +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java @@ -0,0 +1,82 @@ +package ca.bc.gov.open.bambora.payment.starter.managment; + +import ca.bc.gov.open.bambora.payment.starter.BamboraException; +import ca.bc.gov.open.bambora.payment.starter.BamboraProperties; +import ca.bc.gov.open.bambora.payment.starter.managment.models.RecurringPaymentDetails; +import com.sun.jndi.toolkit.url.Uri; + +import java.net.MalformedURLException; + +public class BamboraCardServiceImpl implements BamboraCardService { + + private final BamboraProperties bamboraProperties; + + public BamboraCardServiceImpl(BamboraProperties bamboraProperties) { + this.bamboraProperties = bamboraProperties; + } + + @Override + public Uri setupRecurringPayment(RecurringPaymentDetails recurringPaymentDetails) { + try { + return buildRecurringPaymentUrl(recurringPaymentDetails); + } catch (MalformedURLException e) { + throw new BamboraException("Url construction failed", e.getCause()); + } + } + + + private Uri buildRecurringPaymentUrl(RecurringPaymentDetails recurringPaymentDetails) throws MalformedURLException { + String paramString = BeanstreamConstants.PARAM_PPRDIR_SERVICE_VERSION + + "=" + this.hostedProfileServiceVersion + + "&" + BeanstreamConstants.PARAM_PPRDIR_MERCHANT_ID + + "=" + this.merchantId + + "&" + BeanstreamConstants.PARAM_PPRDIR_LANGUAGE + + "=" + BeanstreamConstants.LanguageType.eng.toString() + + "&" + BeanstreamConstants.PARAM_PPRDIR_OPERATION_TYPE + + "=" + opType.toString() + + "&" + BeanstreamConstants.PARAM_PPRDIR_REF1 + + "=" + echoData + + "&" + BeanstreamConstants.PARAM_PPRDIR_RETURN_URL + + "=" + redirectURL + + "&" + BeanstreamConstants.PARAM_PPRDIR_ORDER_NUMBER + + "=" + orderNumber; + + // if doing an update, supply tyhe customer code. + if ( (null != opType) && (opType == BeanstreamConstants.OperationTypes.M) ) { + paramString = paramString + "&" + BeanstreamConstants.PARAM_PPRDIR_CUSTOMER_CODE + + "=" + endUserId; + } + + //replace spaces with "%20" (do not do a full URL encoding. Doesn't work with beanstream. + paramString = paramString.replace(" ", "%20"); + + //add hash key at end of params + paramString = paramString + this.key; + + // Calculate the MD5 value using the Hash Key set on the Order Settings page (Within Beanstream account). + // How: + // Place the hash key after the last parameter. + // Perform an SHA-1 hash on the text up to the end of the key, then + // Replace the hash key with hashValue=[hash result]. + // Add the result to the hosted service url. + // Note: Hash is calculated on the params ONLY.. Does NOT include the hosted payment page url. + // See http://support.beanstream.com/#docs/about-hash-validation.htm?Highlight=hash for more info. + log.log(Level.INFO, "Calculating MD5 on paramString " + paramString); + String hashed = getHash(paramString); + + // Calculate the expiry based on the minutesToExpire value. + SimpleDateFormat sdfDate = new SimpleDateFormat(BeanstreamConstants.PARAM_TRANS_HASH_EXPIRY_FORMAT); + Calendar cal = Calendar.getInstance(); + cal.setTime(new Date()); + cal.add(Calendar.MINUTE, this.minutesToExpire); + String expiry = sdfDate.format(cal.getTime()); + + // Add hash and expiry to the redirect + paramString = paramString.replace(this.key, + "&" + BeanstreamConstants.PARAM_TRANS_HASH_VALUE + "=" + hashed + + "&" + BeanstreamConstants.PARAM_TRANS_HASH_EXPIRY + "=" + expiry); + + String redirect = this.hostedProfileURL + "?" + paramString; + return new Uri("blarg"); + } +} diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/models/RecurringPaymentDetails.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/models/RecurringPaymentDetails.java new file mode 100644 index 0000000..c06800a --- /dev/null +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/models/RecurringPaymentDetails.java @@ -0,0 +1,64 @@ +package ca.bc.gov.open.bambora.payment.starter.managment.models; + +public class RecurringPaymentDetails { + private String orderNumber; + private String endUserId; + private String echoData; + private String redirectURL; + + public String getOrderNumber() { + return orderNumber; + } + + public String getEndUserId() { + return endUserId; + } + + public String getEchoData() { + return echoData; + } + + public String getRedirectURL() { + return redirectURL; + } + + protected RecurringPaymentDetails(Builder builder) { + this.orderNumber = builder.orderNumber; + this.endUserId = builder.endUserId; + this.echoData = builder.echoData; + this.redirectURL = builder.redirectURL; + } + + public static class Builder { + + private String orderNumber; + private String endUserId; + private String echoData; + private String redirectURL; + + public Builder orderNumber(String orderNumber) { + this.orderNumber = orderNumber; + return this; + } + + public Builder endUserId(String endUserId) { + this.endUserId = endUserId; + return this; + } + + public Builder echoData(String echoData) { + this.echoData = echoData; + return this; + } + + public Builder redirectURL(String redirectURL) { + this.redirectURL = redirectURL; + return this; + } + + public RecurringPaymentDetails create() { + return new RecurringPaymentDetails(this); + } + + } +} diff --git a/src/bambora-payment-starter/src/main/resources/META-INF/spring.factories b/src/bambora-payment-starter/src/main/resources/META-INF/spring.factories new file mode 100644 index 0000000..84a80b3 --- /dev/null +++ b/src/bambora-payment-starter/src/main/resources/META-INF/spring.factories @@ -0,0 +1,2 @@ +org.springframework.boot.autoconfigure.EnableAutoConfiguration=\ + ca.bc.gov.open.bambora.payment.starter.AutoConfiguration diff --git a/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/AutoConfigurationTest.java b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/AutoConfigurationTest.java new file mode 100644 index 0000000..82e463b --- /dev/null +++ b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/AutoConfigurationTest.java @@ -0,0 +1,26 @@ +package ca.bc.gov.open.bambora.payment.starter; + +import ca.bc.gov.open.bambora.payment.starter.managment.BamboraCardService; +import org.junit.jupiter.api.Assertions; +import org.junit.jupiter.api.DisplayName; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.TestInstance; +import org.springframework.boot.test.context.runner.ApplicationContextRunner; + +import static org.assertj.core.api.Assertions.assertThat; + +@TestInstance(TestInstance.Lifecycle.PER_CLASS) +@DisplayName("Test AutoConfiguration") +public class AutoConfigurationTest { + ApplicationContextRunner context = new ApplicationContextRunner(); + + @Test + @DisplayName("Test Beans Exist") + public void testConfigure() { + + context.run(it -> { + Assertions.assertNotNull(assertThat(it).getBean(BamboraCardService.class)); + }); + + } +} diff --git a/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceTest.java b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceTest.java new file mode 100644 index 0000000..658ff28 --- /dev/null +++ b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceTest.java @@ -0,0 +1,4 @@ +package ca.bc.gov.open.bambora.payment.starter.managment; + +public class BamboraCardServiceTest { +} diff --git a/src/pom.xml b/src/pom.xml index fd522dc..d4746d0 100644 --- a/src/pom.xml +++ b/src/pom.xml @@ -23,6 +23,7 @@ spring-bceid-starter spring-starters-bom spring-clamav-starter + bambora-payment-starter @@ -50,6 +51,13 @@ + + payment + + bambora-payment-starter + + + From 82a54c750c265d1d7551e2578849925ea05cec1e Mon Sep 17 00:00:00 2001 From: TayGov Date: Mon, 24 Aug 2020 12:41:25 -0700 Subject: [PATCH 2/5] Adding tests --- .../payment/starter/BamboraConstants.java | 19 +++++ .../payment/starter/BamboraProperties.java | 25 ++++++ .../managment/BamboraCardServiceImpl.java | 84 ++++++++++--------- .../models/RecurringPaymentDetails.java | 4 + .../managment/BamboraCardServiceImplTest.java | 79 +++++++++++++++++ .../managment/BamboraCardServiceTest.java | 4 - 6 files changed, 170 insertions(+), 45 deletions(-) create mode 100644 src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java delete mode 100644 src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceTest.java diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraConstants.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraConstants.java index 6b67d1d..c3031ed 100644 --- a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraConstants.java +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraConstants.java @@ -1,4 +1,23 @@ package ca.bc.gov.open.bambora.payment.starter; public class BamboraConstants { + public static final String PARAM_PPRDIR_SERVICE_VERSION = "serviceVersion"; + public static final String PARAM_PPRDIR_MERCHANT_ID = "merchantId"; + public static final String PARAM_PPRDIR_LANGUAGE = "trnLanguage"; + public static final String PARAM_PPRDIR_OPERATION_TYPE = "operationType"; + public static final String PARAM_PPRDIR_RETURN_URL = "trnReturnURL"; + public static final String PARAM_PPRDIR_ORDER_NUMBER = "trnOrderNumber"; + public static final String PARAM_PPRDIR_CUSTOMER_CODE = "customerCode"; + public static final String PARAM_PPRDIR_REF1 = "ref1"; + public static final String LANGUAGE_TYPE = "eng"; + public static final String PARAM_TRANS_HASH_VALUE = "hashValue"; + public static final String PARAM_TRANS_HASH_EXPIRY = "hashExpiry"; + public static final String PARAM_TRANS_HASH_EXPIRY_FORMAT = "yyyyMMddkkmm"; + + + public enum OperationTypes { + N, M + } + + } diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java index 53f6685..3363f6e 100644 --- a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java @@ -12,6 +12,7 @@ public class BamboraProperties { private String hostedProfileUrl; private String hostedProfileServiceVersion; private String key; + private int minutesToExpiry; public String getMerchantId() { return merchantId; @@ -28,4 +29,28 @@ public String getHostedProfileUrl() { public void setHostedProfileUrl(String hostedProfileUrl) { this.hostedProfileUrl = hostedProfileUrl; } + + public String getHostedProfileServiceVersion() { + return hostedProfileServiceVersion; + } + + public void setHostedProfileServiceVersion(String hostedProfileServiceVersion) { + this.hostedProfileServiceVersion = hostedProfileServiceVersion; + } + + public String getKey() { + return key; + } + + public void setKey(String key) { + this.key = key; + } + + public int getMinutesToExpiry() { + return minutesToExpiry; + } + + public void setMinutesToExpiry(int minutesToExpiry) { + this.minutesToExpiry = minutesToExpiry; + } } diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java index 75abf78..df29caf 100644 --- a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java @@ -1,11 +1,17 @@ package ca.bc.gov.open.bambora.payment.starter.managment; +import ca.bc.gov.open.bambora.payment.starter.BamboraConstants; import ca.bc.gov.open.bambora.payment.starter.BamboraException; import ca.bc.gov.open.bambora.payment.starter.BamboraProperties; import ca.bc.gov.open.bambora.payment.starter.managment.models.RecurringPaymentDetails; import com.sun.jndi.toolkit.url.Uri; +import org.springframework.util.DigestUtils; import java.net.MalformedURLException; +import java.text.MessageFormat; +import java.text.SimpleDateFormat; +import java.util.Calendar; +import java.util.Date; public class BamboraCardServiceImpl implements BamboraCardService { @@ -26,57 +32,53 @@ public Uri setupRecurringPayment(RecurringPaymentDetails recurringPaymentDetails private Uri buildRecurringPaymentUrl(RecurringPaymentDetails recurringPaymentDetails) throws MalformedURLException { - String paramString = BeanstreamConstants.PARAM_PPRDIR_SERVICE_VERSION + - "=" + this.hostedProfileServiceVersion + - "&" + BeanstreamConstants.PARAM_PPRDIR_MERCHANT_ID + - "=" + this.merchantId + - "&" + BeanstreamConstants.PARAM_PPRDIR_LANGUAGE + - "=" + BeanstreamConstants.LanguageType.eng.toString() + - "&" + BeanstreamConstants.PARAM_PPRDIR_OPERATION_TYPE + - "=" + opType.toString() + - "&" + BeanstreamConstants.PARAM_PPRDIR_REF1 + - "=" + echoData + - "&" + BeanstreamConstants.PARAM_PPRDIR_RETURN_URL + - "=" + redirectURL + - "&" + BeanstreamConstants.PARAM_PPRDIR_ORDER_NUMBER + - "=" + orderNumber; - - // if doing an update, supply tyhe customer code. - if ( (null != opType) && (opType == BeanstreamConstants.OperationTypes.M) ) { - paramString = paramString + "&" + BeanstreamConstants.PARAM_PPRDIR_CUSTOMER_CODE + - "=" + endUserId; - } - //replace spaces with "%20" (do not do a full URL encoding. Doesn't work with beanstream. - paramString = paramString.replace(" ", "%20"); + String operationType = (recurringPaymentDetails.getEndUserId() != null ? BamboraConstants.OperationTypes.M.toString() : BamboraConstants.OperationTypes.N.toString()); + + StringBuilder paramString = new StringBuilder(); + + paramString.append(formatBamboraParam("", BamboraConstants.PARAM_PPRDIR_SERVICE_VERSION, bamboraProperties.getHostedProfileServiceVersion())); + + paramString.append(formatBamboraParam("&", BamboraConstants.PARAM_PPRDIR_MERCHANT_ID, bamboraProperties.getMerchantId())); + + paramString.append(formatBamboraParam("&", BamboraConstants.PARAM_PPRDIR_LANGUAGE, BamboraConstants.LANGUAGE_TYPE)); + + paramString.append(formatBamboraParam("&", BamboraConstants.PARAM_PPRDIR_OPERATION_TYPE, operationType)); + + paramString.append(formatBamboraParam("&", BamboraConstants.PARAM_PPRDIR_REF1, recurringPaymentDetails.getEchoData())); + + paramString.append(formatBamboraParam("&", BamboraConstants.PARAM_PPRDIR_RETURN_URL, recurringPaymentDetails.getRedirectURL())); + + paramString.append(formatBamboraParam("&", BamboraConstants.PARAM_PPRDIR_ORDER_NUMBER, recurringPaymentDetails.getRedirectURL())); + + if (operationType.equals(BamboraConstants.OperationTypes.M.toString())) + paramString.append(formatBamboraParam("&", BamboraConstants.PARAM_PPRDIR_CUSTOMER_CODE, recurringPaymentDetails.getEndUserId())); //add hash key at end of params - paramString = paramString + this.key; - - // Calculate the MD5 value using the Hash Key set on the Order Settings page (Within Beanstream account). - // How: - // Place the hash key after the last parameter. - // Perform an SHA-1 hash on the text up to the end of the key, then - // Replace the hash key with hashValue=[hash result]. - // Add the result to the hosted service url. - // Note: Hash is calculated on the params ONLY.. Does NOT include the hosted payment page url. - // See http://support.beanstream.com/#docs/about-hash-validation.htm?Highlight=hash for more info. - log.log(Level.INFO, "Calculating MD5 on paramString " + paramString); - String hashed = getHash(paramString); + paramString.append(bamboraProperties.getKey()); + + String hashed = getHash(paramString.toString()); // Calculate the expiry based on the minutesToExpire value. - SimpleDateFormat sdfDate = new SimpleDateFormat(BeanstreamConstants.PARAM_TRANS_HASH_EXPIRY_FORMAT); + SimpleDateFormat sdfDate = new SimpleDateFormat(BamboraConstants.PARAM_TRANS_HASH_EXPIRY_FORMAT); Calendar cal = Calendar.getInstance(); cal.setTime(new Date()); - cal.add(Calendar.MINUTE, this.minutesToExpire); + cal.add(Calendar.MINUTE, bamboraProperties.getMinutesToExpiry()); String expiry = sdfDate.format(cal.getTime()); // Add hash and expiry to the redirect - paramString = paramString.replace(this.key, - "&" + BeanstreamConstants.PARAM_TRANS_HASH_VALUE + "=" + hashed + - "&" + BeanstreamConstants.PARAM_TRANS_HASH_EXPIRY + "=" + expiry); + String hashedParameter = paramString.toString().replace(bamboraProperties.getKey(), MessageFormat.format("&{0}={1}&{2}={3}", BamboraConstants.PARAM_TRANS_HASH_VALUE, hashed, BamboraConstants.PARAM_TRANS_HASH_EXPIRY, expiry)); + + return new Uri(MessageFormat.format("{0}?{1}", bamboraProperties.getHostedProfileUrl(), hashedParameter)); + + } + + private String formatBamboraParam(String prefix, String key, String value) { + return MessageFormat.format("{0}{1}={2}", prefix, key, value).replace(" ", "%20"); + } - String redirect = this.hostedProfileURL + "?" + paramString; - return new Uri("blarg"); + private String getHash(String message) { + String digest = DigestUtils.md5DigestAsHex(message.getBytes()); + return digest.toUpperCase(); } } diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/models/RecurringPaymentDetails.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/models/RecurringPaymentDetails.java index c06800a..d275fe5 100644 --- a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/models/RecurringPaymentDetails.java +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/models/RecurringPaymentDetails.java @@ -29,6 +29,10 @@ protected RecurringPaymentDetails(Builder builder) { this.redirectURL = builder.redirectURL; } + public static RecurringPaymentDetails.Builder builder() { + return new RecurringPaymentDetails.Builder(); + } + public static class Builder { private String orderNumber; diff --git a/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java new file mode 100644 index 0000000..2baaf8c --- /dev/null +++ b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java @@ -0,0 +1,79 @@ +package ca.bc.gov.open.bambora.payment.starter.managment; + +import ca.bc.gov.open.bambora.payment.starter.BamboraException; +import ca.bc.gov.open.bambora.payment.starter.BamboraProperties; +import ca.bc.gov.open.bambora.payment.starter.managment.models.RecurringPaymentDetails; +import com.sun.jndi.toolkit.url.Uri; +import org.junit.jupiter.api.*; + +@TestInstance(TestInstance.Lifecycle.PER_CLASS) +@DisplayName("Test BamboraCardServiceImpl") +public class BamboraCardServiceImplTest { + private static final String PROFILE_SERVICE_VERSION = "HOSTEDPROFILE"; + private static final String HOSTED_PROFILE_URL = "http://localhost"; + private static final String KEY = "XTN123TNV123"; + private static final String MERCHANT_ID = "123"; + private static final int MINUTES_TO_EXPIRY = 12; + private static final String ORDERNUM = "ORDERNUM"; + private static final String ECHODATA = "ECHODATA"; + private static final String REDIRECTURL = "REDIRECTURL"; + private static final String END_USER_ID = "123"; + private static final String BAMBORA_CLIENT_URL = "http://localhost?serviceVersion=HOSTEDPROFILE&merchantId=123&trnLanguage=eng&operationType=M&ref1=ECHODATA&trnReturnURL=REDIRECTURL&trnOrderNumber=REDIRECTURL&customerCode=123&hashValue=98C48E9A9C45B99698ED7D2ED7194A91&hashExpiry=202008241242"; + private static final String BAMBORA_NEW_URL = "http://localhost?serviceVersion=HOSTEDPROFILE&merchantId=123&trnLanguage=eng&operationType=M&ref1=ECHODATA&trnReturnURL=REDIRECTURL&trnOrderNumber=REDIRECTURL&customerCode=123&hashValue=98C48E9A9C45B99698ED7D2ED7194A91&hashExpiry=202008241242"; + + private BamboraCardServiceImpl sut; + + private BamboraProperties bamboraProperties; + + @BeforeEach + public void init() { + + bamboraProperties = new BamboraProperties(); + bamboraProperties.setHostedProfileServiceVersion(PROFILE_SERVICE_VERSION); + bamboraProperties.setHostedProfileUrl(HOSTED_PROFILE_URL); + bamboraProperties.setKey(KEY); + bamboraProperties.setMerchantId(MERCHANT_ID); + bamboraProperties.setMinutesToExpiry(MINUTES_TO_EXPIRY); + + } + + @Test + @DisplayName("With client create update url") + public void withClientIdCreateUpdateUrl() { + sut = new BamboraCardServiceImpl(bamboraProperties); + + Uri actual = sut.setupRecurringPayment(createPaymentDetail(END_USER_ID)); + + Assertions.assertNotNull(actual); + Assertions.assertEquals(actual.toString(), BAMBORA_CLIENT_URL); + } + + + @Test + @DisplayName("Without client create update url") + public void withoutClientIdCreateUpdateUrl() { + sut = new BamboraCardServiceImpl(bamboraProperties); + + Uri actual = sut.setupRecurringPayment(createPaymentDetail(null)); + + Assertions.assertNotNull(actual); + Assertions.assertEquals(actual.toString(), BAMBORA_NEW_URL); + } + + @Test + @DisplayName("With invalid paramter throw invalid url exception") + public void withInvalidParamterThrowException() { + bamboraProperties.setHostedProfileUrl("NOTAURL"); + sut = new BamboraCardServiceImpl(bamboraProperties); + Assertions.assertThrows(BamboraException.class, () -> sut.setupRecurringPayment(createPaymentDetail(END_USER_ID))); + } + + private RecurringPaymentDetails createPaymentDetail(String endUserId) { + return RecurringPaymentDetails.builder() + .orderNumber(ORDERNUM) + .echoData(ECHODATA) + .redirectURL(REDIRECTURL) + .endUserId(endUserId) + .create(); + } +} diff --git a/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceTest.java b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceTest.java deleted file mode 100644 index 658ff28..0000000 --- a/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceTest.java +++ /dev/null @@ -1,4 +0,0 @@ -package ca.bc.gov.open.bambora.payment.starter.managment; - -public class BamboraCardServiceTest { -} From 612ef7cfc5719ef2ef0f786d751ec4336a61bcf8 Mon Sep 17 00:00:00 2001 From: TayGov Date: Mon, 24 Aug 2020 13:05:06 -0700 Subject: [PATCH 3/5] Update version --- src/bambora-payment-starter/pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/bambora-payment-starter/pom.xml b/src/bambora-payment-starter/pom.xml index 6fbb06c..207794d 100644 --- a/src/bambora-payment-starter/pom.xml +++ b/src/bambora-payment-starter/pom.xml @@ -5,7 +5,7 @@ ca.bc.gov.open bambora-payment-starter - 0.0.1-SNAPSHOT + 0.1.5 1.8 From 5dfc7bf6d7204221a0dbfd3509960ce8ed60861c Mon Sep 17 00:00:00 2001 From: TayGov Date: Mon, 24 Aug 2020 13:08:13 -0700 Subject: [PATCH 4/5] Rename to hashKey --- .../bambora/payment/starter/BamboraProperties.java | 10 +++++----- .../starter/managment/BamboraCardServiceImpl.java | 4 ++-- .../starter/managment/BamboraCardServiceImplTest.java | 2 +- 3 files changed, 8 insertions(+), 8 deletions(-) diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java index 3363f6e..94cc0ec 100644 --- a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/BamboraProperties.java @@ -11,7 +11,7 @@ public class BamboraProperties { private String merchantId; private String hostedProfileUrl; private String hostedProfileServiceVersion; - private String key; + private String hashKey; private int minutesToExpiry; public String getMerchantId() { @@ -38,12 +38,12 @@ public void setHostedProfileServiceVersion(String hostedProfileServiceVersion) { this.hostedProfileServiceVersion = hostedProfileServiceVersion; } - public String getKey() { - return key; + public String getHashKey() { + return hashKey; } - public void setKey(String key) { - this.key = key; + public void setHashKey(String hashKey) { + this.hashKey = hashKey; } public int getMinutesToExpiry() { diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java index df29caf..3d692c4 100644 --- a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java @@ -55,7 +55,7 @@ private Uri buildRecurringPaymentUrl(RecurringPaymentDetails recurringPaymentDet paramString.append(formatBamboraParam("&", BamboraConstants.PARAM_PPRDIR_CUSTOMER_CODE, recurringPaymentDetails.getEndUserId())); //add hash key at end of params - paramString.append(bamboraProperties.getKey()); + paramString.append(bamboraProperties.getHashKey()); String hashed = getHash(paramString.toString()); @@ -67,7 +67,7 @@ private Uri buildRecurringPaymentUrl(RecurringPaymentDetails recurringPaymentDet String expiry = sdfDate.format(cal.getTime()); // Add hash and expiry to the redirect - String hashedParameter = paramString.toString().replace(bamboraProperties.getKey(), MessageFormat.format("&{0}={1}&{2}={3}", BamboraConstants.PARAM_TRANS_HASH_VALUE, hashed, BamboraConstants.PARAM_TRANS_HASH_EXPIRY, expiry)); + String hashedParameter = paramString.toString().replace(bamboraProperties.getHashKey(), MessageFormat.format("&{0}={1}&{2}={3}", BamboraConstants.PARAM_TRANS_HASH_VALUE, hashed, BamboraConstants.PARAM_TRANS_HASH_EXPIRY, expiry)); return new Uri(MessageFormat.format("{0}?{1}", bamboraProperties.getHostedProfileUrl(), hashedParameter)); diff --git a/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java index 2baaf8c..22d7c7a 100644 --- a/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java +++ b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java @@ -31,7 +31,7 @@ public void init() { bamboraProperties = new BamboraProperties(); bamboraProperties.setHostedProfileServiceVersion(PROFILE_SERVICE_VERSION); bamboraProperties.setHostedProfileUrl(HOSTED_PROFILE_URL); - bamboraProperties.setKey(KEY); + bamboraProperties.setHashKey(KEY); bamboraProperties.setMerchantId(MERCHANT_ID); bamboraProperties.setMinutesToExpiry(MINUTES_TO_EXPIRY); From 47e507441cf86c31b445abf9b6da41b92601ecb0 Mon Sep 17 00:00:00 2001 From: TayGov Date: Tue, 25 Aug 2020 07:01:45 -0700 Subject: [PATCH 5/5] Update logic --- src/bambora-payment-starter/pom.xml | 5 +++++ .../payment/starter/AutoConfiguration.java | 4 ++++ .../managment/BamboraCardServiceImpl.java | 21 +++++++------------ .../managment/BamboraCardServiceImplTest.java | 8 +++---- 4 files changed, 21 insertions(+), 17 deletions(-) diff --git a/src/bambora-payment-starter/pom.xml b/src/bambora-payment-starter/pom.xml index 207794d..02316ac 100644 --- a/src/bambora-payment-starter/pom.xml +++ b/src/bambora-payment-starter/pom.xml @@ -41,6 +41,11 @@ commons-lang3 + + commons-codec + commons-codec + + diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/AutoConfiguration.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/AutoConfiguration.java index f411366..0edb517 100644 --- a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/AutoConfiguration.java +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/AutoConfiguration.java @@ -3,8 +3,12 @@ import ca.bc.gov.open.bambora.payment.starter.managment.BamboraCardService; import ca.bc.gov.open.bambora.payment.starter.managment.BamboraCardServiceImpl; import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +@Configuration +@EnableConfigurationProperties(BamboraProperties.class) public class AutoConfiguration { private final BamboraProperties bamboraProperties; diff --git a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java index 3d692c4..8a46071 100644 --- a/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java +++ b/src/bambora-payment-starter/src/main/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImpl.java @@ -5,7 +5,7 @@ import ca.bc.gov.open.bambora.payment.starter.BamboraProperties; import ca.bc.gov.open.bambora.payment.starter.managment.models.RecurringPaymentDetails; import com.sun.jndi.toolkit.url.Uri; -import org.springframework.util.DigestUtils; +import org.apache.commons.codec.digest.DigestUtils; import java.net.MalformedURLException; import java.text.MessageFormat; @@ -54,23 +54,18 @@ private Uri buildRecurringPaymentUrl(RecurringPaymentDetails recurringPaymentDet if (operationType.equals(BamboraConstants.OperationTypes.M.toString())) paramString.append(formatBamboraParam("&", BamboraConstants.PARAM_PPRDIR_CUSTOMER_CODE, recurringPaymentDetails.getEndUserId())); - //add hash key at end of params - paramString.append(bamboraProperties.getHashKey()); + paramString.append(MessageFormat.format("&{0}={1}&{2}={3}", BamboraConstants.PARAM_TRANS_HASH_VALUE, getHash(paramString.toString()), BamboraConstants.PARAM_TRANS_HASH_EXPIRY, getExpiry())); - String hashed = getHash(paramString.toString()); + return new Uri(MessageFormat.format("{0}?{1}", bamboraProperties.getHostedProfileUrl(), paramString.toString())); - // Calculate the expiry based on the minutesToExpire value. + } + + private String getExpiry() { SimpleDateFormat sdfDate = new SimpleDateFormat(BamboraConstants.PARAM_TRANS_HASH_EXPIRY_FORMAT); Calendar cal = Calendar.getInstance(); cal.setTime(new Date()); cal.add(Calendar.MINUTE, bamboraProperties.getMinutesToExpiry()); - String expiry = sdfDate.format(cal.getTime()); - - // Add hash and expiry to the redirect - String hashedParameter = paramString.toString().replace(bamboraProperties.getHashKey(), MessageFormat.format("&{0}={1}&{2}={3}", BamboraConstants.PARAM_TRANS_HASH_VALUE, hashed, BamboraConstants.PARAM_TRANS_HASH_EXPIRY, expiry)); - - return new Uri(MessageFormat.format("{0}?{1}", bamboraProperties.getHostedProfileUrl(), hashedParameter)); - + return sdfDate.format(cal.getTime()); } private String formatBamboraParam(String prefix, String key, String value) { @@ -78,7 +73,7 @@ private String formatBamboraParam(String prefix, String key, String value) { } private String getHash(String message) { - String digest = DigestUtils.md5DigestAsHex(message.getBytes()); + String digest = DigestUtils.md5Hex(MessageFormat.format("{0}{1}", message, bamboraProperties.getHashKey())); return digest.toUpperCase(); } } diff --git a/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java index 22d7c7a..f5faf43 100644 --- a/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java +++ b/src/bambora-payment-starter/src/test/java/ca/bc/gov/open/bambora/payment/starter/managment/BamboraCardServiceImplTest.java @@ -18,8 +18,8 @@ public class BamboraCardServiceImplTest { private static final String ECHODATA = "ECHODATA"; private static final String REDIRECTURL = "REDIRECTURL"; private static final String END_USER_ID = "123"; - private static final String BAMBORA_CLIENT_URL = "http://localhost?serviceVersion=HOSTEDPROFILE&merchantId=123&trnLanguage=eng&operationType=M&ref1=ECHODATA&trnReturnURL=REDIRECTURL&trnOrderNumber=REDIRECTURL&customerCode=123&hashValue=98C48E9A9C45B99698ED7D2ED7194A91&hashExpiry=202008241242"; - private static final String BAMBORA_NEW_URL = "http://localhost?serviceVersion=HOSTEDPROFILE&merchantId=123&trnLanguage=eng&operationType=M&ref1=ECHODATA&trnReturnURL=REDIRECTURL&trnOrderNumber=REDIRECTURL&customerCode=123&hashValue=98C48E9A9C45B99698ED7D2ED7194A91&hashExpiry=202008241242"; + private static final String BAMBORA_CLIENT_URL = "http://localhost?serviceVersion=HOSTEDPROFILE&merchantId=123&trnLanguage=eng&operationType=M&ref1=ECHODATA&trnReturnURL=REDIRECTURL&trnOrderNumber=REDIRECTURL&customerCode=123&hashValue=98C48E9A9C45B99698ED7D2ED7194A91"; + private static final String BAMBORA_NEW_URL = "http://localhost?serviceVersion=HOSTEDPROFILE&merchantId=123&trnLanguage=eng&operationType=N&ref1=ECHODATA&trnReturnURL=REDIRECTURL&trnOrderNumber=REDIRECTURL&hashValue=BB2616ADA96F7C72824835D484F23B9B"; private BamboraCardServiceImpl sut; @@ -45,7 +45,7 @@ public void withClientIdCreateUpdateUrl() { Uri actual = sut.setupRecurringPayment(createPaymentDetail(END_USER_ID)); Assertions.assertNotNull(actual); - Assertions.assertEquals(actual.toString(), BAMBORA_CLIENT_URL); + Assertions.assertTrue(actual.toString().contains(BAMBORA_CLIENT_URL)); } @@ -57,7 +57,7 @@ public void withoutClientIdCreateUpdateUrl() { Uri actual = sut.setupRecurringPayment(createPaymentDetail(null)); Assertions.assertNotNull(actual); - Assertions.assertEquals(actual.toString(), BAMBORA_NEW_URL); + Assertions.assertTrue(actual.toString().contains(BAMBORA_NEW_URL)); } @Test