Created JobContext to have a cross dialog context per job and to be able to set FinTsClientCallback (and later on other objects) on a job basis
This commit is contained in:
parent
d5573817ef
commit
0a9b31b393
|
@ -9,14 +9,16 @@ import net.dankito.banking.fints.response.BankResponse
|
||||||
import net.dankito.banking.fints.response.client.*
|
import net.dankito.banking.fints.response.client.*
|
||||||
import net.dankito.banking.fints.response.segments.*
|
import net.dankito.banking.fints.response.segments.*
|
||||||
import net.dankito.utils.multiplatform.Date
|
import net.dankito.utils.multiplatform.Date
|
||||||
|
import kotlin.jvm.JvmOverloads
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This is the high level FinTS client that groups single low level jobs of [FinTsJobExecutor] to senseful units e.g.
|
* 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.
|
* [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(
|
open class FinTsClient @JvmOverloads constructor(
|
||||||
protected open val jobExecutor: FinTsJobExecutor // TODO: recreate when callback is set to avoid multithreading issues - but use its configured instances like RequestExecutor
|
open var callback: FinTsClientCallback,
|
||||||
|
protected open val jobExecutor: FinTsJobExecutor = FinTsJobExecutor()
|
||||||
) {
|
) {
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
@ -24,16 +26,9 @@ open class FinTsClient(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
constructor(callback: FinTsClientCallback) : this(FinTsJobExecutor(callback))
|
|
||||||
|
|
||||||
|
|
||||||
open val messageLogWithoutSensitiveData: List<MessageLogEntry>
|
open val messageLogWithoutSensitiveData: List<MessageLogEntry>
|
||||||
get() = jobExecutor.messageLogWithoutSensitiveData
|
get() = jobExecutor.messageLogWithoutSensitiveData
|
||||||
|
|
||||||
open fun setCallback(callback: FinTsClientCallback) {
|
|
||||||
jobExecutor.callback = callback
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves information about bank (e.g. supported HBCI versions, FinTS server address,
|
* Retrieves information about bank (e.g. supported HBCI versions, FinTS server address,
|
||||||
|
@ -55,7 +50,7 @@ open class FinTsClient(
|
||||||
* On success [bank] parameter is updated afterwards.
|
* On success [bank] parameter is updated afterwards.
|
||||||
*/
|
*/
|
||||||
open fun getAnonymousBankInfo(bank: BankData, callback: (FinTsClientResponse) -> Unit) {
|
open fun getAnonymousBankInfo(bank: BankData, callback: (FinTsClientResponse) -> Unit) {
|
||||||
jobExecutor.getAnonymousBankInfo(bank) { response ->
|
jobExecutor.getAnonymousBankInfo(JobContext(JobContextType.AnonymousBankInfo, this.callback, bank)) { response ->
|
||||||
callback(FinTsClientResponse(response))
|
callback(FinTsClientResponse(response))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -63,10 +58,11 @@ open class FinTsClient(
|
||||||
|
|
||||||
open fun addAccountAsync(parameter: AddAccountParameter, callback: (AddAccountResponse) -> Unit) {
|
open fun addAccountAsync(parameter: AddAccountParameter, callback: (AddAccountResponse) -> Unit) {
|
||||||
val bank = parameter.bank
|
val bank = parameter.bank
|
||||||
|
val context = JobContext(JobContextType.AddAccount, this.callback, bank)
|
||||||
|
|
||||||
/* First dialog: Get user's basic data like BPD, customer system ID and her TAN methods */
|
/* First dialog: Get user's basic data like BPD, customer system ID and her TAN methods */
|
||||||
|
|
||||||
jobExecutor.retrieveBasicDataLikeUsersTanMethods(bank, parameter.preferredTanMethods, parameter.preferredTanMedium) { newUserInfoResponse ->
|
jobExecutor.retrieveBasicDataLikeUsersTanMethods(context, parameter.preferredTanMethods, parameter.preferredTanMedium) { newUserInfoResponse ->
|
||||||
|
|
||||||
if (newUserInfoResponse.successful == false) { // bank parameter (FinTS server address, ...) already seem to be wrong
|
if (newUserInfoResponse.successful == false) { // bank parameter (FinTS server address, ...) already seem to be wrong
|
||||||
callback(AddAccountResponse(newUserInfoResponse, bank))
|
callback(AddAccountResponse(newUserInfoResponse, bank))
|
||||||
|
@ -75,37 +71,38 @@ open class FinTsClient(
|
||||||
|
|
||||||
/* Second dialog: some banks require that in order to initialize a dialog with strong customer authorization TAN media is required */
|
/* Second dialog: some banks require that in order to initialize a dialog with strong customer authorization TAN media is required */
|
||||||
|
|
||||||
addAccountGetAccountsAndTransactions(parameter, bank, callback)
|
addAccountGetAccountsAndTransactions(context, parameter, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun addAccountGetAccountsAndTransactions(parameter: AddAccountParameter, bank: BankData,
|
protected open fun addAccountGetAccountsAndTransactions(context: JobContext, parameter: AddAccountParameter,
|
||||||
callback: (AddAccountResponse) -> Unit) {
|
callback: (AddAccountResponse) -> Unit) {
|
||||||
|
|
||||||
/* Third dialog: Now we can initialize our first dialog with strong customer authorization. Use it to get UPD and customer's accounts */
|
/* Third dialog: Now we can initialize our first dialog with strong customer authorization. Use it to get UPD and customer's accounts */
|
||||||
|
|
||||||
jobExecutor.getAccounts(bank) { getAccountsResponse ->
|
jobExecutor.getAccounts(context) { getAccountsResponse ->
|
||||||
|
|
||||||
if (getAccountsResponse.successful == false) {
|
if (getAccountsResponse.successful == false) {
|
||||||
callback(AddAccountResponse(getAccountsResponse, bank))
|
callback(AddAccountResponse(getAccountsResponse, context.bank))
|
||||||
return@getAccounts
|
return@getAccounts
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Fourth dialog (if requested): Try to retrieve account balances and transactions of last 90 days without TAN */
|
/* Fourth dialog (if requested): Try to retrieve account balances and transactions of last 90 days without TAN */
|
||||||
|
|
||||||
if (parameter.fetchBalanceAndTransactions) {
|
if (parameter.fetchBalanceAndTransactions) {
|
||||||
addAccountGetAccountBalancesAndTransactions(bank, getAccountsResponse, callback)
|
addAccountGetAccountBalancesAndTransactions(context, getAccountsResponse, callback)
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
val retrievedAccountData = bank.accounts.associateBy( { it }, { RetrievedAccountData.balanceAndTransactionsNotRequestedByUser(it) } )
|
val retrievedAccountData = context.bank.accounts.associateBy( { it }, { RetrievedAccountData.balanceAndTransactionsNotRequestedByUser(it) } )
|
||||||
addAccountDone(bank, getAccountsResponse, retrievedAccountData, callback)
|
addAccountDone(context, getAccountsResponse, retrievedAccountData, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun addAccountGetAccountBalancesAndTransactions(bank: BankData, getAccountsResponse: BankResponse,
|
protected open fun addAccountGetAccountBalancesAndTransactions(context: JobContext, getAccountsResponse: BankResponse,
|
||||||
callback: (AddAccountResponse) -> Unit) {
|
callback: (AddAccountResponse) -> Unit) {
|
||||||
|
|
||||||
|
val bank = context.bank
|
||||||
val retrievedAccountData = bank.accounts.associateBy( { it }, { RetrievedAccountData.unsuccessful(it) } ).toMutableMap()
|
val retrievedAccountData = bank.accounts.associateBy( { it }, { RetrievedAccountData.unsuccessful(it) } ).toMutableMap()
|
||||||
|
|
||||||
val accountsSupportingRetrievingTransactions = bank.accounts.filter { it.supportsRetrievingBalance || it.supportsRetrievingAccountTransactions }
|
val accountsSupportingRetrievingTransactions = bank.accounts.filter { it.supportsRetrievingBalance || it.supportsRetrievingAccountTransactions }
|
||||||
|
@ -113,7 +110,7 @@ open class FinTsClient(
|
||||||
var countRetrievedAccounts = 0
|
var countRetrievedAccounts = 0
|
||||||
|
|
||||||
if (countAccountsSupportingRetrievingTransactions == 0) {
|
if (countAccountsSupportingRetrievingTransactions == 0) {
|
||||||
addAccountDone(bank, getAccountsResponse, retrievedAccountData, callback)
|
addAccountDone(context, getAccountsResponse, retrievedAccountData, callback)
|
||||||
return // no necessary just to make it clearer that code below doesn't get called
|
return // no necessary just to make it clearer that code below doesn't get called
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -127,17 +124,17 @@ open class FinTsClient(
|
||||||
|
|
||||||
countRetrievedAccounts++
|
countRetrievedAccounts++
|
||||||
if (countRetrievedAccounts == countAccountsSupportingRetrievingTransactions) {
|
if (countRetrievedAccounts == countAccountsSupportingRetrievingTransactions) {
|
||||||
addAccountDone(bank, getAccountsResponse, retrievedAccountData, callback)
|
addAccountDone(context, getAccountsResponse, retrievedAccountData, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun addAccountDone(bank: BankData, getAccountsResponse: BankResponse,
|
protected open fun addAccountDone(context: JobContext, getAccountsResponse: BankResponse,
|
||||||
retrievedAccountData: Map<AccountData, RetrievedAccountData>,
|
retrievedAccountData: Map<AccountData, RetrievedAccountData>,
|
||||||
callback: (AddAccountResponse) -> Unit) {
|
callback: (AddAccountResponse) -> Unit) {
|
||||||
|
|
||||||
callback(AddAccountResponse(getAccountsResponse, bank, retrievedAccountData.values.toList()))
|
callback(AddAccountResponse(getAccountsResponse, context.bank, retrievedAccountData.values.toList()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -158,26 +155,34 @@ open class FinTsClient(
|
||||||
|
|
||||||
open fun getTransactionsAsync(parameter: GetTransactionsParameter, bank: BankData, callback: (GetTransactionsResponse) -> Unit) {
|
open fun getTransactionsAsync(parameter: GetTransactionsParameter, bank: BankData, callback: (GetTransactionsResponse) -> Unit) {
|
||||||
|
|
||||||
jobExecutor.getTransactionsAsync(parameter, bank, callback)
|
val context = JobContext(JobContextType.GetTransactions, this.callback, bank, parameter.account)
|
||||||
|
|
||||||
|
jobExecutor.getTransactionsAsync(context, parameter, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun getTanMediaList(bank: BankData, tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle,
|
open fun getTanMediaList(bank: BankData, tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle,
|
||||||
tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien, callback: (GetTanMediaListResponse) -> Unit) {
|
tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien, callback: (GetTanMediaListResponse) -> Unit) {
|
||||||
|
|
||||||
jobExecutor.getTanMediaList(bank, tanMediaKind, tanMediumClass, callback)
|
val context = JobContext(JobContextType.GetTanMedia, this.callback, bank)
|
||||||
|
|
||||||
|
jobExecutor.getTanMediaList(context, tanMediaKind, tanMediumClass, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun changeTanMedium(newActiveTanMedium: TanGeneratorTanMedium, bank: BankData, callback: (FinTsClientResponse) -> Unit) {
|
open fun changeTanMedium(newActiveTanMedium: TanGeneratorTanMedium, bank: BankData, callback: (FinTsClientResponse) -> Unit) {
|
||||||
jobExecutor.changeTanMedium(newActiveTanMedium, bank) { response ->
|
val context = JobContext(JobContextType.ChangeTanMedium, this.callback, bank)
|
||||||
|
|
||||||
|
jobExecutor.changeTanMedium(context, newActiveTanMedium) { response ->
|
||||||
callback(FinTsClientResponse(response))
|
callback(FinTsClientResponse(response))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun doBankTransferAsync(bankTransferData: BankTransferData, bank: BankData, account: AccountData, callback: (FinTsClientResponse) -> Unit) {
|
open fun doBankTransferAsync(bankTransferData: BankTransferData, bank: BankData, account: AccountData, callback: (FinTsClientResponse) -> Unit) {
|
||||||
jobExecutor.doBankTransferAsync(bankTransferData, bank, account, callback)
|
val context = JobContext(JobContextType.TransferMoney, this.callback, bank, account)
|
||||||
|
|
||||||
|
jobExecutor.doBankTransferAsync(context, bankTransferData, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -32,14 +32,14 @@ open class FinTsClientForCustomer(
|
||||||
: this(bank, callback, RequestExecutor(MessageBuilder(), webClient, base64Service))
|
: this(bank, callback, RequestExecutor(MessageBuilder(), webClient, base64Service))
|
||||||
|
|
||||||
|
|
||||||
protected val client = FinTsClient(FinTsJobExecutor(callback, requestExecutor, messageBuilder, mt940Parser, modelMapper, tanMethodSelector, product))
|
protected val client = FinTsClient(callback, FinTsJobExecutor(requestExecutor, messageBuilder, mt940Parser, modelMapper, tanMethodSelector, product))
|
||||||
|
|
||||||
|
|
||||||
open val messageLogWithoutSensitiveData: List<MessageLogEntry>
|
open val messageLogWithoutSensitiveData: List<MessageLogEntry>
|
||||||
get() = client.messageLogWithoutSensitiveData
|
get() = client.messageLogWithoutSensitiveData
|
||||||
|
|
||||||
open fun setCallback(callback: FinTsClientCallback) {
|
open fun setCallback(callback: FinTsClientCallback) {
|
||||||
client.setCallback(callback)
|
client.callback = callback
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -32,7 +32,6 @@ import net.dankito.utils.multiplatform.ObjectReference
|
||||||
* In almost all cases you want to use [FinTsClient] which wraps these business transactions to a higher level API.
|
* In almost all cases you want to use [FinTsClient] which wraps these business transactions to a higher level API.
|
||||||
*/
|
*/
|
||||||
open class FinTsJobExecutor(
|
open class FinTsJobExecutor(
|
||||||
open var callback: FinTsClientCallback,
|
|
||||||
protected open val requestExecutor: RequestExecutor = RequestExecutor(),
|
protected open val requestExecutor: RequestExecutor = RequestExecutor(),
|
||||||
protected open val messageBuilder: MessageBuilder = MessageBuilder(),
|
protected open val messageBuilder: MessageBuilder = MessageBuilder(),
|
||||||
protected open val mt940Parser: IAccountTransactionsParser = Mt940AccountTransactionsParser(),
|
protected open val mt940Parser: IAccountTransactionsParser = Mt940AccountTransactionsParser(),
|
||||||
|
@ -55,30 +54,31 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun getAnonymousBankInfo(bank: BankData, callback: (BankResponse) -> Unit) {
|
open fun getAnonymousBankInfo(context: JobContext, callback: (BankResponse) -> Unit) {
|
||||||
val dialogContext = DialogContext(bank, product)
|
val dialogContext = DialogContext(context.bank, product)
|
||||||
|
context.startNewDialog(dialogContext)
|
||||||
|
|
||||||
val message = messageBuilder.createAnonymousDialogInitMessage(dialogContext)
|
val message = messageBuilder.createAnonymousDialogInitMessage(context)
|
||||||
|
|
||||||
getAndHandleResponseForMessage(message, dialogContext) { response ->
|
getAndHandleResponseForMessage(context, message) { response ->
|
||||||
if (response.successful) {
|
if (response.successful) {
|
||||||
closeAnonymousDialog(dialogContext, response)
|
closeAnonymousDialog(context, response)
|
||||||
}
|
}
|
||||||
|
|
||||||
callback(response)
|
callback(response)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun closeAnonymousDialog(dialogContext: DialogContext, response: BankResponse) {
|
protected open fun closeAnonymousDialog(context: JobContext, response: BankResponse) {
|
||||||
|
|
||||||
// bank already closed dialog -> there's no need to send dialog end message
|
// bank already closed dialog -> there's no need to send dialog end message
|
||||||
if (dialogContext.closeDialog == false || dialogContext.didBankCloseDialog) {
|
if (context.dialog.closeDialog == false || context.dialog.didBankCloseDialog) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
val dialogEndRequestBody = messageBuilder.createAnonymousDialogEndMessage(dialogContext)
|
val dialogEndRequestBody = messageBuilder.createAnonymousDialogEndMessage(context)
|
||||||
|
|
||||||
fireAndForgetMessage(dialogEndRequestBody, dialogContext)
|
fireAndForgetMessage(context, dialogEndRequestBody)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -89,8 +89,10 @@ open class FinTsJobExecutor(
|
||||||
*
|
*
|
||||||
* Be aware this method resets BPD, UPD and selected TAN method!
|
* Be aware this method resets BPD, UPD and selected TAN method!
|
||||||
*/
|
*/
|
||||||
open fun retrieveBasicDataLikeUsersTanMethods(bank: BankData, preferredTanMethods: List<TanMethodType>? = null, preferredTanMedium: String? = null,
|
open fun retrieveBasicDataLikeUsersTanMethods(context: JobContext, preferredTanMethods: List<TanMethodType>? = null, preferredTanMedium: String? = null,
|
||||||
closeDialog: Boolean = false, callback: (BankResponse) -> Unit) {
|
closeDialog: Boolean = false, callback: (BankResponse) -> Unit) {
|
||||||
|
val bank = context.bank
|
||||||
|
|
||||||
// just to ensure settings are in its initial state and that bank sends us bank parameter (BPD),
|
// 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())
|
// user parameter (UPD) and allowed tan methods for user (therefore the resetSelectedTanMethod())
|
||||||
bank.resetBpdVersion()
|
bank.resetBpdVersion()
|
||||||
|
@ -105,21 +107,22 @@ open class FinTsJobExecutor(
|
||||||
|
|
||||||
// this is the only case where Einschritt-TAN-Verfahren is accepted: to get user's TAN methods
|
// 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 dialogContext = DialogContext(bank, product, closeDialog, versionOfSecurityMethod = VersionDesSicherheitsverfahrens.Version_1)
|
||||||
|
context.startNewDialog(dialogContext)
|
||||||
|
|
||||||
val message = messageBuilder.createInitDialogMessage(dialogContext)
|
val message = messageBuilder.createInitDialogMessage(context)
|
||||||
|
|
||||||
getAndHandleResponseForMessage(message, dialogContext) { response ->
|
getAndHandleResponseForMessage(context, message) { response ->
|
||||||
closeDialog(dialogContext)
|
closeDialog(context)
|
||||||
|
|
||||||
handleGetUsersTanMethodsResponse(response, dialogContext) { getTanMethodsResponse ->
|
handleGetUsersTanMethodsResponse(context, response) { getTanMethodsResponse ->
|
||||||
if (bank.tanMethodsAvailableForUser.isEmpty()) { // could not retrieve supported tan methods for user
|
if (bank.tanMethodsAvailableForUser.isEmpty()) { // could not retrieve supported tan methods for user
|
||||||
callback(getTanMethodsResponse)
|
callback(getTanMethodsResponse)
|
||||||
} else {
|
} else {
|
||||||
getUsersTanMethod(bank, preferredTanMethods) {
|
getUsersTanMethod(context, preferredTanMethods) {
|
||||||
if (bank.isTanMethodSelected == false) {
|
if (bank.isTanMethodSelected == false) {
|
||||||
callback(getTanMethodsResponse)
|
callback(getTanMethodsResponse)
|
||||||
} else if (bank.tanMedia.isEmpty() && isJobSupported(bank, CustomerSegmentId.TanMediaList)) { // tan media not retrieved yet
|
} else if (bank.tanMedia.isEmpty() && isJobSupported(bank, CustomerSegmentId.TanMediaList)) { // tan media not retrieved yet
|
||||||
getTanMediaList(bank, TanMedienArtVersion.Alle, TanMediumKlasse.AlleMedien, preferredTanMedium) {
|
getTanMediaList(context, TanMedienArtVersion.Alle, TanMediumKlasse.AlleMedien, preferredTanMedium) {
|
||||||
callback(getTanMethodsResponse) // TODO: judge if bank requires selecting TAN media and if though evaluate getTanMediaListResponse
|
callback(getTanMethodsResponse) // TODO: judge if bank requires selecting TAN media and if though evaluate getTanMediaListResponse
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -131,12 +134,12 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun handleGetUsersTanMethodsResponse(response: BankResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
protected open fun handleGetUsersTanMethodsResponse(context: JobContext, response: BankResponse, callback: (BankResponse) -> Unit) {
|
||||||
val getUsersTanMethodsResponse = GetUserTanMethodsResponse(response)
|
val getUsersTanMethodsResponse = GetUserTanMethodsResponse(response)
|
||||||
|
|
||||||
// even though it is required by specification some banks don't support retrieving user's TAN method by setting TAN method to '999'
|
// 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)) {
|
if (bankDoesNotSupportRetrievingUsersTanMethods(getUsersTanMethodsResponse)) {
|
||||||
getBankDataForNewUserViaAnonymousDialog(dialogContext.bank, callback) // TODO: should not be necessary anymore
|
getBankDataForNewUserViaAnonymousDialog(context, callback) // TODO: should not be necessary anymore
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
callback(getUsersTanMethodsResponse)
|
callback(getUsersTanMethodsResponse)
|
||||||
|
@ -150,8 +153,10 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
|
|
||||||
// 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 getBankDataForNewUserViaAnonymousDialog(bank: BankData, callback: (BankResponse) -> Unit) {
|
protected open fun getBankDataForNewUserViaAnonymousDialog(context: JobContext, callback: (BankResponse) -> Unit) {
|
||||||
getAnonymousBankInfo(bank) { anonymousBankInfoResponse ->
|
getAnonymousBankInfo(context) { anonymousBankInfoResponse ->
|
||||||
|
val bank = context.bank
|
||||||
|
|
||||||
if (anonymousBankInfoResponse.successful == false) {
|
if (anonymousBankInfoResponse.successful == false) {
|
||||||
callback(anonymousBankInfoResponse)
|
callback(anonymousBankInfoResponse)
|
||||||
} else if (bank.tanMethodsSupportedByBank.isEmpty()) { // should only be a theoretical error
|
} else if (bank.tanMethodsSupportedByBank.isEmpty()) { // should only be a theoretical error
|
||||||
|
@ -159,12 +164,13 @@ open class FinTsJobExecutor(
|
||||||
} else {
|
} else {
|
||||||
bank.tanMethodsAvailableForUser = bank.tanMethodsSupportedByBank
|
bank.tanMethodsAvailableForUser = bank.tanMethodsSupportedByBank
|
||||||
|
|
||||||
getUsersTanMethod(bank) { didSelectTanMethod ->
|
getUsersTanMethod(context) { didSelectTanMethod ->
|
||||||
if (didSelectTanMethod) {
|
if (didSelectTanMethod) {
|
||||||
val dialogContext = DialogContext(bank, product)
|
val dialogContext = DialogContext(bank, product)
|
||||||
|
context.startNewDialog(dialogContext)
|
||||||
|
|
||||||
initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext) { initDialogResponse ->
|
initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(context) { initDialogResponse ->
|
||||||
closeDialog(dialogContext)
|
closeDialog(context)
|
||||||
|
|
||||||
callback(initDialogResponse)
|
callback(initDialogResponse)
|
||||||
}
|
}
|
||||||
|
@ -177,48 +183,50 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun getAccounts(bank: BankData, callback: (BankResponse) -> Unit) {
|
open fun getAccounts(context: JobContext, callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
val dialogContext = DialogContext(bank, product, false)
|
val dialogContext = DialogContext(context.bank, product, false)
|
||||||
|
context.startNewDialog(dialogContext)
|
||||||
|
|
||||||
initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext) { response ->
|
initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(context) { response ->
|
||||||
closeDialog(dialogContext)
|
closeDialog(context)
|
||||||
|
|
||||||
callback(response)
|
callback(response)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun getTransactionsAsync(parameter: GetTransactionsParameter, bank: BankData, callback: (GetTransactionsResponse) -> Unit) {
|
open fun getTransactionsAsync(context: JobContext, parameter: GetTransactionsParameter, callback: (GetTransactionsResponse) -> Unit) {
|
||||||
|
|
||||||
val dialogContext = DialogContext(bank, product)
|
val dialogContext = DialogContext(context.bank, product)
|
||||||
|
context.startNewDialog(dialogContext)
|
||||||
|
|
||||||
initDialogWithStrongCustomerAuthentication(dialogContext) { initDialogResponse ->
|
initDialogWithStrongCustomerAuthentication(context) { initDialogResponse ->
|
||||||
|
|
||||||
if (initDialogResponse.successful == false) {
|
if (initDialogResponse.successful == false) {
|
||||||
callback(GetTransactionsResponse(initDialogResponse, RetrievedAccountData.unsuccessfulList(parameter.account)))
|
callback(GetTransactionsResponse(initDialogResponse, RetrievedAccountData.unsuccessfulList(parameter.account)))
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// we now retrieved the fresh account information from FinTS server, use that one
|
// we now retrieved the fresh account information from FinTS server, use that one
|
||||||
parameter.account = getUpdatedAccount(bank, parameter.account)
|
parameter.account = getUpdatedAccount(context, parameter.account)
|
||||||
|
|
||||||
mayGetBalance(parameter, dialogContext) { balanceResponse ->
|
mayGetBalance(context, parameter) { balanceResponse ->
|
||||||
if (dialogContext.didBankCloseDialog) {
|
if (dialogContext.didBankCloseDialog) {
|
||||||
callback(GetTransactionsResponse(balanceResponse ?: initDialogResponse, RetrievedAccountData.unsuccessfulList(parameter.account)))
|
callback(GetTransactionsResponse(balanceResponse ?: initDialogResponse, RetrievedAccountData.unsuccessfulList(parameter.account)))
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
getTransactionsAfterInitAndGetBalance(parameter, dialogContext, balanceResponse, callback)
|
getTransactionsAfterInitAndGetBalance(context, parameter, balanceResponse, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun getUpdatedAccount(bank: BankData, account: AccountData): AccountData {
|
private fun getUpdatedAccount(context: JobContext, account: AccountData): AccountData {
|
||||||
return bank.accounts.firstOrNull { it.accountIdentifier == account.accountIdentifier } ?: account
|
return context.bank.accounts.firstOrNull { it.accountIdentifier == account.accountIdentifier } ?: account
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun getTransactionsAfterInitAndGetBalance(parameter: GetTransactionsParameter, dialogContext: DialogContext,
|
protected open fun getTransactionsAfterInitAndGetBalance(context: JobContext, parameter: GetTransactionsParameter,
|
||||||
balanceResponse: BankResponse?, callback: (GetTransactionsResponse) -> Unit) {
|
balanceResponse: BankResponse?, callback: (GetTransactionsResponse) -> Unit) {
|
||||||
var balance: Money? = balanceResponse?.getFirstSegmentById<BalanceSegment>(InstituteSegmentId.Balance)?.let {
|
var balance: Money? = balanceResponse?.getFirstSegmentById<BalanceSegment>(InstituteSegmentId.Balance)?.let {
|
||||||
Money(it.balance, it.currency)
|
Money(it.balance, it.currency)
|
||||||
|
@ -226,16 +234,16 @@ open class FinTsJobExecutor(
|
||||||
val bookedTransactions = mutableSetOf<AccountTransaction>()
|
val bookedTransactions = mutableSetOf<AccountTransaction>()
|
||||||
val unbookedTransactions = mutableSetOf<Any>()
|
val unbookedTransactions = mutableSetOf<Any>()
|
||||||
|
|
||||||
val message = messageBuilder.createGetTransactionsMessage(parameter, dialogContext)
|
val message = messageBuilder.createGetTransactionsMessage(context, parameter)
|
||||||
|
|
||||||
var remainingMt940String = ""
|
var remainingMt940String = ""
|
||||||
|
|
||||||
dialogContext.abortIfTanIsRequired = parameter.abortIfTanIsRequired
|
context.dialog.abortIfTanIsRequired = parameter.abortIfTanIsRequired
|
||||||
|
|
||||||
dialogContext.chunkedResponseHandler = { response ->
|
context.dialog.chunkedResponseHandler = { response ->
|
||||||
response.getFirstSegmentById<ReceivedAccountTransactions>(InstituteSegmentId.AccountTransactionsMt940)?.let { transactionsSegment ->
|
response.getFirstSegmentById<ReceivedAccountTransactions>(InstituteSegmentId.AccountTransactionsMt940)?.let { transactionsSegment ->
|
||||||
val (chunkTransaction, remainder) = mt940Parser.parseTransactionsChunk(remainingMt940String + transactionsSegment.bookedTransactionsString,
|
val (chunkTransaction, remainder) = mt940Parser.parseTransactionsChunk(remainingMt940String + transactionsSegment.bookedTransactionsString,
|
||||||
dialogContext.bank, parameter.account)
|
context.bank, parameter.account)
|
||||||
|
|
||||||
bookedTransactions.addAll(chunkTransaction)
|
bookedTransactions.addAll(chunkTransaction)
|
||||||
remainingMt940String = remainder
|
remainingMt940String = remainder
|
||||||
|
@ -249,8 +257,8 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
getAndHandleResponseForMessage(message, dialogContext) { response ->
|
getAndHandleResponseForMessage(context, message) { response ->
|
||||||
closeDialog(dialogContext)
|
closeDialog(context)
|
||||||
|
|
||||||
val successful = response.tanRequiredButWeWereToldToAbortIfSo
|
val successful = response.tanRequiredButWeWereToldToAbortIfSo
|
||||||
|| (response.successful && (parameter.alsoRetrieveBalance == false || balance != null))
|
|| (response.successful && (parameter.alsoRetrieveBalance == false || balance != null))
|
||||||
|
@ -267,13 +275,11 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun mayGetBalance(parameter: GetTransactionsParameter, dialogContext: DialogContext, callback: (BankResponse?) -> Unit) {
|
protected open fun mayGetBalance(context: JobContext, parameter: GetTransactionsParameter, callback: (BankResponse?) -> Unit) {
|
||||||
if (parameter.alsoRetrieveBalance && parameter.account.supportsRetrievingBalance) {
|
if (parameter.alsoRetrieveBalance && parameter.account.supportsRetrievingBalance) {
|
||||||
val message = messageBuilder.createGetBalanceMessage(parameter.account, dialogContext)
|
val message = messageBuilder.createGetBalanceMessage(context, parameter.account)
|
||||||
|
|
||||||
getAndHandleResponseForMessage(message, dialogContext) { response ->
|
getAndHandleResponseForMessage(context, message, callback)
|
||||||
callback(response)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
callback(null)
|
callback(null)
|
||||||
|
@ -292,14 +298,16 @@ open class FinTsJobExecutor(
|
||||||
*
|
*
|
||||||
* 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, callback: (FinTsClientResponse) -> Unit) {
|
protected open fun synchronizeCustomerSystemId(context: JobContext, callback: (FinTsClientResponse) -> Unit) {
|
||||||
|
|
||||||
val dialogContext = DialogContext(bank, product)
|
val dialogContext = DialogContext(context.bank, product)
|
||||||
val message = messageBuilder.createSynchronizeCustomerSystemIdMessage(dialogContext)
|
context.startNewDialog(dialogContext)
|
||||||
|
|
||||||
getAndHandleResponseForMessage(message, dialogContext) { response ->
|
val message = messageBuilder.createSynchronizeCustomerSystemIdMessage(context)
|
||||||
|
|
||||||
|
getAndHandleResponseForMessage(context, message) { response ->
|
||||||
if (response.successful) {
|
if (response.successful) {
|
||||||
closeDialog(dialogContext)
|
closeDialog(context)
|
||||||
}
|
}
|
||||||
|
|
||||||
callback(FinTsClientResponse(response))
|
callback(FinTsClientResponse(response))
|
||||||
|
@ -307,18 +315,18 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun getTanMediaList(bank: BankData, tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle, tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien,
|
open fun getTanMediaList(context: JobContext, tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle, tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien,
|
||||||
callback: (GetTanMediaListResponse) -> Unit) {
|
callback: (GetTanMediaListResponse) -> Unit) {
|
||||||
getTanMediaList(bank, tanMediaKind, tanMediumClass, null, callback)
|
getTanMediaList(context, tanMediaKind, tanMediumClass, null, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun getTanMediaList(bank: BankData, tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle, tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien,
|
protected open fun getTanMediaList(context: JobContext, tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle, tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien,
|
||||||
preferredTanMedium: String? = null, callback: (GetTanMediaListResponse) -> Unit) {
|
preferredTanMedium: String? = null, callback: (GetTanMediaListResponse) -> Unit) {
|
||||||
|
|
||||||
sendMessageAndHandleResponse(bank, CustomerSegmentId.TanMediaList, false, { dialogContext ->
|
sendMessageAndHandleResponse(context, CustomerSegmentId.TanMediaList, false, {
|
||||||
messageBuilder.createGetTanMediaListMessage(dialogContext, tanMediaKind, tanMediumClass)
|
messageBuilder.createGetTanMediaListMessage(context, tanMediaKind, tanMediumClass)
|
||||||
}) { response ->
|
}) { response ->
|
||||||
handleGetTanMediaListResponse(response, bank, preferredTanMedium, callback)
|
handleGetTanMediaListResponse(response, context.bank, preferredTanMedium, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -339,80 +347,79 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun changeTanMedium(newActiveTanMedium: TanGeneratorTanMedium, bank: BankData, callback: (BankResponse) -> Unit) {
|
open fun changeTanMedium(context: JobContext, newActiveTanMedium: TanGeneratorTanMedium, callback: (BankResponse) -> Unit) {
|
||||||
|
val bank = context.bank
|
||||||
|
|
||||||
if (bank.changeTanMediumParameters?.enteringAtcAndTanRequired == true) {
|
if (bank.changeTanMediumParameters?.enteringAtcAndTanRequired == true) {
|
||||||
this.callback.enterTanGeneratorAtc(bank, newActiveTanMedium) { enteredAtc ->
|
context.callback.enterTanGeneratorAtc(bank, newActiveTanMedium) { enteredAtc ->
|
||||||
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
|
||||||
callback(BankResponse(false, internalError = message))
|
callback(BankResponse(false, internalError = message))
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
sendChangeTanMediumMessage(bank, newActiveTanMedium, enteredAtc, callback)
|
sendChangeTanMediumMessage(context, newActiveTanMedium, enteredAtc, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
sendChangeTanMediumMessage(bank, newActiveTanMedium, null, callback)
|
sendChangeTanMediumMessage(context, newActiveTanMedium, null, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun sendChangeTanMediumMessage(bank: BankData, newActiveTanMedium: TanGeneratorTanMedium, enteredAtc: EnterTanGeneratorAtcResult?,
|
protected open fun sendChangeTanMediumMessage(context: JobContext, newActiveTanMedium: TanGeneratorTanMedium, enteredAtc: EnterTanGeneratorAtcResult?,
|
||||||
callback: (BankResponse) -> Unit) {
|
callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
sendMessageAndHandleResponse(bank, null, true, { dialogContext ->
|
sendMessageAndHandleResponse(context, null, true, {
|
||||||
messageBuilder.createChangeTanMediumMessage(newActiveTanMedium, dialogContext, enteredAtc?.tan, enteredAtc?.atc)
|
messageBuilder.createChangeTanMediumMessage(context, newActiveTanMedium, enteredAtc?.tan, enteredAtc?.atc)
|
||||||
}) { response ->
|
}, callback)
|
||||||
callback(response)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun doBankTransferAsync(bankTransferData: BankTransferData, bank: BankData, account: AccountData, callback: (FinTsClientResponse) -> Unit) {
|
open fun doBankTransferAsync(context: JobContext, bankTransferData: BankTransferData, callback: (FinTsClientResponse) -> Unit) {
|
||||||
|
|
||||||
sendMessageAndHandleResponse(bank, null, true, { dialogContext ->
|
sendMessageAndHandleResponse(context, null, true, {
|
||||||
val updatedAccount = getUpdatedAccount(bank, account)
|
val updatedAccount = getUpdatedAccount(context, context.account!!)
|
||||||
messageBuilder.createBankTransferMessage(bankTransferData, updatedAccount, dialogContext)
|
messageBuilder.createBankTransferMessage(context, bankTransferData, updatedAccount)
|
||||||
}) { response ->
|
}) { response ->
|
||||||
callback(FinTsClientResponse(response))
|
callback(FinTsClientResponse(response))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected open fun getAndHandleResponseForMessage(message: MessageBuilderResult, dialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
protected open fun getAndHandleResponseForMessage(context: JobContext, message: MessageBuilderResult, callback: (BankResponse) -> Unit) {
|
||||||
requestExecutor.getAndHandleResponseForMessage(message, dialogContext,
|
requestExecutor.getAndHandleResponseForMessage(message, context,
|
||||||
{ tanResponse, bankResponse, tanRequiredCallback ->
|
{ tanResponse, bankResponse, tanRequiredCallback ->
|
||||||
// if we receive a message that tells us a TAN is required below callback doesn't get called for that message -> update data here
|
// if we receive a message that tells us a TAN is required below callback doesn't get called for that message -> update data here
|
||||||
// for Hypovereinsbank it's absolutely necessary to update bank data (more specific: PinInfo / HIPINS) after first strong authentication dialog init response
|
// for Hypovereinsbank it's absolutely necessary to update bank data (more specific: PinInfo / HIPINS) after first strong authentication dialog init response
|
||||||
// as HIPINS differ in anonymous and in authenticated dialog. Anonymous dialog tells us for HKSAL and HKKAZ no TAN is needed
|
// as HIPINS differ in anonymous and in authenticated dialog. Anonymous dialog tells us for HKSAL and HKKAZ no TAN is needed
|
||||||
updateBankAndCustomerDataIfResponseSuccessful(dialogContext, bankResponse)
|
updateBankAndCustomerDataIfResponseSuccessful(context, bankResponse)
|
||||||
handleEnteringTanRequired(tanResponse, bankResponse, dialogContext, tanRequiredCallback)
|
handleEnteringTanRequired(context, tanResponse, bankResponse, tanRequiredCallback)
|
||||||
}) { response ->
|
}) { response ->
|
||||||
// TODO: really update data only on complete successfully response? as it may contain useful information anyway // TODO: extract method for this code part
|
// TODO: really update data only on complete successfully response? as it may contain useful information anyway // TODO: extract method for this code part
|
||||||
updateBankAndCustomerDataIfResponseSuccessful(dialogContext, response)
|
updateBankAndCustomerDataIfResponseSuccessful(context, response)
|
||||||
|
|
||||||
callback(response)
|
callback(response)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun fireAndForgetMessage(message: MessageBuilderResult, dialogContext: DialogContext) {
|
protected open fun fireAndForgetMessage(context: JobContext, message: MessageBuilderResult) {
|
||||||
requestExecutor.fireAndForgetMessage(message, dialogContext)
|
requestExecutor.fireAndForgetMessage(context, message)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected open fun handleEnteringTanRequired(tanResponse: TanResponse, response: BankResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
protected open fun handleEnteringTanRequired(context: JobContext, tanResponse: TanResponse, response: BankResponse, callback: (BankResponse) -> Unit) {
|
||||||
val bank = dialogContext.bank // TODO: copy required data to TanChallenge
|
val bank = context.bank // TODO: copy required data to TanChallenge
|
||||||
val tanChallenge = createTanChallenge(tanResponse, bank)
|
val tanChallenge = createTanChallenge(tanResponse, bank)
|
||||||
|
|
||||||
val userDidCancelEnteringTan = ObjectReference(false)
|
val userDidCancelEnteringTan = ObjectReference(false)
|
||||||
|
|
||||||
this.callback.enterTan(bank, tanChallenge) { enteredTanResult ->
|
context.callback.enterTan(bank, tanChallenge) { enteredTanResult ->
|
||||||
userDidCancelEnteringTan.value = true
|
userDidCancelEnteringTan.value = true
|
||||||
|
|
||||||
handleEnterTanResult(enteredTanResult, tanResponse, response, dialogContext, callback)
|
handleEnterTanResult(context, enteredTanResult, tanResponse, response, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
mayRetrieveAutomaticallyIfUserEnteredDecoupledTan(tanChallenge, tanResponse, userDidCancelEnteringTan, dialogContext)
|
mayRetrieveAutomaticallyIfUserEnteredDecoupledTan(context, tanChallenge, tanResponse, userDidCancelEnteringTan)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun createTanChallenge(tanResponse: TanResponse, bank: BankData): TanChallenge {
|
protected open fun createTanChallenge(tanResponse: TanResponse, bank: BankData): TanChallenge {
|
||||||
|
@ -435,28 +442,28 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun mayRetrieveAutomaticallyIfUserEnteredDecoupledTan(tanChallenge: TanChallenge, tanResponse: TanResponse,
|
protected open fun mayRetrieveAutomaticallyIfUserEnteredDecoupledTan(context: JobContext, tanChallenge: TanChallenge, tanResponse: TanResponse,
|
||||||
userDidCancelEnteringTan: ObjectReference<Boolean>, dialogContext: DialogContext
|
userDidCancelEnteringTan: ObjectReference<Boolean>
|
||||||
) {
|
) {
|
||||||
dialogContext.bank.selectedTanMethod.decoupledParameters?.let { decoupledTanMethodParameters ->
|
context.bank.selectedTanMethod.decoupledParameters?.let { decoupledTanMethodParameters ->
|
||||||
if (tanResponse.tanProcess == TanProcess.AppTan && decoupledTanMethodParameters.periodicStateRequestsAllowed) {
|
if (tanResponse.tanProcess == TanProcess.AppTan && decoupledTanMethodParameters.periodicStateRequestsAllowed) {
|
||||||
automaticallyRetrieveIfUserEnteredDecoupledTan(tanChallenge, userDidCancelEnteringTan, dialogContext)
|
automaticallyRetrieveIfUserEnteredDecoupledTan(context, tanChallenge, userDidCancelEnteringTan)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun automaticallyRetrieveIfUserEnteredDecoupledTan(tanChallenge: TanChallenge, userDidCancelEnteringTan: ObjectReference<Boolean>, dialogContext: DialogContext) {
|
protected open fun automaticallyRetrieveIfUserEnteredDecoupledTan(context: JobContext, tanChallenge: TanChallenge, userDidCancelEnteringTan: ObjectReference<Boolean>) {
|
||||||
log.info("automaticallyRetrieveIfUserEnteredDecoupledTan() called for $tanChallenge")
|
log.info("automaticallyRetrieveIfUserEnteredDecoupledTan() called for $tanChallenge")
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun handleEnterTanResult(enteredTanResult: EnterTanResult, tanResponse: TanResponse, response: BankResponse,
|
protected open fun handleEnterTanResult(context: JobContext, enteredTanResult: EnterTanResult, tanResponse: TanResponse,
|
||||||
dialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
response: BankResponse, callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
if (enteredTanResult.changeTanMethodTo != null) {
|
if (enteredTanResult.changeTanMethodTo != null) {
|
||||||
handleUserAsksToChangeTanMethodAndResendLastMessage(enteredTanResult.changeTanMethodTo, dialogContext, callback)
|
handleUserAsksToChangeTanMethodAndResendLastMessage(context, enteredTanResult.changeTanMethodTo, callback)
|
||||||
}
|
}
|
||||||
else if (enteredTanResult.changeTanMediumTo is TanGeneratorTanMedium) {
|
else if (enteredTanResult.changeTanMediumTo is TanGeneratorTanMedium) {
|
||||||
handleUserAsksToChangeTanMediumAndResendLastMessage(enteredTanResult.changeTanMediumTo, dialogContext,
|
handleUserAsksToChangeTanMediumAndResendLastMessage(context, enteredTanResult.changeTanMediumTo,
|
||||||
enteredTanResult.changeTanMediumResultCallback, callback)
|
enteredTanResult.changeTanMediumResultCallback, callback)
|
||||||
}
|
}
|
||||||
else if (enteredTanResult.enteredTan == null) {
|
else if (enteredTanResult.enteredTan == null) {
|
||||||
|
@ -467,66 +474,67 @@ open class FinTsJobExecutor(
|
||||||
callback(response)
|
callback(response)
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
sendTanToBank(enteredTanResult.enteredTan, tanResponse, dialogContext, callback)
|
sendTanToBank(context, enteredTanResult.enteredTan, tanResponse, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun sendTanToBank(enteredTan: String, tanResponse: TanResponse, dialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
protected open fun sendTanToBank(context: JobContext, enteredTan: String, tanResponse: TanResponse, callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
val message = messageBuilder.createSendEnteredTanMessage(enteredTan, tanResponse, dialogContext)
|
val message = messageBuilder.createSendEnteredTanMessage(context, enteredTan, tanResponse)
|
||||||
|
|
||||||
getAndHandleResponseForMessage(message, dialogContext, callback)
|
getAndHandleResponseForMessage(context, message, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun handleUserAsksToChangeTanMethodAndResendLastMessage(changeTanMethodTo: TanMethod, dialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
protected open fun handleUserAsksToChangeTanMethodAndResendLastMessage(context: JobContext, changeTanMethodTo: TanMethod, callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
dialogContext.bank.selectedTanMethod = changeTanMethodTo
|
context.bank.selectedTanMethod = changeTanMethodTo
|
||||||
|
|
||||||
|
|
||||||
val lastCreatedMessage = dialogContext.currentMessage
|
val lastCreatedMessage = context.dialog.currentMessage
|
||||||
|
|
||||||
lastCreatedMessage?.let { closeDialog(dialogContext) }
|
lastCreatedMessage?.let { closeDialog(context) }
|
||||||
|
|
||||||
resendMessageInNewDialog(lastCreatedMessage, dialogContext, callback)
|
resendMessageInNewDialog(context, lastCreatedMessage, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun handleUserAsksToChangeTanMediumAndResendLastMessage(changeTanMediumTo: TanGeneratorTanMedium,
|
protected open fun handleUserAsksToChangeTanMediumAndResendLastMessage(context: JobContext, changeTanMediumTo: TanGeneratorTanMedium,
|
||||||
dialogContext: DialogContext,
|
|
||||||
changeTanMediumResultCallback: ((FinTsClientResponse) -> Unit)?,
|
changeTanMediumResultCallback: ((FinTsClientResponse) -> Unit)?,
|
||||||
callback: (BankResponse) -> Unit) {
|
callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
val lastCreatedMessage = dialogContext.currentMessage
|
val lastCreatedMessage = context.dialog.currentMessage
|
||||||
|
|
||||||
lastCreatedMessage?.let { closeDialog(dialogContext) }
|
lastCreatedMessage?.let { closeDialog(context) }
|
||||||
|
|
||||||
|
|
||||||
changeTanMedium(changeTanMediumTo, dialogContext.bank) { changeTanMediumResponse ->
|
changeTanMedium(context, changeTanMediumTo) { changeTanMediumResponse ->
|
||||||
changeTanMediumResultCallback?.invoke(FinTsClientResponse(changeTanMediumResponse))
|
changeTanMediumResultCallback?.invoke(FinTsClientResponse(changeTanMediumResponse))
|
||||||
|
|
||||||
if (changeTanMediumResponse.successful == false || lastCreatedMessage == null) {
|
if (changeTanMediumResponse.successful == false || lastCreatedMessage == null) {
|
||||||
callback(changeTanMediumResponse)
|
callback(changeTanMediumResponse)
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
resendMessageInNewDialog(lastCreatedMessage, dialogContext, callback)
|
resendMessageInNewDialog(context, lastCreatedMessage, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected open fun resendMessageInNewDialog(lastCreatedMessage: MessageBuilderResult?, previousDialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
protected open fun resendMessageInNewDialog(context: JobContext, lastCreatedMessage: MessageBuilderResult?, callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
if (lastCreatedMessage != null) { // do not use previousDialogContext.currentMessage as this may is previous dialog's dialog close message
|
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)
|
val previousDialog = context.dialog
|
||||||
|
val newDialogContext = DialogContext(context.bank, previousDialog.product, chunkedResponseHandler = previousDialog.chunkedResponseHandler)
|
||||||
|
context.startNewDialog(newDialogContext)
|
||||||
|
|
||||||
initDialogWithStrongCustomerAuthentication(newDialogContext) { initDialogResponse ->
|
initDialogWithStrongCustomerAuthentication(context) { initDialogResponse ->
|
||||||
if (initDialogResponse.successful == false) {
|
if (initDialogResponse.successful == false) {
|
||||||
callback(initDialogResponse)
|
callback(initDialogResponse)
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
val newMessage = messageBuilder.rebuildMessage(lastCreatedMessage, newDialogContext)
|
val newMessage = messageBuilder.rebuildMessage(context, lastCreatedMessage)
|
||||||
|
|
||||||
getAndHandleResponseForMessage(newMessage, newDialogContext) { response ->
|
getAndHandleResponseForMessage(context, newMessage) { response ->
|
||||||
closeDialog(newDialogContext)
|
closeDialog(context)
|
||||||
|
|
||||||
callback(response)
|
callback(response)
|
||||||
}
|
}
|
||||||
|
@ -540,91 +548,95 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected open fun sendMessageAndHandleResponse(bank: BankData, segmentForNonStrongCustomerAuthenticationTwoStepTanProcess: CustomerSegmentId? = null,
|
protected open fun sendMessageAndHandleResponse(context: JobContext, segmentForNonStrongCustomerAuthenticationTwoStepTanProcess: CustomerSegmentId? = null,
|
||||||
closeDialog: Boolean = true, createMessage: (DialogContext) -> MessageBuilderResult, callback: (BankResponse) -> Unit) {
|
closeDialog: Boolean = true, createMessage: () -> MessageBuilderResult, callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
val dialogContext = DialogContext(bank, product, closeDialog)
|
val dialogContext = DialogContext(context.bank, product, closeDialog)
|
||||||
|
context.startNewDialog(dialogContext)
|
||||||
|
|
||||||
if (segmentForNonStrongCustomerAuthenticationTwoStepTanProcess == null) {
|
if (segmentForNonStrongCustomerAuthenticationTwoStepTanProcess == null) {
|
||||||
initDialogWithStrongCustomerAuthentication(dialogContext) { initDialogResponse ->
|
initDialogWithStrongCustomerAuthentication(context) { initDialogResponse ->
|
||||||
sendMessageAndHandleResponseAfterDialogInitialization(dialogContext, initDialogResponse, createMessage, callback)
|
sendMessageAndHandleResponseAfterDialogInitialization(context, initDialogResponse, createMessage, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
initDialogMessageWithoutStrongCustomerAuthenticationAfterSuccessfulChecks(dialogContext, segmentForNonStrongCustomerAuthenticationTwoStepTanProcess) { initDialogResponse ->
|
initDialogMessageWithoutStrongCustomerAuthenticationAfterSuccessfulChecks(context, segmentForNonStrongCustomerAuthenticationTwoStepTanProcess) { initDialogResponse ->
|
||||||
sendMessageAndHandleResponseAfterDialogInitialization(dialogContext, initDialogResponse, createMessage, callback)
|
sendMessageAndHandleResponseAfterDialogInitialization(context, initDialogResponse, createMessage, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun sendMessageAndHandleResponseAfterDialogInitialization(dialogContext: DialogContext, initDialogResponse: BankResponse, createMessage: (DialogContext) -> MessageBuilderResult, callback: (BankResponse) -> Unit) {
|
private fun sendMessageAndHandleResponseAfterDialogInitialization(context: JobContext, initDialogResponse: BankResponse,
|
||||||
|
createMessage: () -> MessageBuilderResult, callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
if (initDialogResponse.successful == false) {
|
if (initDialogResponse.successful == false) {
|
||||||
callback(initDialogResponse)
|
callback(initDialogResponse)
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
val message = createMessage(dialogContext)
|
val message = createMessage()
|
||||||
|
|
||||||
getAndHandleResponseForMessage(message, dialogContext) { response ->
|
getAndHandleResponseForMessage(context, message) { response ->
|
||||||
closeDialog(dialogContext)
|
closeDialog(context)
|
||||||
|
|
||||||
callback(response)
|
callback(response)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun initDialogWithStrongCustomerAuthentication(dialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
protected open fun initDialogWithStrongCustomerAuthentication(context: JobContext, callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
// we first need to retrieve supported tan methods and jobs before we can do anything
|
// we first need to retrieve supported tan methods and jobs before we can do anything
|
||||||
ensureBasicBankDataRetrieved(dialogContext.bank) { retrieveBasicBankDataResponse ->
|
ensureBasicBankDataRetrieved(context) { retrieveBasicBankDataResponse ->
|
||||||
if (retrieveBasicBankDataResponse.successful == false) {
|
if (retrieveBasicBankDataResponse.successful == false) {
|
||||||
callback(retrieveBasicBankDataResponse)
|
callback(retrieveBasicBankDataResponse)
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// as in the next step we have to supply user's tan method, ensure user selected his or her
|
// as in the next step we have to supply user's tan method, ensure user selected his or her
|
||||||
ensureTanMethodIsSelected(dialogContext.bank) { tanMethodSelectedResponse ->
|
ensureTanMethodIsSelected(context) { tanMethodSelectedResponse ->
|
||||||
if (tanMethodSelectedResponse.successful == false) {
|
if (tanMethodSelectedResponse.successful == false) {
|
||||||
callback(tanMethodSelectedResponse)
|
callback(tanMethodSelectedResponse)
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext, callback)
|
initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(context, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(dialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
protected open fun initDialogWithStrongCustomerAuthenticationAfterSuccessfulPreconditionChecks(context: JobContext, callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
val message = messageBuilder.createInitDialogMessage(dialogContext)
|
val message = messageBuilder.createInitDialogMessage(context)
|
||||||
|
|
||||||
getAndHandleResponseForMessage(message, dialogContext, callback)
|
getAndHandleResponseForMessage(context, message, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun initDialogMessageWithoutStrongCustomerAuthenticationAfterSuccessfulChecks(dialogContext: DialogContext, segmentIdForTwoStepTanProcess: CustomerSegmentId?,
|
protected open fun initDialogMessageWithoutStrongCustomerAuthenticationAfterSuccessfulChecks(context: JobContext, segmentIdForTwoStepTanProcess: CustomerSegmentId?,
|
||||||
callback: (BankResponse) -> Unit) {
|
callback: (BankResponse) -> Unit) {
|
||||||
|
|
||||||
val message = messageBuilder.createInitDialogMessageWithoutStrongCustomerAuthentication(dialogContext, segmentIdForTwoStepTanProcess)
|
val message = messageBuilder.createInitDialogMessageWithoutStrongCustomerAuthentication(context, segmentIdForTwoStepTanProcess)
|
||||||
|
|
||||||
getAndHandleResponseForMessage(message, dialogContext, callback)
|
getAndHandleResponseForMessage(context, message, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun closeDialog(dialogContext: DialogContext) {
|
protected open fun closeDialog(context: JobContext) {
|
||||||
|
|
||||||
// bank already closed dialog -> there's no need to send dialog end message
|
// bank already closed dialog -> there's no need to send dialog end message
|
||||||
if (dialogContext.closeDialog == false || dialogContext.didBankCloseDialog) {
|
if (context.dialog.closeDialog == false || context.dialog.didBankCloseDialog) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
val dialogEndRequestBody = messageBuilder.createDialogEndMessage(dialogContext)
|
val dialogEndRequestBody = messageBuilder.createDialogEndMessage(context)
|
||||||
|
|
||||||
fireAndForgetMessage(dialogEndRequestBody, dialogContext)
|
fireAndForgetMessage(context, dialogEndRequestBody)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected open fun ensureBasicBankDataRetrieved(bank: BankData, callback: (BankResponse) -> Unit) {
|
protected open fun ensureBasicBankDataRetrieved(context: JobContext, callback: (BankResponse) -> Unit) {
|
||||||
|
val bank = context.bank
|
||||||
|
|
||||||
if (bank.tanMethodsSupportedByBank.isEmpty() || bank.supportedJobs.isEmpty()) {
|
if (bank.tanMethodsSupportedByBank.isEmpty() || bank.supportedJobs.isEmpty()) {
|
||||||
retrieveBasicDataLikeUsersTanMethods(bank) { getBankInfoResponse ->
|
retrieveBasicDataLikeUsersTanMethods(context) { getBankInfoResponse ->
|
||||||
if (getBankInfoResponse.successful == false) {
|
if (getBankInfoResponse.successful == false) {
|
||||||
callback(getBankInfoResponse)
|
callback(getBankInfoResponse)
|
||||||
} else if (bank.tanMethodsSupportedByBank.isEmpty() || bank.supportedJobs.isEmpty()) {
|
} else if (bank.tanMethodsSupportedByBank.isEmpty() || bank.supportedJobs.isEmpty()) {
|
||||||
|
@ -640,15 +652,17 @@ open class FinTsJobExecutor(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun ensureTanMethodIsSelected(bank: BankData, callback: (BankResponse) -> Unit) {
|
protected open fun ensureTanMethodIsSelected(context: JobContext, callback: (BankResponse) -> Unit) {
|
||||||
|
val bank = context.bank
|
||||||
|
|
||||||
if (bank.isTanMethodSelected == false) {
|
if (bank.isTanMethodSelected == false) {
|
||||||
if (bank.tanMethodsAvailableForUser.isEmpty()) {
|
if (bank.tanMethodsAvailableForUser.isEmpty()) {
|
||||||
retrieveBasicDataLikeUsersTanMethods(bank) { retrieveBasicDataResponse ->
|
retrieveBasicDataLikeUsersTanMethods(context) { retrieveBasicDataResponse ->
|
||||||
callback(retrieveBasicDataResponse)
|
callback(retrieveBasicDataResponse)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
getUsersTanMethod(bank) {
|
getUsersTanMethod(context) {
|
||||||
callback(createNoTanMethodSelectedResponse(bank))
|
callback(createNoTanMethodSelectedResponse(bank))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -665,7 +679,9 @@ open class FinTsJobExecutor(
|
||||||
return BankResponse(true, noTanMethodSelected = noTanMethodSelected, internalError = errorMessage)
|
return BankResponse(true, noTanMethodSelected = noTanMethodSelected, internalError = errorMessage)
|
||||||
}
|
}
|
||||||
|
|
||||||
open fun getUsersTanMethod(bank: BankData, preferredTanMethods: List<TanMethodType>? = null, done: (Boolean) -> Unit) {
|
open fun getUsersTanMethod(context: JobContext, preferredTanMethods: List<TanMethodType>? = null, done: (Boolean) -> Unit) {
|
||||||
|
val bank = context.bank
|
||||||
|
|
||||||
if (bank.tanMethodsAvailableForUser.size == 1) { // user has only one TAN method -> set it and we're done
|
if (bank.tanMethodsAvailableForUser.size == 1) { // user has only one TAN method -> set it and we're done
|
||||||
bank.selectedTanMethod = bank.tanMethodsAvailableForUser.first()
|
bank.selectedTanMethod = bank.tanMethodsAvailableForUser.first()
|
||||||
done(true)
|
done(true)
|
||||||
|
@ -679,7 +695,7 @@ open class FinTsJobExecutor(
|
||||||
|
|
||||||
// we know user's supported tan methods, now ask user which one to select
|
// we know user's supported tan methods, now ask user which one to select
|
||||||
val suggestedTanMethod = tanMethodSelector.getSuggestedTanMethod(bank.tanMethodsAvailableForUser)
|
val suggestedTanMethod = tanMethodSelector.getSuggestedTanMethod(bank.tanMethodsAvailableForUser)
|
||||||
callback.askUserForTanMethod(bank.tanMethodsAvailableForUser, suggestedTanMethod) { selectedTanMethod ->
|
context.callback.askUserForTanMethod(bank.tanMethodsAvailableForUser, suggestedTanMethod) { selectedTanMethod ->
|
||||||
if (selectedTanMethod != null) {
|
if (selectedTanMethod != null) {
|
||||||
bank.selectedTanMethod = selectedTanMethod
|
bank.selectedTanMethod = selectedTanMethod
|
||||||
done(true)
|
done(true)
|
||||||
|
@ -696,9 +712,9 @@ open class FinTsJobExecutor(
|
||||||
modelMapper.updateBankData(bank, response)
|
modelMapper.updateBankData(bank, response)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun updateBankAndCustomerDataIfResponseSuccessful(dialogContext: DialogContext, response: BankResponse) {
|
protected open fun updateBankAndCustomerDataIfResponseSuccessful(context: JobContext, response: BankResponse) {
|
||||||
if (response.successful) {
|
if (response.successful) {
|
||||||
updateBankAndCustomerData(dialogContext.bank, response)
|
updateBankAndCustomerData(context.bank, response)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -43,7 +43,7 @@ open class RequestExecutor(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun getAndHandleResponseForMessage(message: MessageBuilderResult, dialogContext: DialogContext,
|
open fun getAndHandleResponseForMessage(message: MessageBuilderResult, context: JobContext,
|
||||||
tanRequiredCallback: (TanResponse, BankResponse, callback: (BankResponse) -> Unit) -> Unit, callback: (BankResponse) -> Unit) {
|
tanRequiredCallback: (TanResponse, BankResponse, callback: (BankResponse) -> Unit) -> Unit, callback: (BankResponse) -> Unit) {
|
||||||
if (message.createdMessage == null) {
|
if (message.createdMessage == null) {
|
||||||
log.error("Could not create FinTS message to be sent to bank. isJobAllowed ${message.isJobAllowed}, isJobVersionSupported = ${message.isJobVersionSupported}," +
|
log.error("Could not create FinTS message to be sent to bank. isJobAllowed ${message.isJobAllowed}, isJobVersionSupported = ${message.isJobVersionSupported}," +
|
||||||
|
@ -51,16 +51,16 @@ open class RequestExecutor(
|
||||||
callback(BankResponse(false, messageThatCouldNotBeCreated = message, internalError = "Could not create FinTS message to be sent to bank")) // TODO: translate
|
callback(BankResponse(false, messageThatCouldNotBeCreated = message, internalError = "Could not create FinTS message to be sent to bank")) // TODO: translate
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
getAndHandleResponseForMessage(message.createdMessage, dialogContext) { response ->
|
getAndHandleResponseForMessage(context, message.createdMessage) { response ->
|
||||||
handleMayRequiresTan(response, dialogContext, tanRequiredCallback) { handledResponse ->
|
handleMayRequiresTan(context, response, tanRequiredCallback) { handledResponse ->
|
||||||
// if there's a Aufsetzpunkt (continuationId) set, then response is not complete yet, there's more information to fetch by sending this Aufsetzpunkt
|
// 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 ->
|
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 (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
|
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)
|
context.dialog.chunkedResponseHandler?.invoke(handledResponse)
|
||||||
}
|
}
|
||||||
|
|
||||||
getFollowUpMessageForContinuationId(handledResponse, continuationId, message, dialogContext, tanRequiredCallback) { followUpResponse ->
|
getFollowUpMessageForContinuationId(context, handledResponse, continuationId, message, tanRequiredCallback) { followUpResponse ->
|
||||||
handledResponse.followUpResponse = followUpResponse
|
handledResponse.followUpResponse = followUpResponse
|
||||||
handledResponse.hasFollowUpMessageButCouldNotReceiveIt = handledResponse.followUpResponse == null
|
handledResponse.hasFollowUpMessageButCouldNotReceiveIt = handledResponse.followUpResponse == null
|
||||||
|
|
||||||
|
@ -75,7 +75,7 @@ open class RequestExecutor(
|
||||||
// e.g. response = enter TAN response, but handledResponse is then response after entering TAN, e.g. account transactions
|
// 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
|
// -> chunkedResponseHandler would get called for same handledResponse multiple times
|
||||||
if (response == handledResponse) {
|
if (response == handledResponse) {
|
||||||
dialogContext.chunkedResponseHandler?.invoke(handledResponse)
|
context.dialog.chunkedResponseHandler?.invoke(handledResponse)
|
||||||
}
|
}
|
||||||
|
|
||||||
callback(handledResponse)
|
callback(handledResponse)
|
||||||
|
@ -85,16 +85,17 @@ open class RequestExecutor(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun getAndHandleResponseForMessage(requestBody: String, dialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
protected open fun getAndHandleResponseForMessage(context: JobContext, requestBody: String, callback: (BankResponse) -> Unit) {
|
||||||
addMessageLog(requestBody, MessageLogEntryType.Sent, dialogContext)
|
addMessageLog(context, MessageLogEntryType.Sent, requestBody)
|
||||||
|
|
||||||
getResponseForMessage(requestBody, dialogContext.bank.finTs3ServerAddress) { webResponse ->
|
getResponseForMessage(requestBody, context.bank.finTs3ServerAddress) { webResponse ->
|
||||||
val response = handleResponse(webResponse, dialogContext)
|
val response = handleResponse(context, webResponse)
|
||||||
|
|
||||||
dialogContext.response = response
|
val dialog = context.dialog
|
||||||
|
dialog.response = response
|
||||||
|
|
||||||
response.messageHeader?.let { header -> dialogContext.dialogId = header.dialogId }
|
response.messageHeader?.let { header -> dialog.dialogId = header.dialogId }
|
||||||
dialogContext.didBankCloseDialog = response.didBankCloseDialog
|
dialog.didBankCloseDialog = response.didBankCloseDialog
|
||||||
|
|
||||||
callback(response)
|
callback(response)
|
||||||
}
|
}
|
||||||
|
@ -106,17 +107,17 @@ open class RequestExecutor(
|
||||||
webClient.post(finTs3ServerAddress, encodedRequestBody, "application/octet-stream", IWebClient.DefaultUserAgent, callback)
|
webClient.post(finTs3ServerAddress, encodedRequestBody, "application/octet-stream", IWebClient.DefaultUserAgent, callback)
|
||||||
}
|
}
|
||||||
|
|
||||||
open fun fireAndForgetMessage(message: MessageBuilderResult, dialogContext: DialogContext) {
|
open fun fireAndForgetMessage(context: JobContext, message: MessageBuilderResult) {
|
||||||
message.createdMessage?.let { requestBody ->
|
message.createdMessage?.let { requestBody ->
|
||||||
addMessageLog(requestBody, MessageLogEntryType.Sent, dialogContext)
|
addMessageLog(context, MessageLogEntryType.Sent, requestBody)
|
||||||
|
|
||||||
getResponseForMessage(requestBody, dialogContext.bank.finTs3ServerAddress) { }
|
getResponseForMessage(requestBody, context.bank.finTs3ServerAddress) { }
|
||||||
|
|
||||||
// if really needed add received response to message log here
|
// if really needed add received response to message log here
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun handleResponse(webResponse: WebClientResponse, dialogContext: DialogContext): BankResponse {
|
protected open fun handleResponse(context: JobContext, webResponse: WebClientResponse): BankResponse {
|
||||||
val responseBody = webResponse.body
|
val responseBody = webResponse.body
|
||||||
|
|
||||||
if (webResponse.successful && responseBody != null) {
|
if (webResponse.successful && responseBody != null) {
|
||||||
|
@ -124,18 +125,18 @@ open class RequestExecutor(
|
||||||
try {
|
try {
|
||||||
val decodedResponse = decodeBase64Response(responseBody)
|
val decodedResponse = decodeBase64Response(responseBody)
|
||||||
|
|
||||||
addMessageLog(decodedResponse, MessageLogEntryType.Received, dialogContext)
|
addMessageLog(context, MessageLogEntryType.Received, decodedResponse)
|
||||||
|
|
||||||
return responseParser.parse(decodedResponse)
|
return responseParser.parse(decodedResponse)
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
logError("Could not decode responseBody:\r\n'$responseBody'", dialogContext, e)
|
logError(context, "Could not decode responseBody:\r\n'$responseBody'", e)
|
||||||
|
|
||||||
return BankResponse(false, internalError = e.getAllExceptionMessagesJoined())
|
return BankResponse(false, internalError = e.getAllExceptionMessagesJoined())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
val bank = dialogContext.bank
|
val bank = context.bank
|
||||||
logError("Request to $bank (${bank.finTs3ServerAddress}) failed", dialogContext, webResponse.error)
|
logError(context, "Request to $bank (${bank.finTs3ServerAddress}) failed", webResponse.error)
|
||||||
}
|
}
|
||||||
|
|
||||||
return BankResponse(false, internalError = webResponse.error?.getAllExceptionMessagesJoined())
|
return BankResponse(false, internalError = webResponse.error?.getAllExceptionMessagesJoined())
|
||||||
|
@ -146,23 +147,23 @@ open class RequestExecutor(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected open fun getFollowUpMessageForContinuationId(response: BankResponse, continuationId: String, message: MessageBuilderResult, dialogContext: DialogContext,
|
protected open fun getFollowUpMessageForContinuationId(context: JobContext, response: BankResponse, continuationId: String, message: MessageBuilderResult,
|
||||||
tanRequiredCallback: (TanResponse, BankResponse, callback: (BankResponse) -> Unit) -> Unit,
|
tanRequiredCallback: (TanResponse, BankResponse, callback: (BankResponse) -> Unit) -> Unit,
|
||||||
callback: (BankResponse?) -> Unit) {
|
callback: (BankResponse?) -> Unit) {
|
||||||
|
|
||||||
messageBuilder.rebuildMessageWithContinuationId(message, continuationId, dialogContext)?.let { followUpMessage ->
|
messageBuilder.rebuildMessageWithContinuationId(context, message, continuationId)?.let { followUpMessage ->
|
||||||
getAndHandleResponseForMessage(followUpMessage, dialogContext, tanRequiredCallback, callback)
|
getAndHandleResponseForMessage(followUpMessage, context, tanRequiredCallback, callback)
|
||||||
}
|
}
|
||||||
?: run { callback(null) }
|
?: run { callback(null) }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected open fun handleMayRequiresTan(response: BankResponse, dialogContext: DialogContext,
|
protected open fun handleMayRequiresTan(context: JobContext, response: BankResponse,
|
||||||
tanRequiredCallback: (TanResponse, BankResponse, callback: (BankResponse) -> Unit) -> Unit,
|
tanRequiredCallback: (TanResponse, BankResponse, callback: (BankResponse) -> Unit) -> Unit,
|
||||||
callback: (BankResponse) -> Unit) { // TODO: use response from DialogContext
|
callback: (BankResponse) -> Unit) { // TODO: use response from DialogContext
|
||||||
|
|
||||||
if (response.isStrongAuthenticationRequired) {
|
if (response.isStrongAuthenticationRequired) {
|
||||||
if (dialogContext.abortIfTanIsRequired) {
|
if (context.dialog.abortIfTanIsRequired) {
|
||||||
response.tanRequiredButWeWereToldToAbortIfSo = true
|
response.tanRequiredButWeWereToldToAbortIfSo = true
|
||||||
|
|
||||||
callback(response)
|
callback(response)
|
||||||
|
@ -188,12 +189,12 @@ open class RequestExecutor(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected open fun addMessageLog(message: String, type: MessageLogEntryType, dialogContext: DialogContext) {
|
protected open fun addMessageLog(context: JobContext, type: MessageLogEntryType, message: String) {
|
||||||
messageLogCollector.addMessageLog(message, type, dialogContext.bank)
|
messageLogCollector.addMessageLog(message, type, context.bank)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun logError(message: String, dialogContext: DialogContext, e: Exception?) {
|
protected open fun logError(context: JobContext, message: String, e: Exception?) {
|
||||||
messageLogAppender.logError(message, e, log, dialogContext.bank)
|
messageLogAppender.logError(message, e, log, context.bank)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -58,24 +58,24 @@ open class MessageBuilder(protected val generator: ISegmentNumberGenerator = Seg
|
||||||
übermittelt werden, wenn das Kreditinstitut dies unterstützt.
|
übermittelt werden, wenn das Kreditinstitut dies unterstützt.
|
||||||
(PinTan S. 35)
|
(PinTan S. 35)
|
||||||
*/
|
*/
|
||||||
open fun createAnonymousDialogInitMessage(dialogContext: DialogContext): MessageBuilderResult {
|
open fun createAnonymousDialogInitMessage(context: JobContext): MessageBuilderResult {
|
||||||
|
|
||||||
return createUnsignedMessageBuilderResult(dialogContext, listOf(
|
return createUnsignedMessageBuilderResult(context.dialog, listOf(
|
||||||
IdentifikationsSegment(generator.resetSegmentNumber(1), dialogContext),
|
IdentifikationsSegment(generator.resetSegmentNumber(1), context.dialog),
|
||||||
Verarbeitungsvorbereitung(generator.getNextSegmentNumber(), dialogContext)
|
Verarbeitungsvorbereitung(generator.getNextSegmentNumber(), context.dialog)
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
open fun createAnonymousDialogEndMessage(dialogContext: DialogContext): MessageBuilderResult {
|
open fun createAnonymousDialogEndMessage(context: JobContext): MessageBuilderResult {
|
||||||
|
|
||||||
return createUnsignedMessageBuilderResult(dialogContext, listOf(
|
return createUnsignedMessageBuilderResult(context.dialog, listOf(
|
||||||
Dialogende(generator.resetSegmentNumber(1), dialogContext)
|
Dialogende(generator.resetSegmentNumber(1), context.dialog)
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun createInitDialogMessage(dialogContext: DialogContext): MessageBuilderResult {
|
open fun createInitDialogMessage(context: JobContext): MessageBuilderResult {
|
||||||
return createInitDialogMessage(dialogContext, null)
|
return createInitDialogMessage(context, null)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -112,61 +112,63 @@ open class MessageBuilder(protected val generator: ISegmentNumberGenerator = Seg
|
||||||
|
|
||||||
(PinTan S. 37/38)
|
(PinTan S. 37/38)
|
||||||
*/
|
*/
|
||||||
open fun createInitDialogMessageWithoutStrongCustomerAuthentication(dialogContext: DialogContext, segmentIdForTwoStepTanProcess: CustomerSegmentId?): MessageBuilderResult {
|
open fun createInitDialogMessageWithoutStrongCustomerAuthentication(context: JobContext, segmentIdForTwoStepTanProcess: CustomerSegmentId?): MessageBuilderResult {
|
||||||
return createInitDialogMessage(dialogContext, segmentIdForTwoStepTanProcess)
|
return createInitDialogMessage(context, segmentIdForTwoStepTanProcess)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun createInitDialogMessage(dialogContext: DialogContext, segmentIdForTwoStepTanProcess: CustomerSegmentId?): MessageBuilderResult {
|
protected open fun createInitDialogMessage(context: JobContext, segmentIdForTwoStepTanProcess: CustomerSegmentId?): MessageBuilderResult {
|
||||||
|
val dialog = context.dialog
|
||||||
|
|
||||||
val segments = mutableListOf(
|
val segments = mutableListOf(
|
||||||
IdentifikationsSegment(generator.resetSegmentNumber(2), dialogContext),
|
IdentifikationsSegment(generator.resetSegmentNumber(2), dialog),
|
||||||
Verarbeitungsvorbereitung(generator.getNextSegmentNumber(), dialogContext)
|
Verarbeitungsvorbereitung(generator.getNextSegmentNumber(), dialog)
|
||||||
)
|
)
|
||||||
|
|
||||||
if (segmentIdForTwoStepTanProcess != null) {
|
if (segmentIdForTwoStepTanProcess != null) {
|
||||||
segments.add(createTwoStepTanSegment(segmentIdForTwoStepTanProcess, dialogContext))
|
segments.add(createTwoStepTanSegment(segmentIdForTwoStepTanProcess, dialog))
|
||||||
}
|
}
|
||||||
else if (dialogContext.bank.isTanMethodSelected) {
|
else if (context.bank.isTanMethodSelected) {
|
||||||
segments.add(createTwoStepTanSegment(CustomerSegmentId.Identification, dialogContext))
|
segments.add(createTwoStepTanSegment(CustomerSegmentId.Identification, dialog))
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dialogContext.bank.customerSystemId == KundensystemID.Anonymous) {
|
if (context.bank.customerSystemId == KundensystemID.Anonymous) {
|
||||||
segments.add(Synchronisierung(generator.getNextSegmentNumber(), Synchronisierungsmodus.NeueKundensystemIdZurueckmelden))
|
segments.add(Synchronisierung(generator.getNextSegmentNumber(), Synchronisierungsmodus.NeueKundensystemIdZurueckmelden))
|
||||||
}
|
}
|
||||||
|
|
||||||
return createSignedMessageBuilderResult(dialogContext, segments)
|
return createSignedMessageBuilderResult(dialog, segments)
|
||||||
}
|
}
|
||||||
|
|
||||||
open fun createSynchronizeCustomerSystemIdMessage(dialogContext: DialogContext): MessageBuilderResult {
|
open fun createSynchronizeCustomerSystemIdMessage(context: JobContext): MessageBuilderResult {
|
||||||
|
val dialog = context.dialog
|
||||||
|
|
||||||
return createSignedMessageBuilderResult(dialogContext, listOf(
|
return createSignedMessageBuilderResult(dialog, listOf(
|
||||||
IdentifikationsSegment(generator.resetSegmentNumber(2), dialogContext),
|
IdentifikationsSegment(generator.resetSegmentNumber(2), dialog),
|
||||||
Verarbeitungsvorbereitung(generator.getNextSegmentNumber(), dialogContext),
|
Verarbeitungsvorbereitung(generator.getNextSegmentNumber(), dialog),
|
||||||
createTwoStepTanSegment(CustomerSegmentId.Identification, dialogContext),
|
createTwoStepTanSegment(CustomerSegmentId.Identification, dialog),
|
||||||
Synchronisierung(generator.getNextSegmentNumber(), Synchronisierungsmodus.NeueKundensystemIdZurueckmelden)
|
Synchronisierung(generator.getNextSegmentNumber(), Synchronisierungsmodus.NeueKundensystemIdZurueckmelden)
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
open fun createDialogEndMessage(dialogContext: DialogContext): MessageBuilderResult {
|
open fun createDialogEndMessage(context: JobContext): MessageBuilderResult {
|
||||||
|
|
||||||
return createSignedMessageBuilderResult(dialogContext, listOf(
|
return createSignedMessageBuilderResult(context.dialog, listOf(
|
||||||
Dialogende(generator.resetSegmentNumber(2), dialogContext)
|
Dialogende(generator.resetSegmentNumber(2), context.dialog)
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun createGetTransactionsMessage(parameter: GetTransactionsParameter, dialogContext: DialogContext): MessageBuilderResult {
|
open fun createGetTransactionsMessage(context: JobContext, parameter: GetTransactionsParameter): MessageBuilderResult {
|
||||||
|
|
||||||
val result = supportsGetTransactionsMt940(parameter.account)
|
val result = supportsGetTransactionsMt940(parameter.account)
|
||||||
|
|
||||||
if (result.isJobVersionSupported) {
|
if (result.isJobVersionSupported) {
|
||||||
return createGetTransactionsMessageMt940(result, parameter, dialogContext)
|
return createGetTransactionsMessageMt940(result, parameter, context.dialog)
|
||||||
}
|
}
|
||||||
|
|
||||||
val creditCardResult = supportsGetCreditCardTransactions(parameter.account)
|
val creditCardResult = supportsGetCreditCardTransactions(parameter.account)
|
||||||
|
|
||||||
if (creditCardResult.isJobVersionSupported) {
|
if (creditCardResult.isJobVersionSupported) {
|
||||||
return createGetCreditCardTransactionsMessage(result, parameter, dialogContext)
|
return createGetCreditCardTransactionsMessage(result, parameter, context.dialog)
|
||||||
}
|
}
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
@ -220,19 +222,19 @@ open class MessageBuilder(protected val generator: ISegmentNumberGenerator = Seg
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun createGetBalanceMessage(account: AccountData, dialogContext: DialogContext): MessageBuilderResult {
|
open fun createGetBalanceMessage(context: JobContext, account: AccountData): MessageBuilderResult {
|
||||||
|
|
||||||
val result = supportsGetBalanceMessage(account)
|
val result = supportsGetBalanceMessage(account)
|
||||||
|
|
||||||
if (result.isJobVersionSupported) {
|
if (result.isJobVersionSupported) {
|
||||||
val balanceJob = if (result.isAllowed(5)) SaldenabfrageVersion5(generator.resetSegmentNumber(2), account)
|
val balanceJob = if (result.isAllowed(5)) SaldenabfrageVersion5(generator.resetSegmentNumber(2), account)
|
||||||
else SaldenabfrageVersion7(generator.resetSegmentNumber(2), account, dialogContext.bank)
|
else SaldenabfrageVersion7(generator.resetSegmentNumber(2), account, context.bank)
|
||||||
|
|
||||||
val segments = mutableListOf<Segment>(balanceJob)
|
val segments = mutableListOf<Segment>(balanceJob)
|
||||||
|
|
||||||
addTanSegmentIfRequired(CustomerSegmentId.Balance, dialogContext, segments)
|
addTanSegmentIfRequired(CustomerSegmentId.Balance, context.dialog, segments)
|
||||||
|
|
||||||
return createSignedMessageBuilderResult(dialogContext, segments)
|
return createSignedMessageBuilderResult(context.dialog, segments)
|
||||||
}
|
}
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
@ -247,11 +249,11 @@ open class MessageBuilder(protected val generator: ISegmentNumberGenerator = Seg
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun createGetTanMediaListMessage(dialogContext: DialogContext,
|
open fun createGetTanMediaListMessage(context: JobContext,
|
||||||
tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle,
|
tanMediaKind: TanMedienArtVersion = TanMedienArtVersion.Alle,
|
||||||
tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien): MessageBuilderResult {
|
tanMediumClass: TanMediumKlasse = TanMediumKlasse.AlleMedien): MessageBuilderResult {
|
||||||
|
|
||||||
val result = getSupportedVersionsOfJobForBank(CustomerSegmentId.TanMediaList, dialogContext.bank, listOf(2, 3, 4, 5))
|
val result = getSupportedVersionsOfJobForBank(CustomerSegmentId.TanMediaList, context.bank, listOf(2, 3, 4, 5))
|
||||||
|
|
||||||
if (result.isJobVersionSupported) {
|
if (result.isJobVersionSupported) {
|
||||||
val segments = listOf(
|
val segments = listOf(
|
||||||
|
@ -259,31 +261,31 @@ open class MessageBuilder(protected val generator: ISegmentNumberGenerator = Seg
|
||||||
generator.resetSegmentNumber(2), tanMediaKind, tanMediumClass)
|
generator.resetSegmentNumber(2), tanMediaKind, tanMediumClass)
|
||||||
)
|
)
|
||||||
|
|
||||||
return createSignedMessageBuilderResult(dialogContext, segments)
|
return createSignedMessageBuilderResult(context.dialog, segments)
|
||||||
}
|
}
|
||||||
|
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: no HKTAN needed?
|
// TODO: no HKTAN needed?
|
||||||
open fun createChangeTanMediumMessage(newActiveTanMedium: TanGeneratorTanMedium, dialogContext: DialogContext,
|
open fun createChangeTanMediumMessage(context: JobContext, newActiveTanMedium: TanGeneratorTanMedium,
|
||||||
tan: String? = null, atc: Int? = null): MessageBuilderResult {
|
tan: String? = null, atc: Int? = null): MessageBuilderResult {
|
||||||
|
|
||||||
val result = getSupportedVersionsOfJobForBank(CustomerSegmentId.ChangeTanMedium, dialogContext.bank, listOf(1, 2, 3))
|
val result = getSupportedVersionsOfJobForBank(CustomerSegmentId.ChangeTanMedium, context.bank, listOf(1, 2, 3))
|
||||||
|
|
||||||
if (result.isJobVersionSupported) {
|
if (result.isJobVersionSupported) {
|
||||||
val segments = listOf(
|
val segments = listOf(
|
||||||
TanGeneratorTanMediumAnOderUmmelden(result.getHighestAllowedVersion!!, generator.resetSegmentNumber(2),
|
TanGeneratorTanMediumAnOderUmmelden(result.getHighestAllowedVersion!!, generator.resetSegmentNumber(2),
|
||||||
dialogContext.bank, newActiveTanMedium, tan, atc)
|
context.bank, newActiveTanMedium, tan, atc)
|
||||||
)
|
)
|
||||||
|
|
||||||
return createSignedMessageBuilderResult(dialogContext, segments)
|
return createSignedMessageBuilderResult(context.dialog, segments)
|
||||||
}
|
}
|
||||||
|
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
open fun createSendEnteredTanMessage(enteredTan: String, tanResponse: TanResponse, dialogContext: DialogContext): MessageBuilderResult {
|
open fun createSendEnteredTanMessage(context: JobContext, enteredTan: String, tanResponse: TanResponse): MessageBuilderResult {
|
||||||
|
|
||||||
val tanProcess = if (tanResponse.tanProcess == TanProcess.TanProcess1) TanProcess.TanProcess1 else TanProcess.TanProcess2
|
val tanProcess = if (tanResponse.tanProcess == TanProcess.TanProcess1) TanProcess.TanProcess1 else TanProcess.TanProcess2
|
||||||
|
|
||||||
|
@ -292,23 +294,23 @@ open class MessageBuilder(protected val generator: ISegmentNumberGenerator = Seg
|
||||||
tanResponse.jobHashValue, tanResponse.jobReference, false, null, tanResponse.tanMediaIdentifier)
|
tanResponse.jobHashValue, tanResponse.jobReference, false, null, tanResponse.tanMediaIdentifier)
|
||||||
)
|
)
|
||||||
|
|
||||||
return createSignedMessageBuilderResult(createSignedMessage(dialogContext, enteredTan, segments), dialogContext, segments)
|
return createSignedMessageBuilderResult(createSignedMessage(context.dialog, enteredTan, segments), context.dialog, segments)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun createBankTransferMessage(data: BankTransferData, account: AccountData, dialogContext: DialogContext): MessageBuilderResult {
|
open fun createBankTransferMessage(context: JobContext, data: BankTransferData, account: AccountData): MessageBuilderResult {
|
||||||
|
|
||||||
val segmentId = if (data.realTimeTransfer) CustomerSegmentId.SepaRealTimeTransfer else CustomerSegmentId.SepaBankTransfer
|
val segmentId = if (data.realTimeTransfer) CustomerSegmentId.SepaRealTimeTransfer else CustomerSegmentId.SepaBankTransfer
|
||||||
|
|
||||||
val (result, urn) = supportsBankTransferAndSepaVersion(dialogContext.bank, account, segmentId)
|
val (result, urn) = supportsBankTransferAndSepaVersion(context.bank, account, segmentId)
|
||||||
|
|
||||||
if (result.isJobVersionSupported && urn != null) {
|
if (result.isJobVersionSupported && urn != null) {
|
||||||
val segments = mutableListOf<Segment>(SepaBankTransferBase(segmentId, generator.resetSegmentNumber(2),
|
val segments = mutableListOf<Segment>(SepaBankTransferBase(segmentId, generator.resetSegmentNumber(2),
|
||||||
urn, dialogContext.bank.customerName, account, dialogContext.bank.bic, data))
|
urn, context.bank.customerName, account, context.bank.bic, data))
|
||||||
|
|
||||||
addTanSegmentIfRequired(segmentId, dialogContext, segments)
|
addTanSegmentIfRequired(segmentId, context.dialog, segments)
|
||||||
|
|
||||||
return createSignedMessageBuilderResult(dialogContext, segments)
|
return createSignedMessageBuilderResult(context.dialog, segments)
|
||||||
}
|
}
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
@ -342,7 +344,7 @@ open class MessageBuilder(protected val generator: ISegmentNumberGenerator = Seg
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun rebuildMessageWithContinuationId(message: MessageBuilderResult, continuationId: String, dialogContext: DialogContext): MessageBuilderResult? {
|
open fun rebuildMessageWithContinuationId(context: JobContext, message: MessageBuilderResult, continuationId: String): MessageBuilderResult? {
|
||||||
|
|
||||||
// val copiedSegments = message.messageBodySegments.map { }
|
// val copiedSegments = message.messageBodySegments.map { }
|
||||||
val aufsetzpunkte = message.messageBodySegments.flatMap { it.dataElementsAndGroups }.filterIsInstance<Aufsetzpunkt>()
|
val aufsetzpunkte = message.messageBodySegments.flatMap { it.dataElementsAndGroups }.filterIsInstance<Aufsetzpunkt>()
|
||||||
|
@ -354,12 +356,12 @@ open class MessageBuilder(protected val generator: ISegmentNumberGenerator = Seg
|
||||||
|
|
||||||
aufsetzpunkte.forEach { it.resetContinuationId(continuationId) }
|
aufsetzpunkte.forEach { it.resetContinuationId(continuationId) }
|
||||||
|
|
||||||
return rebuildMessage(message, dialogContext)
|
return rebuildMessage(context, message)
|
||||||
}
|
}
|
||||||
|
|
||||||
open fun rebuildMessage(message: MessageBuilderResult, dialogContext: DialogContext): MessageBuilderResult {
|
open fun rebuildMessage(context: JobContext, message: MessageBuilderResult): MessageBuilderResult {
|
||||||
|
|
||||||
return createSignedMessageBuilderResult(dialogContext, message.messageBodySegments)
|
return createSignedMessageBuilderResult(context.dialog, message.messageBodySegments)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun createSignedMessageBuilderResult(dialogContext: DialogContext, segments: List<Segment>): MessageBuilderResult {
|
protected open fun createSignedMessageBuilderResult(dialogContext: DialogContext, segments: List<Segment>): MessageBuilderResult {
|
||||||
|
@ -518,7 +520,8 @@ open class MessageBuilder(protected val generator: ISegmentNumberGenerator = Seg
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun createTwoStepTanSegment(segmentId: CustomerSegmentId, dialogContext: DialogContext): ZweiSchrittTanEinreichung {
|
protected open fun createTwoStepTanSegment(segmentId: CustomerSegmentId, dialogContext: DialogContext): ZweiSchrittTanEinreichung {
|
||||||
return ZweiSchrittTanEinreichung(generator.getNextSegmentNumber(), TanProcess.TanProcess4, segmentId, tanMediaIdentifier = getTanMediaIdentifierIfRequired(dialogContext))
|
return ZweiSchrittTanEinreichung(generator.getNextSegmentNumber(), TanProcess.TanProcess4, segmentId,
|
||||||
|
tanMediaIdentifier = getTanMediaIdentifierIfRequired(dialogContext))
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun getTanMediaIdentifierIfRequired(dialogContext: DialogContext): String? {
|
protected open fun getTanMediaIdentifierIfRequired(dialogContext: DialogContext): String? {
|
||||||
|
|
|
@ -0,0 +1,24 @@
|
||||||
|
package net.dankito.banking.fints.model
|
||||||
|
|
||||||
|
import net.dankito.banking.fints.callback.FinTsClientCallback
|
||||||
|
|
||||||
|
|
||||||
|
class JobContext(
|
||||||
|
val type: JobContextType,
|
||||||
|
val callback: FinTsClientCallback,
|
||||||
|
val bank: BankData,
|
||||||
|
/**
|
||||||
|
* Only set if the current context is for a specific account (like get account's transactions).
|
||||||
|
*/
|
||||||
|
val account: AccountData? = null
|
||||||
|
) {
|
||||||
|
|
||||||
|
|
||||||
|
lateinit var dialog: DialogContext
|
||||||
|
|
||||||
|
|
||||||
|
fun startNewDialog(dialog: DialogContext) {
|
||||||
|
this.dialog = dialog
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,18 @@
|
||||||
|
package net.dankito.banking.fints.model
|
||||||
|
|
||||||
|
|
||||||
|
enum class JobContextType {
|
||||||
|
|
||||||
|
AnonymousBankInfo,
|
||||||
|
|
||||||
|
GetTanMedia,
|
||||||
|
|
||||||
|
ChangeTanMedium,
|
||||||
|
|
||||||
|
AddAccount,
|
||||||
|
|
||||||
|
GetTransactions,
|
||||||
|
|
||||||
|
TransferMoney
|
||||||
|
|
||||||
|
}
|
|
@ -1,5 +1,6 @@
|
||||||
package net.dankito.banking.fints
|
package net.dankito.banking.fints
|
||||||
|
|
||||||
|
import net.dankito.banking.fints.callback.SimpleFinTsClientCallback
|
||||||
import net.dankito.banking.fints.messages.datenelemente.abgeleiteteformate.Datum
|
import net.dankito.banking.fints.messages.datenelemente.abgeleiteteformate.Datum
|
||||||
import net.dankito.banking.fints.messages.datenelemente.abgeleiteteformate.Laenderkennzeichen
|
import net.dankito.banking.fints.messages.datenelemente.abgeleiteteformate.Laenderkennzeichen
|
||||||
import net.dankito.banking.fints.messages.datenelemente.implementierte.Dialogsprache
|
import net.dankito.banking.fints.messages.datenelemente.implementierte.Dialogsprache
|
||||||
|
@ -61,6 +62,15 @@ abstract class FinTsTestBase {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
protected open fun createContext(dialogId: String = DialogContext.InitialDialogId): JobContext {
|
||||||
|
val dialogContext = DialogContext(Bank, Product, dialogId = dialogId)
|
||||||
|
|
||||||
|
val context = JobContext(JobContextType.AnonymousBankInfo, SimpleFinTsClientCallback(), Bank)
|
||||||
|
context.startNewDialog(dialogContext)
|
||||||
|
|
||||||
|
return context
|
||||||
|
}
|
||||||
|
|
||||||
protected open fun createDialogId(): String {
|
protected open fun createDialogId(): String {
|
||||||
return UUID.random().replace("-", "")
|
return UUID.random().replace("-", "")
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,10 +47,10 @@ class MessageBuilderTest : FinTsTestBase() {
|
||||||
fun createAnonymousDialogInitMessage() {
|
fun createAnonymousDialogInitMessage() {
|
||||||
|
|
||||||
// given
|
// given
|
||||||
val dialogContext = DialogContext(Bank, Product)
|
val context = createContext()
|
||||||
|
|
||||||
// when
|
// when
|
||||||
val result = underTest.createAnonymousDialogInitMessage(dialogContext).createdMessage
|
val result = underTest.createAnonymousDialogInitMessage(context).createdMessage
|
||||||
|
|
||||||
// then
|
// then
|
||||||
expect(result).toBe(
|
expect(result).toBe(
|
||||||
|
@ -66,10 +66,10 @@ class MessageBuilderTest : FinTsTestBase() {
|
||||||
|
|
||||||
// given
|
// given
|
||||||
val dialogId = createDialogId()
|
val dialogId = createDialogId()
|
||||||
val dialogContext = DialogContext(Bank, Product, dialogId = dialogId)
|
val context = createContext(dialogId)
|
||||||
|
|
||||||
// when
|
// when
|
||||||
val result = underTest.createAnonymousDialogEndMessage(dialogContext).createdMessage ?: ""
|
val result = underTest.createAnonymousDialogEndMessage(context).createdMessage ?: ""
|
||||||
|
|
||||||
// then
|
// then
|
||||||
expect(normalizeBinaryData(result)).toBe(normalizeBinaryData(
|
expect(normalizeBinaryData(result)).toBe(normalizeBinaryData(
|
||||||
|
@ -84,10 +84,10 @@ class MessageBuilderTest : FinTsTestBase() {
|
||||||
fun createDialogInitMessage() {
|
fun createDialogInitMessage() {
|
||||||
|
|
||||||
// given
|
// given
|
||||||
val dialogContext = DialogContext(Bank, Product)
|
val context = createContext()
|
||||||
|
|
||||||
// when
|
// when
|
||||||
val result = underTest.createSynchronizeCustomerSystemIdMessage(dialogContext).createdMessage ?: ""
|
val result = underTest.createSynchronizeCustomerSystemIdMessage(context).createdMessage ?: ""
|
||||||
|
|
||||||
// then
|
// then
|
||||||
expect(normalizeBinaryData(result)).toBe(normalizeBinaryData(
|
expect(normalizeBinaryData(result)).toBe(normalizeBinaryData(
|
||||||
|
@ -108,10 +108,10 @@ class MessageBuilderTest : FinTsTestBase() {
|
||||||
|
|
||||||
// given
|
// given
|
||||||
val dialogId = createDialogId()
|
val dialogId = createDialogId()
|
||||||
val dialogContext = DialogContext(Bank, Product, dialogId = dialogId)
|
val context = createContext(dialogId)
|
||||||
|
|
||||||
// when
|
// when
|
||||||
val result = underTest.createDialogEndMessage(dialogContext).createdMessage ?: ""
|
val result = underTest.createDialogEndMessage(context).createdMessage ?: ""
|
||||||
|
|
||||||
// then
|
// then
|
||||||
expect(normalizeBinaryData(result)).toBe(normalizeBinaryData(
|
expect(normalizeBinaryData(result)).toBe(normalizeBinaryData(
|
||||||
|
@ -129,10 +129,10 @@ class MessageBuilderTest : FinTsTestBase() {
|
||||||
fun createGetTransactionsMessage_JobIsNotAllowed() {
|
fun createGetTransactionsMessage_JobIsNotAllowed() {
|
||||||
|
|
||||||
// given
|
// given
|
||||||
val dialogContext = DialogContext(Bank, Product)
|
val context = createContext()
|
||||||
|
|
||||||
// when
|
// when
|
||||||
val result = underTest.createGetTransactionsMessage(GetTransactionsParameter(Account), dialogContext)
|
val result = underTest.createGetTransactionsMessage(context, GetTransactionsParameter(Account))
|
||||||
|
|
||||||
// then
|
// then
|
||||||
expect(result.isJobAllowed).toBe(false)
|
expect(result.isJobAllowed).toBe(false)
|
||||||
|
@ -147,10 +147,11 @@ class MessageBuilderTest : FinTsTestBase() {
|
||||||
Bank.supportedJobs = listOf(getTransactionsJob)
|
Bank.supportedJobs = listOf(getTransactionsJob)
|
||||||
val account = AccountData(CustomerId, null, BankCountryCode, BankCode, null, CustomerId, AccountType.Girokonto, "EUR", "", null, null, listOf(getTransactionsJob.jobName), listOf(getTransactionsJobWithPreviousVersion))
|
val account = AccountData(CustomerId, null, BankCountryCode, BankCode, null, CustomerId, AccountType.Girokonto, "EUR", "", null, null, listOf(getTransactionsJob.jobName), listOf(getTransactionsJobWithPreviousVersion))
|
||||||
Bank.addAccount(account)
|
Bank.addAccount(account)
|
||||||
val dialogContext = DialogContext(Bank, Product)
|
|
||||||
|
val context = createContext()
|
||||||
|
|
||||||
// when
|
// when
|
||||||
val result = underTest.createGetTransactionsMessage(GetTransactionsParameter(account), dialogContext)
|
val result = underTest.createGetTransactionsMessage(context, GetTransactionsParameter(account))
|
||||||
|
|
||||||
// then
|
// then
|
||||||
expect(result.isJobAllowed).toBe(true)
|
expect(result.isJobAllowed).toBe(true)
|
||||||
|
@ -166,14 +167,15 @@ class MessageBuilderTest : FinTsTestBase() {
|
||||||
Bank.pinInfo = PinInfo(getTransactionsJob, null, null, null, null, null, listOf(JobTanConfiguration(CustomerSegmentId.AccountTransactionsMt940.id, true)))
|
Bank.pinInfo = PinInfo(getTransactionsJob, null, null, null, null, null, listOf(JobTanConfiguration(CustomerSegmentId.AccountTransactionsMt940.id, true)))
|
||||||
val account = AccountData(CustomerId, null, BankCountryCode, BankCode, null, CustomerId, AccountType.Girokonto, "EUR", "", null, null, listOf(getTransactionsJob.jobName), listOf(getTransactionsJob))
|
val account = AccountData(CustomerId, null, BankCountryCode, BankCode, null, CustomerId, AccountType.Girokonto, "EUR", "", null, null, listOf(getTransactionsJob.jobName), listOf(getTransactionsJob))
|
||||||
Bank.addAccount(account)
|
Bank.addAccount(account)
|
||||||
val dialogContext = DialogContext(Bank, Product)
|
|
||||||
|
val context = createContext()
|
||||||
|
|
||||||
val fromDate = Date(2019, Month.August, 6)
|
val fromDate = Date(2019, Month.August, 6)
|
||||||
val toDate = Date(2019, Month.October, 21)
|
val toDate = Date(2019, Month.October, 21)
|
||||||
val maxCountEntries = 99
|
val maxCountEntries = 99
|
||||||
|
|
||||||
// when
|
// when
|
||||||
val result = underTest.createGetTransactionsMessage(GetTransactionsParameter(account, false, fromDate, toDate, maxCountEntries), dialogContext)
|
val result = underTest.createGetTransactionsMessage(context, GetTransactionsParameter(account, false, fromDate, toDate, maxCountEntries))
|
||||||
|
|
||||||
// then
|
// then
|
||||||
expect(result.createdMessage).notToBeNull()
|
expect(result.createdMessage).notToBeNull()
|
||||||
|
@ -198,7 +200,8 @@ class MessageBuilderTest : FinTsTestBase() {
|
||||||
Bank.pinInfo = PinInfo(getTransactionsJob, null, null, null, null, null, listOf(JobTanConfiguration(CustomerSegmentId.AccountTransactionsMt940.id, true)))
|
Bank.pinInfo = PinInfo(getTransactionsJob, null, null, null, null, null, listOf(JobTanConfiguration(CustomerSegmentId.AccountTransactionsMt940.id, true)))
|
||||||
val account = AccountData(CustomerId, null, BankCountryCode, BankCode, null, CustomerId, AccountType.Girokonto, "EUR", "", null, null, listOf(getTransactionsJob.jobName), listOf(getTransactionsJob))
|
val account = AccountData(CustomerId, null, BankCountryCode, BankCode, null, CustomerId, AccountType.Girokonto, "EUR", "", null, null, listOf(getTransactionsJob.jobName), listOf(getTransactionsJob))
|
||||||
Bank.addAccount(account)
|
Bank.addAccount(account)
|
||||||
val dialogContext = DialogContext(Bank, Product)
|
|
||||||
|
val context = createContext()
|
||||||
|
|
||||||
val fromDate = Date(2019, Month.August, 6)
|
val fromDate = Date(2019, Month.August, 6)
|
||||||
val toDate = Date(2019, Month.October, 21)
|
val toDate = Date(2019, Month.October, 21)
|
||||||
|
@ -206,7 +209,8 @@ class MessageBuilderTest : FinTsTestBase() {
|
||||||
val continuationId = "9345-10-26-11.52.15.693455"
|
val continuationId = "9345-10-26-11.52.15.693455"
|
||||||
|
|
||||||
// when
|
// when
|
||||||
val result = underTest.createGetTransactionsMessage(GetTransactionsParameter(account, false, fromDate, toDate, maxCountEntries, false), dialogContext) // TODO: test Aufsetzpunkt / continuationId
|
val result = underTest.createGetTransactionsMessage(context, // TODO: test Aufsetzpunkt / continuationId
|
||||||
|
GetTransactionsParameter(account, false, fromDate, toDate, maxCountEntries, false))
|
||||||
|
|
||||||
// then
|
// then
|
||||||
expect(result.createdMessage).notToBeNull()
|
expect(result.createdMessage).notToBeNull()
|
||||||
|
|
|
@ -13,6 +13,7 @@ import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.SmsAb
|
||||||
import net.dankito.banking.fints.model.*
|
import net.dankito.banking.fints.model.*
|
||||||
import net.dankito.banking.bankfinder.BankInfo
|
import net.dankito.banking.bankfinder.BankInfo
|
||||||
import net.dankito.banking.fints.FinTsJobExecutor
|
import net.dankito.banking.fints.FinTsJobExecutor
|
||||||
|
import net.dankito.banking.fints.callback.SimpleFinTsClientCallback
|
||||||
import net.dankito.banking.fints.model.mapper.ModelMapper
|
import net.dankito.banking.fints.model.mapper.ModelMapper
|
||||||
import net.dankito.banking.fints.response.BankResponse
|
import net.dankito.banking.fints.response.BankResponse
|
||||||
import net.dankito.banking.fints.response.segments.SepaAccountInfoParameters
|
import net.dankito.banking.fints.response.segments.SepaAccountInfoParameters
|
||||||
|
@ -57,10 +58,10 @@ class BanksFinTsDetailsRetriever {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private val jobExecutor = object : FinTsJobExecutor(NoOpFinTsClientCallback(), modelMapper = modelMapper) {
|
private val jobExecutor = object : FinTsJobExecutor(modelMapper = modelMapper) {
|
||||||
|
|
||||||
fun getAndHandleResponseForMessagePublic(message: MessageBuilderResult, dialogContext: DialogContext, callback: (BankResponse) -> Unit) {
|
fun getAndHandleResponseForMessagePublic(context: JobContext, message: MessageBuilderResult, callback: (BankResponse) -> Unit) {
|
||||||
getAndHandleResponseForMessage(message, dialogContext, callback)
|
getAndHandleResponseForMessage(context, message, callback)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -127,12 +128,15 @@ class BanksFinTsDetailsRetriever {
|
||||||
|
|
||||||
private fun getAnonymousBankInfo(bank: BankData): BankResponse {
|
private fun getAnonymousBankInfo(bank: BankData): BankResponse {
|
||||||
val dialogContext = DialogContext(bank, product)
|
val dialogContext = DialogContext(bank, product)
|
||||||
val requestBody = messageBuilder.createAnonymousDialogInitMessage(dialogContext)
|
val context = JobContext(JobContextType.AnonymousBankInfo, SimpleFinTsClientCallback(), bank)
|
||||||
|
context.startNewDialog(dialogContext)
|
||||||
|
|
||||||
|
val requestBody = messageBuilder.createAnonymousDialogInitMessage(context)
|
||||||
|
|
||||||
val anonymousBankInfoResponse = AtomicReference<BankResponse>()
|
val anonymousBankInfoResponse = AtomicReference<BankResponse>()
|
||||||
val countDownLatch = CountDownLatch(1)
|
val countDownLatch = CountDownLatch(1)
|
||||||
|
|
||||||
jobExecutor.getAndHandleResponseForMessagePublic(requestBody, dialogContext) {
|
jobExecutor.getAndHandleResponseForMessagePublic(context, requestBody) {
|
||||||
anonymousBankInfoResponse.set(it)
|
anonymousBankInfoResponse.set(it)
|
||||||
countDownLatch.countDown()
|
countDownLatch.countDown()
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue