155 lines
5.8 KiB
Kotlin
155 lines
5.8 KiB
Kotlin
package net.dankito.banking
|
|
|
|
import net.dankito.banking.ui.BankingClientCallback
|
|
import net.dankito.banking.ui.IBankingClient
|
|
import net.dankito.banking.ui.model.Account
|
|
import net.dankito.banking.ui.model.BankAccount
|
|
import net.dankito.banking.ui.model.parameters.GetTransactionsParameter
|
|
import net.dankito.banking.ui.model.parameters.TransferMoneyData
|
|
import net.dankito.banking.ui.model.responses.AddAccountResponse
|
|
import net.dankito.banking.ui.model.responses.BankingClientResponse
|
|
import net.dankito.banking.ui.model.responses.GetTransactionsResponse
|
|
import net.dankito.fints.FinTsClientForCustomer
|
|
import net.dankito.fints.callback.FinTsClientCallback
|
|
import net.dankito.fints.messages.datenelemente.implementierte.tan.TanGeneratorTanMedium
|
|
import net.dankito.fints.model.*
|
|
import net.dankito.fints.model.mapper.BankDataMapper
|
|
import net.dankito.fints.util.IBase64Service
|
|
import net.dankito.utils.IThreadPool
|
|
import net.dankito.utils.ThreadPool
|
|
import net.dankito.utils.serialization.JacksonJsonSerializer
|
|
import net.dankito.utils.web.client.IWebClient
|
|
import net.dankito.utils.web.client.OkHttpWebClient
|
|
import org.slf4j.LoggerFactory
|
|
import java.io.File
|
|
|
|
|
|
open class fints4javaBankingClient(
|
|
bankInfo: BankInfo,
|
|
customerId: String,
|
|
pin: String,
|
|
protected val dataFolder: File,
|
|
webClient: IWebClient = OkHttpWebClient(),
|
|
base64Service: IBase64Service,
|
|
threadPool: IThreadPool = ThreadPool(),
|
|
callback: BankingClientCallback
|
|
|
|
) : IBankingClient {
|
|
|
|
companion object {
|
|
val fints4javaClientDataFilename = "fints4javaClientData.json"
|
|
|
|
private val log = LoggerFactory.getLogger(fints4javaBankingClient::class.java)
|
|
}
|
|
|
|
|
|
protected val mapper = net.dankito.banking.mapper.fints4javaModelMapper()
|
|
|
|
protected val bankDataMapper = BankDataMapper()
|
|
|
|
protected val serializer = JacksonJsonSerializer()
|
|
|
|
|
|
protected val bank = bankDataMapper.mapFromBankInfo(bankInfo)
|
|
|
|
protected val customer = CustomerData(customerId, pin)
|
|
|
|
protected var account: Account = mapper.mapAccount(customer, bank) // temporary save temp account, we update with data from server response like BankAccounts later
|
|
|
|
|
|
protected val client = FinTsClientForCustomer(bank, customer, webClient, base64Service, threadPool, object : FinTsClientCallback {
|
|
|
|
override fun askUserForTanProcedure(supportedTanProcedures: List<TanProcedure>, suggestedTanProcedure: TanProcedure?): TanProcedure? {
|
|
// we simply return suggestedTanProcedure as even so it's not user's preferred TAN procedure she still can select it in EnterTanDialog
|
|
return suggestedTanProcedure
|
|
}
|
|
|
|
override fun enterTan(customer: CustomerData, tanChallenge: TanChallenge): EnterTanResult {
|
|
mapper.updateTanMediaAndProcedures(account, customer)
|
|
|
|
val result = callback.enterTan(account, mapper.mapTanChallenge(tanChallenge))
|
|
|
|
return mapper.mapEnterTanResult(result, customer)
|
|
}
|
|
|
|
override fun enterTanGeneratorAtc(customer: CustomerData, tanMedium: TanGeneratorTanMedium): EnterTanGeneratorAtcResult {
|
|
mapper.updateTanMediaAndProcedures(account, customer)
|
|
|
|
val result = callback.enterTanGeneratorAtc(mapper.mapTanMedium(tanMedium))
|
|
|
|
return mapper.mapEnterTanGeneratorAtcResult(result)
|
|
}
|
|
|
|
})
|
|
|
|
|
|
override fun addAccountAsync(callback: (AddAccountResponse) -> Unit) {
|
|
client.addAccountAsync { response ->
|
|
this.account = mapper.mapAccount(customer, bank)
|
|
val mappedResponse = mapper.mapResponse(account, response)
|
|
|
|
saveData()
|
|
|
|
callback(mappedResponse)
|
|
}
|
|
}
|
|
|
|
override fun getTransactionsAsync(bankAccount: BankAccount, parameter: GetTransactionsParameter, callback: (GetTransactionsResponse) -> Unit) {
|
|
val account = mapper.findAccountForBankAccount(customer, bankAccount)
|
|
|
|
if (account == null) {
|
|
callback(GetTransactionsResponse(false, "Cannot find account for ${bankAccount.identifier}")) // TODO: translate
|
|
}
|
|
else {
|
|
client.getTransactionsAsync(net.dankito.fints.model.GetTransactionsParameter(parameter.alsoRetrieveBalance, parameter.fromDate, parameter.toDate), account) { response ->
|
|
|
|
val mappedResponse = mapper.mapResponse(bankAccount, response)
|
|
|
|
saveData()
|
|
|
|
callback(mappedResponse)
|
|
}
|
|
}
|
|
}
|
|
|
|
override fun transferMoneyAsync(data: TransferMoneyData, bankAccount: BankAccount, callback: (BankingClientResponse) -> Unit) {
|
|
val account = mapper.findAccountForBankAccount(customer, bankAccount)
|
|
|
|
if (account == null) {
|
|
callback(BankingClientResponse(false, "Cannot find account for ${bankAccount.identifier}")) // TODO: translate
|
|
}
|
|
else {
|
|
val mappedData = BankTransferData(data.creditorName, data.creditorIban, data.creditorBic, data.amount, data.usage)
|
|
|
|
client.doBankTransferAsync(mappedData, account) { response ->
|
|
saveData()
|
|
|
|
callback(mapper.mapResponse(response))
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
override fun restoreData() {
|
|
val deserializedCustomer = serializer.deserializeObject(getFints4javaClientDataFile(), CustomerData::class.java)
|
|
|
|
deserializedCustomer?.let {
|
|
mapper.updateCustomer(customer, deserializedCustomer)
|
|
|
|
account = mapper.mapAccount(customer, bank)
|
|
}
|
|
}
|
|
|
|
protected open fun saveData() {
|
|
try {
|
|
serializer.serializeObject(customer, getFints4javaClientDataFile())
|
|
} catch (e: Exception) {
|
|
log.error("Could not save customer data for $customer", e)
|
|
}
|
|
}
|
|
|
|
protected open fun getFints4javaClientDataFile(): File {
|
|
return File(dataFolder, "${bank.bankCode}_${customer.customerId}_$fints4javaClientDataFilename")
|
|
}
|
|
|
|
} |