2020-01-02 21:39:02 +00:00
package net.dankito.banking
2020-06-03 15:49:29 +00:00
import net.dankito.banking.extensions.toIonspinBigDecimal
import net.dankito.banking.extensions.toJavaBigDecimal
import com.soywiz.klock.jvm.toDate
import net.dankito.banking.extensions.toKlockDate
2020-01-02 23:35:36 +00:00
import net.dankito.banking.ui.BankingClientCallback
2020-01-02 21:39:02 +00:00
import net.dankito.banking.ui.IBankingClient
2020-01-02 23:35:36 +00:00
import net.dankito.banking.ui.model.Account
2020-01-02 21:39:02 +00:00
import net.dankito.banking.ui.model.BankAccount
2020-05-16 20:51:51 +00:00
import net.dankito.banking.ui.model.MessageLogEntry
2020-01-02 21:39:02 +00:00
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
2020-05-18 19:23:03 +00:00
import net.dankito.banking.fints.FinTsClientForCustomer
import net.dankito.banking.fints.callback.FinTsClientCallback
import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.TanGeneratorTanMedium
import net.dankito.banking.fints.model.*
2020-06-03 15:49:29 +00:00
import net.dankito.banking.mapper.BankDataMapper
2020-05-18 19:23:03 +00:00
import net.dankito.banking.fints.util.IBase64Service
2020-06-03 15:49:29 +00:00
import net.dankito.banking.fints.util.PureKotlinBase64Service
import net.dankito.banking.fints.util.IThreadPool
import net.dankito.banking.fints.util.JavaThreadPool
2020-01-25 19:29:44 +00:00
import net.dankito.utils.serialization.JacksonJsonSerializer
2020-06-03 15:49:29 +00:00
import net.dankito.banking.fints.webclient.IWebClient
import net.dankito.banking.fints.webclient.KtorWebClient
import net.dankito.banking.bankfinder.BankInfo
2020-01-25 19:29:44 +00:00
import org.slf4j.LoggerFactory
import java.io.File
2020-01-02 21:39:02 +00:00
2020-05-16 18:19:42 +00:00
open class fints4kBankingClient (
2020-01-02 21:39:02 +00:00
bankInfo : BankInfo ,
customerId : String ,
pin : String ,
2020-01-25 19:29:44 +00:00
protected val dataFolder : File ,
2020-06-03 15:49:29 +00:00
webClient : IWebClient = KtorWebClient ( ) ,
base64Service : IBase64Service = PureKotlinBase64Service ( ) ,
threadPool : IThreadPool = JavaThreadPool ( ) ,
2020-01-02 23:35:36 +00:00
callback : BankingClientCallback
2020-01-02 21:39:02 +00:00
) : IBankingClient {
2020-01-25 19:29:44 +00:00
companion object {
2020-05-16 18:19:42 +00:00
val fints4kClientDataFilename = " fints4kClientData.json "
2020-01-25 19:29:44 +00:00
2020-05-16 18:19:42 +00:00
private val log = LoggerFactory . getLogger ( fints4kBankingClient :: class . java )
2020-06-03 15:49:29 +00:00
init {
net . dankito . banking . fints . util . log . LoggerFactory . loggerFactory = net . dankito . banking . fints . util . log . Slf4jLoggerFactory ( )
}
2020-01-25 19:29:44 +00:00
}
2020-05-16 18:19:42 +00:00
protected val mapper = net . dankito . banking . mapper . fints4kModelMapper ( )
2020-01-02 21:39:02 +00:00
protected val bankDataMapper = BankDataMapper ( )
2020-01-25 19:29:44 +00:00
protected val serializer = JacksonJsonSerializer ( )
2020-01-02 21:39:02 +00:00
protected val bank = bankDataMapper . mapFromBankInfo ( bankInfo )
protected val customer = CustomerData ( customerId , pin )
2020-04-28 16:17:47 +00:00
protected var account : Account = mapper . mapAccount ( customer , bank ) // temporary save temp account, we update with data from server response like BankAccounts later
2020-01-02 23:35:36 +00:00
2020-06-03 15:49:29 +00:00
protected val client = FinTsClientForCustomer ( bank , customer , object : FinTsClientCallback {
2020-01-19 15:50:26 +00:00
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
}
2020-01-02 23:35:36 +00:00
override fun enterTan ( customer : CustomerData , tanChallenge : TanChallenge ) : EnterTanResult {
2020-01-19 15:22:43 +00:00
mapper . updateTanMediaAndProcedures ( account , customer )
2020-01-02 23:35:36 +00:00
val result = callback . enterTan ( account , mapper . mapTanChallenge ( tanChallenge ) )
return mapper . mapEnterTanResult ( result , customer )
}
override fun enterTanGeneratorAtc ( customer : CustomerData , tanMedium : TanGeneratorTanMedium ) : EnterTanGeneratorAtcResult {
2020-01-19 15:22:43 +00:00
mapper . updateTanMediaAndProcedures ( account , customer )
2020-01-02 23:35:36 +00:00
val result = callback . enterTanGeneratorAtc ( mapper . mapTanMedium ( tanMedium ) )
return mapper . mapEnterTanGeneratorAtcResult ( result )
}
2020-06-03 15:49:29 +00:00
} , webClient , base64Service , threadPool )
2020-01-02 21:39:02 +00:00
2020-05-16 20:51:51 +00:00
override val messageLogWithoutSensitiveData : List < MessageLogEntry >
2020-06-03 15:49:29 +00:00
get ( ) = client . messageLogWithoutSensitiveData . map { MessageLogEntry ( it . message , it . time . toDate ( ) , account ) }
2020-05-16 20:51:51 +00:00
2020-01-02 21:39:02 +00:00
override fun addAccountAsync ( callback : ( AddAccountResponse ) -> Unit ) {
client . addAccountAsync { response ->
2020-01-02 23:35:36 +00:00
this . account = mapper . mapAccount ( customer , bank )
2020-01-02 21:39:02 +00:00
val mappedResponse = mapper . mapResponse ( account , response )
2020-01-25 19:29:44 +00:00
saveData ( )
2020-01-02 21:39:02 +00:00
callback ( mappedResponse )
}
}
override fun getTransactionsAsync ( bankAccount : BankAccount , parameter : GetTransactionsParameter , callback : ( GetTransactionsResponse ) -> Unit ) {
2020-01-20 23:54:06 +00:00
val account = mapper . findAccountForBankAccount ( customer , bankAccount )
2020-01-02 21:39:02 +00:00
2020-01-20 23:54:06 +00:00
if ( account == null ) {
2020-05-25 17:05:56 +00:00
callback ( GetTransactionsResponse ( bankAccount , false , " Cannot find account for ${bankAccount.identifier} " ) ) // TODO: translate
2020-01-20 23:54:06 +00:00
}
else {
2020-06-03 15:49:29 +00:00
client . getTransactionsAsync ( GetTransactionsParameter ( parameter . alsoRetrieveBalance , parameter . fromDate ?. toKlockDate ( ) , parameter . toDate ?. toKlockDate ( ) , null , parameter . abortIfTanIsRequired ,
2020-05-20 22:29:53 +00:00
{ parameter . retrievedChunkListener ?. invoke ( mapper . mapTransactions ( bankAccount , it ) ) } ) , account ) { response ->
2020-01-02 21:39:02 +00:00
2020-01-20 23:54:06 +00:00
val mappedResponse = mapper . mapResponse ( bankAccount , response )
2020-01-25 19:29:44 +00:00
saveData ( )
2020-01-20 23:54:06 +00:00
callback ( mappedResponse )
}
2020-01-02 21:39:02 +00:00
}
}
2020-01-20 23:54:06 +00:00
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 {
2020-06-03 15:49:29 +00:00
val mappedData = BankTransferData ( data . creditorName , data . creditorIban , data . creditorBic , data . amount . toIonspinBigDecimal ( ) , data . usage , data . instantPayment )
2020-01-02 21:39:02 +00:00
2020-01-20 23:54:06 +00:00
client . doBankTransferAsync ( mappedData , account ) { response ->
2020-01-25 19:29:44 +00:00
saveData ( )
2020-01-20 23:54:06 +00:00
callback ( mapper . mapResponse ( response ) )
}
2020-01-02 21:39:02 +00:00
}
}
2020-01-19 15:22:43 +00:00
2020-01-25 19:29:44 +00:00
override fun restoreData ( ) {
2020-05-16 18:19:42 +00:00
val deserializedCustomer = serializer . deserializeObject ( getFints4kClientDataFile ( ) , CustomerData :: class . java )
2020-01-25 19:29:44 +00:00
deserializedCustomer ?. let {
mapper . updateCustomer ( customer , deserializedCustomer )
2020-04-28 16:17:47 +00:00
account = mapper . mapAccount ( customer , bank )
2020-01-25 19:29:44 +00:00
}
}
protected open fun saveData ( ) {
try {
2020-05-16 18:19:42 +00:00
serializer . serializeObject ( customer , getFints4kClientDataFile ( ) )
2020-01-25 19:29:44 +00:00
} catch ( e : Exception ) {
log . error ( " Could not save customer data for $customer " , e )
}
}
2020-05-16 18:19:42 +00:00
protected open fun getFints4kClientDataFile ( ) : File {
return File ( dataFolder , " ${bank.bankCode} _ ${customer.customerId} _ $fints4kClientDataFilename " )
2020-01-25 19:29:44 +00:00
}
2020-01-02 21:39:02 +00:00
}