Made IWebClient and FinTsClient async so that on iOS everything but the web request can run on the main thread -> we don't need to freeze objects to pass them between threads

This commit is contained in:
dankito 2020-07-15 21:40:06 +02:00
parent 27dd8c98cc
commit 8e221f17ce
5 changed files with 375 additions and 327 deletions

View File

@ -74,7 +74,7 @@ open class FinTsClient(
open fun getAnonymousBankInfoAsync(bank: BankData, callback: (FinTsClientResponse) -> Unit) { open fun getAnonymousBankInfoAsync(bank: BankData, callback: (FinTsClientResponse) -> Unit) {
GlobalScope.launch { GlobalScope.launch {
callback(getAnonymousBankInfo(bank)) getAnonymousBankInfo(bank, callback)
} }
} }
@ -84,20 +84,20 @@ open class FinTsClient(
* *
* On success [bank] parameter is updated afterwards. * On success [bank] parameter is updated afterwards.
*/ */
open fun getAnonymousBankInfo(bank: BankData): FinTsClientResponse { open fun getAnonymousBankInfo(bank: BankData, callback: (FinTsClientResponse) -> Unit) {
val dialogContext = DialogContext(bank, CustomerData.Anonymous, product) val dialogContext = DialogContext(bank, CustomerData.Anonymous, product)
val message = messageBuilder.createAnonymousDialogInitMessage(dialogContext) val message = messageBuilder.createAnonymousDialogInitMessage(dialogContext)
val response = getAndHandleResponseForMessage(message, dialogContext) getAndHandleResponseForMessage(message, dialogContext) { response ->
if (response.successful) {
updateBankData(bank, response)
if (response.successful) { closeAnonymousDialog(dialogContext, response)
updateBankData(bank, response) }
closeAnonymousDialog(dialogContext, response) callback(FinTsClientResponse(response))
} }
return FinTsClientResponse(response)
} }
protected open fun closeAnonymousDialog(dialogContext: DialogContext, response: Response) { protected open fun closeAnonymousDialog(dialogContext: DialogContext, response: Response) {
@ -113,7 +113,7 @@ open class FinTsClient(
} }
open fun getBankAndCustomerInfoForNewUser(bank: BankData, customer: CustomerData): AddAccountResponse { open fun getBankAndCustomerInfoForNewUser(bank: BankData, customer: CustomerData, callback: (AddAccountResponse) -> Unit) {
// just to ensure settings are in its initial state and that bank sends use bank parameter (BPD), // just to ensure settings are in its initial state and that bank sends use bank parameter (BPD),
// user parameter (UPD) and allowed tan procedures for user (therefore the resetSelectedTanProcedure()) // user parameter (UPD) and allowed tan procedures for user (therefore the resetSelectedTanProcedure())
bank.resetBpdVersion() bank.resetBpdVersion()
@ -128,16 +128,17 @@ open class FinTsClient(
val dialogContext = DialogContext(bank, customer, product) val dialogContext = DialogContext(bank, customer, product)
val initDialogResponse = initDialogAfterSuccessfulChecks(dialogContext) initDialogAfterSuccessfulChecks(dialogContext) { initDialogResponse ->
closeDialog(dialogContext)
closeDialog(dialogContext) // even though it is required by specification some banks don't support retrieving user's TAN procedure by setting TAN procedure to '999'
if (bankDoesNotSupportRetrievingUsersTanProcedures(initDialogResponse)) {
// even though it is required by specification some banks don't support retrieving user's TAN procedure by setting TAN procedure to '999' getBankAndCustomerInfoForNewUserViaAnonymousDialog(bank, customer, callback)
if (bankDoesNotSupportRetrievingUsersTanProcedures(initDialogResponse)) { }
return getBankAndCustomerInfoForNewUserViaAnonymousDialog(bank, customer) else {
callback(AddAccountResponse(initDialogResponse, bank, customer))
}
} }
return AddAccountResponse(initDialogResponse, bank, customer)
} }
protected open fun bankDoesNotSupportRetrievingUsersTanProcedures(response: Response): Boolean { protected open fun bankDoesNotSupportRetrievingUsersTanProcedures(response: Response): Boolean {
@ -147,27 +148,28 @@ open class FinTsClient(
} }
// TODO: this is only a quick fix. Find a better and general solution // TODO: this is only a quick fix. Find a better and general solution
protected open fun getBankAndCustomerInfoForNewUserViaAnonymousDialog(bank: BankData, customer: CustomerData): AddAccountResponse { protected open fun getBankAndCustomerInfoForNewUserViaAnonymousDialog(bank: BankData, customer: CustomerData, callback: (AddAccountResponse) -> Unit) {
val anonymousBankInfoResponse = getAnonymousBankInfo(bank) getAnonymousBankInfo(bank) { anonymousBankInfoResponse ->
if (anonymousBankInfoResponse.isSuccessful == false) {
callback(AddAccountResponse(anonymousBankInfoResponse.toResponse(), bank, customer))
}
else if (bank.supportedTanProcedures.isEmpty()) { // should only be a theoretical error
callback(AddAccountResponse(Response(true,
exception = Exception("Die TAN Verfahren der Bank konnten nicht ermittelt werden")), bank, customer)) // TODO: translate
}
else {
customer.supportedTanProcedures = bank.supportedTanProcedures
getUsersTanProcedure(customer)
if (anonymousBankInfoResponse.isSuccessful == false) { val dialogContext = DialogContext(bank, customer, product)
return AddAccountResponse(anonymousBankInfoResponse.toResponse(), bank, customer)
initDialogAfterSuccessfulChecks(dialogContext) { initDialogResponse ->
closeDialog(dialogContext)
callback(AddAccountResponse(initDialogResponse, bank, customer))
}
}
} }
else if (bank.supportedTanProcedures.isEmpty()) { // should only be a theoretical error
return AddAccountResponse(Response(true,
exception = Exception("Die TAN Verfahren der Bank konnten nicht ermittelt werden")), bank, customer) // TODO: translate
}
customer.supportedTanProcedures = bank.supportedTanProcedures
getUsersTanProcedure(customer)
val dialogContext = DialogContext(bank, customer, product)
val initDialogResponse = initDialogAfterSuccessfulChecks(dialogContext)
closeDialog(dialogContext)
return AddAccountResponse(initDialogResponse, bank, customer)
} }
@ -182,21 +184,21 @@ open class FinTsClient(
* *
* If you change customer system id during a dialog your messages get rejected by bank institute. * If you change customer system id during a dialog your messages get rejected by bank institute.
*/ */
protected open fun synchronizeCustomerSystemId(bank: BankData, customer: CustomerData): FinTsClientResponse { protected open fun synchronizeCustomerSystemId(bank: BankData, customer: CustomerData, callback: (FinTsClientResponse) -> Unit) {
val dialogContext = DialogContext(bank, customer, product) val dialogContext = DialogContext(bank, customer, product)
val message = messageBuilder.createSynchronizeCustomerSystemIdMessage(dialogContext) val message = messageBuilder.createSynchronizeCustomerSystemIdMessage(dialogContext)
val response = getAndHandleResponseForMessage(message, dialogContext) getAndHandleResponseForMessage(message, dialogContext) { response ->
if (response.successful) {
updateBankData(bank, response)
updateCustomerData(customer, bank, response)
if (response.successful) { closeDialog(dialogContext)
updateBankData(bank, response) }
updateCustomerData(customer, bank, response)
closeDialog(dialogContext) callback(FinTsClientResponse(response))
} }
return FinTsClientResponse(response)
} }
@ -204,59 +206,76 @@ open class FinTsClient(
callback: (AddAccountResponse) -> Unit) { callback: (AddAccountResponse) -> Unit) {
GlobalScope.launch { GlobalScope.launch {
callback(addAccount(bank, customer)) addAccount(bank, customer, callback)
} }
} }
open fun addAccount(bank: BankData, customer: CustomerData): AddAccountResponse { open fun addAccount(bank: BankData, customer: CustomerData, callback: (AddAccountResponse) -> Unit) {
val originalAreWeThatGentleToCloseDialogs = areWeThatGentleToCloseDialogs val originalAreWeThatGentleToCloseDialogs = areWeThatGentleToCloseDialogs
areWeThatGentleToCloseDialogs = false areWeThatGentleToCloseDialogs = false
/* First dialog: Get user's basic data like her TAN procedures */ /* First dialog: Get user's basic data like her TAN procedures */
val newUserInfoResponse = getBankAndCustomerInfoForNewUser(bank, customer) getBankAndCustomerInfoForNewUser(bank, customer) { newUserInfoResponse ->
if (newUserInfoResponse.isSuccessful == false) { // bank parameter (FinTS server address, ...) already seem to be wrong if (newUserInfoResponse.isSuccessful == false) { // bank parameter (FinTS server address, ...) already seem to be wrong
return newUserInfoResponse callback(newUserInfoResponse)
}
// do not ask user for tan at this stage
var didOverwriteUserUnselectedTanProcedure = false
if (customer.isTanProcedureSelected == false && customer.supportedTanProcedures.isNotEmpty()) {
if (customer.supportedTanProcedures.size == 1) { // user has only one TAN procedure -> set it and we're done
customer.selectedTanProcedure = customer.supportedTanProcedures.first()
} }
else {
didOverwriteUserUnselectedTanProcedure = true
customer.selectedTanProcedure = selectSuggestedTanProcedure(customer) ?: customer.supportedTanProcedures.first() // do not ask user for tan at this stage
} var didOverwriteUserUnselectedTanProcedure = false
} if (customer.isTanProcedureSelected == false && customer.supportedTanProcedures.isNotEmpty()) {
if (customer.supportedTanProcedures.size == 1) { // user has only one TAN procedure -> set it and we're done
/* Second dialog: Get customer system ID - done now in getBankAndCustomerInfoForNewUser(), we try to make it without having to open an extra dialog */ customer.selectedTanProcedure = customer.supportedTanProcedures.first()
}
else {
/* Third dialog: Get customer TAN media list - last step that can and must be done without strong customer authorization */ didOverwriteUserUnselectedTanProcedure = true
customer.selectedTanProcedure = selectSuggestedTanProcedure(customer) ?: customer.supportedTanProcedures.first()
getTanMediaList(bank, customer, TanMedienArtVersion.Alle, TanMediumKlasse.AlleMedien) }
}
/* Fourth dialog: Try to retrieve account transactions of last 90 days without TAN */
/* Second dialog: Get customer system ID - done now in getBankAndCustomerInfoForNewUser(), we try to make it without having to open an extra dialog */
// also check if retrieving account transactions of last 90 days without tan is supported (and thereby may retrieve first account transactions)
val transactionsOfLast90DaysResponses = mutableListOf<GetTransactionsResponse>()
val balances = mutableMapOf<AccountData, Money>() /* Third dialog: Get customer TAN media list - last step that can and must be done without strong customer authorization */
customer.accounts.forEach { account ->
if (account.supportsFeature(AccountFeature.RetrieveAccountTransactions)) { getTanMediaList(bank, customer, TanMedienArtVersion.Alle, TanMediumKlasse.AlleMedien) {
val response = tryGetTransactionsOfLast90DaysWithoutTan(bank, customer, account, false)
transactionsOfLast90DaysResponses.add(response)
response.balance?.let { balances.put(account, it) } /* Fourth dialog: Try to retrieve account transactions of last 90 days without TAN */
// also check if retrieving account transactions of last 90 days without tan is supported (and thereby may retrieve first account transactions)
val transactionsOfLast90DaysResponses = mutableListOf<GetTransactionsResponse>()
val balances = mutableMapOf<AccountData, Money>()
val countAccountSupportingRetrievingTransactions = customer.accounts.filter { it.supportsFeature(AccountFeature.RetrieveAccountTransactions) }.size
var countRetrievedAccounts = 0
customer.accounts.forEach { account ->
if (account.supportsFeature(AccountFeature.RetrieveAccountTransactions)) {
tryGetTransactionsOfLast90DaysWithoutTan(bank, customer, account, false) { response ->
transactionsOfLast90DaysResponses.add(response)
response.balance?.let { balances.put(account, it) }
countRetrievedAccounts++
if (countRetrievedAccounts == countAccountSupportingRetrievingTransactions) {
addAccountAfterRetrievingTransactions(bank, customer, newUserInfoResponse, didOverwriteUserUnselectedTanProcedure, originalAreWeThatGentleToCloseDialogs,
transactionsOfLast90DaysResponses, balances, callback)
}
}
}
}
} }
} }
}
protected open fun addAccountAfterRetrievingTransactions(bank: BankData, customer: CustomerData, newUserInfoResponse: AddAccountResponse,
didOverwriteUserUnselectedTanProcedure: Boolean, originalAreWeThatGentleToCloseDialogs: Boolean,
transactionsOfLast90DaysResponses: MutableList<GetTransactionsResponse>,
balances: MutableMap<AccountData, Money>, callback: (AddAccountResponse) -> Unit) {
if (didOverwriteUserUnselectedTanProcedure) { if (didOverwriteUserUnselectedTanProcedure) {
customer.resetSelectedTanProcedure() customer.resetSelectedTanProcedure()
} }
@ -267,8 +286,8 @@ open class FinTsClient(
val unbookedTransactions = transactionsOfLast90DaysResponses.flatMap { it.unbookedTransactions } val unbookedTransactions = transactionsOfLast90DaysResponses.flatMap { it.unbookedTransactions }
val bookedTransactions = transactionsOfLast90DaysResponses.flatMap { it.bookedTransactions } val bookedTransactions = transactionsOfLast90DaysResponses.flatMap { it.bookedTransactions }
return AddAccountResponse(newUserInfoResponse.toResponse(), bank, customer, callback(AddAccountResponse(newUserInfoResponse.toResponse(), bank, customer,
supportsRetrievingTransactionsOfLast90DaysWithoutTan, bookedTransactions, unbookedTransactions, balances) supportsRetrievingTransactionsOfLast90DaysWithoutTan, bookedTransactions, unbookedTransactions, balances))
} }
@ -278,66 +297,67 @@ open class FinTsClient(
* *
* Check if bank supports this. * Check if bank supports this.
*/ */
open fun tryGetTransactionsOfLast90DaysWithoutTan(bank: BankData, customer: CustomerData, account: AccountData): GetTransactionsResponse { open fun tryGetTransactionsOfLast90DaysWithoutTan(bank: BankData, customer: CustomerData, account: AccountData, callback: (GetTransactionsResponse) -> Unit) {
return tryGetTransactionsOfLast90DaysWithoutTan(bank, customer, account, false) tryGetTransactionsOfLast90DaysWithoutTan(bank, customer, account, false, callback)
} }
protected open fun tryGetTransactionsOfLast90DaysWithoutTan(bank: BankData, customer: CustomerData, account: AccountData, protected open fun tryGetTransactionsOfLast90DaysWithoutTan(bank: BankData, customer: CustomerData, account: AccountData,
hasRetrievedTransactionsWithTanJustBefore: Boolean): GetTransactionsResponse { hasRetrievedTransactionsWithTanJustBefore: Boolean, callback: (GetTransactionsResponse) -> Unit) {
val now = Date() val now = Date()
val ninetyDaysAgo = Date(now.millisSinceEpoch - NinetyDaysMillis) val ninetyDaysAgo = Date(now.millisSinceEpoch - NinetyDaysMillis)
val response = getTransactions(GetTransactionsParameter(account.supportsFeature(AccountFeature.RetrieveBalance), ninetyDaysAgo, abortIfTanIsRequired = true), bank, customer, account) getTransactions(GetTransactionsParameter(account.supportsFeature(AccountFeature.RetrieveBalance), ninetyDaysAgo, abortIfTanIsRequired = true), bank, customer, account) { response ->
account.triedToRetrieveTransactionsOfLast90DaysWithoutTan = true
if (response.isSuccessful) {
account.triedToRetrieveTransactionsOfLast90DaysWithoutTan = true if (response.isStrongAuthenticationRequired == false || hasRetrievedTransactionsWithTanJustBefore) {
// TODO: make use of supportsRetrievingTransactionsOfLast90DaysWithoutTan in UI e.g. in updateAccountsTransactionsIfNoTanIsRequiredAsync()
if (response.isSuccessful) { account.supportsRetrievingTransactionsOfLast90DaysWithoutTan = !!! response.isStrongAuthenticationRequired
if (response.isStrongAuthenticationRequired == false || hasRetrievedTransactionsWithTanJustBefore) { }
// TODO: make use of supportsRetrievingTransactionsOfLast90DaysWithoutTan in UI e.g. in updateAccountsTransactionsIfNoTanIsRequiredAsync()
account.supportsRetrievingTransactionsOfLast90DaysWithoutTan = !!! response.isStrongAuthenticationRequired
} }
}
return response callback(response)
}
} }
open fun getTransactionsAsync(parameter: GetTransactionsParameter, bank: BankData, open fun getTransactionsAsync(parameter: GetTransactionsParameter, bank: BankData,
customer: CustomerData, account: AccountData, callback: (GetTransactionsResponse) -> Unit) { customer: CustomerData, account: AccountData, callback: (GetTransactionsResponse) -> Unit) {
GlobalScope.launch { GlobalScope.launch {
callback(getTransactions(parameter, bank, customer, account)) getTransactions(parameter, bank, customer, account, callback)
} }
} }
open fun getTransactions(parameter: GetTransactionsParameter, bank: BankData, open fun getTransactions(parameter: GetTransactionsParameter, bank: BankData,
customer: CustomerData, account: AccountData): GetTransactionsResponse { customer: CustomerData, account: AccountData, callback: (GetTransactionsResponse) -> Unit) {
val dialogContext = DialogContext(bank, customer, product) val dialogContext = DialogContext(bank, customer, product)
val initDialogResponse = initDialog(dialogContext) initDialog(dialogContext) { initDialogResponse ->
if (initDialogResponse.successful == false) { if (initDialogResponse.successful == false) {
return GetTransactionsResponse(initDialogResponse) callback(GetTransactionsResponse(initDialogResponse))
}
var balance: Money? = null
if (parameter.alsoRetrieveBalance && account.supportsFeature(AccountFeature.RetrieveBalance)) {
val balanceResponse = getBalanceAfterDialogInit(account, dialogContext)
if (balanceResponse.successful == false && balanceResponse.couldCreateMessage == true) { // don't break here if required HKSAL message is not implemented
closeDialog(dialogContext)
return GetTransactionsResponse(balanceResponse)
} }
else {
balanceResponse.getFirstSegmentById<BalanceSegment>(InstituteSegmentId.Balance)?.let { mayGetBalance(parameter, account, dialogContext) { balanceResponse ->
balance = Money(it.balance, it.currency) if (balanceResponse.successful == false && balanceResponse.couldCreateMessage == true) { // don't break here if required HKSAL message is not implemented
closeDialog(dialogContext)
callback(GetTransactionsResponse(balanceResponse))
}
else {
getTransactionsAfterInitAndGetBalance(parameter, account, dialogContext, balanceResponse, callback)
}
}
} }
} }
}
protected open fun getTransactionsAfterInitAndGetBalance(parameter: GetTransactionsParameter, account: AccountData, dialogContext: DialogContext,
balanceResponse: Response, callback: (GetTransactionsResponse) -> Unit) {
val balance: Money? = balanceResponse.getFirstSegmentById<BalanceSegment>(InstituteSegmentId.Balance)?.let {
Money(it.balance, it.currency)
}
val message = messageBuilder.createGetTransactionsMessage(parameter, account, dialogContext) val message = messageBuilder.createGetTransactionsMessage(parameter, account, dialogContext)
@ -357,28 +377,34 @@ open class FinTsClient(
} }
} }
val response = getAndHandleResponseForMessage(message, dialogContext) getAndHandleResponseForMessage(message, dialogContext) { response ->
closeDialog(dialogContext)
closeDialog(dialogContext)
// just retrieved all transactions -> check if retrieving that ones of last 90 days is possible without entering TAN // just retrieved all transactions -> check if retrieving that ones of last 90 days is possible without entering TAN
if (account.supportsRetrievingTransactionsOfLast90DaysWithoutTan == null && if (account.supportsRetrievingTransactionsOfLast90DaysWithoutTan == null &&
response.successful && bookedTransactions.isNotEmpty() && parameter.fromDate == null) { response.successful && bookedTransactions.isNotEmpty() && parameter.fromDate == null) {
tryGetTransactionsOfLast90DaysWithoutTan(bank, customer, account, true) tryGetTransactionsOfLast90DaysWithoutTan(dialogContext.bank, dialogContext.customer, account, true) { }
}
callback(GetTransactionsResponse(
response,
bookedTransactions,
listOf(), // TODO: implement parsing MT942
balance
)
)
} }
return GetTransactionsResponse(response,
bookedTransactions,
listOf(), // TODO: implement parsing MT942
balance)
} }
protected open fun getBalanceAfterDialogInit(account: AccountData, dialogContext: DialogContext): Response { protected open fun mayGetBalance(parameter: GetTransactionsParameter, account: AccountData, dialogContext: DialogContext, callback: (Response) -> Unit) {
if (parameter.alsoRetrieveBalance && account.supportsFeature(AccountFeature.RetrieveBalance)) {
val message = messageBuilder.createGetBalanceMessage(account, dialogContext)
val message = messageBuilder.createGetBalanceMessage(account, dialogContext) getAndHandleResponseForMessage(message, dialogContext) { response ->
callback(response)
return getAndHandleResponseForMessage(message, dialogContext) }
}
} }
@ -388,50 +414,50 @@ open class FinTsClient(
callback: (GetTanMediaListResponse) -> Unit) { callback: (GetTanMediaListResponse) -> Unit) {
GlobalScope.launch { GlobalScope.launch {
callback(getTanMediaList(bank, customer)) getTanMediaList(bank, customer, tanMediaKind, tanMediumClass, callback)
} }
} }
open fun getTanMediaList(bank: BankData, customer: CustomerData, tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle, open fun getTanMediaList(bank: BankData, customer: CustomerData, tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle,
tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien): GetTanMediaListResponse { tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien, callback: (GetTanMediaListResponse) -> Unit) {
val response = sendMessageAndHandleResponse(bank, customer) { dialogContext -> sendMessageAndHandleResponse(bank, customer, true, { dialogContext ->
messageBuilder.createGetTanMediaListMessage(dialogContext, tanMediaKind, tanMediumClass) messageBuilder.createGetTanMediaListMessage(dialogContext, tanMediaKind, tanMediumClass)
}) { response ->
// TAN media list (= TAN generator list) is only returned for users with chipTAN TAN procedures
val tanMediaList = if (response.successful == false ) null
else response.getFirstSegmentById<TanMediaList>(InstituteSegmentId.TanMediaList)
tanMediaList?.let {
customer.tanMedia = it.tanMedia
}
callback(GetTanMediaListResponse(response, tanMediaList))
} }
// TAN media list (= TAN generator list) is only returned for users with chipTAN TAN procedures
val tanMediaList = if (response.successful == false ) null
else response.getFirstSegmentById<TanMediaList>(InstituteSegmentId.TanMediaList)
tanMediaList?.let {
customer.tanMedia = it.tanMedia
}
return GetTanMediaListResponse(response, tanMediaList)
} }
open fun changeTanMedium(newActiveTanMedium: TanGeneratorTanMedium, bank: BankData, customer: CustomerData): FinTsClientResponse { open fun changeTanMedium(newActiveTanMedium: TanGeneratorTanMedium, bank: BankData, customer: CustomerData, callback: (FinTsClientResponse) -> Unit) {
var enteredAtc: EnterTanGeneratorAtcResult? = null var enteredAtc: EnterTanGeneratorAtcResult? = null
if (bank.changeTanMediumParameters?.enteringAtcAndTanRequired == true) { if (bank.changeTanMediumParameters?.enteringAtcAndTanRequired == true) {
enteredAtc = callback.enterTanGeneratorAtc(customer, newActiveTanMedium) enteredAtc = this.callback.enterTanGeneratorAtc(customer, newActiveTanMedium)
if (enteredAtc.hasAtcBeenEntered == false) { if (enteredAtc.hasAtcBeenEntered == false) {
val message = "Bank requires to enter ATC and TAN in order to change TAN medium." // TODO: translate val message = "Bank requires to enter ATC and TAN in order to change TAN medium." // TODO: translate
return FinTsClientResponse(Response(false, exception = Exception(message))) callback(FinTsClientResponse(Response(false, exception = Exception(message))))
return
} }
} }
val response = sendMessageAndHandleResponse(bank, customer, false) { dialogContext -> sendMessageAndHandleResponse(bank, customer, false, { dialogContext ->
messageBuilder.createChangeTanMediumMessage(newActiveTanMedium, dialogContext, messageBuilder.createChangeTanMediumMessage(newActiveTanMedium, dialogContext,
enteredAtc?.tan, enteredAtc?.atc) enteredAtc?.tan, enteredAtc?.atc)
}) { response ->
callback(FinTsClientResponse(response))
} }
return FinTsClientResponse(response)
} }
@ -439,74 +465,78 @@ open class FinTsClient(
customer: CustomerData, account: AccountData, callback: (FinTsClientResponse) -> Unit) { customer: CustomerData, account: AccountData, callback: (FinTsClientResponse) -> Unit) {
GlobalScope.launch { GlobalScope.launch {
callback(doBankTransfer(bankTransferData, bank, customer, account)) doBankTransfer(bankTransferData, bank, customer, account, callback)
} }
} }
open fun doBankTransfer(bankTransferData: BankTransferData, bank: BankData, open fun doBankTransfer(bankTransferData: BankTransferData, bank: BankData,
customer: CustomerData, account: AccountData): FinTsClientResponse { customer: CustomerData, account: AccountData, callback: (FinTsClientResponse) -> Unit) {
val response = sendMessageAndHandleResponse(bank, customer) { dialogContext -> sendMessageAndHandleResponse(bank, customer, true, { dialogContext ->
messageBuilder.createBankTransferMessage(bankTransferData, account, dialogContext) messageBuilder.createBankTransferMessage(bankTransferData, account, dialogContext)
}) { response ->
callback(FinTsClientResponse(response))
} }
return FinTsClientResponse(response)
} }
protected open fun sendMessageAndHandleResponse(bank: BankData, customer: CustomerData, messageMayRequiresTan: Boolean = true, protected open fun sendMessageAndHandleResponse(bank: BankData, customer: CustomerData, messageMayRequiresTan: Boolean = true,
createMessage: (DialogContext) -> MessageBuilderResult): Response { createMessage: (DialogContext) -> MessageBuilderResult, callback: (Response) -> Unit) {
val dialogContext = DialogContext(bank, customer, product) val dialogContext = DialogContext(bank, customer, product)
val initDialogResponse = initDialog(dialogContext) initDialog(dialogContext) { initDialogResponse ->
if (initDialogResponse.successful == false) {
callback(initDialogResponse)
}
else {
val message = createMessage(dialogContext)
if (initDialogResponse.successful == false) { getAndHandleResponseForMessage(message, dialogContext) { response ->
return initDialogResponse closeDialog(dialogContext)
callback(response)
}
}
} }
val message = createMessage(dialogContext)
val response = getAndHandleResponseForMessage(message, dialogContext)
closeDialog(dialogContext)
return response
} }
protected open fun initDialog(dialogContext: DialogContext): Response { protected open fun initDialog(dialogContext: DialogContext, callback: (Response) -> Unit) {
// we first need to retrieve supported tan procedures and jobs before we can do anything // we first need to retrieve supported tan procedures and jobs before we can do anything
val retrieveBasicBankDataResponse = ensureBasicBankDataRetrieved(dialogContext.bank, dialogContext.customer) ensureBasicBankDataRetrieved(dialogContext.bank, dialogContext.customer) { retrieveBasicBankDataResponse ->
if (retrieveBasicBankDataResponse.successful == false) { if (retrieveBasicBankDataResponse.successful == false) {
return retrieveBasicBankDataResponse callback(retrieveBasicBankDataResponse)
}
else {
// as in the next step we have to supply user's tan procedure, ensure user selected his or her
ensureTanProcedureIsSelected(dialogContext.bank, dialogContext.customer) { tanProcedureSelectedResponse ->
if (tanProcedureSelectedResponse.successful == false) {
callback(tanProcedureSelectedResponse)
}
else {
initDialogAfterSuccessfulChecks(dialogContext, callback)
}
}
}
} }
// as in the next step we have to supply user's tan procedure, ensure user selected his or her
val tanProcedureSelectedResponse = ensureTanProcedureIsSelected(dialogContext.bank, dialogContext.customer)
if (tanProcedureSelectedResponse.successful == false) {
return tanProcedureSelectedResponse
}
return initDialogAfterSuccessfulChecks(dialogContext)
} }
protected open fun initDialogAfterSuccessfulChecks(dialogContext: DialogContext): Response { protected open fun initDialogAfterSuccessfulChecks(dialogContext: DialogContext, callback: (Response) -> Unit) {
val message = messageBuilder.createInitDialogMessage(dialogContext) val message = messageBuilder.createInitDialogMessage(dialogContext)
val response = GetUserTanProceduresResponse(getAndHandleResponseForMessage(message, dialogContext)) getAndHandleResponseForMessage(message, dialogContext) { unmappedResponse ->
dialogContext.response = response val response = GetUserTanProceduresResponse(unmappedResponse)
dialogContext.response = response
if (response.successful) { if (response.successful) {
updateBankData(dialogContext.bank, response) updateBankData(dialogContext.bank, response)
updateCustomerData(dialogContext.customer, dialogContext.bank, response) updateCustomerData(dialogContext.customer, dialogContext.bank, response)
}
callback(response)
} }
return response
} }
protected open fun closeDialog(dialogContext: DialogContext) { protected open fun closeDialog(dialogContext: DialogContext) {
@ -522,35 +552,45 @@ open class FinTsClient(
} }
protected open fun ensureBasicBankDataRetrieved(bank: BankData, customer: CustomerData): Response { protected open fun ensureBasicBankDataRetrieved(bank: BankData, customer: CustomerData, callback: (Response) -> Unit) {
if (bank.supportedTanProcedures.isEmpty() || bank.supportedJobs.isEmpty()) { if (bank.supportedTanProcedures.isEmpty() || bank.supportedJobs.isEmpty()) {
val getBankInfoResponse = getBankAndCustomerInfoForNewUser(bank, customer) getBankAndCustomerInfoForNewUser(bank, customer) { getBankInfoResponse ->
if (getBankInfoResponse.isSuccessful == false || bank.supportedTanProcedures.isEmpty()
|| bank.supportedJobs.isEmpty()) {
if (getBankInfoResponse.isSuccessful == false || bank.supportedTanProcedures.isEmpty() callback(Response(false, exception =
|| bank.supportedJobs.isEmpty()) { Exception("Could not retrieve basic bank data like supported tan procedures or supported jobs"))) // TODO: translate // TODO: add as messageToShowToUser
}
return Response(false, exception = else {
Exception("Could not retrieve basic bank data like supported tan procedures or supported jobs")) // TODO: translate // TODO: add as messageToShowToUser callback(Response(true))
}
} }
} }
else {
return Response(true) callback(Response(true))
}
} }
protected open fun ensureTanProcedureIsSelected(bank: BankData, customer: CustomerData): Response { protected open fun ensureTanProcedureIsSelected(bank: BankData, customer: CustomerData, callback: (Response) -> Unit) {
if (customer.isTanProcedureSelected == false) { if (customer.isTanProcedureSelected == false) {
if (customer.supportedTanProcedures.isEmpty()) { if (customer.supportedTanProcedures.isEmpty()) {
getBankAndCustomerInfoForNewUser(bank, customer) getBankAndCustomerInfoForNewUser(bank, customer) {
if (customer.supportedTanProcedures.isEmpty()) { // could not retrieve supported tan procedures for user
callback(Response(false, noTanProcedureSelected = true))
}
else {
getUsersTanProcedure(customer)
callback(Response(customer.isTanProcedureSelected, noTanProcedureSelected = !!!customer.isTanProcedureSelected))
}
}
} }
else {
if (customer.supportedTanProcedures.isEmpty()) { // could not retrieve supported tan procedures for user callback(Response(customer.isTanProcedureSelected, noTanProcedureSelected = !!!customer.isTanProcedureSelected))
return Response(false, noTanProcedureSelected = true)
} }
getUsersTanProcedure(customer)
} }
else {
return Response(customer.isTanProcedureSelected, noTanProcedureSelected = !!!customer.isTanProcedureSelected) callback(Response(customer.isTanProcedureSelected, noTanProcedureSelected = !!!customer.isTanProcedureSelected))
}
} }
protected open fun getUsersTanProcedure(customer: CustomerData) { protected open fun getUsersTanProcedure(customer: CustomerData) {
@ -573,55 +613,64 @@ open class FinTsClient(
} }
protected open fun getAndHandleResponseForMessage(message: MessageBuilderResult, dialogContext: DialogContext): Response { protected open fun getAndHandleResponseForMessage(message: MessageBuilderResult, dialogContext: DialogContext, callback: (Response) -> Unit) {
val response = if (message.createdMessage == null) Response(false, messageCreationError = message) if (message.createdMessage == null) {
else getAndHandleResponseForMessage(message.createdMessage, dialogContext) callback(Response(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)
}
val handledResponse = handleMayRequiresTan(response, dialogContext) getFollowUpMessageForContinuationId(handledResponse, continuationId, message, dialogContext) { followUpResponse ->
handledResponse.followUpResponse = followUpResponse
handledResponse.hasFollowUpMessageButCouldNotReceiveIt = handledResponse.followUpResponse == null
// if there's a Aufsetzpunkt (continuationId) set, then response is not complete yet, there's more information to fetch by sending this Aufsetzpunkt callback(handledResponse)
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) ?: run {
dialogContext.chunkedResponseHandler?.invoke(handledResponse)
callback(handledResponse)
}
} }
handledResponse.followUpResponse = getFollowUpMessageForContinuationId(handledResponse, continuationId, message, dialogContext)
handledResponse.hasFollowUpMessageButCouldNotReceiveIt = handledResponse.followUpResponse == null
} }
} }
?: run { dialogContext.chunkedResponseHandler?.invoke(handledResponse) }
return handledResponse
} }
protected open fun getAndHandleResponseForMessage(requestBody: String, dialogContext: DialogContext): Response { protected open fun getAndHandleResponseForMessage(requestBody: String, dialogContext: DialogContext, callback: (Response) -> Unit) {
addMessageLog(requestBody, MessageLogEntryType.Sent, dialogContext) addMessageLog(requestBody, MessageLogEntryType.Sent, dialogContext)
val webResponse = getResponseForMessage(requestBody, dialogContext.bank.finTs3ServerAddress) getResponseForMessage(requestBody, dialogContext.bank.finTs3ServerAddress) { webResponse ->
val response = handleResponse(webResponse, dialogContext)
val response = handleResponse(webResponse, dialogContext) dialogContext.response = response
dialogContext.response = response response.messageHeader?.let { header -> dialogContext.dialogId = header.dialogId }
dialogContext.didBankCloseDialog = response.didBankCloseDialog
response.messageHeader?.let { header -> dialogContext.dialogId = header.dialogId } callback(response)
dialogContext.didBankCloseDialog = response.didBankCloseDialog }
return response
} }
protected open fun getResponseForMessage(requestBody: String, finTs3ServerAddress: String): WebClientResponse { protected open fun getResponseForMessage(requestBody: String, finTs3ServerAddress: String, callback: (WebClientResponse) -> Unit) {
val encodedRequestBody = base64Service.encode(requestBody) val encodedRequestBody = base64Service.encode(requestBody)
return webClient.post(finTs3ServerAddress, encodedRequestBody, "application/octet-stream") webClient.post(finTs3ServerAddress, encodedRequestBody, "application/octet-stream", IWebClient.DefaultUserAgent, callback)
} }
protected open fun fireAndForgetMessage(message: MessageBuilderResult, dialogContext: DialogContext) { protected open fun fireAndForgetMessage(message: MessageBuilderResult, dialogContext: DialogContext) {
message.createdMessage?.let { requestBody -> message.createdMessage?.let { requestBody ->
addMessageLog(requestBody, MessageLogEntryType.Sent, dialogContext) addMessageLog(requestBody, MessageLogEntryType.Sent, dialogContext)
getResponseForMessage(requestBody, dialogContext.bank.finTs3ServerAddress) getResponseForMessage(requestBody, dialogContext.bank.finTs3ServerAddress) { }
// if really needed add received response to message log here // if really needed add received response to message log here
} }
@ -658,13 +707,12 @@ open class FinTsClient(
protected open fun getFollowUpMessageForContinuationId(response: Response, continuationId: String, message: MessageBuilderResult, protected open fun getFollowUpMessageForContinuationId(response: Response, continuationId: String, message: MessageBuilderResult,
dialogContext: DialogContext): Response? { dialogContext: DialogContext, callback: (Response?) -> Unit) {
messageBuilder.rebuildMessageWithContinuationId(message, continuationId, dialogContext)?.let { followUpMessage -> messageBuilder.rebuildMessageWithContinuationId(message, continuationId, dialogContext)?.let { followUpMessage ->
return getAndHandleResponseForMessage(followUpMessage, dialogContext) getAndHandleResponseForMessage(followUpMessage, dialogContext, callback)
} }
?: run { callback(null) }
return null
} }
@ -717,26 +765,29 @@ open class FinTsClient(
} }
protected open fun handleMayRequiresTan(response: Response, dialogContext: DialogContext): Response { // TODO: use response from DialogContext protected open fun handleMayRequiresTan(response: Response, dialogContext: DialogContext, callback: (Response) -> Unit) { // TODO: use response from DialogContext
if (response.isStrongAuthenticationRequired) { if (response.isStrongAuthenticationRequired) {
if (dialogContext.abortIfTanIsRequired) { if (dialogContext.abortIfTanIsRequired) {
response.tanRequiredButWeWereToldToAbortIfSo = true response.tanRequiredButWeWereToldToAbortIfSo = true
return response callback(response)
return
} }
response.tanResponse?.let { tanResponse -> response.tanResponse?.let { tanResponse ->
val customer = dialogContext.customer val customer = dialogContext.customer
val enteredTanResult = callback.enterTan(customer, createTanChallenge(tanResponse, customer)) val enteredTanResult = this.callback.enterTan(customer, createTanChallenge(tanResponse, customer))
if (enteredTanResult.changeTanProcedureTo != null) { if (enteredTanResult.changeTanProcedureTo != null) {
return handleUserAsksToChangeTanProcedureAndResendLastMessage(enteredTanResult.changeTanProcedureTo, handleUserAsksToChangeTanProcedureAndResendLastMessage(enteredTanResult.changeTanProcedureTo,
dialogContext) dialogContext, callback)
return
} }
else if (enteredTanResult.changeTanMediumTo is TanGeneratorTanMedium) { else if (enteredTanResult.changeTanMediumTo is TanGeneratorTanMedium) {
return handleUserAsksToChangeTanMediumAndResendLastMessage(enteredTanResult.changeTanMediumTo, handleUserAsksToChangeTanMediumAndResendLastMessage(enteredTanResult.changeTanMediumTo,
dialogContext, enteredTanResult.changeTanMediumResultCallback) dialogContext, enteredTanResult.changeTanMediumResultCallback, callback)
return
} }
else if (enteredTanResult.enteredTan == null) { else if (enteredTanResult.enteredTan == null) {
// i tried to send a HKTAN with cancelJob = true but then i saw there are no tan procedures that support cancellation (at least not at my bank) // i tried to send a HKTAN with cancelJob = true but then i saw there are no tan procedures that support cancellation (at least not at my bank)
@ -744,7 +795,8 @@ open class FinTsClient(
response.tanRequiredButUserDidNotEnterOne = true response.tanRequiredButUserDidNotEnterOne = true
} }
else { else {
return sendTanToBank(enteredTanResult.enteredTan, tanResponse, dialogContext) sendTanToBank(enteredTanResult.enteredTan, tanResponse, dialogContext, callback)
return
} }
} }
} }
@ -756,7 +808,7 @@ open class FinTsClient(
// TODO: also check '9931 Sperrung des Kontos nach %1 Fehlversuchen' -> if %1 == 3 synchronize TAN generator // 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 // as it's quite unrealistic that user entered TAN wrong three times, in most cases TAN generator is not synchronized
return response callback(response)
} }
protected open fun createTanChallenge(tanResponse: TanResponse, customer: CustomerData): TanChallenge { protected open fun createTanChallenge(tanResponse: TanResponse, customer: CustomerData): TanChallenge {
@ -778,14 +830,14 @@ open class FinTsClient(
} }
} }
protected open fun sendTanToBank(enteredTan: String, tanResponse: TanResponse, dialogContext: DialogContext): Response { protected open fun sendTanToBank(enteredTan: String, tanResponse: TanResponse, dialogContext: DialogContext, callback: (Response) -> Unit) {
val message = messageBuilder.createSendEnteredTanMessage(enteredTan, tanResponse, dialogContext) val message = messageBuilder.createSendEnteredTanMessage(enteredTan, tanResponse, dialogContext)
return getAndHandleResponseForMessage(message, dialogContext) getAndHandleResponseForMessage(message, dialogContext, callback)
} }
protected open fun handleUserAsksToChangeTanProcedureAndResendLastMessage(changeTanProcedureTo: TanProcedure, dialogContext: DialogContext): Response { protected open fun handleUserAsksToChangeTanProcedureAndResendLastMessage(changeTanProcedureTo: TanProcedure, dialogContext: DialogContext, callback: (Response) -> Unit) {
dialogContext.customer.selectedTanProcedure = changeTanProcedureTo dialogContext.customer.selectedTanProcedure = changeTanProcedureTo
@ -794,53 +846,55 @@ open class FinTsClient(
lastCreatedMessage?.let { closeDialog(dialogContext) } lastCreatedMessage?.let { closeDialog(dialogContext) }
return resendMessageInNewDialog(lastCreatedMessage, dialogContext) resendMessageInNewDialog(lastCreatedMessage, dialogContext, callback)
} }
protected open fun handleUserAsksToChangeTanMediumAndResendLastMessage(changeTanMediumTo: TanGeneratorTanMedium, protected open fun handleUserAsksToChangeTanMediumAndResendLastMessage(changeTanMediumTo: TanGeneratorTanMedium,
dialogContext: DialogContext, dialogContext: DialogContext,
changeTanMediumResultCallback: ((FinTsClientResponse) -> Unit)?): Response { changeTanMediumResultCallback: ((FinTsClientResponse) -> Unit)?,
callback: (Response) -> Unit) {
val lastCreatedMessage = dialogContext.currentMessage val lastCreatedMessage = dialogContext.currentMessage
lastCreatedMessage?.let { closeDialog(dialogContext) } lastCreatedMessage?.let { closeDialog(dialogContext) }
val changeTanMediumResponse = changeTanMedium(changeTanMediumTo, dialogContext.bank, dialogContext.customer) changeTanMedium(changeTanMediumTo, dialogContext.bank, dialogContext.customer) { changeTanMediumResponse ->
changeTanMediumResultCallback?.invoke(changeTanMediumResponse)
changeTanMediumResultCallback?.invoke(changeTanMediumResponse) if (changeTanMediumResponse.isSuccessful == false || lastCreatedMessage == null) {
callback(changeTanMediumResponse.toResponse())
if (changeTanMediumResponse.isSuccessful == false || lastCreatedMessage == null) { }
return changeTanMediumResponse.toResponse() else {
resendMessageInNewDialog(lastCreatedMessage, dialogContext, callback)
}
} }
return resendMessageInNewDialog(lastCreatedMessage, dialogContext)
} }
protected open fun resendMessageInNewDialog(lastCreatedMessage: MessageBuilderResult?, previousDialogContext: DialogContext): Response { protected open fun resendMessageInNewDialog(lastCreatedMessage: MessageBuilderResult?, previousDialogContext: DialogContext, callback: (Response) -> Unit) {
lastCreatedMessage?.let { // do not use previousDialogContext.currentMessage as this may is previous dialog's dialog close message lastCreatedMessage?.let { // do not use previousDialogContext.currentMessage as this may is previous dialog's dialog close message
val newDialogContext = DialogContext(previousDialogContext.bank, previousDialogContext.customer, previousDialogContext.product, chunkedResponseHandler = previousDialogContext.chunkedResponseHandler) val newDialogContext = DialogContext(previousDialogContext.bank, previousDialogContext.customer, previousDialogContext.product, chunkedResponseHandler = previousDialogContext.chunkedResponseHandler)
val initDialogResponse = initDialog(newDialogContext) initDialog(newDialogContext) { initDialogResponse ->
if (initDialogResponse.successful == false) { if (initDialogResponse.successful == false) {
return initDialogResponse callback(initDialogResponse)
}
else {
val newMessage = messageBuilder.rebuildMessage(lastCreatedMessage, newDialogContext)
getAndHandleResponseForMessage(newMessage, newDialogContext) { response ->
closeDialog(newDialogContext)
callback(response)
}
}
} }
val newMessage = messageBuilder.rebuildMessage(lastCreatedMessage, newDialogContext)
val response = getAndHandleResponseForMessage(newMessage, newDialogContext)
closeDialog(newDialogContext)
return response
} }
val errorMessage = "There's no last action (like retrieve account transactions, transfer money, ...) to re-send with new TAN procedure. Probably an internal programming error." // TODO: translate val errorMessage = "There's no last action (like retrieve account transactions, transfer money, ...) to re-send with new TAN procedure. Probably an internal programming error." // TODO: translate
return Response(false, exception = Exception(errorMessage)) // should never come to this callback(Response(false, exception = Exception(errorMessage))) // should never come to this
} }

View File

@ -38,8 +38,8 @@ open class FinTsClientForCustomer(
client.addAccountAsync(bank, customer, callback) client.addAccountAsync(bank, customer, callback)
} }
open fun addAccount(): AddAccountResponse { open fun addAccount(callback: (AddAccountResponse) -> Unit) {
return client.addAccount(bank, customer) client.addAccount(bank, customer, callback)
} }
@ -47,8 +47,8 @@ open class FinTsClientForCustomer(
client.getTransactionsAsync(parameter, bank, customer, account, callback) client.getTransactionsAsync(parameter, bank, customer, account, callback)
} }
open fun getTransactions(parameter: GetTransactionsParameter, account: AccountData): GetTransactionsResponse { open fun getTransactions(parameter: GetTransactionsParameter, account: AccountData, callback: (GetTransactionsResponse) -> Unit) {
return client.getTransactions(parameter, bank, customer, account) client.getTransactions(parameter, bank, customer, account, callback)
} }
@ -56,8 +56,8 @@ open class FinTsClientForCustomer(
client.doBankTransferAsync(bankTransferData, bank, customer, account, callback) client.doBankTransferAsync(bankTransferData, bank, customer, account, callback)
} }
open fun doBankTransfer(bankTransferData: BankTransferData, account: AccountData): FinTsClientResponse { open fun doBankTransfer(bankTransferData: BankTransferData, account: AccountData, callback: (FinTsClientResponse) -> Unit) {
return client.doBankTransfer(bankTransferData, bank, customer, account) client.doBankTransfer(bankTransferData, bank, customer, account, callback)
} }
} }

View File

@ -20,6 +20,6 @@ interface IWebClient {
} }
fun post(url: String, body: String, contentType: String = "application/octet-stream", userAgent: String = DefaultUserAgent) : WebClientResponse fun post(url: String, body: String, contentType: String = "application/octet-stream", userAgent: String = DefaultUserAgent, callback: (WebClientResponse) -> Unit)
} }

View File

@ -34,29 +34,21 @@ open class KtorWebClient : IWebClient {
} }
override fun post(url: String, body: String, contentType: String, userAgent: String): WebClientResponse { override fun post(url: String, body: String, contentType: String, userAgent: String, callback: (WebClientResponse) -> Unit) {
try { GlobalScope.async {
val job = GlobalScope.async { try {
try { val clientResponse = client.post<HttpResponse>(url) {
val clientResponse = client.post<HttpResponse>(url) { this.body = TextContent(body, contentType = ContentType.Application.OctetStream)
this.body = TextContent(body, contentType = ContentType.Application.OctetStream)
}
val responseBody = clientResponse.readText()
WebClientResponse(clientResponse.status.value == 200, clientResponse.status.value, body = responseBody)
} catch (e: Exception) {
log.error(e) { "Could not send request to url '$url'" }
WebClientResponse(false, error = e)
} }
val responseBody = clientResponse.readText()
callback(WebClientResponse(clientResponse.status.value == 200, clientResponse.status.value, body = responseBody))
} catch (e: Exception) {
log.error(e) { "Could not send request to url '$url'" }
callback(WebClientResponse(false, error = e))
} }
while (job.isCompleted == false) { } // let's warm the CPU to get suspend function synchronous (runBlocking is not available in common projects)
return job.getCompleted()
} catch (e: Exception) {
return WebClientResponse(false, error = e)
} }
} }

View File

@ -87,15 +87,16 @@ open class FinTsClientTestBase {
fun getAnonymousBankInfo() { fun getAnonymousBankInfo() {
// when // when
val result = underTest.getAnonymousBankInfo(BankDataAnonymous) underTest.getAnonymousBankInfo(BankDataAnonymous) { result ->
// then // then
expect(result.isSuccessful).isTrue() expect(result.isSuccessful).isTrue()
expect(BankDataAnonymous.supportedHbciVersions).isNotEmpty() expect(BankDataAnonymous.supportedHbciVersions).isNotEmpty()
expect(BankDataAnonymous.supportedTanProcedures).isNotEmpty() expect(BankDataAnonymous.supportedTanProcedures).isNotEmpty()
expect(BankDataAnonymous.supportedJobs).isNotEmpty() expect(BankDataAnonymous.supportedJobs).isNotEmpty()
expect(BankDataAnonymous.supportedLanguages).isNotEmpty() expect(BankDataAnonymous.supportedLanguages).isNotEmpty()
expect(BankDataAnonymous.name).isNotEmpty() expect(BankDataAnonymous.name).isNotEmpty()
}
} }
@ -166,7 +167,7 @@ open class FinTsClientTestBase {
// this test is only senseful for accounts using chipTAN / TAN generator as TAN procedure // this test is only senseful for accounts using chipTAN / TAN generator as TAN procedure
underTest.getAnonymousBankInfo(Bank) underTest.getAnonymousBankInfo(Bank) { }
val supportsRetrievingTanMedia = Bank.supportedJobs.firstOrNull { it.jobName == "HKTAB" } != null val supportsRetrievingTanMedia = Bank.supportedJobs.firstOrNull { it.jobName == "HKTAB" } != null
@ -178,17 +179,17 @@ open class FinTsClientTestBase {
// when // when
val result = underTest.getTanMediaList(Bank, Customer, TanMedienArtVersion.Alle, TanMediumKlasse.AlleMedien) underTest.getTanMediaList(Bank, Customer, TanMedienArtVersion.Alle, TanMediumKlasse.AlleMedien) { result ->
// then
expect(result.isSuccessful).isTrue()
// then expect(result.tanMediaList).notToBeNull()
expect(result.isSuccessful).isTrue() expect(result.tanMediaList!!.usageOption).toBe(TanEinsatzOption.KundeKannGenauEinMediumZuEinerZeitNutzen) // TODO: may adjust to your value
expect(result.tanMediaList!!.tanMedia).isNotEmpty()
expect(result.tanMediaList).notToBeNull() expect(Customer.tanMedia).isNotEmpty()
expect(result.tanMediaList!!.usageOption).toBe(TanEinsatzOption.KundeKannGenauEinMediumZuEinerZeitNutzen) // TODO: may adjust to your value }
expect(result.tanMediaList!!.tanMedia).isNotEmpty()
expect(Customer.tanMedia).isNotEmpty()
} }
@Ignore // only works with banks that don't support HKTAB version 5 @Ignore // only works with banks that don't support HKTAB version 5
@ -197,7 +198,7 @@ open class FinTsClientTestBase {
// when // when
expect { expect {
underTest.getTanMediaList(Bank, Customer, TanMedienArtVersion.Alle, TanMediumKlasse.BilateralVereinbart) underTest.getTanMediaList(Bank, Customer, TanMedienArtVersion.Alle, TanMediumKlasse.BilateralVereinbart) { }
}.toThrow<UnsupportedOperationException>() }.toThrow<UnsupportedOperationException>()
@ -226,10 +227,11 @@ open class FinTsClientTestBase {
// when // when
val result = underTest.doBankTransfer(BankTransferData, Bank, Customer, account) underTest.doBankTransfer(BankTransferData, Bank, Customer, account) { result ->
// then // then
expect(result.isSuccessful).isTrue() expect(result.isSuccessful).isTrue()
}
} }
} }