Implemented checkIfAccountExists() which retrieves user's data and may even first account transactions without asking user for her/his tan procedure
This commit is contained in:
parent
1444d93860
commit
d03c0eb003
|
@ -36,6 +36,8 @@ open class FinTsClient @JvmOverloads constructor(
|
||||||
) {
|
) {
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
const val NinetyDaysAgoMilliseconds = 90 * 24 * 60 * 60 * 1000L
|
||||||
|
|
||||||
private val log = LoggerFactory.getLogger(FinTsClient::class.java)
|
private val log = LoggerFactory.getLogger(FinTsClient::class.java)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -109,6 +111,48 @@ open class FinTsClient @JvmOverloads constructor(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
open fun checkIfAccountExistsAsync(bank: BankData, customer: CustomerData,
|
||||||
|
callback: (FinTsClientResponse) -> Unit) {
|
||||||
|
|
||||||
|
threadPool.runAsync {
|
||||||
|
callback(checkIfAccountExists(bank, customer))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
open fun checkIfAccountExists(bank: BankData, customer: CustomerData): FinTsClientResponse {
|
||||||
|
|
||||||
|
val newUserInfoResponse = getBankAndCustomerInfoForNewUser(bank, customer)
|
||||||
|
|
||||||
|
if (newUserInfoResponse.isSuccessful == false) { // bank parameter (FinTS server address, ...) already seem to be wrong
|
||||||
|
return newUserInfoResponse
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// do not ask user for tan at this stage
|
||||||
|
var didOverwriteUserUnselectedTanProcedure = false
|
||||||
|
if (customer.isTanProcedureSelected == false && customer.supportedTanProcedures.isNotEmpty()) {
|
||||||
|
|
||||||
|
didOverwriteUserUnselectedTanProcedure = true
|
||||||
|
customer.selectedTanProcedure = customer.supportedTanProcedures.first()
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
val synchronizeCustomerResponse = synchronizeCustomerSystemId(bank, customer)
|
||||||
|
|
||||||
|
// also check if retrieving account transactions of last 90 days without tan is supported (and thereby may retrieve first account transactions)
|
||||||
|
val transactionsOfLast90DaysResponse =
|
||||||
|
tryGetTransactionsOfLast90DaysWithoutTan(bank, customer, true)
|
||||||
|
|
||||||
|
|
||||||
|
if (didOverwriteUserUnselectedTanProcedure) {
|
||||||
|
customer.resetSelectedTanProcedure()
|
||||||
|
}
|
||||||
|
|
||||||
|
return if (transactionsOfLast90DaysResponse.isSuccessful) transactionsOfLast90DaysResponse
|
||||||
|
else synchronizeCustomerResponse
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Some banks support that according to PSD2 account transactions may be retrieved without
|
* Some banks support that according to PSD2 account transactions may be retrieved without
|
||||||
* a TAN (= no strong customer authorization needed).
|
* a TAN (= no strong customer authorization needed).
|
||||||
|
@ -117,7 +161,7 @@ open class FinTsClient @JvmOverloads constructor(
|
||||||
callback: (GetTransactionsResponse) -> Unit) {
|
callback: (GetTransactionsResponse) -> Unit) {
|
||||||
|
|
||||||
threadPool.runAsync {
|
threadPool.runAsync {
|
||||||
callback(tryGetTransactionsOfLast90DaysWithoutTan(bank, customer, false))
|
callback(tryGetTransactionsOfLast90DaysWithoutTan(bank, customer))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -133,12 +177,12 @@ open class FinTsClient @JvmOverloads constructor(
|
||||||
protected open fun tryGetTransactionsOfLast90DaysWithoutTan(bank: BankData, customer: CustomerData,
|
protected open fun tryGetTransactionsOfLast90DaysWithoutTan(bank: BankData, customer: CustomerData,
|
||||||
skipSettingCustomerFlag: Boolean): GetTransactionsResponse {
|
skipSettingCustomerFlag: Boolean): GetTransactionsResponse {
|
||||||
|
|
||||||
val ninetyDaysAgoMilliseconds = 90 * 24 * 60 * 60 * 1000L
|
val ninetyDaysAgo = Date(Date().time - NinetyDaysAgoMilliseconds)
|
||||||
val ninetyDaysAgo = Date(Date().time - ninetyDaysAgoMilliseconds)
|
|
||||||
|
|
||||||
val response = getTransactions(
|
val response = getTransactions(
|
||||||
GetTransactionsParameter(false, ninetyDaysAgo), bank, customer)
|
GetTransactionsParameter(false, ninetyDaysAgo), bank, customer)
|
||||||
|
|
||||||
|
|
||||||
customer.triedToRetrieveTransactionsOfLast90DaysWithoutTan = true
|
customer.triedToRetrieveTransactionsOfLast90DaysWithoutTan = true
|
||||||
|
|
||||||
if (response.isSuccessful) {
|
if (response.isSuccessful) {
|
||||||
|
@ -162,8 +206,6 @@ open class FinTsClient @JvmOverloads constructor(
|
||||||
open fun getTransactions(parameter: GetTransactionsParameter, bank: BankData,
|
open fun getTransactions(parameter: GetTransactionsParameter, bank: BankData,
|
||||||
customer: CustomerData): GetTransactionsResponse {
|
customer: CustomerData): GetTransactionsResponse {
|
||||||
|
|
||||||
// synchronizeCustomerSystemIdIfNotDoneYet(bank, customer) // even though specification says this is required it can be omitted
|
|
||||||
|
|
||||||
if (customer.supportsRetrievingTransactionsOfLast90DaysWithoutTan == null &&
|
if (customer.supportsRetrievingTransactionsOfLast90DaysWithoutTan == null &&
|
||||||
customer.triedToRetrieveTransactionsOfLast90DaysWithoutTan == false &&
|
customer.triedToRetrieveTransactionsOfLast90DaysWithoutTan == false &&
|
||||||
parameter.fromDate == null) {
|
parameter.fromDate == null) {
|
||||||
|
|
|
@ -10,19 +10,29 @@ import net.dankito.fints.util.Java8Base64Service
|
||||||
import org.assertj.core.api.Assertions.assertThat
|
import org.assertj.core.api.Assertions.assertThat
|
||||||
import org.junit.Ignore
|
import org.junit.Ignore
|
||||||
import org.junit.Test
|
import org.junit.Test
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean
|
||||||
|
|
||||||
|
|
||||||
@Ignore // not an automatic test, supply your settings below
|
@Ignore // not an automatic test, supply your settings below
|
||||||
class FinTsClientTest {
|
class FinTsClientTest {
|
||||||
|
|
||||||
|
private val didAskUserForTanProcedure = AtomicBoolean(false)
|
||||||
|
|
||||||
|
private val didAskUserToEnterTan = AtomicBoolean(false)
|
||||||
|
|
||||||
|
|
||||||
private val callback = object : FinTsClientCallback {
|
private val callback = object : FinTsClientCallback {
|
||||||
|
|
||||||
override fun askUserForTanProcedure(supportedTanProcedures: List<TanProcedure>): TanProcedure? {
|
override fun askUserForTanProcedure(supportedTanProcedures: List<TanProcedure>): TanProcedure? {
|
||||||
|
didAskUserForTanProcedure.set(true)
|
||||||
|
|
||||||
// TODO: if entering TAN is required select your tan procedure here
|
// TODO: if entering TAN is required select your tan procedure here
|
||||||
return supportedTanProcedures.first()
|
return supportedTanProcedures.first()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun enterTan(tanChallenge: TanChallenge): String? {
|
override fun enterTan(tanChallenge: TanChallenge): String? {
|
||||||
|
didAskUserToEnterTan.set(true)
|
||||||
|
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -41,11 +51,8 @@ class FinTsClientTest {
|
||||||
private val BankDataAnonymous = BankData("10070000", Laenderkennzeichen.Germany, "https://fints.deutsche-bank.de/")
|
private val BankDataAnonymous = BankData("10070000", Laenderkennzeichen.Germany, "https://fints.deutsche-bank.de/")
|
||||||
|
|
||||||
// TODO: add your settings here:
|
// TODO: add your settings here:
|
||||||
private val Bank = BankData("", Laenderkennzeichen.Germany, "", bic = "")
|
private val Bank = BankData("", Laenderkennzeichen.Germany, "")
|
||||||
private val Customer = CustomerData("", "", iban = "")
|
private val Customer = CustomerData("", "")
|
||||||
|
|
||||||
// transfer 1 cent to yourself. Transferring money to oneself also doesn't require to enter a TAN according to PSD2
|
|
||||||
private val BankTransferData = BankTransferData(Customer.name, Customer.iban!!, Bank.bic!!, 0.01.toBigDecimal(), "Give it to me baby")
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -65,6 +72,34 @@ class FinTsClientTest {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun checkIfAccountExists() {
|
||||||
|
|
||||||
|
// when
|
||||||
|
val result = underTest.checkIfAccountExists(Bank, Customer)
|
||||||
|
|
||||||
|
// then
|
||||||
|
assertThat(result.isSuccessful).isTrue()
|
||||||
|
|
||||||
|
assertThat(didAskUserForTanProcedure).isFalse()
|
||||||
|
|
||||||
|
assertThat(Bank.name).isNotEmpty()
|
||||||
|
assertThat(Bank.supportedJobs).isNotEmpty() // supported jobs are now known
|
||||||
|
assertThat(Bank.supportedTanProcedures).isNotEmpty() // supported tan procedures are now known
|
||||||
|
assertThat(Bank.supportedHbciVersions).isNotEmpty() // supported HBIC versions are now known
|
||||||
|
assertThat(Bank.supportedLanguages).isNotEmpty() // supported languages are now known
|
||||||
|
|
||||||
|
assertThat(Customer.name).isNotEmpty()
|
||||||
|
assertThat(Customer.iban).isNotNull()
|
||||||
|
assertThat(Customer.supportedTanProcedures).isNotEmpty()
|
||||||
|
assertThat(Customer.selectedLanguage).isNotEqualTo(Dialogsprache.Default) // language is set now
|
||||||
|
assertThat(Customer.customerSystemId).isNotEqualTo(KundensystemStatus.SynchronizingCustomerSystemId) // customer system id is now set
|
||||||
|
assertThat(Customer.customerSystemStatus).isEqualTo(KundensystemStatusWerte.Benoetigt) // customerSystemStatus is set now
|
||||||
|
assertThat(Customer.accounts).isNotEmpty() // accounts are now known
|
||||||
|
assertThat(Customer.accounts.first().allowedJobs).isNotEmpty() // allowed jobs are now known
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun synchronizeCustomerSystemId() {
|
fun synchronizeCustomerSystemId() {
|
||||||
|
|
||||||
|
@ -97,6 +132,17 @@ class FinTsClientTest {
|
||||||
@Test
|
@Test
|
||||||
fun testBankTransfer() {
|
fun testBankTransfer() {
|
||||||
|
|
||||||
|
// given
|
||||||
|
underTest.checkIfAccountExists(Bank, Customer)
|
||||||
|
|
||||||
|
// now BIC and IBAN should be set
|
||||||
|
assertThat(Bank.bic).describedAs("Bank's BIC should now be set").isNotNull()
|
||||||
|
assertThat(Customer.iban).describedAs("Customer's IBAN should now be set").isNotNull()
|
||||||
|
|
||||||
|
// transfer 1 cent to yourself. Transferring money to oneself also doesn't require to enter a TAN according to PSD2
|
||||||
|
val BankTransferData = BankTransferData(Customer.name, Customer.iban!!, Bank.bic!!, 0.01.toBigDecimal(), "Give it to me baby")
|
||||||
|
|
||||||
|
|
||||||
// when
|
// when
|
||||||
val result = underTest.doBankTransfer(BankTransferData, Bank, Customer)
|
val result = underTest.doBankTransfer(BankTransferData, Bank, Customer)
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue