BankingClient/fints4k/src/test/kotlin/net/dankito/banking/fints/FinTsClientTest.kt

225 lines
9.2 KiB
Kotlin

package net.dankito.banking.fints
import net.dankito.banking.fints.banks.InMemoryBankFinder
import net.dankito.banking.fints.callback.FinTsClientCallback
import net.dankito.banking.fints.messages.datenelemente.abgeleiteteformate.Laenderkennzeichen
import net.dankito.banking.fints.messages.datenelemente.implementierte.Dialogsprache
import net.dankito.banking.fints.messages.datenelemente.implementierte.KundensystemStatus
import net.dankito.banking.fints.messages.datenelemente.implementierte.KundensystemStatusWerte
import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.TanEinsatzOption
import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.TanGeneratorTanMedium
import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.TanMedienArtVersion
import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.TanMediumKlasse
import net.dankito.banking.fints.messages.segmente.id.CustomerSegmentId
import net.dankito.banking.fints.model.*
import net.dankito.banking.fints.model.mapper.BankDataMapper
import net.dankito.banking.fints.response.client.FinTsClientResponse
import net.dankito.banking.fints.util.Java8Base64Service
import org.assertj.core.api.Assertions.assertThat
import org.junit.Assert
import org.junit.Ignore
import org.junit.Test
import java.time.LocalDateTime
import java.time.format.DateTimeFormatter
import java.util.*
import java.util.concurrent.atomic.AtomicBoolean
@Ignore // not an automatic test, supply your settings below
class FinTsClientTest {
companion object {
val DateTimeFormatForUniqueBankTransferUsage = DateTimeFormatter.ofPattern("yyyy-MM-dd'T'HH:mm:ss.SSS")
}
private val didAskUserForTanProcedure = AtomicBoolean(false)
private val didAskUserToEnterTan = AtomicBoolean(false)
private val callback = object : FinTsClientCallback {
override fun askUserForTanProcedure(supportedTanProcedures: List<TanProcedure>, suggestedTanProcedure: TanProcedure?): TanProcedure? {
return suggestedTanProcedure // simply return suggestedTanProcedure as in most cases it's the best fitting one
}
override fun enterTan(customer: CustomerData, tanChallenge: TanChallenge): EnterTanResult {
didAskUserToEnterTan.set(true)
return EnterTanResult.userDidNotEnterTan()
}
override fun enterTanGeneratorAtc(customer: CustomerData, tanMedium: TanGeneratorTanMedium): EnterTanGeneratorAtcResult {
Assert.fail("Bank asks you to synchronize your TAN generator for card ${tanMedium.cardNumber} " +
"(card sequence number ${tanMedium.cardSequenceNumber}). Please do this via your online banking portal or Banking UI.")
return EnterTanGeneratorAtcResult.userDidNotEnterAtc() // should actually never be called
}
}
private val underTest = object : FinTsClient(callback, Java8Base64Service()) {
fun testSynchronizeCustomerSystemId(bank: BankData, customer: CustomerData): FinTsClientResponse {
return synchronizeCustomerSystemId(bank, customer)
}
}
private val BankDataAnonymous = BankData("10070000", Laenderkennzeichen.Germany, "https://fints.deutsche-bank.de/", "DEUTDEBBXXX")
// TODO: add your settings here:
private val bankInfo = InMemoryBankFinder().findBankByBankCode("<your bank code (BLZ) here>").first()
private val Bank = BankDataMapper().mapFromBankInfo(bankInfo)
private val Customer = CustomerData("<your customer id (Kontonummer) here>", "<your PIN here>")
@Test
fun getAnonymousBankInfo() {
// when
val result = underTest.getAnonymousBankInfo(BankDataAnonymous)
// then
assertThat(result.isSuccessful).isTrue()
assertThat(BankDataAnonymous.supportedHbciVersions).isNotEmpty()
assertThat(BankDataAnonymous.supportedTanProcedures).isNotEmpty()
assertThat(BankDataAnonymous.supportedJobs).isNotEmpty()
assertThat(BankDataAnonymous.supportedLanguages).isNotEmpty()
assertThat(BankDataAnonymous.name).isNotEmpty()
}
@Test
fun addAccount() {
// when
val result = underTest.addAccount(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.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
fun synchronizeCustomerSystemId() {
// when
val result = underTest.testSynchronizeCustomerSystemId(Bank, Customer)
// then
assertThat(result.isSuccessful).isTrue()
assertThat(Customer.customerSystemId).isNotEqualTo(KundensystemStatus.SynchronizingCustomerSystemId) // customer system id is now set
assertThat(Customer.selectedLanguage).isNotEqualTo(Dialogsprache.Default) // language is set now
assertThat(Customer.customerSystemStatus).isEqualTo(KundensystemStatusWerte.Benoetigt) // customerSystemStatus is set now
}
@Test
fun getTransactions() {
// given
underTest.addAccount(Bank, Customer) // retrieve basic data, e.g. accounts
val account = Customer.accounts.firstOrNull { it.supportsFeature(AccountFeature.RetrieveAccountTransactions) }
assertThat(account).describedAs("We need at least one account that supports retrieving account transactions (${CustomerSegmentId.AccountTransactionsMt940.id})").isNotNull()
// when
// some banks support retrieving account transactions of last 90 days without TAN
val result = underTest.tryGetTransactionsOfLast90DaysWithoutTan(Bank, Customer, account!!)
// then
assertThat(result.isSuccessful).isTrue()
assertThat(result.bookedTransactions).isNotEmpty()
}
@Test
fun getTanMediaList() {
// this test is only senseful for accounts using chipTAN / TAN generator as TAN procedure
underTest.getAnonymousBankInfo(Bank)
val supportsRetrievingTanMedia = Bank.supportedJobs.firstOrNull { it.jobName == "HKTAB" } != null
if (supportsRetrievingTanMedia == false) { // accounts with appTAN, pushTAN, smsTAN, ... would fail here -> simply return
return
}
assertThat(Customer.tanMedia).isEmpty()
// when
val result = underTest.getTanMediaList(Bank, Customer, TanMedienArtVersion.Alle, TanMediumKlasse.AlleMedien)
// then
assertThat(result.isSuccessful).isTrue()
assertThat(result.tanMediaList).isNotNull()
assertThat(result.tanMediaList!!.usageOption).isEqualByComparingTo(TanEinsatzOption.KundeKannGenauEinMediumZuEinerZeitNutzen) // TODO: may adjust to your value
assertThat(result.tanMediaList!!.tanMedia).isNotEmpty()
assertThat(Customer.tanMedia).isNotEmpty()
}
@Ignore // only works with banks that don't support HKTAB version 5
@Test(expected = UnsupportedOperationException::class)
fun getTanMediaList_UnsupportedTanMediumClass() {
// when
underTest.getTanMediaList(Bank, Customer, TanMedienArtVersion.Alle, TanMediumKlasse.BilateralVereinbart)
// then
// exception gets thrown
}
@Test
fun testBankTransfer() {
// given
underTest.addAccount(Bank, Customer) // retrieve basic data, e.g. accounts
// we need at least one account that supports cash transfer
val account = Customer.accounts.firstOrNull { it.supportsFeature(AccountFeature.TransferMoney) }
assertThat(account).describedAs("We need at least one account that supports cash transfer (${CustomerSegmentId.SepaBankTransfer.id})").isNotNull()
// IBAN should be set
assertThat(account?.iban).describedAs("Account IBAN must 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, account?.iban!!, Bank.bic, 0.01.toBigDecimal(),
"${DateTimeFormatForUniqueBankTransferUsage.format(LocalDateTime.now())} Test transaction ${UUID.randomUUID()}")
// when
val result = underTest.doBankTransfer(BankTransferData, Bank, Customer, account)
// then
assertThat(result.isSuccessful).isTrue()
}
}