From b6a0e48fd7f45bab2bf08b911aad82a9f601b612 Mon Sep 17 00:00:00 2001 From: dankito Date: Mon, 21 Dec 2020 20:36:25 +0100 Subject: [PATCH] =?UTF-8?q?Extracted=20FinTsJobExecutor=20to=20separate=20?= =?UTF-8?q?high=20level=20methods=20of=20FinTsClient=20that=20group=20mult?= =?UTF-8?q?iple=20low=20level=20jobs=20from=20FinTsJobExecutor=20that=20ex?= =?UTF-8?q?ecutes=20that=20low=20level=20jobs=20(=3D=20FinTS=20Gesch=C3=A4?= =?UTF-8?q?ftsvorf=C3=A4lle)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../net/dankito/banking/fints/FinTsClient.kt | 1132 +--------------- .../banking/fints/FinTsClientForCustomer.kt | 2 +- .../dankito/banking/fints/FinTsJobExecutor.kt | 1157 +++++++++++++++++ .../banking/fints/FinTsClientTestBase.kt | 2 +- .../bankdetails/BanksFinTsDetailsRetriever.kt | 9 +- 5 files changed, 1183 insertions(+), 1119 deletions(-) create mode 100644 fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsJobExecutor.kt diff --git a/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsClient.kt b/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsClient.kt index 458d24b9..fdbd64fc 100644 --- a/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsClient.kt +++ b/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsClient.kt @@ -3,48 +3,21 @@ package net.dankito.banking.fints import kotlinx.coroutines.GlobalScope import kotlinx.coroutines.launch import net.dankito.banking.fints.callback.FinTsClientCallback -import net.dankito.banking.fints.log.IMessageLogAppender -import net.dankito.banking.fints.log.MessageLogCollector -import net.dankito.banking.fints.messages.MessageBuilder -import net.dankito.banking.fints.messages.MessageBuilderResult -import net.dankito.banking.fints.messages.datenelemente.implementierte.Dialogsprache -import net.dankito.banking.fints.messages.datenelemente.implementierte.KundensystemStatusWerte -import net.dankito.banking.fints.messages.datenelemente.implementierte.signatur.Sicherheitsfunktion -import net.dankito.banking.fints.messages.datenelemente.implementierte.signatur.VersionDesSicherheitsverfahrens import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.* import net.dankito.banking.fints.messages.segmente.id.CustomerSegmentId -import net.dankito.banking.fints.messages.segmente.id.ISegmentId import net.dankito.banking.fints.model.* -import net.dankito.banking.fints.response.InstituteSegmentId import net.dankito.banking.fints.response.BankResponse -import net.dankito.banking.fints.response.ResponseParser import net.dankito.banking.fints.response.client.* import net.dankito.banking.fints.response.segments.* -import net.dankito.banking.fints.tan.FlickerCodeDecoder -import net.dankito.banking.fints.tan.TanImageDecoder -import net.dankito.banking.fints.transactions.IAccountTransactionsParser -import net.dankito.banking.fints.transactions.Mt940AccountTransactionsParser -import net.dankito.banking.fints.util.IBase64Service -import net.dankito.banking.fints.util.PureKotlinBase64Service -import net.dankito.utils.multiplatform.log.Logger -import net.dankito.utils.multiplatform.log.LoggerFactory -import net.dankito.banking.fints.webclient.IWebClient -import net.dankito.banking.fints.webclient.KtorWebClient -import net.dankito.banking.fints.webclient.WebClientResponse import net.dankito.utils.multiplatform.Date -import net.dankito.utils.multiplatform.getInnerExceptionMessage -import net.dankito.utils.multiplatform.ObjectReference +/** + * This is the high level FinTS client that groups single low level jobs of [FinTsJobExecutor] to senseful units e.g. + * [addAccountAsync] gets user's TAN methods, user's TAN media, user's bank accounts and may even current balance and account transactions of last 90 days. + */ open class FinTsClient( - protected val callback: FinTsClientCallback, - protected val webClient: IWebClient = KtorWebClient(), - protected val base64Service: IBase64Service = PureKotlinBase64Service(), - protected val messageBuilder: MessageBuilder = MessageBuilder(), - protected val responseParser: ResponseParser = ResponseParser(), - protected val mt940Parser: IAccountTransactionsParser = Mt940AccountTransactionsParser(), - protected val messageLogCollector: MessageLogCollector = MessageLogCollector(), - protected val product: ProductData = ProductData("15E53C26816138699C7B6A3E8", "1.0.0") // TODO: get version dynamically + protected open val jobExecutor: FinTsJobExecutor ) { companion object { @@ -52,28 +25,14 @@ open class FinTsClient( const val OneDayMillis = 24 * 60 * 60 * 1000L const val NinetyDaysMillis = 90 * OneDayMillis - - - private val log = LoggerFactory.getLogger(FinTsClient::class) } + constructor(callback: FinTsClientCallback) : this(FinTsJobExecutor(callback)) + + open val messageLogWithoutSensitiveData: List - get() = messageLogCollector.messageLogWithoutSensitiveData - - protected open val messageLogAppender: IMessageLogAppender = object : IMessageLogAppender { - - override fun logError(message: String, e: Exception?, logger: Logger?, bank: BankData?) { - messageLogCollector.logError(message, e, logger, bank) - } - - } - - - init { - responseParser.logAppender = messageLogAppender - mt940Parser.logAppender = messageLogAppender - } + get() = jobExecutor.messageLogWithoutSensitiveData /** @@ -96,166 +55,7 @@ open class FinTsClient( * On success [bank] parameter is updated afterwards. */ open fun getAnonymousBankInfo(bank: BankData, callback: (FinTsClientResponse) -> Unit) { - getAnonymousBankInfoInternal(bank) { response -> - callback(FinTsClientResponse(response)) - } - } - - protected open fun getAnonymousBankInfoInternal(bank: BankData, callback: (BankResponse) -> Unit) { - val dialogContext = DialogContext(bank, product) - - val message = messageBuilder.createAnonymousDialogInitMessage(dialogContext) - - getAndHandleResponseForMessage(message, dialogContext) { response -> - if (response.successful) { - updateBankData(bank, response) - - closeAnonymousDialog(dialogContext, response) - } - - callback(response) - } - } - - protected open fun closeAnonymousDialog(dialogContext: DialogContext, response: BankResponse) { - - // bank already closed dialog -> there's no need to send dialog end message - if (dialogContext.closeDialog == false || dialogContext.didBankCloseDialog) { - return - } - - val dialogEndRequestBody = messageBuilder.createAnonymousDialogEndMessage(dialogContext) - - fireAndForgetMessage(dialogEndRequestBody, dialogContext) - } - - - open fun getUsersTanMethods(bank: BankData, callback: (FinTsClientResponse) -> Unit) { - getUsersTanMethodsInternal(bank, true) { - callback(FinTsClientResponse(it)) - } - } - - protected open fun getUsersTanMethodsInternal(bank: BankData, closeDialog: Boolean = false, callback: (BankResponse) -> Unit) { - // just to ensure settings are in its initial state and that bank sends us bank parameter (BPD), - // user parameter (UPD) and allowed tan methods for user (therefore the resetSelectedTanMethod()) - bank.resetBpdVersion() - bank.resetUpdVersion() - /** - * Sind dem Kundenprodukt die konkreten, für den Benutzer zugelassenen Sicherheitsverfahren nicht bekannt, so können - * diese über eine Dialoginitialisierung mit Sicherheitsfunktion=999 angefordert werden. Die konkreten Verfahren - * werden dann über den Rückmeldungscode=3920 zurückgemeldet. Im Rahmen dieses Prozesses darf keine UPD - * zurückgeliefert werden und die Durchführung anderer Geschäftsvorfälle ist in einem solchen Dialog nicht erlaubt. - */ - bank.resetSelectedTanMethod() - - // this is the only case where Einschritt-TAN-Verfahren is accepted: to get user's TAN methods - val dialogContext = DialogContext(bank, product, closeDialog, versionOfSecurityMethod = VersionDesSicherheitsverfahrens.Version_1) - - val message = messageBuilder.createInitDialogMessage(dialogContext) - - getAndHandleResponseForMessage(message, dialogContext) { response -> - closeDialog(dialogContext) - - handleGetUsersTanMethodsResponse(response, dialogContext, callback) - } - } - - protected open fun handleGetUsersTanMethodsResponse(response: BankResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { - val getUsersTanMethodsResponse = GetUserTanMethodsResponse(response) - - if (getUsersTanMethodsResponse.successful) { // TODO: really update data only on complete successfully response? as it may contain useful information anyway // TODO: extract method for this code part - updateBankData(dialogContext.bank, getUsersTanMethodsResponse) - updateCustomerData(dialogContext.bank, getUsersTanMethodsResponse) - } - - // even though it is required by specification some banks don't support retrieving user's TAN method by setting TAN method to '999' - if (bankDoesNotSupportRetrievingUsersTanMethods(getUsersTanMethodsResponse)) { - getBankDataForNewUserViaAnonymousDialog(dialogContext.bank, callback) // TODO: should not be necessary anymore - } - else { - callback(getUsersTanMethodsResponse) - } - } - - protected open fun bankDoesNotSupportRetrievingUsersTanMethods(response: BankResponse): Boolean { - return response.successful == false && - response.segmentFeedbacks.flatMap { it.feedbacks }.firstOrNull { it.responseCode == 9200 && - it.message == "Gewähltes Zwei-Schritt-Verfahren nicht unterstützt." } != null - } - - // TODO: this is only a quick fix. Find a better and general solution - protected open fun getBankDataForNewUserViaAnonymousDialog(bank: BankData, callback: (BankResponse) -> Unit) { - getAnonymousBankInfoInternal(bank) { anonymousBankInfoResponse -> - if (anonymousBankInfoResponse.successful == false) { - callback(anonymousBankInfoResponse) - } - else if (bank.tanMethodSupportedByBank.isEmpty()) { // should only be a theoretical error - callback(BankResponse(true, - errorMessage = "Die TAN Verfahren der Bank konnten nicht ermittelt werden")) // TODO: translate - } - else { - bank.tanMethodsAvailableForUser = bank.tanMethodSupportedByBank - getUsersTanMethod(bank) { didSelectTanMethod -> - if (didSelectTanMethod) { - val dialogContext = DialogContext(bank, product) - - initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext) { initDialogResponse -> - closeDialog(dialogContext) - - callback(initDialogResponse) - } - } - else { - callback(BankResponse(false)) - } - } - } - } - } - - - protected open fun getAccounts(bank: BankData, callback: (BankResponse) -> Unit) { - - val dialogContext = DialogContext(bank, product, false) - - initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext) { response -> - closeDialog(dialogContext) - - if (response.successful) { - updateBankData(bank, response) - updateCustomerData(bank, response) - } - - callback(response) - } - } - - - /** - * According to specification synchronizing customer system id is required: - * "Die Kundensystem-ID ist beim HBCI RAH- / RDH- sowie dem PIN/TAN-Verfahren erforderlich." - * - * But as tests show this can be omitted. - * - * But when you do it, this has to be done in an extra dialog as dialog has to be initialized - * with retrieved customer system id. - * - * If you change customer system id during a dialog your messages get rejected by bank institute. - */ - protected open fun synchronizeCustomerSystemId(bank: BankData, callback: (FinTsClientResponse) -> Unit) { - - val dialogContext = DialogContext(bank, product) - val message = messageBuilder.createSynchronizeCustomerSystemIdMessage(dialogContext) - - getAndHandleResponseForMessage(message, dialogContext) { response -> - if (response.successful) { - updateBankData(bank, response) - updateCustomerData(bank, response) - - closeDialog(dialogContext) - } - + jobExecutor.getAnonymousBankInfo(bank) { response -> callback(FinTsClientResponse(response)) } } @@ -266,14 +66,14 @@ open class FinTsClient( /* First dialog: Get user's basic data like BPD, customer system ID and her TAN methods */ - getUsersTanMethodsInternal(bank) { newUserInfoResponse -> + jobExecutor.retrieveBasicDataLikeUsersTanMethods(bank) { newUserInfoResponse -> if (newUserInfoResponse.successful == false) { // bank parameter (FinTS server address, ...) already seem to be wrong callback(AddAccountResponse(newUserInfoResponse, bank)) - return@getUsersTanMethodsInternal + return@retrieveBasicDataLikeUsersTanMethods } - getUsersTanMethod(bank) { didSelectTanMethod -> + jobExecutor.getUsersTanMethod(bank) { didSelectTanMethod -> if (didSelectTanMethod == false) { callback(AddAccountResponse(BankResponse(false), bank)) @@ -282,7 +82,7 @@ open class FinTsClient( /* Second dialog: some banks require that in order to initialize a dialog with strong customer authorization TAN media is required */ - if (isJobSupported(bank, CustomerSegmentId.TanMediaList)) { + if (jobExecutor.isJobSupported(bank, CustomerSegmentId.TanMediaList)) { getTanMediaList(bank, TanMedienArtVersion.Alle, TanMediumKlasse.AlleMedien) { addAccountGetAccountsAndTransactions(parameter, bank, callback) } @@ -298,7 +98,7 @@ open class FinTsClient( /* Third dialog: Now we can initialize our first dialog with strong customer authorization. Use it to get UPD and customer's accounts */ - getAccounts(bank) { getAccountsResponse -> + jobExecutor.getAccounts(bank) { getAccountsResponse -> if (getAccountsResponse.successful == false) { callback(AddAccountResponse(getAccountsResponse, bank)) @@ -366,920 +166,26 @@ open class FinTsClient( open fun getTransactionsAsync(parameter: GetTransactionsParameter, bank: BankData, callback: (GetTransactionsResponse) -> Unit) { - val dialogContext = DialogContext(bank, product) - - initDialogWithStrongCustomerAuthentication(dialogContext) { initDialogResponse -> - - if (initDialogResponse.successful == false) { - callback(GetTransactionsResponse(initDialogResponse, RetrievedAccountData.unsuccessfulList(parameter.account))) - } - else { - mayGetBalance(parameter, dialogContext) { balanceResponse -> - if (dialogContext.didBankCloseDialog) { - callback(GetTransactionsResponse(balanceResponse, RetrievedAccountData.unsuccessfulList(parameter.account))) - } - else { - getTransactionsAfterInitAndGetBalance(parameter, dialogContext, balanceResponse, callback) - } - } - } - } - } - - protected open fun getTransactionsAfterInitAndGetBalance(parameter: GetTransactionsParameter, dialogContext: DialogContext, - balanceResponse: BankResponse, callback: (GetTransactionsResponse) -> Unit) { - var balance: Money? = balanceResponse.getFirstSegmentById(InstituteSegmentId.Balance)?.let { - Money(it.balance, it.currency) - } - val bookedTransactions = mutableSetOf() - val unbookedTransactions = mutableSetOf() - - val message = messageBuilder.createGetTransactionsMessage(parameter, dialogContext) - - var remainingMt940String = "" - - dialogContext.abortIfTanIsRequired = parameter.abortIfTanIsRequired - - dialogContext.chunkedResponseHandler = { response -> - response.getFirstSegmentById(InstituteSegmentId.AccountTransactionsMt940)?.let { transactionsSegment -> - val (chunkTransaction, remainder) = mt940Parser.parseTransactionsChunk(remainingMt940String + transactionsSegment.bookedTransactionsString, - dialogContext.bank, parameter.account) - - bookedTransactions.addAll(chunkTransaction) - remainingMt940String = remainder - - parameter.retrievedChunkListener?.invoke(bookedTransactions) - } - - response.getFirstSegmentById(InstituteSegmentId.CreditCardTransactions)?.let { creditCardTransactionsSegment -> - balance = Money(creditCardTransactionsSegment.balance.amount, creditCardTransactionsSegment.balance.currency ?: "EUR") - bookedTransactions.addAll(creditCardTransactionsSegment.transactions.map { AccountTransaction(parameter.account, it.amount, it.description, it.bookingDate, it.transactionDescriptionBase ?: "", null, null, "", it.valueDate) }) - } - } - - getAndHandleResponseForMessage(message, dialogContext) { response -> - closeDialog(dialogContext) - - val successful = response.successful && (parameter.alsoRetrieveBalance == false || balance != null) - val fromDate = parameter.fromDate - ?: parameter.account.countDaysForWhichTransactionsAreKept?.let { Date(Date.today.millisSinceEpoch - it * OneDayMillis) } - ?: bookedTransactions.map { it.valueDate }.sortedBy { it.millisSinceEpoch }.firstOrNull() - val retrievedData = RetrievedAccountData(parameter.account, successful, balance, bookedTransactions, unbookedTransactions, fromDate, parameter.toDate ?: Date.today, response.errorMessage) - - callback(GetTransactionsResponse(response, listOf(retrievedData), - if (parameter.maxCountEntries != null) parameter.isSettingMaxCountEntriesAllowedByBank else null - )) - } - } - - protected open fun mayGetBalance(parameter: GetTransactionsParameter, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { - if (parameter.alsoRetrieveBalance && parameter.account.supportsRetrievingBalance) { - val message = messageBuilder.createGetBalanceMessage(parameter.account, dialogContext) - - getAndHandleResponseForMessage(message, dialogContext) { response -> - callback(response) - } - } - else { - callback(BankResponse(false, errorMessage = "Either not requested to get balance or account does not support retrieving balance. " + - "Should retrieve balance = ${parameter.alsoRetrieveBalance}, account supports retrieving balance = ${parameter.account.supportsRetrievingBalance}.")) - } + jobExecutor.getTransactionsAsync(parameter, bank, callback) } open fun getTanMediaList(bank: BankData, tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle, tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien, callback: (GetTanMediaListResponse) -> Unit) { - sendMessageAndHandleResponse(bank, CustomerSegmentId.TanMediaList, false, { dialogContext -> - messageBuilder.createGetTanMediaListMessage(dialogContext, tanMediaKind, tanMediumClass) - }) { response -> - handleGetTanMediaListResponse(response, bank, callback) - } - } - - private fun handleGetTanMediaListResponse(response: BankResponse, bank: BankData, callback: (GetTanMediaListResponse) -> Unit) { - // TAN media list (= TAN generator list) is only returned for users with chipTAN TAN methods - val tanMediaList = if (response.successful == false) null - else response.getFirstSegmentById(InstituteSegmentId.TanMediaList) - - tanMediaList?.let { - bank.tanMedia = it.tanMedia - } - - callback(GetTanMediaListResponse(response, tanMediaList)) + jobExecutor.getTanMediaList(bank, tanMediaKind, tanMediumClass, callback) } open fun changeTanMedium(newActiveTanMedium: TanGeneratorTanMedium, bank: BankData, callback: (FinTsClientResponse) -> Unit) { - changeTanMediumInternal(newActiveTanMedium, bank) { response -> + jobExecutor.changeTanMedium(newActiveTanMedium, bank) { response -> callback(FinTsClientResponse(response)) } } - protected open fun changeTanMediumInternal(newActiveTanMedium: TanGeneratorTanMedium, bank: BankData, callback: (BankResponse) -> Unit) { - - if (bank.changeTanMediumParameters?.enteringAtcAndTanRequired == true) { - this.callback.enterTanGeneratorAtc(bank, newActiveTanMedium) { enteredAtc -> - if (enteredAtc.hasAtcBeenEntered == false) { - val message = "Bank requires to enter ATC and TAN in order to change TAN medium." // TODO: translate - callback(BankResponse(false, errorMessage = message)) - } - else { - sendChangeTanMediumMessage(bank, newActiveTanMedium, enteredAtc, callback) - } - } - } - else { - sendChangeTanMediumMessage(bank, newActiveTanMedium, null, callback) - } - } - - protected open fun sendChangeTanMediumMessage(bank: BankData, newActiveTanMedium: TanGeneratorTanMedium, enteredAtc: EnterTanGeneratorAtcResult?, - callback: (BankResponse) -> Unit) { - - sendMessageAndHandleResponse(bank, null, true, { dialogContext -> - messageBuilder.createChangeTanMediumMessage(newActiveTanMedium, dialogContext, enteredAtc?.tan, enteredAtc?.atc) - }) { response -> - callback(response) - } - } - open fun doBankTransferAsync(bankTransferData: BankTransferData, bank: BankData, account: AccountData, callback: (FinTsClientResponse) -> Unit) { - - sendMessageAndHandleResponse(bank, null, true, { dialogContext -> - messageBuilder.createBankTransferMessage(bankTransferData, account, dialogContext) - }) { response -> - callback(FinTsClientResponse(response)) - } - } - - - protected open fun sendMessageAndHandleResponse(bank: BankData, segmentForNonStrongCustomerAuthenticationTwoStepTanProcess: CustomerSegmentId? = null, - closeDialog: Boolean = true, createMessage: (DialogContext) -> MessageBuilderResult, callback: (BankResponse) -> Unit) { - - val dialogContext = DialogContext(bank, product, closeDialog) - - if (segmentForNonStrongCustomerAuthenticationTwoStepTanProcess == null) { - initDialogWithStrongCustomerAuthentication(dialogContext) { initDialogResponse -> - sendMessageAndHandleResponseAfterDialogInitialization(dialogContext, initDialogResponse, createMessage, callback) - } - } - else { - initDialogMessageWithoutStrongCustomerAuthenticationAfterSuccessfulChecks(dialogContext, segmentForNonStrongCustomerAuthenticationTwoStepTanProcess) { initDialogResponse -> - sendMessageAndHandleResponseAfterDialogInitialization(dialogContext, initDialogResponse, createMessage, callback) - } - } - } - - private fun sendMessageAndHandleResponseAfterDialogInitialization(dialogContext: DialogContext, initDialogResponse: BankResponse, createMessage: (DialogContext) -> MessageBuilderResult, callback: (BankResponse) -> Unit) { - - if (initDialogResponse.successful == false) { - callback(initDialogResponse) - } - else { - val message = createMessage(dialogContext) - - getAndHandleResponseForMessage(message, dialogContext) { response -> - closeDialog(dialogContext) - - callback(response) - } - } - } - - protected open fun initDialogWithStrongCustomerAuthentication(dialogContext: DialogContext, callback: (BankResponse) -> Unit) { - - // we first need to retrieve supported tan methods and jobs before we can do anything - ensureBasicBankDataRetrieved(dialogContext.bank) { retrieveBasicBankDataResponse -> - if (retrieveBasicBankDataResponse.successful == false) { - callback(retrieveBasicBankDataResponse) - } - else { - // as in the next step we have to supply user's tan method, ensure user selected his or her - ensureTanMethodIsSelected(dialogContext.bank) { tanMethodSelectedResponse -> - if (tanMethodSelectedResponse.successful == false) { - callback(tanMethodSelectedResponse) - } - else { - initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext, callback) - } - } - } - } - } - - protected open fun initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext: DialogContext, callback: (BankResponse) -> Unit) { - - val message = messageBuilder.createInitDialogMessage(dialogContext) - - getAndHandleResponseForMessage(message, dialogContext) { response -> - - if (response.successful) { - updateBankData(dialogContext.bank, response) - updateCustomerData(dialogContext.bank, response) - } - - callback(response) - } - } - - protected open fun initDialogMessageWithoutStrongCustomerAuthenticationAfterSuccessfulChecks(dialogContext: DialogContext, segmentIdForTwoStepTanProcess: CustomerSegmentId?, - callback: (BankResponse) -> Unit) { - - val message = messageBuilder.createInitDialogMessageWithoutStrongCustomerAuthentication(dialogContext, segmentIdForTwoStepTanProcess) - - getAndHandleResponseForMessage(message, dialogContext) { response -> - if (response.successful) { - updateBankData(dialogContext.bank, response) - updateCustomerData(dialogContext.bank, response) - } - - callback(response) - } - } - - protected open fun closeDialog(dialogContext: DialogContext) { - - // bank already closed dialog -> there's no need to send dialog end message - if (dialogContext.closeDialog == false || dialogContext.didBankCloseDialog) { - return - } - - val dialogEndRequestBody = messageBuilder.createDialogEndMessage(dialogContext) - - fireAndForgetMessage(dialogEndRequestBody, dialogContext) - } - - - protected open fun ensureBasicBankDataRetrieved(bank: BankData, callback: (BankResponse) -> Unit) { - if (bank.tanMethodSupportedByBank.isEmpty() || bank.supportedJobs.isEmpty()) { - getUsersTanMethodsInternal(bank) { getBankInfoResponse -> - if (getBankInfoResponse.successful == false || bank.tanMethodSupportedByBank.isEmpty() - || bank.supportedJobs.isEmpty()) { - - callback(BankResponse(false, errorMessage = - "Could not retrieve basic bank data like supported tan methods or supported jobs")) // TODO: translate // TODO: add as messageToShowToUser - } - else { - callback(BankResponse(true)) - } - } - } - else { - callback(BankResponse(true)) - } - } - - protected open fun ensureTanMethodIsSelected(bank: BankData, callback: (BankResponse) -> Unit) { - if (bank.isTanMethodSelected == false) { - if (bank.tanMethodsAvailableForUser.isEmpty()) { - getUsersTanMethodsInternal(bank) { - if (bank.tanMethodsAvailableForUser.isEmpty()) { // could not retrieve supported tan methods for user - callback(BankResponse(false, noTanMethodSelected = true)) - } - else { - getUsersTanMethod(bank) { - callback(BankResponse(bank.isTanMethodSelected, noTanMethodSelected = !!!bank.isTanMethodSelected)) - } - } - } - } - else { - getUsersTanMethod(bank) { - callback(BankResponse(bank.isTanMethodSelected, noTanMethodSelected = !!!bank.isTanMethodSelected)) - } - } - } - else { - callback(BankResponse(bank.isTanMethodSelected, noTanMethodSelected = !!!bank.isTanMethodSelected)) - } - } - - protected open fun getUsersTanMethod(bank: BankData, done: (Boolean) -> Unit) { - if (bank.tanMethodsAvailableForUser.size == 1) { // user has only one TAN method -> set it and we're done - bank.selectedTanMethod = bank.tanMethodsAvailableForUser.first() - done(true) - } - else { - // we know user's supported tan methods, now ask user which one to select - callback.askUserForTanMethod(bank.tanMethodsAvailableForUser, selectSuggestedTanMethod(bank)) { selectedTanMethod -> - if (selectedTanMethod != null) { - bank.selectedTanMethod = selectedTanMethod - done(true) - } - else { - done(false) - } - } - } - } - - protected open fun selectSuggestedTanMethod(bank: BankData): TanMethod? { - return bank.tanMethodsAvailableForUser.firstOrNull { it.type != TanMethodType.ChipTanUsb && it.type != TanMethodType.SmsTan && it.type != TanMethodType.ChipTanManuell } - ?: bank.tanMethodsAvailableForUser.firstOrNull { it.type != TanMethodType.ChipTanUsb && it.type != TanMethodType.SmsTan } - ?: bank.tanMethodsAvailableForUser.firstOrNull { it.type != TanMethodType.ChipTanUsb } - ?: bank.tanMethodsAvailableForUser.firstOrNull() - } - - - protected open fun getAndHandleResponseForMessage(message: MessageBuilderResult, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { - if (message.createdMessage == null) { - callback(BankResponse(false, messageCreationError = message)) - } - else { - getAndHandleResponseForMessage(message.createdMessage, dialogContext) { response -> - handleMayRequiresTan(response, dialogContext) { handledResponse -> - // if there's a Aufsetzpunkt (continuationId) set, then response is not complete yet, there's more information to fetch by sending this Aufsetzpunkt - handledResponse.aufsetzpunkt?.let { continuationId -> - if (handledResponse.followUpResponse == null) { // for re-sent messages followUpResponse is already set and dialog already closed -> would be overwritten with an error response that dialog is closed - if (message.isSendEnteredTanMessage() == false) { // for sending TAN no follow up message can be created -> filter out, otherwise chunkedResponseHandler would get called twice for same response - dialogContext.chunkedResponseHandler?.invoke(handledResponse) - } - - getFollowUpMessageForContinuationId(handledResponse, continuationId, message, dialogContext) { followUpResponse -> - handledResponse.followUpResponse = followUpResponse - handledResponse.hasFollowUpMessageButCouldNotReceiveIt = handledResponse.followUpResponse == null - - callback(handledResponse) - } - } - else { - callback(handledResponse) - } - } - ?: run { - // e.g. response = enter TAN response, but handledResponse is then response after entering TAN, e.g. account transactions - // -> chunkedResponseHandler would get called for same handledResponse multiple times - if (response == handledResponse) { - dialogContext.chunkedResponseHandler?.invoke(handledResponse) - } - - callback(handledResponse) - } - } - } - } - } - - protected open fun getAndHandleResponseForMessage(requestBody: String, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { - addMessageLog(requestBody, MessageLogEntryType.Sent, dialogContext) - - getResponseForMessage(requestBody, dialogContext.bank.finTs3ServerAddress) { webResponse -> - val response = handleResponse(webResponse, dialogContext) - - dialogContext.response = response - - response.messageHeader?.let { header -> dialogContext.dialogId = header.dialogId } - dialogContext.didBankCloseDialog = response.didBankCloseDialog - - callback(response) - } - } - - protected open fun getResponseForMessage(requestBody: String, finTs3ServerAddress: String, callback: (WebClientResponse) -> Unit) { - val encodedRequestBody = base64Service.encode(requestBody) - - webClient.post(finTs3ServerAddress, encodedRequestBody, "application/octet-stream", IWebClient.DefaultUserAgent, callback) - } - - protected open fun fireAndForgetMessage(message: MessageBuilderResult, dialogContext: DialogContext) { - message.createdMessage?.let { requestBody -> - addMessageLog(requestBody, MessageLogEntryType.Sent, dialogContext) - - getResponseForMessage(requestBody, dialogContext.bank.finTs3ServerAddress) { } - - // if really needed add received response to message log here - } - } - - protected open fun handleResponse(webResponse: WebClientResponse, dialogContext: DialogContext): BankResponse { - val responseBody = webResponse.body - - if (webResponse.successful && responseBody != null) { - - try { - val decodedResponse = decodeBase64Response(responseBody) - - addMessageLog(decodedResponse, MessageLogEntryType.Received, dialogContext) - - return responseParser.parse(decodedResponse) - } catch (e: Exception) { - logError("Could not decode responseBody:\r\n'$responseBody'", dialogContext, e) - - return BankResponse(false, errorMessage = e.getInnerExceptionMessage()) - } - } - else { - val bank = dialogContext.bank - logError("Request to $bank (${bank.finTs3ServerAddress}) failed", dialogContext, webResponse.error) - } - - return BankResponse(false, errorMessage = webResponse.error?.getInnerExceptionMessage()) - } - - protected open fun decodeBase64Response(responseBody: String): String { - return base64Service.decode(responseBody.replace("\r", "").replace("\n", "")) - } - - - protected open fun getFollowUpMessageForContinuationId(response: BankResponse, continuationId: String, message: MessageBuilderResult, - dialogContext: DialogContext, callback: (BankResponse?) -> Unit) { - - messageBuilder.rebuildMessageWithContinuationId(message, continuationId, dialogContext)?.let { followUpMessage -> - getAndHandleResponseForMessage(followUpMessage, dialogContext, callback) - } - ?: run { callback(null) } - } - - - protected open fun addMessageLog(message: String, type: MessageLogEntryType, dialogContext: DialogContext) { - messageLogCollector.addMessageLog(message, type, dialogContext.bank) - } - - protected open fun logError(message: String, dialogContext: DialogContext, e: Exception?) { - messageLogAppender.logError(message, e, log, dialogContext.bank) - } - - - protected open fun handleMayRequiresTan(response: BankResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { // TODO: use response from DialogContext - - if (response.isStrongAuthenticationRequired) { - if (dialogContext.abortIfTanIsRequired) { - response.tanRequiredButWeWereToldToAbortIfSo = true - - callback(response) - return - } - else if (response.tanResponse != null) { - response.tanResponse?.let { tanResponse -> - handleEnteringTanRequired(tanResponse, response, dialogContext, callback) - } - - return - } - } - - // TODO: check if response contains '3931 TAN-Generator gesperrt, Synchronisierung erforderlich' or - // '3933 TAN-Generator gesperrt, Synchronisierung erforderlich Kartennummer ##########' message, - // call callback.enterAtc() and implement and call HKTSY job (p. 77) - - // TODO: also check '9931 Sperrung des Kontos nach %1 Fehlversuchen' -> if %1 == 3 synchronize TAN generator - // as it's quite unrealistic that user entered TAN wrong three times, in most cases TAN generator is not synchronized - - callback(response) - } - - protected open fun handleEnteringTanRequired(tanResponse: TanResponse, response: BankResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { - val bank = dialogContext.bank // TODO: copy required data to TanChallenge - val tanChallenge = createTanChallenge(tanResponse, bank) - - val userDidCancelEnteringTan = ObjectReference(false) - - this.callback.enterTan(bank, tanChallenge) { enteredTanResult -> - userDidCancelEnteringTan.value = true - - handleEnterTanResult(enteredTanResult, tanResponse, response, dialogContext, callback) - } - - mayRetrieveAutomaticallyIfUserEnteredDecoupledTan(tanChallenge, tanResponse, userDidCancelEnteringTan, dialogContext) - } - - protected open fun createTanChallenge(tanResponse: TanResponse, bank: BankData): TanChallenge { - // TODO: is this true for all tan methods? - val messageToShowToUser = tanResponse.challenge ?: "" - val challenge = tanResponse.challengeHHD_UC ?: "" - val tanMethod = bank.selectedTanMethod - - return when (tanMethod.type) { - TanMethodType.ChipTanFlickercode -> - FlickerCodeTanChallenge(FlickerCodeDecoder().decodeChallenge(challenge, tanMethod.hhdVersion ?: HHDVersion.HHD_1_4), // HHD 1.4 is currently the most used version - messageToShowToUser, challenge, tanMethod, tanResponse.tanMediaIdentifier) - - TanMethodType.ChipTanQrCode, TanMethodType.ChipTanPhotoTanMatrixCode, - TanMethodType.QrCode, TanMethodType.photoTan -> - ImageTanChallenge(TanImageDecoder().decodeChallenge(challenge), messageToShowToUser, challenge, tanMethod, tanResponse.tanMediaIdentifier) - - else -> TanChallenge(messageToShowToUser, challenge, tanMethod, tanResponse.tanMediaIdentifier) - } - } - - protected open fun mayRetrieveAutomaticallyIfUserEnteredDecoupledTan(tanChallenge: TanChallenge, tanResponse: TanResponse, - userDidCancelEnteringTan: ObjectReference, dialogContext: DialogContext) { - dialogContext.bank.selectedTanMethod.decoupledParameters?.let { decoupledTanMethodParameters -> - if (tanResponse.tanProcess == TanProcess.AppTan && decoupledTanMethodParameters.periodicStateRequestsAllowed) { - automaticallyRetrieveIfUserEnteredDecoupledTan(tanChallenge, userDidCancelEnteringTan, dialogContext) - } - } - } - - protected open fun automaticallyRetrieveIfUserEnteredDecoupledTan(tanChallenge: TanChallenge, userDidCancelEnteringTan: ObjectReference, dialogContext: DialogContext) { - log.info("automaticallyRetrieveIfUserEnteredDecoupledTan() called for $tanChallenge") - } - - protected open fun handleEnterTanResult(enteredTanResult: EnterTanResult, tanResponse: TanResponse, response: BankResponse, - dialogContext: DialogContext, callback: (BankResponse) -> Unit) { - - if (enteredTanResult.changeTanMethodTo != null) { - handleUserAsksToChangeTanMethodAndResendLastMessage(enteredTanResult.changeTanMethodTo, dialogContext, callback) - } - else if (enteredTanResult.changeTanMediumTo is TanGeneratorTanMedium) { - handleUserAsksToChangeTanMediumAndResendLastMessage(enteredTanResult.changeTanMediumTo, dialogContext, - enteredTanResult.changeTanMediumResultCallback, callback) - } - else if (enteredTanResult.enteredTan == null) { - // i tried to send a HKTAN with cancelJob = true but then i saw there are no tan methods that support cancellation (at least not at my bank) - // but it's not required anyway, tan times out after some time. Simply don't respond anything and close dialog - response.tanRequiredButUserDidNotEnterOne = true - - callback(response) - } - else { - sendTanToBank(enteredTanResult.enteredTan, tanResponse, dialogContext, callback) - } - } - - protected open fun sendTanToBank(enteredTan: String, tanResponse: TanResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { - - val message = messageBuilder.createSendEnteredTanMessage(enteredTan, tanResponse, dialogContext) - - getAndHandleResponseForMessage(message, dialogContext, callback) - } - - protected open fun handleUserAsksToChangeTanMethodAndResendLastMessage(changeTanMethodTo: TanMethod, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { - - dialogContext.bank.selectedTanMethod = changeTanMethodTo - - - val lastCreatedMessage = dialogContext.currentMessage - - lastCreatedMessage?.let { closeDialog(dialogContext) } - - resendMessageInNewDialog(lastCreatedMessage, dialogContext, callback) - } - - protected open fun handleUserAsksToChangeTanMediumAndResendLastMessage(changeTanMediumTo: TanGeneratorTanMedium, - dialogContext: DialogContext, - changeTanMediumResultCallback: ((FinTsClientResponse) -> Unit)?, - callback: (BankResponse) -> Unit) { - - val lastCreatedMessage = dialogContext.currentMessage - - lastCreatedMessage?.let { closeDialog(dialogContext) } - - - changeTanMediumInternal(changeTanMediumTo, dialogContext.bank) { changeTanMediumResponse -> - changeTanMediumResultCallback?.invoke(FinTsClientResponse(changeTanMediumResponse)) - - if (changeTanMediumResponse.successful == false || lastCreatedMessage == null) { - callback(changeTanMediumResponse) - } - else { - resendMessageInNewDialog(lastCreatedMessage, dialogContext, callback) - } - } - } - - - protected open fun resendMessageInNewDialog(lastCreatedMessage: MessageBuilderResult?, previousDialogContext: DialogContext, callback: (BankResponse) -> Unit) { - - if (lastCreatedMessage != null) { // do not use previousDialogContext.currentMessage as this may is previous dialog's dialog close message - val newDialogContext = DialogContext(previousDialogContext.bank, previousDialogContext.product, chunkedResponseHandler = previousDialogContext.chunkedResponseHandler) - - initDialogWithStrongCustomerAuthentication(newDialogContext) { initDialogResponse -> - if (initDialogResponse.successful == false) { - callback(initDialogResponse) - } - else { - val newMessage = messageBuilder.rebuildMessage(lastCreatedMessage, newDialogContext) - - getAndHandleResponseForMessage(newMessage, newDialogContext) { response -> - closeDialog(newDialogContext) - - callback(response) - } - } - } - } - else { - val errorMessage = "There's no last action (like retrieve account transactions, transfer money, ...) to re-send with new TAN method. Probably an internal programming error." // TODO: translate - callback(BankResponse(false, errorMessage = errorMessage)) // should never come to this - } - } - - - protected open fun updateBankData(bank: BankData, response: BankResponse) { - response.getFirstSegmentById(InstituteSegmentId.BankParameters)?.let { bankParameters -> - bank.bpdVersion = bankParameters.bpdVersion - bank.bankCode = bankParameters.bankCode - bank.countryCode = bankParameters.bankCountryCode - bank.countMaxJobsPerMessage = bankParameters.countMaxJobsPerMessage - bank.supportedHbciVersions = bankParameters.supportedHbciVersions - bank.supportedLanguages = bankParameters.supportedLanguages - -// bank.bic = bankParameters. // TODO: where's the BIC? - } - - response.getFirstSegmentById(InstituteSegmentId.PinInfo)?.let { pinInfo -> - bank.pinInfo = pinInfo - } - - response.getFirstSegmentById(InstituteSegmentId.TanInfo)?.let { tanInfo -> - bank.tanMethodSupportedByBank = mapToTanMethods(tanInfo) - } - - response.getFirstSegmentById(InstituteSegmentId.CommunicationInfo)?.let { communicationInfo -> - communicationInfo.parameters.firstOrNull { it.type == Kommunikationsdienst.Https }?.address?.let { address -> - bank.finTs3ServerAddress = if (address.startsWith("https://", true)) address else "https://$address" - } - } - - response.getFirstSegmentById(InstituteSegmentId.SepaAccountInfo)?.let { sepaAccountInfo -> - sepaAccountInfo.account.bic?.let { - bank.bic = it // TODO: really set BIC on bank then? - } - } - - response.getFirstSegmentById(InstituteSegmentId.SepaAccountInfo)?.let { sepaAccountInfo -> - sepaAccountInfo.account.bic?.let { - bank.bic = it // TODO: really set BIC on bank then? - } - } - - response.getFirstSegmentById(InstituteSegmentId.ChangeTanMediaParameters)?.let { parameters -> - bank.changeTanMediumParameters = parameters - } - - if (response.supportedJobs.isNotEmpty()) { - bank.supportedJobs = response.supportedJobs - } - } - - protected open fun updateCustomerData(bank: BankData, response: BankResponse) { - response.getFirstSegmentById(InstituteSegmentId.BankParameters)?.let { bankParameters -> - // TODO: ask user if there is more than one supported language? But it seems that almost all banks only support German. - if (bank.selectedLanguage == Dialogsprache.Default && bankParameters.supportedLanguages.isNotEmpty()) { - bank.selectedLanguage = bankParameters.supportedLanguages.first() - } - } - - response.getFirstSegmentById(InstituteSegmentId.Synchronization)?.let { synchronization -> - synchronization.customerSystemId?.let { - bank.customerSystemId = it - - bank.customerSystemStatus = KundensystemStatusWerte.Benoetigt // TODO: didn't find out for sure yet, but i think i read somewhere, that this has to be set when customerSystemId is set - } - } - - response.getSegmentsById(InstituteSegmentId.AccountInfo).forEach { accountInfo -> - var accountHolderName = accountInfo.accountHolderName1 - accountInfo.accountHolderName2?.let { - accountHolderName += it // TODO: add a whitespace in between? - } - bank.customerName = accountHolderName - - findExistingAccount(bank, accountInfo)?.let { account -> - // TODO: update AccountData. But can this ever happen that an account changes? - } - ?: run { - val newAccount = AccountData(accountInfo.accountIdentifier, accountInfo.subAccountAttribute, - accountInfo.bankCountryCode, accountInfo.bankCode, accountInfo.iban, accountInfo.customerId, - mapAccountType(accountInfo), accountInfo.currency, accountHolderName, accountInfo.productName, - accountInfo.accountLimit, accountInfo.allowedJobNames) - - bank.supportedJobs.filterIsInstance().sortedByDescending { it.segmentVersion }.firstOrNull { newAccount.allowedJobNames.contains(it.jobName) }?.let { transactionsParameters -> - newAccount.countDaysForWhichTransactionsAreKept = transactionsParameters.countDaysForWhichTransactionsAreKept - } - - bank.addAccount(newAccount) - } - - // TODO: may also make use of other info - } - - response.getFirstSegmentById(InstituteSegmentId.SepaAccountInfo)?.let { sepaAccountInfo -> - // TODO: make use of information - sepaAccountInfo.account.iban?.let { - - } - } - - response.getFirstSegmentById(InstituteSegmentId.UserParameters)?.let { userParameters -> - bank.updVersion = userParameters.updVersion - - if (bank.customerName.isEmpty()) { - userParameters.username?.let { - bank.customerName = it - } - } - - // TODO: may also make use of other info - } - - response.getFirstSegmentById(InstituteSegmentId.CommunicationInfo)?.let { communicationInfo -> - if (bank.selectedLanguage != communicationInfo.defaultLanguage) { - bank.selectedLanguage = communicationInfo.defaultLanguage - } - } - - val supportedJobs = response.supportedJobs - if (supportedJobs.isNotEmpty()) { // if allowedJobsForBank is empty than bank didn't send any allowed job - for (account in bank.accounts) { - setAllowedJobsForAccount(bank, account, supportedJobs) - } - } - else if (bank.supportedJobs.isNotEmpty()) { - for (account in bank.accounts) { - if (account.allowedJobs.isEmpty()) { - setAllowedJobsForAccount(bank, account, bank.supportedJobs) - } - } - } - - if (response.supportedTanMethodsForUser.isNotEmpty()) { - bank.tanMethodsAvailableForUser = response.supportedTanMethodsForUser.mapNotNull { findTanMethod(it, bank) } - - if (bank.tanMethodsAvailableForUser.firstOrNull { it.securityFunction == bank.selectedTanMethod.securityFunction } == null) { // supportedTanMethods don't contain selectedTanMethod anymore - bank.resetSelectedTanMethod() - } - } - } - - protected open fun findTanMethod(securityFunction: Sicherheitsfunktion, bank: BankData): TanMethod? { - return bank.tanMethodSupportedByBank.firstOrNull { it.securityFunction == securityFunction } - } - - protected open fun setAllowedJobsForAccount(bank: BankData, account: AccountData, supportedJobs: List) { - val allowedJobsForAccount = mutableListOf() - - for (job in supportedJobs) { - if (isJobSupported(account, job)) { - allowedJobsForAccount.add(job) - } - } - - account.allowedJobs = allowedJobsForAccount - - account.setSupportsFeature(AccountFeature.RetrieveAccountTransactions, messageBuilder.supportsGetTransactions(account)) - account.setSupportsFeature(AccountFeature.RetrieveBalance, messageBuilder.supportsGetBalance(account)) - account.setSupportsFeature(AccountFeature.TransferMoney, messageBuilder.supportsBankTransfer(bank, account)) - account.setSupportsFeature(AccountFeature.RealTimeTransfer, messageBuilder.supportsSepaRealTimeTransfer(bank, account)) - } - - protected open fun mapToTanMethods(tanInfo: TanInfo): List { - return tanInfo.tanProcedureParameters.methodParameters.mapNotNull { - mapToTanMethod(it) - } - } - - protected open fun mapToTanMethod(parameters: TanMethodParameters): TanMethod? { - val methodName = parameters.methodName - - // we filter out iTAN and Einschritt-Verfahren as they are not permitted anymore according to PSD2 - if (methodName.toLowerCase() == "itan") { - return null - } - - return TanMethod(methodName, parameters.securityFunction, - mapToTanMethodType(parameters) ?: TanMethodType.EnterTan, mapHhdVersion(parameters), - parameters.maxTanInputLength, parameters.allowedTanFormat, - parameters.nameOfTanMediumRequired == BezeichnungDesTanMediumsErforderlich.BezeichnungDesTanMediumsMussAngegebenWerden, - mapDecoupledTanMethodParameters(parameters)) - } - - protected open fun mapToTanMethodType(parameters: TanMethodParameters): TanMethodType? { - val name = parameters.methodName.toLowerCase() - - return when { - // names are like 'chipTAN (comfort) manuell', 'Smart(-)TAN plus (manuell)' and - // technical identification is 'HHD'. Exception: there's one that states itself as 'chipTAN (Manuell)' - // but its DkTanMethod is set to 'HHDOPT1' -> handle ChipTanManuell before ChipTanFlickercode - parameters.dkTanMethod == DkTanMethod.HHD || name.contains("manuell") -> - TanMethodType.ChipTanManuell - - // names are like 'chipTAN optisch/comfort', 'SmartTAN (plus) optic/USB', 'chipTAN (Flicker)' and - // technical identification is 'HHDOPT1' - parameters.dkTanMethod == DkTanMethod.HHDOPT1 || - tanMethodNameContains(name, "optisch", "optic", "comfort", "flicker") -> - TanMethodType.ChipTanFlickercode - - // 'Smart-TAN plus optisch / USB' seems to be a Flickertan method -> test for 'optisch' first - name.contains("usb") -> TanMethodType.ChipTanUsb - - // QRTAN+ from 1822 direct has nothing to do with chipTAN QR. - name.contains("qr") -> { - if (tanMethodNameContains(name, "chipTAN", "Smart")) TanMethodType.ChipTanQrCode - else TanMethodType.QrCode - } - - // photoTAN from Commerzbank (comdirect), Deutsche Bank, norisbank has nothing to do with chipTAN photo - name.contains("photo") -> { - // e.g. 'Smart-TAN photo' / description 'Challenge' - if (tanMethodNameContains(name, "chipTAN", "Smart")) TanMethodType.ChipTanPhotoTanMatrixCode - // e.g. 'photoTAN-Verfahren', description 'Freigabe durch photoTAN' - else TanMethodType.photoTan - } - - tanMethodNameContains(name, "SMS", "mobile", "mTAN") -> TanMethodType.SmsTan - - // 'flateXSecure' identifies itself as 'PPTAN' instead of 'AppTAN' - // 'activeTAN-Verfahren' can actually be used either with an app or a reader; it's like chipTAN QR but without a chip card - parameters.dkTanMethod == DkTanMethod.App - || tanMethodNameContains(name, "push", "app", "BestSign", "SecureGo", "TAN2go", "activeTAN", "easyTAN", "SecurePlus", "TAN+") - || technicalTanMethodIdentificationContains(parameters, "SECURESIGN", "PPTAN") -> - TanMethodType.AppTan - - // we filter out iTAN and Einschritt-Verfahren as they are not permitted anymore according to PSD2 - else -> null - } - } - - protected open fun mapHhdVersion(parameters: TanMethodParameters): HHDVersion? { - return when { - technicalTanMethodIdentificationContains(parameters, "HHD1.4") -> HHDVersion.HHD_1_4 - technicalTanMethodIdentificationContains(parameters, "HHD1.3") -> HHDVersion.HHD_1_3 - parameters.versionDkTanMethod?.contains("1.4") == true -> HHDVersion.HHD_1_4 - parameters.versionDkTanMethod?.contains("1.3") == true -> HHDVersion.HHD_1_4 - else -> null - } - } - - protected open fun tanMethodNameContains(name: String, vararg namesToTest: String): Boolean { - namesToTest.forEach { nameToTest -> - if (name.contains(nameToTest.toLowerCase())) { - return true - } - } - - return false - } - - protected open fun technicalTanMethodIdentificationContains(parameters: TanMethodParameters, vararg valuesToTest: String): Boolean { - valuesToTest.forEach { valueToTest -> - if (parameters.technicalTanMethodIdentification.contains(valueToTest, true)) { - return true - } - } - - return false - } - - protected open fun mapDecoupledTanMethodParameters(parameters: TanMethodParameters): DecoupledTanMethodParameters? { - parameters.manualConfirmationAllowedForDecoupled?.let { manualConfirmationAllowed -> - return DecoupledTanMethodParameters( - manualConfirmationAllowed, - parameters.periodicStateRequestsAllowedForDecoupled ?: false, // this and the following values are all set when manualConfirmationAllowedForDecoupled is set - parameters.maxNumberOfStateRequestsForDecoupled ?: 0, - parameters.initialDelayInSecondsForStateRequestsForDecoupled ?: Int.MAX_VALUE, - parameters.delayInSecondsForNextStateRequestsForDecoupled ?: Int.MAX_VALUE - ) - } - - return null - } - - - protected open fun isJobSupported(bank: BankData, segmentId: ISegmentId): Boolean { - return bank.supportedJobs.map { it.jobName }.contains(segmentId.id) - } - - protected open fun isJobSupported(account: AccountData, supportedJob: JobParameters): Boolean { - for (allowedJobName in account.allowedJobNames) { - if (allowedJobName == supportedJob.jobName) { - return true - } - } - - return false - } - - protected open fun findExistingAccount(bank: BankData, accountInfo: AccountInfo): AccountData? { - bank.accounts.forEach { account -> - if (account.accountIdentifier == accountInfo.accountIdentifier - && account.productName == accountInfo.productName) { - - return account - } - } - - return null - } - - protected open fun mapAccountType(accountInfo: AccountInfo): AccountType? { - if (accountInfo.accountType == null || accountInfo.accountType == AccountType.Sonstige) { - accountInfo.productName?.let { name -> - // comdirect doesn't set account type field but names its bank accounts according to them like 'Girokonto', 'Tagesgeldkonto', ... - return when { - name.contains("Girokonto", true) -> AccountType.Girokonto - name.contains("Festgeld", true) -> AccountType.Festgeldkonto - name.contains("Tagesgeld", true) -> AccountType.Sparkonto // learnt something new today: according to Wikipedia some direct banks offer a modern version of saving accounts as 'Tagesgeldkonto' - name.contains("Kreditkarte", true) -> AccountType.Kreditkartenkonto - else -> accountInfo.accountType - } - } - } - - return accountInfo.accountType + jobExecutor.doBankTransferAsync(bankTransferData, bank, account, callback) } } \ No newline at end of file diff --git a/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsClientForCustomer.kt b/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsClientForCustomer.kt index df0fea19..2e36dbe7 100644 --- a/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsClientForCustomer.kt +++ b/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsClientForCustomer.kt @@ -28,7 +28,7 @@ open class FinTsClientForCustomer( product: ProductData = ProductData("15E53C26816138699C7B6A3E8", "1.0.0") // TODO: get version dynamically){} ) { - protected val client = FinTsClient(callback, webClient, base64Service, messageBuilder, responseParser, mt940Parser, messageLogCollector, product) + protected val client = FinTsClient(FinTsJobExecutor(callback, webClient, base64Service, messageBuilder, responseParser, mt940Parser, messageLogCollector, product)) open val messageLogWithoutSensitiveData: List diff --git a/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsJobExecutor.kt b/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsJobExecutor.kt new file mode 100644 index 00000000..9661be6f --- /dev/null +++ b/fints4k/src/commonMain/kotlin/net/dankito/banking/fints/FinTsJobExecutor.kt @@ -0,0 +1,1157 @@ +package net.dankito.banking.fints + +import net.dankito.banking.fints.callback.FinTsClientCallback +import net.dankito.banking.fints.log.IMessageLogAppender +import net.dankito.banking.fints.log.MessageLogCollector +import net.dankito.banking.fints.messages.MessageBuilder +import net.dankito.banking.fints.messages.MessageBuilderResult +import net.dankito.banking.fints.messages.datenelemente.implementierte.Dialogsprache +import net.dankito.banking.fints.messages.datenelemente.implementierte.KundensystemStatusWerte +import net.dankito.banking.fints.messages.datenelemente.implementierte.signatur.Sicherheitsfunktion +import net.dankito.banking.fints.messages.datenelemente.implementierte.signatur.VersionDesSicherheitsverfahrens +import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.* +import net.dankito.banking.fints.messages.segmente.id.CustomerSegmentId +import net.dankito.banking.fints.messages.segmente.id.ISegmentId +import net.dankito.banking.fints.model.* +import net.dankito.banking.fints.response.BankResponse +import net.dankito.banking.fints.response.InstituteSegmentId +import net.dankito.banking.fints.response.ResponseParser +import net.dankito.banking.fints.response.client.FinTsClientResponse +import net.dankito.banking.fints.response.client.GetTanMediaListResponse +import net.dankito.banking.fints.response.client.GetTransactionsResponse +import net.dankito.banking.fints.response.client.GetUserTanMethodsResponse +import net.dankito.banking.fints.response.segments.* +import net.dankito.banking.fints.tan.FlickerCodeDecoder +import net.dankito.banking.fints.tan.TanImageDecoder +import net.dankito.banking.fints.transactions.IAccountTransactionsParser +import net.dankito.banking.fints.transactions.Mt940AccountTransactionsParser +import net.dankito.banking.fints.util.IBase64Service +import net.dankito.banking.fints.util.PureKotlinBase64Service +import net.dankito.utils.multiplatform.log.Logger +import net.dankito.utils.multiplatform.log.LoggerFactory +import net.dankito.banking.fints.webclient.IWebClient +import net.dankito.banking.fints.webclient.KtorWebClient +import net.dankito.banking.fints.webclient.WebClientResponse +import net.dankito.utils.multiplatform.Date +import net.dankito.utils.multiplatform.getInnerExceptionMessage +import net.dankito.utils.multiplatform.ObjectReference + + +/** + * Low level class that executes concrete business transactions (= FinTS Geschäftsvorfälle). + * + * In almost all cases you want to use [FinTsClient] which wraps these business transactions to a higher level API. + */ +open class FinTsJobExecutor( + protected open val callback: FinTsClientCallback, + protected open val webClient: IWebClient = KtorWebClient(), + protected open val base64Service: IBase64Service = PureKotlinBase64Service(), + protected open val messageBuilder: MessageBuilder = MessageBuilder(), + protected open val responseParser: ResponseParser = ResponseParser(), + protected open val mt940Parser: IAccountTransactionsParser = Mt940AccountTransactionsParser(), + protected open val messageLogCollector: MessageLogCollector = MessageLogCollector(), + protected open val product: ProductData = ProductData("15E53C26816138699C7B6A3E8", "1.0.0") // TODO: get version dynamically +) { + + companion object { + private val log = LoggerFactory.getLogger(FinTsJobExecutor::class) + } + + + open val messageLogWithoutSensitiveData: List + get() = messageLogCollector.messageLogWithoutSensitiveData + + protected open val messageLogAppender: IMessageLogAppender = object : IMessageLogAppender { + + override fun logError(message: String, e: Exception?, logger: Logger?, bank: BankData?) { + messageLogCollector.logError(message, e, logger, bank) + } + + } + + + init { + responseParser.logAppender = messageLogAppender + mt940Parser.logAppender = messageLogAppender + } + + + open fun getAnonymousBankInfo(bank: BankData, callback: (BankResponse) -> Unit) { + val dialogContext = DialogContext(bank, product) + + val message = messageBuilder.createAnonymousDialogInitMessage(dialogContext) + + getAndHandleResponseForMessage(message, dialogContext) { response -> + if (response.successful) { + updateBankData(bank, response) + + closeAnonymousDialog(dialogContext, response) + } + + callback(response) + } + } + + protected open fun closeAnonymousDialog(dialogContext: DialogContext, response: BankResponse) { + + // bank already closed dialog -> there's no need to send dialog end message + if (dialogContext.closeDialog == false || dialogContext.didBankCloseDialog) { + return + } + + val dialogEndRequestBody = messageBuilder.createAnonymousDialogEndMessage(dialogContext) + + fireAndForgetMessage(dialogEndRequestBody, dialogContext) + } + + + /** + * Retrieves basic data like user's TAN methods, ... in an not authenticated Init dialog. + * + * This is the first step to do when adding a new account as for almost all other jobs user's selected TAN method has to be specified. + * + * Be aware this method resets BPD, UPD and selected TAN method! + */ + open fun retrieveBasicDataLikeUsersTanMethods(bank: BankData, closeDialog: Boolean = false, callback: (BankResponse) -> Unit) { + // just to ensure settings are in its initial state and that bank sends us bank parameter (BPD), + // user parameter (UPD) and allowed tan methods for user (therefore the resetSelectedTanMethod()) + bank.resetBpdVersion() + bank.resetUpdVersion() + /** + * Sind dem Kundenprodukt die konkreten, für den Benutzer zugelassenen Sicherheitsverfahren nicht bekannt, so können + * diese über eine Dialoginitialisierung mit Sicherheitsfunktion=999 angefordert werden. Die konkreten Verfahren + * werden dann über den Rückmeldungscode=3920 zurückgemeldet. Im Rahmen dieses Prozesses darf keine UPD + * zurückgeliefert werden und die Durchführung anderer Geschäftsvorfälle ist in einem solchen Dialog nicht erlaubt. + */ + bank.resetSelectedTanMethod() + + // this is the only case where Einschritt-TAN-Verfahren is accepted: to get user's TAN methods + val dialogContext = DialogContext(bank, product, closeDialog, versionOfSecurityMethod = VersionDesSicherheitsverfahrens.Version_1) + + val message = messageBuilder.createInitDialogMessage(dialogContext) + + getAndHandleResponseForMessage(message, dialogContext) { response -> + closeDialog(dialogContext) + + handleGetUsersTanMethodsResponse(response, dialogContext, callback) + } + } + + protected open fun handleGetUsersTanMethodsResponse(response: BankResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { + val getUsersTanMethodsResponse = GetUserTanMethodsResponse(response) + + if (getUsersTanMethodsResponse.successful) { // TODO: really update data only on complete successfully response? as it may contain useful information anyway // TODO: extract method for this code part + updateBankData(dialogContext.bank, getUsersTanMethodsResponse) + updateCustomerData(dialogContext.bank, getUsersTanMethodsResponse) + } + + // even though it is required by specification some banks don't support retrieving user's TAN method by setting TAN method to '999' + if (bankDoesNotSupportRetrievingUsersTanMethods(getUsersTanMethodsResponse)) { + getBankDataForNewUserViaAnonymousDialog(dialogContext.bank, callback) // TODO: should not be necessary anymore + } + else { + callback(getUsersTanMethodsResponse) + } + } + + protected open fun bankDoesNotSupportRetrievingUsersTanMethods(response: BankResponse): Boolean { + return response.successful == false && + response.segmentFeedbacks.flatMap { it.feedbacks }.firstOrNull { it.responseCode == 9200 && + it.message == "Gewähltes Zwei-Schritt-Verfahren nicht unterstützt." } != null + } + + // TODO: this is only a quick fix. Find a better and general solution + protected open fun getBankDataForNewUserViaAnonymousDialog(bank: BankData, callback: (BankResponse) -> Unit) { + getAnonymousBankInfo(bank) { anonymousBankInfoResponse -> + if (anonymousBankInfoResponse.successful == false) { + callback(anonymousBankInfoResponse) + } + else if (bank.tanMethodSupportedByBank.isEmpty()) { // should only be a theoretical error + callback( + BankResponse(true, + errorMessage = "Die TAN Verfahren der Bank konnten nicht ermittelt werden") + ) // TODO: translate + } + else { + bank.tanMethodsAvailableForUser = bank.tanMethodSupportedByBank + getUsersTanMethod(bank) { didSelectTanMethod -> + if (didSelectTanMethod) { + val dialogContext = DialogContext(bank, product) + + initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext) { initDialogResponse -> + closeDialog(dialogContext) + + callback(initDialogResponse) + } + } + else { + callback(BankResponse(false)) + } + } + } + } + } + + + open fun getAccounts(bank: BankData, callback: (BankResponse) -> Unit) { + + val dialogContext = DialogContext(bank, product, false) + + initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext) { response -> + closeDialog(dialogContext) + + if (response.successful) { + updateBankData(bank, response) + updateCustomerData(bank, response) + } + + callback(response) + } + } + + + open fun getTransactionsAsync(parameter: GetTransactionsParameter, bank: BankData, callback: (GetTransactionsResponse) -> Unit) { + + val dialogContext = DialogContext(bank, product) + + initDialogWithStrongCustomerAuthentication(dialogContext) { initDialogResponse -> + + if (initDialogResponse.successful == false) { + callback(GetTransactionsResponse(initDialogResponse, RetrievedAccountData.unsuccessfulList(parameter.account))) + } + else { + mayGetBalance(parameter, dialogContext) { balanceResponse -> + if (dialogContext.didBankCloseDialog) { + callback(GetTransactionsResponse(balanceResponse, RetrievedAccountData.unsuccessfulList(parameter.account))) + } + else { + getTransactionsAfterInitAndGetBalance(parameter, dialogContext, balanceResponse, callback) + } + } + } + } + } + + protected open fun getTransactionsAfterInitAndGetBalance(parameter: GetTransactionsParameter, dialogContext: DialogContext, + balanceResponse: BankResponse, callback: (GetTransactionsResponse) -> Unit) { + var balance: Money? = balanceResponse.getFirstSegmentById(InstituteSegmentId.Balance)?.let { + Money(it.balance, it.currency) + } + val bookedTransactions = mutableSetOf() + val unbookedTransactions = mutableSetOf() + + val message = messageBuilder.createGetTransactionsMessage(parameter, dialogContext) + + var remainingMt940String = "" + + dialogContext.abortIfTanIsRequired = parameter.abortIfTanIsRequired + + dialogContext.chunkedResponseHandler = { response -> + response.getFirstSegmentById(InstituteSegmentId.AccountTransactionsMt940)?.let { transactionsSegment -> + val (chunkTransaction, remainder) = mt940Parser.parseTransactionsChunk(remainingMt940String + transactionsSegment.bookedTransactionsString, + dialogContext.bank, parameter.account) + + bookedTransactions.addAll(chunkTransaction) + remainingMt940String = remainder + + parameter.retrievedChunkListener?.invoke(bookedTransactions) + } + + response.getFirstSegmentById(InstituteSegmentId.CreditCardTransactions)?.let { creditCardTransactionsSegment -> + balance = Money(creditCardTransactionsSegment.balance.amount, creditCardTransactionsSegment.balance.currency ?: "EUR") + bookedTransactions.addAll(creditCardTransactionsSegment.transactions.map { AccountTransaction(parameter.account, it.amount, it.description, it.bookingDate, it.transactionDescriptionBase ?: "", null, null, "", it.valueDate) }) + } + } + + getAndHandleResponseForMessage(message, dialogContext) { response -> + closeDialog(dialogContext) + + val successful = response.successful && (parameter.alsoRetrieveBalance == false || balance != null) + val fromDate = parameter.fromDate + ?: parameter.account.countDaysForWhichTransactionsAreKept?.let { Date(Date.today.millisSinceEpoch - it * FinTsClient.OneDayMillis) } + ?: bookedTransactions.map { it.valueDate }.sortedBy { it.millisSinceEpoch }.firstOrNull() + val retrievedData = RetrievedAccountData(parameter.account, successful, balance, bookedTransactions, unbookedTransactions, fromDate, parameter.toDate ?: Date.today, response.errorMessage) + + callback( + GetTransactionsResponse(response, listOf(retrievedData), + if (parameter.maxCountEntries != null) parameter.isSettingMaxCountEntriesAllowedByBank else null + ) + ) + } + } + + protected open fun mayGetBalance(parameter: GetTransactionsParameter, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { + if (parameter.alsoRetrieveBalance && parameter.account.supportsRetrievingBalance) { + val message = messageBuilder.createGetBalanceMessage(parameter.account, dialogContext) + + getAndHandleResponseForMessage(message, dialogContext) { response -> + callback(response) + } + } + else { + callback(BankResponse(false, errorMessage = "Either not requested to get balance or account does not support retrieving balance. " + + "Should retrieve balance = ${parameter.alsoRetrieveBalance}, account supports retrieving balance = ${parameter.account.supportsRetrievingBalance}.")) + } + } + + + /** + * According to specification synchronizing customer system id is required: + * "Die Kundensystem-ID ist beim HBCI RAH- / RDH- sowie dem PIN/TAN-Verfahren erforderlich." + * + * But as tests show this can be omitted. + * + * But when you do it, this has to be done in an extra dialog as dialog has to be initialized + * with retrieved customer system id. + * + * If you change customer system id during a dialog your messages get rejected by bank institute. + */ + protected open fun synchronizeCustomerSystemId(bank: BankData, callback: (FinTsClientResponse) -> Unit) { + + val dialogContext = DialogContext(bank, product) + val message = messageBuilder.createSynchronizeCustomerSystemIdMessage(dialogContext) + + getAndHandleResponseForMessage(message, dialogContext) { response -> + if (response.successful) { + updateBankData(bank, response) + updateCustomerData(bank, response) + + closeDialog(dialogContext) + } + + callback(FinTsClientResponse(response)) + } + } + + + open fun getTanMediaList(bank: BankData, tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle, + tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien, callback: (GetTanMediaListResponse) -> Unit) { + + sendMessageAndHandleResponse(bank, CustomerSegmentId.TanMediaList, false, { dialogContext -> + messageBuilder.createGetTanMediaListMessage(dialogContext, tanMediaKind, tanMediumClass) + }) { response -> + handleGetTanMediaListResponse(response, bank, callback) + } + } + + protected open fun handleGetTanMediaListResponse(response: BankResponse, bank: BankData, callback: (GetTanMediaListResponse) -> Unit) { + // TAN media list (= TAN generator list) is only returned for users with chipTAN TAN methods + val tanMediaList = if (response.successful == false) null + else response.getFirstSegmentById(InstituteSegmentId.TanMediaList) + + tanMediaList?.let { + bank.tanMedia = it.tanMedia + } + + callback(GetTanMediaListResponse(response, tanMediaList)) + } + + + open fun changeTanMedium(newActiveTanMedium: TanGeneratorTanMedium, bank: BankData, callback: (BankResponse) -> Unit) { + + if (bank.changeTanMediumParameters?.enteringAtcAndTanRequired == true) { + this.callback.enterTanGeneratorAtc(bank, newActiveTanMedium) { enteredAtc -> + if (enteredAtc.hasAtcBeenEntered == false) { + val message = "Bank requires to enter ATC and TAN in order to change TAN medium." // TODO: translate + callback(BankResponse(false, errorMessage = message)) + } + else { + sendChangeTanMediumMessage(bank, newActiveTanMedium, enteredAtc, callback) + } + } + } + else { + sendChangeTanMediumMessage(bank, newActiveTanMedium, null, callback) + } + } + + protected open fun sendChangeTanMediumMessage(bank: BankData, newActiveTanMedium: TanGeneratorTanMedium, enteredAtc: EnterTanGeneratorAtcResult?, + callback: (BankResponse) -> Unit) { + + sendMessageAndHandleResponse(bank, null, true, { dialogContext -> + messageBuilder.createChangeTanMediumMessage(newActiveTanMedium, dialogContext, enteredAtc?.tan, enteredAtc?.atc) + }) { response -> + callback(response) + } + } + + + open fun doBankTransferAsync(bankTransferData: BankTransferData, bank: BankData, account: AccountData, callback: (FinTsClientResponse) -> Unit) { + + sendMessageAndHandleResponse(bank, null, true, { dialogContext -> + messageBuilder.createBankTransferMessage(bankTransferData, account, dialogContext) + }) { response -> + callback(FinTsClientResponse(response)) + } + } + + + protected open fun getAndHandleResponseForMessage(message: MessageBuilderResult, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { + if (message.createdMessage == null) { + callback(BankResponse(false, messageCreationError = message)) + } + else { + getAndHandleResponseForMessage(message.createdMessage, dialogContext) { response -> + handleMayRequiresTan(response, dialogContext) { handledResponse -> + // if there's a Aufsetzpunkt (continuationId) set, then response is not complete yet, there's more information to fetch by sending this Aufsetzpunkt + handledResponse.aufsetzpunkt?.let { continuationId -> + if (handledResponse.followUpResponse == null) { // for re-sent messages followUpResponse is already set and dialog already closed -> would be overwritten with an error response that dialog is closed + if (message.isSendEnteredTanMessage() == false) { // for sending TAN no follow up message can be created -> filter out, otherwise chunkedResponseHandler would get called twice for same response + dialogContext.chunkedResponseHandler?.invoke(handledResponse) + } + + getFollowUpMessageForContinuationId(handledResponse, continuationId, message, dialogContext) { followUpResponse -> + handledResponse.followUpResponse = followUpResponse + handledResponse.hasFollowUpMessageButCouldNotReceiveIt = handledResponse.followUpResponse == null + + callback(handledResponse) + } + } + else { + callback(handledResponse) + } + } + ?: run { + // e.g. response = enter TAN response, but handledResponse is then response after entering TAN, e.g. account transactions + // -> chunkedResponseHandler would get called for same handledResponse multiple times + if (response == handledResponse) { + dialogContext.chunkedResponseHandler?.invoke(handledResponse) + } + + callback(handledResponse) + } + } + } + } + } + + protected open fun getAndHandleResponseForMessage(requestBody: String, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { + addMessageLog(requestBody, MessageLogEntryType.Sent, dialogContext) + + getResponseForMessage(requestBody, dialogContext.bank.finTs3ServerAddress) { webResponse -> + val response = handleResponse(webResponse, dialogContext) + + dialogContext.response = response + + response.messageHeader?.let { header -> dialogContext.dialogId = header.dialogId } + dialogContext.didBankCloseDialog = response.didBankCloseDialog + + callback(response) + } + } + + protected open fun getResponseForMessage(requestBody: String, finTs3ServerAddress: String, callback: (WebClientResponse) -> Unit) { + val encodedRequestBody = base64Service.encode(requestBody) + + webClient.post(finTs3ServerAddress, encodedRequestBody, "application/octet-stream", IWebClient.DefaultUserAgent, callback) + } + + protected open fun fireAndForgetMessage(message: MessageBuilderResult, dialogContext: DialogContext) { + message.createdMessage?.let { requestBody -> + addMessageLog(requestBody, MessageLogEntryType.Sent, dialogContext) + + getResponseForMessage(requestBody, dialogContext.bank.finTs3ServerAddress) { } + + // if really needed add received response to message log here + } + } + + protected open fun handleResponse(webResponse: WebClientResponse, dialogContext: DialogContext): BankResponse { + val responseBody = webResponse.body + + if (webResponse.successful && responseBody != null) { + + try { + val decodedResponse = decodeBase64Response(responseBody) + + addMessageLog(decodedResponse, MessageLogEntryType.Received, dialogContext) + + return responseParser.parse(decodedResponse) + } catch (e: Exception) { + logError("Could not decode responseBody:\r\n'$responseBody'", dialogContext, e) + + return BankResponse(false, errorMessage = e.getInnerExceptionMessage()) + } + } + else { + val bank = dialogContext.bank + logError("Request to $bank (${bank.finTs3ServerAddress}) failed", dialogContext, webResponse.error) + } + + return BankResponse(false, errorMessage = webResponse.error?.getInnerExceptionMessage()) + } + + protected open fun decodeBase64Response(responseBody: String): String { + return base64Service.decode(responseBody.replace("\r", "").replace("\n", "")) + } + + + protected open fun getFollowUpMessageForContinuationId(response: BankResponse, continuationId: String, message: MessageBuilderResult, + dialogContext: DialogContext, callback: (BankResponse?) -> Unit) { + + messageBuilder.rebuildMessageWithContinuationId(message, continuationId, dialogContext)?.let { followUpMessage -> + getAndHandleResponseForMessage(followUpMessage, dialogContext, callback) + } + ?: run { callback(null) } + } + + + protected open fun handleMayRequiresTan(response: BankResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { // TODO: use response from DialogContext + + if (response.isStrongAuthenticationRequired) { + if (dialogContext.abortIfTanIsRequired) { + response.tanRequiredButWeWereToldToAbortIfSo = true + + callback(response) + return + } + else if (response.tanResponse != null) { + response.tanResponse?.let { tanResponse -> + handleEnteringTanRequired(tanResponse, response, dialogContext, callback) + } + + return + } + } + + // TODO: check if response contains '3931 TAN-Generator gesperrt, Synchronisierung erforderlich' or + // '3933 TAN-Generator gesperrt, Synchronisierung erforderlich Kartennummer ##########' message, + // call callback.enterAtc() and implement and call HKTSY job (p. 77) + + // TODO: also check '9931 Sperrung des Kontos nach %1 Fehlversuchen' -> if %1 == 3 synchronize TAN generator + // as it's quite unrealistic that user entered TAN wrong three times, in most cases TAN generator is not synchronized + + callback(response) + } + + protected open fun handleEnteringTanRequired(tanResponse: TanResponse, response: BankResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { + val bank = dialogContext.bank // TODO: copy required data to TanChallenge + val tanChallenge = createTanChallenge(tanResponse, bank) + + val userDidCancelEnteringTan = ObjectReference(false) + + this.callback.enterTan(bank, tanChallenge) { enteredTanResult -> + userDidCancelEnteringTan.value = true + + handleEnterTanResult(enteredTanResult, tanResponse, response, dialogContext, callback) + } + + mayRetrieveAutomaticallyIfUserEnteredDecoupledTan(tanChallenge, tanResponse, userDidCancelEnteringTan, dialogContext) + } + + protected open fun createTanChallenge(tanResponse: TanResponse, bank: BankData): TanChallenge { + // TODO: is this true for all tan methods? + val messageToShowToUser = tanResponse.challenge ?: "" + val challenge = tanResponse.challengeHHD_UC ?: "" + val tanMethod = bank.selectedTanMethod + + return when (tanMethod.type) { + TanMethodType.ChipTanFlickercode -> + FlickerCodeTanChallenge( + FlickerCodeDecoder().decodeChallenge(challenge, tanMethod.hhdVersion ?: HHDVersion.HHD_1_4), // HHD 1.4 is currently the most used version + messageToShowToUser, challenge, tanMethod, tanResponse.tanMediaIdentifier) + + TanMethodType.ChipTanQrCode, TanMethodType.ChipTanPhotoTanMatrixCode, + TanMethodType.QrCode, TanMethodType.photoTan -> + ImageTanChallenge(TanImageDecoder().decodeChallenge(challenge), messageToShowToUser, challenge, tanMethod, tanResponse.tanMediaIdentifier) + + else -> TanChallenge(messageToShowToUser, challenge, tanMethod, tanResponse.tanMediaIdentifier) + } + } + + protected open fun mayRetrieveAutomaticallyIfUserEnteredDecoupledTan(tanChallenge: TanChallenge, tanResponse: TanResponse, + userDidCancelEnteringTan: ObjectReference, dialogContext: DialogContext) { + dialogContext.bank.selectedTanMethod.decoupledParameters?.let { decoupledTanMethodParameters -> + if (tanResponse.tanProcess == TanProcess.AppTan && decoupledTanMethodParameters.periodicStateRequestsAllowed) { + automaticallyRetrieveIfUserEnteredDecoupledTan(tanChallenge, userDidCancelEnteringTan, dialogContext) + } + } + } + + protected open fun automaticallyRetrieveIfUserEnteredDecoupledTan(tanChallenge: TanChallenge, userDidCancelEnteringTan: ObjectReference, dialogContext: DialogContext) { + log.info("automaticallyRetrieveIfUserEnteredDecoupledTan() called for $tanChallenge") + } + + protected open fun handleEnterTanResult(enteredTanResult: EnterTanResult, tanResponse: TanResponse, response: BankResponse, + dialogContext: DialogContext, callback: (BankResponse) -> Unit) { + + if (enteredTanResult.changeTanMethodTo != null) { + handleUserAsksToChangeTanMethodAndResendLastMessage(enteredTanResult.changeTanMethodTo, dialogContext, callback) + } + else if (enteredTanResult.changeTanMediumTo is TanGeneratorTanMedium) { + handleUserAsksToChangeTanMediumAndResendLastMessage(enteredTanResult.changeTanMediumTo, dialogContext, + enteredTanResult.changeTanMediumResultCallback, callback) + } + else if (enteredTanResult.enteredTan == null) { + // i tried to send a HKTAN with cancelJob = true but then i saw there are no tan methods that support cancellation (at least not at my bank) + // but it's not required anyway, tan times out after some time. Simply don't respond anything and close dialog + response.tanRequiredButUserDidNotEnterOne = true + + callback(response) + } + else { + sendTanToBank(enteredTanResult.enteredTan, tanResponse, dialogContext, callback) + } + } + + protected open fun sendTanToBank(enteredTan: String, tanResponse: TanResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { + + val message = messageBuilder.createSendEnteredTanMessage(enteredTan, tanResponse, dialogContext) + + getAndHandleResponseForMessage(message, dialogContext, callback) + } + + protected open fun handleUserAsksToChangeTanMethodAndResendLastMessage(changeTanMethodTo: TanMethod, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { + + dialogContext.bank.selectedTanMethod = changeTanMethodTo + + + val lastCreatedMessage = dialogContext.currentMessage + + lastCreatedMessage?.let { closeDialog(dialogContext) } + + resendMessageInNewDialog(lastCreatedMessage, dialogContext, callback) + } + + protected open fun handleUserAsksToChangeTanMediumAndResendLastMessage(changeTanMediumTo: TanGeneratorTanMedium, + dialogContext: DialogContext, + changeTanMediumResultCallback: ((FinTsClientResponse) -> Unit)?, + callback: (BankResponse) -> Unit) { + + val lastCreatedMessage = dialogContext.currentMessage + + lastCreatedMessage?.let { closeDialog(dialogContext) } + + + changeTanMedium(changeTanMediumTo, dialogContext.bank) { changeTanMediumResponse -> + changeTanMediumResultCallback?.invoke(FinTsClientResponse(changeTanMediumResponse)) + + if (changeTanMediumResponse.successful == false || lastCreatedMessage == null) { + callback(changeTanMediumResponse) + } + else { + resendMessageInNewDialog(lastCreatedMessage, dialogContext, callback) + } + } + } + + + protected open fun resendMessageInNewDialog(lastCreatedMessage: MessageBuilderResult?, previousDialogContext: DialogContext, callback: (BankResponse) -> Unit) { + + if (lastCreatedMessage != null) { // do not use previousDialogContext.currentMessage as this may is previous dialog's dialog close message + val newDialogContext = DialogContext(previousDialogContext.bank, previousDialogContext.product, chunkedResponseHandler = previousDialogContext.chunkedResponseHandler) + + initDialogWithStrongCustomerAuthentication(newDialogContext) { initDialogResponse -> + if (initDialogResponse.successful == false) { + callback(initDialogResponse) + } + else { + val newMessage = messageBuilder.rebuildMessage(lastCreatedMessage, newDialogContext) + + getAndHandleResponseForMessage(newMessage, newDialogContext) { response -> + closeDialog(newDialogContext) + + callback(response) + } + } + } + } + else { + val errorMessage = "There's no last action (like retrieve account transactions, transfer money, ...) to re-send with new TAN method. Probably an internal programming error." // TODO: translate + callback(BankResponse(false, errorMessage = errorMessage)) // should never come to this + } + } + + + protected open fun sendMessageAndHandleResponse(bank: BankData, segmentForNonStrongCustomerAuthenticationTwoStepTanProcess: CustomerSegmentId? = null, + closeDialog: Boolean = true, createMessage: (DialogContext) -> MessageBuilderResult, callback: (BankResponse) -> Unit) { + + val dialogContext = DialogContext(bank, product, closeDialog) + + if (segmentForNonStrongCustomerAuthenticationTwoStepTanProcess == null) { + initDialogWithStrongCustomerAuthentication(dialogContext) { initDialogResponse -> + sendMessageAndHandleResponseAfterDialogInitialization(dialogContext, initDialogResponse, createMessage, callback) + } + } + else { + initDialogMessageWithoutStrongCustomerAuthenticationAfterSuccessfulChecks(dialogContext, segmentForNonStrongCustomerAuthenticationTwoStepTanProcess) { initDialogResponse -> + sendMessageAndHandleResponseAfterDialogInitialization(dialogContext, initDialogResponse, createMessage, callback) + } + } + } + + private fun sendMessageAndHandleResponseAfterDialogInitialization(dialogContext: DialogContext, initDialogResponse: BankResponse, createMessage: (DialogContext) -> MessageBuilderResult, callback: (BankResponse) -> Unit) { + + if (initDialogResponse.successful == false) { + callback(initDialogResponse) + } + else { + val message = createMessage(dialogContext) + + getAndHandleResponseForMessage(message, dialogContext) { response -> + closeDialog(dialogContext) + + callback(response) + } + } + } + + protected open fun initDialogWithStrongCustomerAuthentication(dialogContext: DialogContext, callback: (BankResponse) -> Unit) { + + // we first need to retrieve supported tan methods and jobs before we can do anything + ensureBasicBankDataRetrieved(dialogContext.bank) { retrieveBasicBankDataResponse -> + if (retrieveBasicBankDataResponse.successful == false) { + callback(retrieveBasicBankDataResponse) + } + else { + // as in the next step we have to supply user's tan method, ensure user selected his or her + ensureTanMethodIsSelected(dialogContext.bank) { tanMethodSelectedResponse -> + if (tanMethodSelectedResponse.successful == false) { + callback(tanMethodSelectedResponse) + } + else { + initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext, callback) + } + } + } + } + } + + protected open fun initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext: DialogContext, callback: (BankResponse) -> Unit) { + + val message = messageBuilder.createInitDialogMessage(dialogContext) + + getAndHandleResponseForMessage(message, dialogContext) { response -> + + if (response.successful) { + updateBankData(dialogContext.bank, response) + updateCustomerData(dialogContext.bank, response) + } + + callback(response) + } + } + + protected open fun initDialogMessageWithoutStrongCustomerAuthenticationAfterSuccessfulChecks(dialogContext: DialogContext, segmentIdForTwoStepTanProcess: CustomerSegmentId?, + callback: (BankResponse) -> Unit) { + + val message = messageBuilder.createInitDialogMessageWithoutStrongCustomerAuthentication(dialogContext, segmentIdForTwoStepTanProcess) + + getAndHandleResponseForMessage(message, dialogContext) { response -> + if (response.successful) { + updateBankData(dialogContext.bank, response) + updateCustomerData(dialogContext.bank, response) + } + + callback(response) + } + } + + protected open fun closeDialog(dialogContext: DialogContext) { + + // bank already closed dialog -> there's no need to send dialog end message + if (dialogContext.closeDialog == false || dialogContext.didBankCloseDialog) { + return + } + + val dialogEndRequestBody = messageBuilder.createDialogEndMessage(dialogContext) + + fireAndForgetMessage(dialogEndRequestBody, dialogContext) + } + + + protected open fun ensureBasicBankDataRetrieved(bank: BankData, callback: (BankResponse) -> Unit) { + if (bank.tanMethodSupportedByBank.isEmpty() || bank.supportedJobs.isEmpty()) { + retrieveBasicDataLikeUsersTanMethods(bank) { getBankInfoResponse -> + if (getBankInfoResponse.successful == false || bank.tanMethodSupportedByBank.isEmpty() + || bank.supportedJobs.isEmpty()) { + + callback(BankResponse(false, errorMessage = + "Could not retrieve basic bank data like supported tan methods or supported jobs")) // TODO: translate // TODO: add as messageToShowToUser + } + else { + callback(BankResponse(true)) + } + } + } + else { + callback(BankResponse(true)) + } + } + + protected open fun ensureTanMethodIsSelected(bank: BankData, callback: (BankResponse) -> Unit) { + if (bank.isTanMethodSelected == false) { + if (bank.tanMethodsAvailableForUser.isEmpty()) { + retrieveBasicDataLikeUsersTanMethods(bank) { + if (bank.tanMethodsAvailableForUser.isEmpty()) { // could not retrieve supported tan methods for user + callback(BankResponse(false, noTanMethodSelected = true)) + } + else { + getUsersTanMethod(bank) { + callback(BankResponse(bank.isTanMethodSelected, noTanMethodSelected = !!!bank.isTanMethodSelected)) + } + } + } + } + else { + getUsersTanMethod(bank) { + callback(BankResponse(bank.isTanMethodSelected, noTanMethodSelected = !!!bank.isTanMethodSelected)) + } + } + } + else { + callback(BankResponse(bank.isTanMethodSelected, noTanMethodSelected = !!!bank.isTanMethodSelected)) + } + } + + open fun getUsersTanMethod(bank: BankData, done: (Boolean) -> Unit) { + if (bank.tanMethodsAvailableForUser.size == 1) { // user has only one TAN method -> set it and we're done + bank.selectedTanMethod = bank.tanMethodsAvailableForUser.first() + done(true) + } + else { + // we know user's supported tan methods, now ask user which one to select + callback.askUserForTanMethod(bank.tanMethodsAvailableForUser, selectSuggestedTanMethod(bank)) { selectedTanMethod -> + if (selectedTanMethod != null) { + bank.selectedTanMethod = selectedTanMethod + done(true) + } + else { + done(false) + } + } + } + } + + protected open fun selectSuggestedTanMethod(bank: BankData): TanMethod? { + return bank.tanMethodsAvailableForUser.firstOrNull { it.type != TanMethodType.ChipTanUsb && it.type != TanMethodType.SmsTan && it.type != TanMethodType.ChipTanManuell } + ?: bank.tanMethodsAvailableForUser.firstOrNull { it.type != TanMethodType.ChipTanUsb && it.type != TanMethodType.SmsTan } + ?: bank.tanMethodsAvailableForUser.firstOrNull { it.type != TanMethodType.ChipTanUsb } + ?: bank.tanMethodsAvailableForUser.firstOrNull() + } + + + protected open fun updateBankData(bank: BankData, response: BankResponse) { + response.getFirstSegmentById(InstituteSegmentId.BankParameters)?.let { bankParameters -> + bank.bpdVersion = bankParameters.bpdVersion + bank.bankCode = bankParameters.bankCode + bank.countryCode = bankParameters.bankCountryCode + bank.countMaxJobsPerMessage = bankParameters.countMaxJobsPerMessage + bank.supportedHbciVersions = bankParameters.supportedHbciVersions + bank.supportedLanguages = bankParameters.supportedLanguages + +// bank.bic = bankParameters. // TODO: where's the BIC? + } + + response.getFirstSegmentById(InstituteSegmentId.PinInfo)?.let { pinInfo -> + bank.pinInfo = pinInfo + } + + response.getFirstSegmentById(InstituteSegmentId.TanInfo)?.let { tanInfo -> + bank.tanMethodSupportedByBank = mapToTanMethods(tanInfo) + } + + response.getFirstSegmentById(InstituteSegmentId.CommunicationInfo)?.let { communicationInfo -> + communicationInfo.parameters.firstOrNull { it.type == Kommunikationsdienst.Https }?.address?.let { address -> + bank.finTs3ServerAddress = if (address.startsWith("https://", true)) address else "https://$address" + } + } + + response.getFirstSegmentById(InstituteSegmentId.SepaAccountInfo)?.let { sepaAccountInfo -> + sepaAccountInfo.account.bic?.let { + bank.bic = it // TODO: really set BIC on bank then? + } + } + + response.getFirstSegmentById(InstituteSegmentId.SepaAccountInfo)?.let { sepaAccountInfo -> + sepaAccountInfo.account.bic?.let { + bank.bic = it // TODO: really set BIC on bank then? + } + } + + response.getFirstSegmentById(InstituteSegmentId.ChangeTanMediaParameters)?.let { parameters -> + bank.changeTanMediumParameters = parameters + } + + if (response.supportedJobs.isNotEmpty()) { + bank.supportedJobs = response.supportedJobs + } + } + + protected open fun updateCustomerData(bank: BankData, response: BankResponse) { + response.getFirstSegmentById(InstituteSegmentId.BankParameters)?.let { bankParameters -> + // TODO: ask user if there is more than one supported language? But it seems that almost all banks only support German. + if (bank.selectedLanguage == Dialogsprache.Default && bankParameters.supportedLanguages.isNotEmpty()) { + bank.selectedLanguage = bankParameters.supportedLanguages.first() + } + } + + response.getFirstSegmentById(InstituteSegmentId.Synchronization)?.let { synchronization -> + synchronization.customerSystemId?.let { + bank.customerSystemId = it + + bank.customerSystemStatus = KundensystemStatusWerte.Benoetigt // TODO: didn't find out for sure yet, but i think i read somewhere, that this has to be set when customerSystemId is set + } + } + + response.getSegmentsById(InstituteSegmentId.AccountInfo).forEach { accountInfo -> + var accountHolderName = accountInfo.accountHolderName1 + accountInfo.accountHolderName2?.let { + accountHolderName += it // TODO: add a whitespace in between? + } + bank.customerName = accountHolderName + + findExistingAccount(bank, accountInfo)?.let { account -> + // TODO: update AccountData. But can this ever happen that an account changes? + } + ?: run { + val newAccount = AccountData(accountInfo.accountIdentifier, accountInfo.subAccountAttribute, + accountInfo.bankCountryCode, accountInfo.bankCode, accountInfo.iban, accountInfo.customerId, + mapAccountType(accountInfo), accountInfo.currency, accountHolderName, accountInfo.productName, + accountInfo.accountLimit, accountInfo.allowedJobNames) + + bank.supportedJobs.filterIsInstance().sortedByDescending { it.segmentVersion }.firstOrNull { newAccount.allowedJobNames.contains(it.jobName) }?.let { transactionsParameters -> + newAccount.countDaysForWhichTransactionsAreKept = transactionsParameters.countDaysForWhichTransactionsAreKept + } + + bank.addAccount(newAccount) + } + + // TODO: may also make use of other info + } + + response.getFirstSegmentById(InstituteSegmentId.SepaAccountInfo)?.let { sepaAccountInfo -> + // TODO: make use of information + sepaAccountInfo.account.iban?.let { + + } + } + + response.getFirstSegmentById(InstituteSegmentId.UserParameters)?.let { userParameters -> + bank.updVersion = userParameters.updVersion + + if (bank.customerName.isEmpty()) { + userParameters.username?.let { + bank.customerName = it + } + } + + // TODO: may also make use of other info + } + + response.getFirstSegmentById(InstituteSegmentId.CommunicationInfo)?.let { communicationInfo -> + if (bank.selectedLanguage != communicationInfo.defaultLanguage) { + bank.selectedLanguage = communicationInfo.defaultLanguage + } + } + + val supportedJobs = response.supportedJobs + if (supportedJobs.isNotEmpty()) { // if allowedJobsForBank is empty than bank didn't send any allowed job + for (account in bank.accounts) { + setAllowedJobsForAccount(bank, account, supportedJobs) + } + } + else if (bank.supportedJobs.isNotEmpty()) { + for (account in bank.accounts) { + if (account.allowedJobs.isEmpty()) { + setAllowedJobsForAccount(bank, account, bank.supportedJobs) + } + } + } + + if (response.supportedTanMethodsForUser.isNotEmpty()) { + bank.tanMethodsAvailableForUser = response.supportedTanMethodsForUser.mapNotNull { findTanMethod(it, bank) } + + if (bank.tanMethodsAvailableForUser.firstOrNull { it.securityFunction == bank.selectedTanMethod.securityFunction } == null) { // supportedTanMethods don't contain selectedTanMethod anymore + bank.resetSelectedTanMethod() + } + } + } + + protected open fun findTanMethod(securityFunction: Sicherheitsfunktion, bank: BankData): TanMethod? { + return bank.tanMethodSupportedByBank.firstOrNull { it.securityFunction == securityFunction } + } + + protected open fun setAllowedJobsForAccount(bank: BankData, account: AccountData, supportedJobs: List) { + val allowedJobsForAccount = mutableListOf() + + for (job in supportedJobs) { + if (isJobSupported(account, job)) { + allowedJobsForAccount.add(job) + } + } + + account.allowedJobs = allowedJobsForAccount + + account.setSupportsFeature(AccountFeature.RetrieveAccountTransactions, messageBuilder.supportsGetTransactions(account)) + account.setSupportsFeature(AccountFeature.RetrieveBalance, messageBuilder.supportsGetBalance(account)) + account.setSupportsFeature(AccountFeature.TransferMoney, messageBuilder.supportsBankTransfer(bank, account)) + account.setSupportsFeature(AccountFeature.RealTimeTransfer, messageBuilder.supportsSepaRealTimeTransfer(bank, account)) + } + + protected open fun mapToTanMethods(tanInfo: TanInfo): List { + return tanInfo.tanProcedureParameters.methodParameters.mapNotNull { + mapToTanMethod(it) + } + } + + protected open fun mapToTanMethod(parameters: TanMethodParameters): TanMethod? { + val methodName = parameters.methodName + + // we filter out iTAN and Einschritt-Verfahren as they are not permitted anymore according to PSD2 + if (methodName.toLowerCase() == "itan") { + return null + } + + return TanMethod(methodName, parameters.securityFunction, + mapToTanMethodType(parameters) ?: TanMethodType.EnterTan, mapHhdVersion(parameters), + parameters.maxTanInputLength, parameters.allowedTanFormat, + parameters.nameOfTanMediumRequired == BezeichnungDesTanMediumsErforderlich.BezeichnungDesTanMediumsMussAngegebenWerden, + mapDecoupledTanMethodParameters(parameters)) + } + + protected open fun mapToTanMethodType(parameters: TanMethodParameters): TanMethodType? { + val name = parameters.methodName.toLowerCase() + + return when { + // names are like 'chipTAN (comfort) manuell', 'Smart(-)TAN plus (manuell)' and + // technical identification is 'HHD'. Exception: there's one that states itself as 'chipTAN (Manuell)' + // but its DkTanMethod is set to 'HHDOPT1' -> handle ChipTanManuell before ChipTanFlickercode + parameters.dkTanMethod == DkTanMethod.HHD || name.contains("manuell") -> + TanMethodType.ChipTanManuell + + // names are like 'chipTAN optisch/comfort', 'SmartTAN (plus) optic/USB', 'chipTAN (Flicker)' and + // technical identification is 'HHDOPT1' + parameters.dkTanMethod == DkTanMethod.HHDOPT1 || + tanMethodNameContains(name, "optisch", "optic", "comfort", "flicker") -> + TanMethodType.ChipTanFlickercode + + // 'Smart-TAN plus optisch / USB' seems to be a Flickertan method -> test for 'optisch' first + name.contains("usb") -> TanMethodType.ChipTanUsb + + // QRTAN+ from 1822 direct has nothing to do with chipTAN QR. + name.contains("qr") -> { + if (tanMethodNameContains(name, "chipTAN", "Smart")) TanMethodType.ChipTanQrCode + else TanMethodType.QrCode + } + + // photoTAN from Commerzbank (comdirect), Deutsche Bank, norisbank has nothing to do with chipTAN photo + name.contains("photo") -> { + // e.g. 'Smart-TAN photo' / description 'Challenge' + if (tanMethodNameContains(name, "chipTAN", "Smart")) TanMethodType.ChipTanPhotoTanMatrixCode + // e.g. 'photoTAN-Verfahren', description 'Freigabe durch photoTAN' + else TanMethodType.photoTan + } + + tanMethodNameContains(name, "SMS", "mobile", "mTAN") -> TanMethodType.SmsTan + + // 'flateXSecure' identifies itself as 'PPTAN' instead of 'AppTAN' + // 'activeTAN-Verfahren' can actually be used either with an app or a reader; it's like chipTAN QR but without a chip card + parameters.dkTanMethod == DkTanMethod.App + || tanMethodNameContains(name, "push", "app", "BestSign", "SecureGo", "TAN2go", "activeTAN", "easyTAN", "SecurePlus", "TAN+") + || technicalTanMethodIdentificationContains(parameters, "SECURESIGN", "PPTAN") -> + TanMethodType.AppTan + + // we filter out iTAN and Einschritt-Verfahren as they are not permitted anymore according to PSD2 + else -> null + } + } + + protected open fun mapHhdVersion(parameters: TanMethodParameters): HHDVersion? { + return when { + technicalTanMethodIdentificationContains(parameters, "HHD1.4") -> HHDVersion.HHD_1_4 + technicalTanMethodIdentificationContains(parameters, "HHD1.3") -> HHDVersion.HHD_1_3 + parameters.versionDkTanMethod?.contains("1.4") == true -> HHDVersion.HHD_1_4 + parameters.versionDkTanMethod?.contains("1.3") == true -> HHDVersion.HHD_1_4 + else -> null + } + } + + protected open fun tanMethodNameContains(name: String, vararg namesToTest: String): Boolean { + namesToTest.forEach { nameToTest -> + if (name.contains(nameToTest.toLowerCase())) { + return true + } + } + + return false + } + + protected open fun technicalTanMethodIdentificationContains(parameters: TanMethodParameters, vararg valuesToTest: String): Boolean { + valuesToTest.forEach { valueToTest -> + if (parameters.technicalTanMethodIdentification.contains(valueToTest, true)) { + return true + } + } + + return false + } + + protected open fun mapDecoupledTanMethodParameters(parameters: TanMethodParameters): DecoupledTanMethodParameters? { + parameters.manualConfirmationAllowedForDecoupled?.let { manualConfirmationAllowed -> + return DecoupledTanMethodParameters( + manualConfirmationAllowed, + parameters.periodicStateRequestsAllowedForDecoupled ?: false, // this and the following values are all set when manualConfirmationAllowedForDecoupled is set + parameters.maxNumberOfStateRequestsForDecoupled ?: 0, + parameters.initialDelayInSecondsForStateRequestsForDecoupled ?: Int.MAX_VALUE, + parameters.delayInSecondsForNextStateRequestsForDecoupled ?: Int.MAX_VALUE + ) + } + + return null + } + + + open fun isJobSupported(bank: BankData, segmentId: ISegmentId): Boolean { + return bank.supportedJobs.map { it.jobName }.contains(segmentId.id) + } + + open fun isJobSupported(account: AccountData, supportedJob: JobParameters): Boolean { + for (allowedJobName in account.allowedJobNames) { + if (allowedJobName == supportedJob.jobName) { + return true + } + } + + return false + } + + protected open fun findExistingAccount(bank: BankData, accountInfo: AccountInfo): AccountData? { + bank.accounts.forEach { account -> + if (account.accountIdentifier == accountInfo.accountIdentifier + && account.productName == accountInfo.productName) { + + return account + } + } + + return null + } + + protected open fun mapAccountType(accountInfo: AccountInfo): AccountType? { + if (accountInfo.accountType == null || accountInfo.accountType == AccountType.Sonstige) { + accountInfo.productName?.let { name -> + // comdirect doesn't set account type field but names its bank accounts according to them like 'Girokonto', 'Tagesgeldkonto', ... + return when { + name.contains("Girokonto", true) -> AccountType.Girokonto + name.contains("Festgeld", true) -> AccountType.Festgeldkonto + name.contains("Tagesgeld", true) -> AccountType.Sparkonto // learnt something new today: according to Wikipedia some direct banks offer a modern version of saving accounts as 'Tagesgeldkonto' + name.contains("Kreditkarte", true) -> AccountType.Kreditkartenkonto + else -> accountInfo.accountType + } + } + } + + return accountInfo.accountType + } + + + protected open fun addMessageLog(message: String, type: MessageLogEntryType, dialogContext: DialogContext) { + messageLogCollector.addMessageLog(message, type, dialogContext.bank) + } + + protected open fun logError(message: String, dialogContext: DialogContext, e: Exception?) { + messageLogAppender.logError(message, e, log, dialogContext.bank) + } + +} \ No newline at end of file diff --git a/fints4k/src/jvm6Test/kotlin/net/dankito/banking/fints/FinTsClientTestBase.kt b/fints4k/src/jvm6Test/kotlin/net/dankito/banking/fints/FinTsClientTestBase.kt index 1f4a6f52..f1660c6d 100644 --- a/fints4k/src/jvm6Test/kotlin/net/dankito/banking/fints/FinTsClientTestBase.kt +++ b/fints4k/src/jvm6Test/kotlin/net/dankito/banking/fints/FinTsClientTestBase.kt @@ -75,7 +75,7 @@ open class FinTsClientTestBase { } - private val underTest = FinTsClient(callback, KtorWebClient(), PureKotlinBase64Service()) + private val underTest = FinTsClient(callback) private val BankDataAnonymous = BankData.anonymous("10070000", "https://fints.deutsche-bank.de/", "DEUTDEBBXXX") diff --git a/fints4k/src/jvm6Test/kotlin/net/dankito/banking/fints/bankdetails/BanksFinTsDetailsRetriever.kt b/fints4k/src/jvm6Test/kotlin/net/dankito/banking/fints/bankdetails/BanksFinTsDetailsRetriever.kt index ee8ced08..f7377c21 100644 --- a/fints4k/src/jvm6Test/kotlin/net/dankito/banking/fints/bankdetails/BanksFinTsDetailsRetriever.kt +++ b/fints4k/src/jvm6Test/kotlin/net/dankito/banking/fints/bankdetails/BanksFinTsDetailsRetriever.kt @@ -13,6 +13,7 @@ import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.Bezei import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.SmsAbbuchungskontoErforderlich import net.dankito.banking.fints.model.* import net.dankito.banking.bankfinder.BankInfo +import net.dankito.banking.fints.FinTsJobExecutor import net.dankito.banking.fints.response.BankResponse import net.dankito.banking.fints.response.segments.SepaAccountInfoParameters import net.dankito.banking.fints.response.segments.TanInfo @@ -49,7 +50,7 @@ class BanksFinTsDetailsRetriever { private val messageBuilder = MessageBuilder() - private val finTsClient = object : FinTsClient(NoOpFinTsClientCallback(), KtorWebClient(), PureKotlinBase64Service()) { + private val jobExecutor = object : FinTsJobExecutor(NoOpFinTsClientCallback()) { fun getAndHandleResponseForMessagePublic(message: MessageBuilderResult, dialogContext: DialogContext, callback: (BankResponse) -> Unit) { getAndHandleResponseForMessage(message, dialogContext, callback) @@ -132,14 +133,14 @@ class BanksFinTsDetailsRetriever { val anonymousBankInfoResponse = AtomicReference() val countDownLatch = CountDownLatch(1) - finTsClient.getAndHandleResponseForMessagePublic(requestBody, dialogContext) { + jobExecutor.getAndHandleResponseForMessagePublic(requestBody, dialogContext) { anonymousBankInfoResponse.set(it) countDownLatch.countDown() } countDownLatch.await(30, TimeUnit.SECONDS) - finTsClient.updateBankDataPublic(bank, anonymousBankInfoResponse.get()) + jobExecutor.updateBankDataPublic(bank, anonymousBankInfoResponse.get()) return anonymousBankInfoResponse.get() } @@ -211,7 +212,7 @@ class BanksFinTsDetailsRetriever { tanMethodParameter[methodParameter.methodName]?.add(methodParameter) } - val tanMethodType = finTsClient.mapToTanMethodTypePublic(methodParameter) + val tanMethodType = jobExecutor.mapToTanMethodTypePublic(methodParameter) if (tanMethodTypes.containsKey(tanMethodType) == false) { tanMethodTypes.put(tanMethodType, mutableSetOf(methodParameter)) }