Compare commits
No commits in common. "62276e2a022a35928a9bcd845f7c0a1ccd9c7b6f" and "b35f1146a5fab7f097f16ee0e63bb23bd658cad2" have entirely different histories.
62276e2a02
...
b35f1146a5
|
@ -53,7 +53,6 @@ kotlin {
|
||||||
linuxX64()
|
linuxX64()
|
||||||
mingwX64()
|
mingwX64()
|
||||||
|
|
||||||
iosX64()
|
|
||||||
iosArm64()
|
iosArm64()
|
||||||
iosSimulatorArm64()
|
iosSimulatorArm64()
|
||||||
macosX64()
|
macosX64()
|
||||||
|
|
|
@ -1,33 +1,14 @@
|
||||||
package net.codinux.banking.client
|
package net.codinux.banking.client
|
||||||
|
|
||||||
import net.codinux.banking.client.model.options.RetrieveTransactions
|
|
||||||
import net.codinux.banking.client.model.request.GetAccountDataRequest
|
import net.codinux.banking.client.model.request.GetAccountDataRequest
|
||||||
import net.codinux.banking.client.model.response.GetAccountDataResponse
|
import net.codinux.banking.client.model.response.GetAccountDataResponse
|
||||||
import net.codinux.banking.client.model.response.Response
|
import net.codinux.banking.client.model.response.Response
|
||||||
|
|
||||||
interface BankingClient {
|
interface BankingClient {
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves account data like customer name, her bank accounts, their balance and account transactions.
|
|
||||||
*
|
|
||||||
* By default the account transactions of the last 90 days are retrieved (which in most cases doesn't require a
|
|
||||||
* TAN according to PSD2).
|
|
||||||
*
|
|
||||||
* If you like to retrieve the transactions of a different period, use the method overload that takes a
|
|
||||||
* [GetAccountDataRequest] parameter and set [RetrieveTransactions] and may to and from date in its options object.
|
|
||||||
*/
|
|
||||||
suspend fun getAccountDataAsync(bankCode: String, loginName: String, password: String) =
|
suspend fun getAccountDataAsync(bankCode: String, loginName: String, password: String) =
|
||||||
getAccountDataAsync(GetAccountDataRequest(bankCode, loginName, password))
|
getAccountDataAsync(GetAccountDataRequest(bankCode, loginName, password))
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves account data like customer name, her bank accounts, their balance and account transactions.
|
|
||||||
*
|
|
||||||
* By default the account transactions of the last 90 days are retrieved (which in most cases doesn't require a
|
|
||||||
* TAN according to PSD2).
|
|
||||||
*
|
|
||||||
* If you like to retrieve the transactions of a different period, set [RetrieveTransactions] and may to and from
|
|
||||||
* date in [GetAccountDataRequest.options].
|
|
||||||
*/
|
|
||||||
suspend fun getAccountDataAsync(request: GetAccountDataRequest): Response<GetAccountDataResponse>
|
suspend fun getAccountDataAsync(request: GetAccountDataRequest): Response<GetAccountDataResponse>
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,6 +1,5 @@
|
||||||
package net.codinux.banking.client
|
package net.codinux.banking.client
|
||||||
|
|
||||||
import net.codinux.banking.client.model.MessageLogEntry
|
|
||||||
import net.codinux.banking.client.model.tan.EnterTanResult
|
import net.codinux.banking.client.model.tan.EnterTanResult
|
||||||
import net.codinux.banking.client.model.tan.TanChallenge
|
import net.codinux.banking.client.model.tan.TanChallenge
|
||||||
|
|
||||||
|
@ -8,11 +7,4 @@ interface BankingClientCallback {
|
||||||
|
|
||||||
fun enterTan(tanChallenge: TanChallenge, callback: (EnterTanResult) -> Unit)
|
fun enterTan(tanChallenge: TanChallenge, callback: (EnterTanResult) -> Unit)
|
||||||
|
|
||||||
/**
|
|
||||||
* An optional method, for sure not available for all client implementations (currently only for FinTs4kBankingClient).
|
|
||||||
*
|
|
||||||
* Gets fired when a FinTS message get sent to bank server, a FinTS message is received from bank server or an error occurred.
|
|
||||||
*/
|
|
||||||
fun messageLogAdded(messageLogEntry: MessageLogEntry)
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -0,0 +1,14 @@
|
||||||
|
package net.codinux.banking.client
|
||||||
|
|
||||||
|
import net.codinux.banking.client.model.options.GetAccountDataOptions
|
||||||
|
import net.codinux.banking.client.model.response.GetAccountDataResponse
|
||||||
|
import net.codinux.banking.client.model.response.Response
|
||||||
|
|
||||||
|
interface BankingClientForCustomer {
|
||||||
|
|
||||||
|
// for languages not supporting default parameters (Java, Swift, JS, ...)
|
||||||
|
suspend fun getAccountDataAsync() = getAccountDataAsync(GetAccountDataOptions())
|
||||||
|
|
||||||
|
suspend fun getAccountDataAsync(options: GetAccountDataOptions): Response<GetAccountDataResponse>
|
||||||
|
|
||||||
|
}
|
|
@ -4,10 +4,10 @@ import net.codinux.banking.client.model.AccountCredentials
|
||||||
import net.codinux.banking.client.model.options.GetAccountDataOptions
|
import net.codinux.banking.client.model.options.GetAccountDataOptions
|
||||||
import net.codinux.banking.client.model.request.GetAccountDataRequest
|
import net.codinux.banking.client.model.request.GetAccountDataRequest
|
||||||
|
|
||||||
abstract class BankingClientForUserBase(
|
abstract class BankingClientForCustomerBase(
|
||||||
protected val credentials: AccountCredentials,
|
protected val credentials: AccountCredentials,
|
||||||
protected val client: BankingClient
|
protected val client: BankingClient
|
||||||
) : BankingClientForUser {
|
) : BankingClientForCustomer {
|
||||||
|
|
||||||
override suspend fun getAccountDataAsync(options: GetAccountDataOptions) =
|
override suspend fun getAccountDataAsync(options: GetAccountDataOptions) =
|
||||||
client.getAccountDataAsync(GetAccountDataRequest(credentials, options))
|
client.getAccountDataAsync(GetAccountDataRequest(credentials, options))
|
|
@ -1,33 +0,0 @@
|
||||||
package net.codinux.banking.client
|
|
||||||
|
|
||||||
import net.codinux.banking.client.model.options.GetAccountDataOptions
|
|
||||||
import net.codinux.banking.client.model.options.RetrieveTransactions
|
|
||||||
import net.codinux.banking.client.model.response.GetAccountDataResponse
|
|
||||||
import net.codinux.banking.client.model.response.Response
|
|
||||||
|
|
||||||
interface BankingClientForUser {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves account data like customer name, her bank accounts, their balance and account transactions.
|
|
||||||
*
|
|
||||||
* By default the account transactions of the last 90 days are retrieved (which in most cases doesn't require a
|
|
||||||
* TAN according to PSD2).
|
|
||||||
*
|
|
||||||
* If you like to retrieve the transactions of a different period, use the method overload that takes a
|
|
||||||
* [GetAccountDataOptions] parameter and set [RetrieveTransactions] and may to and from date.
|
|
||||||
*/
|
|
||||||
// for languages not supporting default parameters (Java, Swift, JS, ...)
|
|
||||||
suspend fun getAccountDataAsync() = getAccountDataAsync(GetAccountDataOptions())
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves account data like customer name, her bank accounts, their balance and account transactions.
|
|
||||||
*
|
|
||||||
* By default the account transactions of the last 90 days are retrieved (which in most cases doesn't require a
|
|
||||||
* TAN according to PSD2).
|
|
||||||
*
|
|
||||||
* If you like to retrieve the transactions of a different period, set [GetAccountDataOptions.retrieveTransactions]
|
|
||||||
* and may [GetAccountDataOptions.retrieveTransactionsFrom] and [GetAccountDataOptions.retrieveTransactionsTo].
|
|
||||||
*/
|
|
||||||
suspend fun getAccountDataAsync(options: GetAccountDataOptions): Response<GetAccountDataResponse>
|
|
||||||
|
|
||||||
}
|
|
|
@ -4,7 +4,7 @@ import net.codinux.banking.client.model.options.GetAccountDataOptions
|
||||||
import net.codinux.banking.client.model.response.GetAccountDataResponse
|
import net.codinux.banking.client.model.response.GetAccountDataResponse
|
||||||
import net.codinux.banking.client.model.response.Response
|
import net.codinux.banking.client.model.response.Response
|
||||||
|
|
||||||
interface BlockingBankingClientForUser {
|
interface BlockingBankingClientForCustomer {
|
||||||
|
|
||||||
// for languages not supporting default parameters (Java, Swift, JS, ...)
|
// for languages not supporting default parameters (Java, Swift, JS, ...)
|
||||||
fun getAccountData() = getAccountData(GetAccountDataOptions())
|
fun getAccountData() = getAccountData(GetAccountDataOptions())
|
|
@ -4,10 +4,10 @@ import net.codinux.banking.client.model.AccountCredentials
|
||||||
import net.codinux.banking.client.model.options.GetAccountDataOptions
|
import net.codinux.banking.client.model.options.GetAccountDataOptions
|
||||||
import net.codinux.banking.client.model.request.GetAccountDataRequest
|
import net.codinux.banking.client.model.request.GetAccountDataRequest
|
||||||
|
|
||||||
abstract class BlockingBankingClientForUserBase(
|
abstract class BlockingBankingClientForCustomerBase(
|
||||||
protected val credentials: AccountCredentials,
|
protected val credentials: AccountCredentials,
|
||||||
protected val client: BlockingBankingClient
|
protected val client: BlockingBankingClient
|
||||||
) : BlockingBankingClientForUser {
|
) : BlockingBankingClientForCustomer {
|
||||||
|
|
||||||
override fun getAccountData(options: GetAccountDataOptions) =
|
override fun getAccountData(options: GetAccountDataOptions) =
|
||||||
client.getAccountData(GetAccountDataRequest(credentials, options))
|
client.getAccountData(GetAccountDataRequest(credentials, options))
|
|
@ -1,11 +1,9 @@
|
||||||
package net.codinux.banking.client
|
package net.codinux.banking.client
|
||||||
|
|
||||||
import net.codinux.banking.client.model.MessageLogEntry
|
|
||||||
import net.codinux.banking.client.model.tan.EnterTanResult
|
import net.codinux.banking.client.model.tan.EnterTanResult
|
||||||
import net.codinux.banking.client.model.tan.TanChallenge
|
import net.codinux.banking.client.model.tan.TanChallenge
|
||||||
|
|
||||||
open class SimpleBankingClientCallback(
|
open class SimpleBankingClientCallback(
|
||||||
protected val messageLogAdded: ((MessageLogEntry) -> Unit)? = null,
|
|
||||||
protected val enterTan: ((tanChallenge: TanChallenge, callback: (EnterTanResult) -> Unit) -> Unit)? = null
|
protected val enterTan: ((tanChallenge: TanChallenge, callback: (EnterTanResult) -> Unit) -> Unit)? = null
|
||||||
) : BankingClientCallback {
|
) : BankingClientCallback {
|
||||||
|
|
||||||
|
@ -17,8 +15,4 @@ open class SimpleBankingClientCallback(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun messageLogAdded(messageLogEntry: MessageLogEntry) {
|
|
||||||
messageLogAdded?.invoke(messageLogEntry)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -12,10 +12,10 @@ fun BankingClient.getAccountData(request: GetAccountDataRequest) = runBlocking {
|
||||||
this@getAccountData.getAccountDataAsync(request)
|
this@getAccountData.getAccountDataAsync(request)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun BankingClientForUser.getAccountData() = runBlocking {
|
fun BankingClientForCustomer.getAccountData() = runBlocking {
|
||||||
this@getAccountData.getAccountDataAsync()
|
this@getAccountData.getAccountDataAsync()
|
||||||
}
|
}
|
||||||
|
|
||||||
fun BankingClientForUser.getAccountData(options: GetAccountDataOptions) = runBlocking {
|
fun BankingClientForCustomer.getAccountData(options: GetAccountDataOptions) = runBlocking {
|
||||||
this@getAccountData.getAccountDataAsync(options)
|
this@getAccountData.getAccountDataAsync(options)
|
||||||
}
|
}
|
|
@ -12,10 +12,10 @@ fun BankingClient.getAccountData(request: GetAccountDataRequest) = runBlocking {
|
||||||
this@getAccountData.getAccountDataAsync(request)
|
this@getAccountData.getAccountDataAsync(request)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun BankingClientForUser.getAccountData() = runBlocking {
|
fun BankingClientForCustomer.getAccountData() = runBlocking {
|
||||||
this@getAccountData.getAccountDataAsync()
|
this@getAccountData.getAccountDataAsync()
|
||||||
}
|
}
|
||||||
|
|
||||||
fun BankingClientForUser.getAccountData(options: GetAccountDataOptions) = runBlocking {
|
fun BankingClientForCustomer.getAccountData(options: GetAccountDataOptions) = runBlocking {
|
||||||
this@getAccountData.getAccountDataAsync(options)
|
this@getAccountData.getAccountDataAsync(options)
|
||||||
}
|
}
|
|
@ -63,7 +63,6 @@ kotlin {
|
||||||
linuxX64()
|
linuxX64()
|
||||||
mingwX64()
|
mingwX64()
|
||||||
|
|
||||||
iosX64()
|
|
||||||
iosArm64()
|
iosArm64()
|
||||||
iosSimulatorArm64()
|
iosSimulatorArm64()
|
||||||
macosX64()
|
macosX64()
|
||||||
|
|
|
@ -59,7 +59,6 @@ open class AccountTransaction(
|
||||||
val relatedReferenceNumber: String? = null,
|
val relatedReferenceNumber: String? = null,
|
||||||
|
|
||||||
var userSetDisplayName: String? = null,
|
var userSetDisplayName: String? = null,
|
||||||
var category: String? = null,
|
|
||||||
var notes: String? = null,
|
var notes: String? = null,
|
||||||
) {
|
) {
|
||||||
override fun toString() = "${valueDate.dayOfMonth}.${valueDate.monthNumber}.${valueDate.year} ${amount.toString().padStart(4, ' ')} ${if (currency == "EUR") "€" else currency} ${otherPartyName ?: ""} - $reference"
|
override fun toString() = "${valueDate.dayOfMonth}.${valueDate.monthNumber}.${valueDate.year} ${amount.toString().padStart(4, ' ')} ${if (currency == "EUR") "€" else currency} ${otherPartyName ?: ""} - $reference"
|
||||||
|
|
|
@ -25,8 +25,8 @@ open class BankAccount(
|
||||||
var haveAllTransactionsBeenRetrieved: Boolean = false,
|
var haveAllTransactionsBeenRetrieved: Boolean = false,
|
||||||
val countDaysForWhichTransactionsAreKept: Int? = null,
|
val countDaysForWhichTransactionsAreKept: Int? = null,
|
||||||
|
|
||||||
open val bookedTransactions: MutableList<AccountTransaction> = mutableListOf(),
|
val bookedTransactions: MutableList<AccountTransaction> = mutableListOf(),
|
||||||
open val unbookedTransactions: MutableList<UnbookedAccountTransaction> = mutableListOf(),
|
val unbookedTransactions: MutableList<UnbookedAccountTransaction> = mutableListOf(),
|
||||||
|
|
||||||
var userSetDisplayName: String? = null,
|
var userSetDisplayName: String? = null,
|
||||||
var displayIndex: Int = 0,
|
var displayIndex: Int = 0,
|
||||||
|
|
|
@ -1,12 +0,0 @@
|
||||||
package net.codinux.banking.client.model
|
|
||||||
|
|
||||||
import net.codinux.banking.client.model.config.NoArgConstructor
|
|
||||||
|
|
||||||
@NoArgConstructor
|
|
||||||
open class BankAccountIdentifier(
|
|
||||||
val identifier: String,
|
|
||||||
val subAccountNumber: String?,
|
|
||||||
val iban: String?
|
|
||||||
) {
|
|
||||||
override fun toString() = "$identifier, $iban"
|
|
||||||
}
|
|
|
@ -0,0 +1,34 @@
|
||||||
|
package net.codinux.banking.client.model
|
||||||
|
|
||||||
|
import net.codinux.banking.client.model.config.NoArgConstructor
|
||||||
|
|
||||||
|
@NoArgConstructor
|
||||||
|
open class CustomerAccount(
|
||||||
|
val bankCode: String,
|
||||||
|
var loginName: String,
|
||||||
|
/**
|
||||||
|
* User may decides to not save password .
|
||||||
|
*/
|
||||||
|
var password: String?,
|
||||||
|
|
||||||
|
val bankName: String,
|
||||||
|
val bic: String,
|
||||||
|
|
||||||
|
val customerName: String,
|
||||||
|
val userId: String = loginName,
|
||||||
|
|
||||||
|
val accounts: List<BankAccount> = emptyList(),
|
||||||
|
|
||||||
|
var bankingGroup: BankingGroup? = null,
|
||||||
|
var iconUrl: String? = null,
|
||||||
|
) {
|
||||||
|
|
||||||
|
var wrongCredentialsEntered: Boolean = false
|
||||||
|
|
||||||
|
var userSetDisplayName: String? = null
|
||||||
|
var displayIndex: Int = 0
|
||||||
|
|
||||||
|
|
||||||
|
override fun toString() = "$bankName $loginName, ${accounts.size} accounts"
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,16 @@
|
||||||
|
package net.codinux.banking.client.model
|
||||||
|
|
||||||
|
import net.codinux.banking.client.model.config.NoArgConstructor
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Contains only the basic info of a [CustomerAccount], just enough that a client application can display it to the user
|
||||||
|
* and the user knows exactly which [CustomerAccount] is meant / referred.
|
||||||
|
*/
|
||||||
|
@NoArgConstructor
|
||||||
|
open class CustomerAccountViewInfo(
|
||||||
|
val bankCode: String,
|
||||||
|
var loginName: String,
|
||||||
|
val bankName: String
|
||||||
|
) {
|
||||||
|
override fun toString() = "$bankCode $bankName $loginName"
|
||||||
|
}
|
|
@ -1,12 +0,0 @@
|
||||||
package net.codinux.banking.client.model
|
|
||||||
|
|
||||||
import kotlinx.datetime.Clock
|
|
||||||
import kotlinx.datetime.Instant
|
|
||||||
|
|
||||||
open class MessageLogEntry(
|
|
||||||
open val type: MessageLogEntryType,
|
|
||||||
open val message: String,
|
|
||||||
open val messageTrace: String? = null,
|
|
||||||
open val error: Throwable? = null,
|
|
||||||
open val time: Instant = Clock.System.now()
|
|
||||||
)
|
|
|
@ -1,9 +0,0 @@
|
||||||
package net.codinux.banking.client.model
|
|
||||||
|
|
||||||
enum class MessageLogEntryType {
|
|
||||||
Sent,
|
|
||||||
|
|
||||||
Received,
|
|
||||||
|
|
||||||
Error
|
|
||||||
}
|
|
|
@ -1,65 +0,0 @@
|
||||||
package net.codinux.banking.client.model
|
|
||||||
|
|
||||||
import net.codinux.banking.client.model.config.JsonIgnore
|
|
||||||
import net.codinux.banking.client.model.config.NoArgConstructor
|
|
||||||
import net.codinux.banking.client.model.tan.TanMedium
|
|
||||||
import net.codinux.banking.client.model.tan.TanMethod
|
|
||||||
|
|
||||||
@Suppress("RUNTIME_ANNOTATION_NOT_SUPPORTED")
|
|
||||||
@NoArgConstructor
|
|
||||||
open class UserAccount(
|
|
||||||
val bankCode: String,
|
|
||||||
var loginName: String,
|
|
||||||
/**
|
|
||||||
* User may decides to not save password .
|
|
||||||
*/
|
|
||||||
var password: String?,
|
|
||||||
|
|
||||||
val bankName: String,
|
|
||||||
val bic: String,
|
|
||||||
|
|
||||||
val customerName: String,
|
|
||||||
val userId: String = loginName,
|
|
||||||
|
|
||||||
open val accounts: List<BankAccount> = emptyList(),
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Identifier of selected TanMethod.
|
|
||||||
*
|
|
||||||
* As [tanMethods] also contains selected TanMethod, we didn't want to duplicate this object. Use
|
|
||||||
* [selectedTanMethod] to get selected TanMethod or iterate over [tanMethods] and filter selected one by this id.
|
|
||||||
*/
|
|
||||||
val selectedTanMethodId: String? = null,
|
|
||||||
open val tanMethods: List<TanMethod> = listOf(),
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Identifier of selected TanMedium.
|
|
||||||
*
|
|
||||||
* As [tanMedia] also contains selected TanMedium, we didn't want to duplicate this object. Use [selectedTanMedium]
|
|
||||||
* to get selected TanMedium or iterate over [tanMedia] and filter selected one by this medium name.
|
|
||||||
*/
|
|
||||||
val selectedTanMediumName: String? = null,
|
|
||||||
open val tanMedia: List<TanMedium> = listOf(),
|
|
||||||
|
|
||||||
var bankingGroup: BankingGroup? = null,
|
|
||||||
var iconUrl: String? = null,
|
|
||||||
) {
|
|
||||||
|
|
||||||
var wrongCredentialsEntered: Boolean = false
|
|
||||||
|
|
||||||
var userSetDisplayName: String? = null
|
|
||||||
var displayIndex: Int = 0
|
|
||||||
|
|
||||||
|
|
||||||
@get:JsonIgnore
|
|
||||||
val selectedTanMethod: TanMethod
|
|
||||||
get() = tanMethods.first { it.identifier == selectedTanMethodId }
|
|
||||||
|
|
||||||
@get:JsonIgnore
|
|
||||||
val selectedTanMedium: TanMedium?
|
|
||||||
get() = tanMedia.firstOrNull { it.mediumName == selectedTanMediumName }
|
|
||||||
|
|
||||||
|
|
||||||
override fun toString() = "$bankName $loginName, ${accounts.size} accounts"
|
|
||||||
|
|
||||||
}
|
|
|
@ -1,17 +0,0 @@
|
||||||
package net.codinux.banking.client.model
|
|
||||||
|
|
||||||
import net.codinux.banking.client.model.config.NoArgConstructor
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Contains only the basic info of a [UserAccount], just enough that a client application can display it to the user
|
|
||||||
* and the user knows exactly which [UserAccount] is meant / referred.
|
|
||||||
*/
|
|
||||||
@NoArgConstructor
|
|
||||||
open class UserAccountViewInfo(
|
|
||||||
val bankCode: String,
|
|
||||||
var loginName: String,
|
|
||||||
val bankName: String,
|
|
||||||
val bankingGroup: BankingGroup? = null
|
|
||||||
) {
|
|
||||||
override fun toString() = "$bankCode $bankName $loginName"
|
|
||||||
}
|
|
|
@ -1,45 +1,15 @@
|
||||||
package net.codinux.banking.client.model.options
|
package net.codinux.banking.client.model.options
|
||||||
|
|
||||||
import kotlinx.datetime.LocalDate
|
import kotlinx.datetime.LocalDate
|
||||||
import net.codinux.banking.client.model.BankAccountIdentifier
|
|
||||||
import net.codinux.banking.client.model.config.NoArgConstructor
|
import net.codinux.banking.client.model.config.NoArgConstructor
|
||||||
import net.codinux.banking.client.model.tan.TanMethodType
|
|
||||||
|
|
||||||
@NoArgConstructor
|
@NoArgConstructor
|
||||||
open class GetAccountDataOptions(
|
open class GetAccountDataOptions(
|
||||||
|
val retrieveBalance: Boolean = true,
|
||||||
val retrieveTransactions: RetrieveTransactions = RetrieveTransactions.OfLast90Days,
|
val retrieveTransactions: RetrieveTransactions = RetrieveTransactions.OfLast90Days,
|
||||||
val retrieveTransactionsFrom: LocalDate? = null,
|
val retrieveTransactionsFrom: LocalDate? = null,
|
||||||
val retrieveTransactionsTo: LocalDate? = null,
|
val retrieveTransactionsTo: LocalDate? = null,
|
||||||
|
val abortIfTanIsRequired: Boolean = false
|
||||||
val retrieveBalance: Boolean = true,
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Account(s) may should be excluded from data retrieval, so this option enabled to set for which accounts data
|
|
||||||
* should be retrieved.
|
|
||||||
*/
|
|
||||||
val accounts: List<BankAccountIdentifier> = emptyList(),
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Specifies which [TanMethodType] should be preferred when having to choose between multiple available for user
|
|
||||||
* without requesting the user to choose one.
|
|
||||||
*
|
|
||||||
* By default we don't ask the user which TanMethod she prefers but choose one that could match best. If she really
|
|
||||||
* likes to use a different one, she can select another one in EnterTanDialog.
|
|
||||||
*
|
|
||||||
* By default we prefer non visual TanMethods (like AppTan and SMS) over image based TanMethods (like QR-code and
|
|
||||||
* photoTan) and exclude ChipTanUsb, which is not supported by application, and Flickercode, which is hard to
|
|
||||||
* implement and therefore most applications have not implemented.
|
|
||||||
*
|
|
||||||
* Console apps can only handle non visual TanMethods.
|
|
||||||
* But also graphical applications prefer non visual TanMethods as then they only have to display a text field to input
|
|
||||||
* TAN, and then image based TanMethods as then they additionally only have to display an image.
|
|
||||||
*/
|
|
||||||
val preferredTanMethods: List<TanMethodType>? = TanMethodType.NonVisualOrImageBased,
|
|
||||||
|
|
||||||
val abortIfTanIsRequired: Boolean = false,
|
|
||||||
|
|
||||||
// there's also the option preferredTanMedium, but can hardly find a use case for it as we
|
|
||||||
// cannot know the TanMedium name upfront. In most cases there's only one TanMedium (per TanMethod) anyway.
|
|
||||||
) {
|
) {
|
||||||
override fun toString(): String {
|
override fun toString(): String {
|
||||||
return "retrieveBalance=$retrieveBalance, retrieveTransactions=$retrieveTransactions, abortIfTanIsRequired=$abortIfTanIsRequired"
|
return "retrieveBalance=$retrieveBalance, retrieveTransactions=$retrieveTransactions, abortIfTanIsRequired=$abortIfTanIsRequired"
|
||||||
|
|
|
@ -1,20 +1,20 @@
|
||||||
package net.codinux.banking.client.model.response
|
package net.codinux.banking.client.model.response
|
||||||
|
|
||||||
import net.codinux.banking.client.model.AccountTransaction
|
import net.codinux.banking.client.model.AccountTransaction
|
||||||
import net.codinux.banking.client.model.UserAccount
|
import net.codinux.banking.client.model.CustomerAccount
|
||||||
import net.codinux.banking.client.model.config.JsonIgnore
|
import net.codinux.banking.client.model.config.JsonIgnore
|
||||||
import net.codinux.banking.client.model.config.NoArgConstructor
|
import net.codinux.banking.client.model.config.NoArgConstructor
|
||||||
|
|
||||||
@Suppress("RUNTIME_ANNOTATION_NOT_SUPPORTED")
|
@Suppress("RUNTIME_ANNOTATION_NOT_SUPPORTED")
|
||||||
@NoArgConstructor
|
@NoArgConstructor
|
||||||
open class GetAccountDataResponse(
|
open class GetAccountDataResponse(
|
||||||
val user: UserAccount
|
val customer: CustomerAccount
|
||||||
) {
|
) {
|
||||||
|
|
||||||
@get:JsonIgnore
|
@get:JsonIgnore
|
||||||
val bookedTransactions: List<AccountTransaction>
|
val bookedTransactions: List<AccountTransaction>
|
||||||
get() = user.accounts.flatMap { it.bookedTransactions }.sortedByDescending { it.valueDate }
|
get() = customer.accounts.flatMap { it.bookedTransactions }.sortedByDescending { it.valueDate }
|
||||||
|
|
||||||
|
|
||||||
override fun toString() = user.toString()
|
override fun toString() = customer.toString()
|
||||||
}
|
}
|
|
@ -1,12 +1,10 @@
|
||||||
package net.codinux.banking.client.model.tan
|
package net.codinux.banking.client.model.tan
|
||||||
|
|
||||||
import net.codinux.banking.client.model.BankAccountIdentifier
|
|
||||||
import net.codinux.banking.client.model.config.NoArgConstructor
|
import net.codinux.banking.client.model.config.NoArgConstructor
|
||||||
|
|
||||||
@NoArgConstructor
|
@NoArgConstructor
|
||||||
open class MobilePhoneTanMedium(
|
open class MobilePhoneTanMedium(
|
||||||
val phoneNumber: String?,
|
val phoneNumber: String?
|
||||||
val concealedPhoneNumber: String? = null
|
|
||||||
) {
|
) {
|
||||||
override fun toString() = phoneNumber ?: "No phone number"
|
override fun toString() = phoneNumber ?: "No phone number"
|
||||||
}
|
}
|
|
@ -1,61 +1,24 @@
|
||||||
package net.codinux.banking.client.model.tan
|
package net.codinux.banking.client.model.tan
|
||||||
|
|
||||||
import net.codinux.banking.client.model.BankAccountViewInfo
|
import net.codinux.banking.client.model.BankAccountViewInfo
|
||||||
import net.codinux.banking.client.model.UserAccount
|
import net.codinux.banking.client.model.CustomerAccountViewInfo
|
||||||
import net.codinux.banking.client.model.UserAccountViewInfo
|
|
||||||
import net.codinux.banking.client.model.config.JsonIgnore
|
|
||||||
import net.codinux.banking.client.model.config.NoArgConstructor
|
import net.codinux.banking.client.model.config.NoArgConstructor
|
||||||
|
|
||||||
@Suppress("RUNTIME_ANNOTATION_NOT_SUPPORTED")
|
|
||||||
@NoArgConstructor
|
@NoArgConstructor
|
||||||
open class TanChallenge(
|
open class TanChallenge(
|
||||||
val type: TanChallengeType,
|
val type: TanChallengeType,
|
||||||
val forAction: ActionRequiringTan,
|
val forAction: ActionRequiringTan,
|
||||||
val messageToShowToUser: String,
|
val messageToShowToUser: String,
|
||||||
|
val tanMethod: TanMethod,
|
||||||
/**
|
|
||||||
* Identifier of selected TanMethod.
|
|
||||||
*
|
|
||||||
* As [availableTanMethods] also contains selected TanMethod, we didn't want to duplicate this object. Use
|
|
||||||
* [selectedTanMethod] to get selected TanMethod or iterate over [availableTanMethods] and filter selected one by this id.
|
|
||||||
*/
|
|
||||||
val selectedTanMethodId: String,
|
|
||||||
/**
|
|
||||||
* When adding an account, frontend has no UserAccount object in BankingClientCallback to know which TanMethods are
|
|
||||||
* available for User.
|
|
||||||
* Also on other calls to bank server, bank server may returned an updated list of available TanMethods, so that
|
|
||||||
* [UserAccount] may contains an outdated list of available TanMethods.
|
|
||||||
*
|
|
||||||
* Therefore i added list with up to date TanMethods here to ensure EnterTanDialog can display user's up to date TanMethods.
|
|
||||||
*/
|
|
||||||
val availableTanMethods: List<TanMethod>,
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Identifier of selected TanMedium.
|
|
||||||
*
|
|
||||||
* As [availableTanMedia] also contains selected TanMedium, we didn't want to duplicate this object. Use
|
|
||||||
* [selectedTanMedium] to get selected TanMedium or iterate over [availableTanMedia] and filter selected one by this medium name.
|
|
||||||
*/
|
|
||||||
val selectedTanMediumName: String? = null,
|
|
||||||
val availableTanMedia: List<TanMedium> = emptyList(),
|
|
||||||
|
|
||||||
val tanImage: TanImage? = null,
|
val tanImage: TanImage? = null,
|
||||||
val flickerCode: FlickerCode? = null,
|
val flickerCode: FlickerCode? = null,
|
||||||
val user: UserAccountViewInfo,
|
val customer: CustomerAccountViewInfo,
|
||||||
val account: BankAccountViewInfo? = null
|
val account: BankAccountViewInfo? = null
|
||||||
|
// TODO: add availableTanMethods, selectedTanMedium, availableTanMedia
|
||||||
) {
|
) {
|
||||||
|
|
||||||
@get:JsonIgnore
|
|
||||||
val selectedTanMethod: TanMethod
|
|
||||||
get() = availableTanMethods.first { it.identifier == selectedTanMethodId }
|
|
||||||
|
|
||||||
@get:JsonIgnore
|
|
||||||
val selectedTanMedium: TanMedium?
|
|
||||||
get() = availableTanMedia.firstOrNull { it.mediumName == selectedTanMediumName }
|
|
||||||
|
|
||||||
|
|
||||||
override fun toString(): String {
|
override fun toString(): String {
|
||||||
return "$selectedTanMethod $forAction: $messageToShowToUser" + when (type) {
|
return "$tanMethod $forAction: $messageToShowToUser" + when (type) {
|
||||||
TanChallengeType.EnterTan -> ""
|
TanChallengeType.EnterTan -> ""
|
||||||
TanChallengeType.Image -> ", Image: $tanImage"
|
TanChallengeType.Image -> ", Image: $tanImage"
|
||||||
TanChallengeType.Flickercode -> ", FlickerCode: $flickerCode"
|
TanChallengeType.Flickercode -> ", FlickerCode: $flickerCode"
|
||||||
|
|
|
@ -1,15 +1,10 @@
|
||||||
package net.codinux.banking.client.model.tan
|
package net.codinux.banking.client.model.tan
|
||||||
|
|
||||||
import kotlinx.datetime.LocalDate
|
|
||||||
import net.codinux.banking.client.model.config.NoArgConstructor
|
import net.codinux.banking.client.model.config.NoArgConstructor
|
||||||
|
|
||||||
@NoArgConstructor
|
@NoArgConstructor
|
||||||
open class TanGeneratorTanMedium(
|
open class TanGeneratorTanMedium(
|
||||||
val cardNumber: String,
|
val cardNumber: String
|
||||||
val cardSequenceNumber: String? = null,
|
|
||||||
val cardType: Int? = null,
|
|
||||||
val validFrom: LocalDate? = null,
|
|
||||||
val validTo: LocalDate? = null
|
|
||||||
) {
|
) {
|
||||||
override fun toString() = cardNumber
|
override fun toString() = cardNumber
|
||||||
}
|
}
|
|
@ -5,7 +5,7 @@ import net.codinux.banking.client.model.config.NoArgConstructor
|
||||||
@NoArgConstructor
|
@NoArgConstructor
|
||||||
open class TanMedium(
|
open class TanMedium(
|
||||||
val type: TanMediumType,
|
val type: TanMediumType,
|
||||||
val mediumName: String?,
|
val displayName: String,
|
||||||
val status: TanMediumStatus,
|
val status: TanMediumStatus,
|
||||||
/**
|
/**
|
||||||
* Only set if [type] is [TanMediumType.TanGenerator].
|
* Only set if [type] is [TanMediumType.TanGenerator].
|
||||||
|
@ -16,5 +16,5 @@ open class TanMedium(
|
||||||
*/
|
*/
|
||||||
val mobilePhone: MobilePhoneTanMedium? = null
|
val mobilePhone: MobilePhoneTanMedium? = null
|
||||||
) {
|
) {
|
||||||
override fun toString() = "$mediumName $status"
|
override fun toString() = "$displayName $status"
|
||||||
}
|
}
|
|
@ -1,25 +1,7 @@
|
||||||
package net.codinux.banking.client.model.tan
|
package net.codinux.banking.client.model.tan
|
||||||
|
|
||||||
enum class TanMediumStatus {
|
enum class TanMediumStatus {
|
||||||
|
|
||||||
/**
|
|
||||||
* Die Bank zeigt an, dass es eine TAN-Verifikation gegen dieses Medium vornimmt.
|
|
||||||
*/
|
|
||||||
Used,
|
Used,
|
||||||
|
|
||||||
/**
|
Available
|
||||||
* Das Medium kann genutzt werden, muss aber zuvor mit „TAN-Generator an- bzw. ummelden (HKTAU)“ aktiv gemeldet werden.
|
|
||||||
*/
|
|
||||||
Available,
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Mit der ersten Nutzung der Folgekarte wird die zur Zeit aktive Karte gesperrt.
|
|
||||||
*/
|
|
||||||
ActiveFollowUpCard,
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Das Medium kann mit dem Geschäftsvorfall „TAN-Medium an- bzw. ummelden (HKTAU)“ aktiv gemeldet werden.
|
|
||||||
* Die aktuelle Karte kann dann nicht mehr genutzt werden.
|
|
||||||
*/
|
|
||||||
AvailableFollowUpCard
|
|
||||||
}
|
}
|
|
@ -20,21 +20,4 @@ enum class TanMethodType {
|
||||||
photoTan,
|
photoTan,
|
||||||
|
|
||||||
QrCode
|
QrCode
|
||||||
;
|
|
||||||
|
|
||||||
|
|
||||||
companion object {
|
|
||||||
|
|
||||||
val NonVisual = listOf(TanMethodType.AppTan, TanMethodType.SmsTan, TanMethodType.ChipTanManuell, TanMethodType.EnterTan)
|
|
||||||
|
|
||||||
val ImageBased = listOf(TanMethodType.QrCode, TanMethodType.ChipTanQrCode, TanMethodType.photoTan, TanMethodType.ChipTanPhotoTanMatrixCode)
|
|
||||||
|
|
||||||
val NonVisualOrImageBased = buildList {
|
|
||||||
addAll(listOf(TanMethodType.AppTan, TanMethodType.SmsTan, TanMethodType.EnterTan))
|
|
||||||
addAll(ImageBased)
|
|
||||||
addAll(listOf(TanMethodType.ChipTanManuell)) // this is quite inconvenient for user, so i added it at last
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,5 +1,3 @@
|
||||||
package net.codinux.banking.client.model.config
|
package net.codinux.banking.client.model.config
|
||||||
|
|
||||||
@Target(AnnotationTarget.ANNOTATION_CLASS, AnnotationTarget.FUNCTION, AnnotationTarget.CONSTRUCTOR, AnnotationTarget.FIELD, AnnotationTarget.PROPERTY_GETTER)
|
|
||||||
@Retention(AnnotationRetention.RUNTIME)
|
|
||||||
actual annotation class JsonIgnore
|
actual annotation class JsonIgnore
|
|
@ -1,5 +1,3 @@
|
||||||
package net.codinux.banking.client.model.config
|
package net.codinux.banking.client.model.config
|
||||||
|
|
||||||
@Target(AnnotationTarget.ANNOTATION_CLASS, AnnotationTarget.FUNCTION, AnnotationTarget.CONSTRUCTOR, AnnotationTarget.FIELD, AnnotationTarget.PROPERTY_GETTER)
|
|
||||||
@Retention(AnnotationRetention.RUNTIME)
|
|
||||||
actual annotation class JsonIgnore
|
actual annotation class JsonIgnore
|
|
@ -1,5 +1,3 @@
|
||||||
package net.codinux.banking.client.model.config
|
package net.codinux.banking.client.model.config
|
||||||
|
|
||||||
@Target(AnnotationTarget.ANNOTATION_CLASS, AnnotationTarget.FUNCTION, AnnotationTarget.CONSTRUCTOR, AnnotationTarget.FIELD, AnnotationTarget.PROPERTY_GETTER)
|
|
||||||
@Retention(AnnotationRetention.RUNTIME)
|
|
||||||
actual annotation class JsonIgnore
|
actual annotation class JsonIgnore
|
|
@ -9,6 +9,11 @@ plugins {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
mavenLocal()
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
kotlin {
|
kotlin {
|
||||||
jvmToolchain(8)
|
jvmToolchain(8)
|
||||||
|
|
||||||
|
@ -49,13 +54,12 @@ kotlin {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// wasmJs() // ktor is not available for wasmJs yet
|
// wasmJs()
|
||||||
|
|
||||||
|
|
||||||
linuxX64()
|
linuxX64()
|
||||||
mingwX64()
|
mingwX64()
|
||||||
|
|
||||||
iosX64()
|
|
||||||
iosArm64()
|
iosArm64()
|
||||||
iosSimulatorArm64()
|
iosSimulatorArm64()
|
||||||
macosX64()
|
macosX64()
|
||||||
|
@ -77,7 +81,7 @@ kotlin {
|
||||||
dependencies {
|
dependencies {
|
||||||
api(project(":BankingClient"))
|
api(project(":BankingClient"))
|
||||||
|
|
||||||
api("net.codinux.banking:fints4k:1.0.0-Alpha-12")
|
api("net.codinux.banking:fints4k:1.0.0-Alpha-11")
|
||||||
|
|
||||||
api("org.jetbrains.kotlinx:kotlinx-datetime:$kotlinxDateTimeVersion")
|
api("org.jetbrains.kotlinx:kotlinx-datetime:$kotlinxDateTimeVersion")
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,13 +1,11 @@
|
||||||
package net.codinux.banking.client.fints4k
|
package net.codinux.banking.client.fints4k
|
||||||
|
|
||||||
import net.codinux.banking.client.BankingClientCallback
|
import net.codinux.banking.client.BankingClientCallback
|
||||||
import net.codinux.banking.client.model.MessageLogEntryType
|
import net.dankito.banking.fints.callback.FinTsClientCallback
|
||||||
import net.codinux.banking.fints.callback.FinTsClientCallback
|
import net.dankito.banking.fints.messages.datenelemente.implementierte.tan.TanGeneratorTanMedium
|
||||||
import net.codinux.banking.fints.messages.datenelemente.implementierte.tan.TanGeneratorTanMedium
|
import net.dankito.banking.fints.model.BankData
|
||||||
import net.codinux.banking.fints.model.BankData
|
import net.dankito.banking.fints.model.EnterTanGeneratorAtcResult
|
||||||
import net.codinux.banking.fints.model.EnterTanGeneratorAtcResult
|
import net.dankito.banking.fints.model.TanMethod
|
||||||
import net.codinux.banking.fints.model.MessageLogEntry
|
|
||||||
import net.codinux.banking.fints.model.TanMethod
|
|
||||||
|
|
||||||
open class BridgeFintTsToBankingClientCallback(
|
open class BridgeFintTsToBankingClientCallback(
|
||||||
protected val bankingClientCallback: BankingClientCallback,
|
protected val bankingClientCallback: BankingClientCallback,
|
||||||
|
@ -18,7 +16,7 @@ open class BridgeFintTsToBankingClientCallback(
|
||||||
return suggestedTanMethod
|
return suggestedTanMethod
|
||||||
}
|
}
|
||||||
|
|
||||||
override suspend fun enterTan(tanChallenge: net.codinux.banking.fints.model.TanChallenge) {
|
override suspend fun enterTan(tanChallenge: net.dankito.banking.fints.model.TanChallenge) {
|
||||||
bankingClientCallback.enterTan(mapper.mapTanChallenge(tanChallenge)) { enterTanResult ->
|
bankingClientCallback.enterTan(mapper.mapTanChallenge(tanChallenge)) { enterTanResult ->
|
||||||
if (enterTanResult.enteredTan != null) {
|
if (enterTanResult.enteredTan != null) {
|
||||||
tanChallenge.userEnteredTan(enterTanResult.enteredTan!!)
|
tanChallenge.userEnteredTan(enterTanResult.enteredTan!!)
|
||||||
|
@ -31,14 +29,4 @@ open class BridgeFintTsToBankingClientCallback(
|
||||||
override suspend fun enterTanGeneratorAtc(bank: BankData, tanMedium: TanGeneratorTanMedium): EnterTanGeneratorAtcResult {
|
override suspend fun enterTanGeneratorAtc(bank: BankData, tanMedium: TanGeneratorTanMedium): EnterTanGeneratorAtcResult {
|
||||||
return EnterTanGeneratorAtcResult.userDidNotEnterAtc()
|
return EnterTanGeneratorAtcResult.userDidNotEnterAtc()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun messageLogAdded(messageLogEntry: MessageLogEntry) {
|
|
||||||
val mapped = net.codinux.banking.client.model.MessageLogEntry(
|
|
||||||
MessageLogEntryType.valueOf(messageLogEntry.type.name),
|
|
||||||
messageLogEntry.message, messageLogEntry.messageTrace,
|
|
||||||
messageLogEntry.error, messageLogEntry.time
|
|
||||||
)
|
|
||||||
|
|
||||||
bankingClientCallback.messageLogAdded(mapped)
|
|
||||||
}
|
|
||||||
}
|
}
|
|
@ -1,17 +0,0 @@
|
||||||
package net.codinux.banking.client.fints4k
|
|
||||||
|
|
||||||
import net.codinux.banking.client.BankingClientCallback
|
|
||||||
import net.codinux.banking.client.BankingClientForUserBase
|
|
||||||
import net.codinux.banking.client.model.AccountCredentials
|
|
||||||
import net.codinux.banking.fints.config.FinTsClientConfiguration
|
|
||||||
|
|
||||||
open class FinTs4KBankingClientForUser(credentials: AccountCredentials, config: FinTsClientConfiguration = FinTsClientConfiguration(), callback: BankingClientCallback)
|
|
||||||
: BankingClientForUserBase(credentials, FinTs4kBankingClient(config, callback)) {
|
|
||||||
|
|
||||||
constructor(bankCode: String, loginName: String, password: String, callback: BankingClientCallback)
|
|
||||||
: this(bankCode, loginName, password, FinTsClientConfiguration(), callback)
|
|
||||||
|
|
||||||
constructor(bankCode: String, loginName: String, password: String, config: FinTsClientConfiguration = FinTsClientConfiguration(), callback: BankingClientCallback)
|
|
||||||
: this(AccountCredentials(bankCode, loginName, password), config, callback)
|
|
||||||
|
|
||||||
}
|
|
|
@ -6,20 +6,15 @@ import net.codinux.banking.client.model.options.GetAccountDataOptions
|
||||||
import net.codinux.banking.client.model.request.GetAccountDataRequest
|
import net.codinux.banking.client.model.request.GetAccountDataRequest
|
||||||
import net.codinux.banking.client.model.response.GetAccountDataResponse
|
import net.codinux.banking.client.model.response.GetAccountDataResponse
|
||||||
import net.codinux.banking.client.model.response.Response
|
import net.codinux.banking.client.model.response.Response
|
||||||
import net.codinux.banking.fints.FinTsClient
|
import net.dankito.banking.fints.FinTsClient
|
||||||
import net.codinux.banking.fints.config.FinTsClientConfiguration
|
|
||||||
|
|
||||||
open class FinTs4kBankingClient(
|
open class FinTs4kBankingClient(
|
||||||
config: FinTsClientConfiguration = FinTsClientConfiguration(),
|
|
||||||
callback: BankingClientCallback
|
callback: BankingClientCallback
|
||||||
) : BankingClient {
|
) : BankingClient {
|
||||||
|
|
||||||
constructor(callback: BankingClientCallback) : this(FinTsClientConfiguration(), callback)
|
private val mapper = FinTs4kMapper()
|
||||||
|
|
||||||
|
private val client = FinTsClient(BridgeFintTsToBankingClientCallback(callback, mapper))
|
||||||
protected val mapper = FinTs4kMapper()
|
|
||||||
|
|
||||||
protected val client = FinTsClient(config, BridgeFintTsToBankingClientCallback(callback, mapper))
|
|
||||||
|
|
||||||
|
|
||||||
override suspend fun getAccountDataAsync(request: GetAccountDataRequest): Response<GetAccountDataResponse> {
|
override suspend fun getAccountDataAsync(request: GetAccountDataRequest): Response<GetAccountDataResponse> {
|
||||||
|
|
|
@ -0,0 +1,13 @@
|
||||||
|
package net.codinux.banking.client.fints4k
|
||||||
|
|
||||||
|
import net.codinux.banking.client.BankingClientCallback
|
||||||
|
import net.codinux.banking.client.BankingClientForCustomerBase
|
||||||
|
import net.codinux.banking.client.model.AccountCredentials
|
||||||
|
|
||||||
|
open class FinTs4kBankingClientForCustomer(credentials: AccountCredentials, callback: BankingClientCallback)
|
||||||
|
: BankingClientForCustomerBase(credentials, FinTs4kBankingClient(callback)) {
|
||||||
|
|
||||||
|
constructor(bankCode: String, loginName: String, password: String, callback: BankingClientCallback)
|
||||||
|
: this(AccountCredentials(bankCode, loginName, password), callback)
|
||||||
|
|
||||||
|
}
|
|
@ -15,72 +15,53 @@ import net.dankito.banking.client.model.BankAccountIdentifierImpl
|
||||||
import net.dankito.banking.client.model.parameter.GetAccountDataParameter
|
import net.dankito.banking.client.model.parameter.GetAccountDataParameter
|
||||||
import net.dankito.banking.client.model.parameter.RetrieveTransactions
|
import net.dankito.banking.client.model.parameter.RetrieveTransactions
|
||||||
import net.dankito.banking.client.model.response.ErrorCode
|
import net.dankito.banking.client.model.response.ErrorCode
|
||||||
import net.codinux.banking.fints.mapper.FinTsModelMapper
|
import net.dankito.banking.fints.mapper.FinTsModelMapper
|
||||||
import net.codinux.banking.fints.model.*
|
import net.dankito.banking.fints.model.*
|
||||||
import net.codinux.banking.fints.messages.datenelemente.implementierte.tan.TanMedium
|
|
||||||
import net.codinux.banking.fints.messages.datenelemente.implementierte.tan.MobilePhoneTanMedium
|
|
||||||
import net.codinux.banking.fints.messages.datenelemente.implementierte.tan.TanGeneratorTanMedium
|
|
||||||
import net.codinux.banking.fints.messages.datenelemente.implementierte.tan.TanMediumStatus
|
|
||||||
import net.dankito.banking.banklistcreator.prettifier.BankingGroupMapper
|
|
||||||
import kotlin.io.encoding.Base64
|
import kotlin.io.encoding.Base64
|
||||||
import kotlin.io.encoding.ExperimentalEncodingApi
|
import kotlin.io.encoding.ExperimentalEncodingApi
|
||||||
|
|
||||||
open class FinTs4kMapper {
|
open class FinTs4kMapper {
|
||||||
|
|
||||||
protected val fintsModelMapper = FinTsModelMapper()
|
private val fintsModelMapper = FinTsModelMapper()
|
||||||
|
|
||||||
protected val bankingGroupMapper = BankingGroupMapper()
|
|
||||||
|
|
||||||
|
|
||||||
open fun mapToGetAccountDataParameter(credentials: AccountCredentials, options: GetAccountDataOptions) = GetAccountDataParameter(
|
fun mapToGetAccountDataParameter(credentials: AccountCredentials, options: GetAccountDataOptions) = GetAccountDataParameter(
|
||||||
credentials.bankCode, credentials.loginName, credentials.password,
|
credentials.bankCode, credentials.loginName, credentials.password,
|
||||||
options.accounts.map { BankAccountIdentifierImpl(it.identifier, it.subAccountNumber, it.iban) },
|
options.accounts.map { BankAccountIdentifierImpl(it.identifier, it.subAccountNumber, it.iban) },
|
||||||
options.retrieveBalance,
|
options.retrieveBalance,
|
||||||
RetrieveTransactions.valueOf(options.retrieveTransactions.name), options.retrieveTransactionsFrom, options.retrieveTransactionsTo,
|
RetrieveTransactions.valueOf(options.retrieveTransactions.name), options.retrieveTransactionsFrom, options.retrieveTransactionsTo,
|
||||||
preferredTanMethods = options.preferredTanMethods?.map { mapTanMethodType(it) },
|
|
||||||
abortIfTanIsRequired = options.abortIfTanIsRequired
|
abortIfTanIsRequired = options.abortIfTanIsRequired
|
||||||
)
|
)
|
||||||
|
|
||||||
protected open fun mapTanMethodType(type: TanMethodType): net.codinux.banking.fints.model.TanMethodType =
|
|
||||||
net.codinux.banking.fints.model.TanMethodType.valueOf(type.name)
|
|
||||||
|
|
||||||
|
fun map(response: net.dankito.banking.client.model.response.GetAccountDataResponse): Response<GetAccountDataResponse> {
|
||||||
open fun map(response: net.dankito.banking.client.model.response.GetAccountDataResponse): Response<GetAccountDataResponse> {
|
|
||||||
return if (response.successful && response.customerAccount != null) {
|
return if (response.successful && response.customerAccount != null) {
|
||||||
Response.success(GetAccountDataResponse(mapUser(response.customerAccount!!)))
|
Response.success(GetAccountDataResponse(mapCustomer(response.customerAccount!!)))
|
||||||
} else {
|
} else {
|
||||||
mapError(response)
|
mapError(response)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
open fun mapToUserAccountViewInfo(bank: BankData): UserAccountViewInfo = UserAccountViewInfo(
|
fun mapToCustomerAccountViewInfo(bank: BankData): CustomerAccountViewInfo = CustomerAccountViewInfo(
|
||||||
bank.bankCode, bank.customerId, bank.bankName, getBankingGroup(bank.bankName, bank.bic)
|
bank.bankCode, bank.customerId, bank.bankName
|
||||||
)
|
)
|
||||||
|
|
||||||
open fun mapToBankAccountViewInfo(account: AccountData): BankAccountViewInfo = BankAccountViewInfo(
|
fun mapToBankAccountViewInfo(account: AccountData): BankAccountViewInfo = BankAccountViewInfo(
|
||||||
account.accountIdentifier, account.subAccountAttribute,
|
account.accountIdentifier, account.subAccountAttribute,
|
||||||
mapAccountType(fintsModelMapper.map(account.accountType)),
|
mapAccountType(fintsModelMapper.map(account.accountType)),
|
||||||
account.iban, account.productName
|
account.iban, account.productName
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
protected open fun mapUser(user: net.dankito.banking.client.model.CustomerAccount) = UserAccount(
|
private fun mapCustomer(customer: net.dankito.banking.client.model.CustomerAccount): CustomerAccount = CustomerAccount(
|
||||||
user.bankCode, user.loginName, user.password,
|
customer.bankCode, customer.loginName, customer.password,
|
||||||
user.bankName, user.bic, user.customerName, user.userId,
|
customer.bankName, customer.bic, customer.customerName, customer.userId,
|
||||||
user.accounts.map { mapAccount(it) },
|
customer.accounts.map { mapAccount(it) }
|
||||||
|
|
||||||
user.selectedTanMethod?.securityFunction?.code, user.tanMethods.map { mapTanMethod(it) },
|
|
||||||
user.selectedTanMedium?.mediumName, user.tanMedia.map { mapTanMedium(it) },
|
|
||||||
|
|
||||||
getBankingGroup(user.bankName, user.bic)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
protected open fun getBankingGroup(bankName: String, bic: String): BankingGroup? =
|
|
||||||
bankingGroupMapper.getBankingGroup(bankName, bic)
|
|
||||||
|
|
||||||
|
private fun mapAccount(account: net.dankito.banking.client.model.BankAccount): BankAccount = BankAccount(
|
||||||
protected open fun mapAccount(account: net.dankito.banking.client.model.BankAccount) = BankAccount(
|
|
||||||
account.identifier, account.accountHolderName, mapAccountType(account.type), account.iban, account.subAccountNumber,
|
account.identifier, account.accountHolderName, mapAccountType(account.type), account.iban, account.subAccountNumber,
|
||||||
account.productName, account.currency, account.accountLimit, account.isAccountTypeSupportedByApplication,
|
account.productName, account.currency, account.accountLimit, account.isAccountTypeSupportedByApplication,
|
||||||
mapFeatures(account),
|
mapFeatures(account),
|
||||||
|
@ -90,10 +71,10 @@ open class FinTs4kMapper {
|
||||||
bookedTransactions = account.bookedTransactions.map { mapTransaction(it) }.toMutableList()
|
bookedTransactions = account.bookedTransactions.map { mapTransaction(it) }.toMutableList()
|
||||||
)
|
)
|
||||||
|
|
||||||
protected open fun mapAccountType(type: net.dankito.banking.client.model.BankAccountType): BankAccountType =
|
private fun mapAccountType(type: net.dankito.banking.client.model.BankAccountType): BankAccountType =
|
||||||
BankAccountType.valueOf(type.name)
|
BankAccountType.valueOf(type.name)
|
||||||
|
|
||||||
protected open fun mapFeatures(account: net.dankito.banking.client.model.BankAccount): Set<BankAccountFeatures> = buildSet {
|
private fun mapFeatures(account: net.dankito.banking.client.model.BankAccount): Set<BankAccountFeatures> = buildSet {
|
||||||
if (account.supportsRetrievingBalance) {
|
if (account.supportsRetrievingBalance) {
|
||||||
add(BankAccountFeatures.RetrieveBalance)
|
add(BankAccountFeatures.RetrieveBalance)
|
||||||
}
|
}
|
||||||
|
@ -109,112 +90,66 @@ open class FinTs4kMapper {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
protected open fun mapTransaction(transaction: net.dankito.banking.client.model.AccountTransaction): AccountTransaction = AccountTransaction(
|
private fun mapTransaction(transaction: net.dankito.banking.client.model.AccountTransaction): AccountTransaction = AccountTransaction(
|
||||||
mapAmount(transaction.amount), transaction.amount.currency.code, transaction.unparsedReference,
|
mapAmount(transaction.amount), transaction.amount.currency.code, transaction.unparsedReference,
|
||||||
transaction.bookingDate, transaction.valueDate,
|
transaction.bookingDate, transaction.valueDate,
|
||||||
transaction.otherPartyName, transaction.otherPartyBankCode, transaction.otherPartyAccountId,
|
transaction.otherPartyName, transaction.otherPartyBankCode, transaction.otherPartyAccountId,
|
||||||
transaction.bookingText, null,
|
transaction.bookingText, null,
|
||||||
transaction.statementNumber, transaction.sequenceNumber,
|
transaction.statementNumber, transaction.sequenceNumber,
|
||||||
mapNullableAmount(transaction.openingBalance), mapNullableAmount(transaction.closingBalance),
|
mapNullableAmount(transaction.openingBalance), mapNullableAmount(transaction.closingBalance),
|
||||||
|
// TODO: map other properties
|
||||||
transaction.endToEndReference, transaction.customerReference, transaction.mandateReference,
|
|
||||||
transaction.creditorIdentifier, transaction.originatorsIdentificationCode,
|
|
||||||
transaction.compensationAmount, transaction.originalAmount,
|
|
||||||
transaction.sepaReference,
|
|
||||||
transaction.deviantOriginator, transaction.deviantRecipient,
|
|
||||||
transaction.referenceWithNoSpecialType, transaction.primaNotaNumber, transaction.textKeySupplement,
|
|
||||||
|
|
||||||
transaction.currencyType, transaction.bookingKey,
|
|
||||||
transaction.referenceForTheAccountOwner, transaction.referenceOfTheAccountServicingInstitution,
|
|
||||||
transaction.supplementaryDetails,
|
|
||||||
|
|
||||||
transaction.transactionReferenceNumber, transaction.relatedReferenceNumber
|
|
||||||
)
|
)
|
||||||
|
|
||||||
protected open fun mapNullableAmount(amount: Money?) = amount?.let { mapAmount(it) }
|
private fun mapNullableAmount(amount: Money?) = amount?.let { mapAmount(it) }
|
||||||
|
|
||||||
protected open fun mapAmount(amount: Money) = Amount.fromString(amount.amount.string.replace(',', '.'))
|
private fun mapAmount(amount: Money) = Amount.fromString(amount.amount.string.replace(',', '.'))
|
||||||
|
|
||||||
|
|
||||||
open fun mapTanChallenge(challenge: net.codinux.banking.fints.model.TanChallenge): TanChallenge {
|
fun mapTanChallenge(challenge: net.dankito.banking.fints.model.TanChallenge): TanChallenge {
|
||||||
val type = mapTanChallengeType(challenge)
|
val type = mapTanChallengeType(challenge)
|
||||||
val action = mapActionRequiringTan(challenge.forAction)
|
val action = mapActionRequiringTan(challenge.forAction)
|
||||||
|
val tanMethod = mapTanMethod(challenge.tanMethod)
|
||||||
val tanMethods = challenge.bank.tanMethodsAvailableForUser.map { mapTanMethod(it) }
|
val customer = mapToCustomerAccountViewInfo(challenge.bank)
|
||||||
val selectedTanMethodId = challenge.tanMethod.securityFunction.code
|
|
||||||
|
|
||||||
val tanMedia = challenge.bank.tanMedia.map { mapTanMedium(it) }
|
|
||||||
val selectedTanMediumName = challenge.bank.selectedTanMedium?.mediumName
|
|
||||||
|
|
||||||
val user = mapToUserAccountViewInfo(challenge.bank)
|
|
||||||
val account = challenge.account?.let { mapToBankAccountViewInfo(it) }
|
val account = challenge.account?.let { mapToBankAccountViewInfo(it) }
|
||||||
|
|
||||||
val tanImage = if (challenge is ImageTanChallenge) mapTanImage(challenge.image) else null
|
val tanImage = if (challenge is ImageTanChallenge) mapTanImage(challenge.image) else null
|
||||||
val flickerCode = if (challenge is FlickerCodeTanChallenge) mapFlickerCode(challenge.flickerCode) else null
|
val flickerCode = if (challenge is FlickerCodeTanChallenge) mapFlickerCode(challenge.flickerCode) else null
|
||||||
|
|
||||||
return TanChallenge(type, action, challenge.messageToShowToUser, selectedTanMethodId, tanMethods, selectedTanMediumName, tanMedia, tanImage, flickerCode, user, account)
|
return TanChallenge(type, action, challenge.messageToShowToUser, tanMethod, tanImage, flickerCode, customer, account)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun mapTanChallengeType(challenge: net.codinux.banking.fints.model.TanChallenge): TanChallengeType = when {
|
private fun mapTanChallengeType(challenge: net.dankito.banking.fints.model.TanChallenge): TanChallengeType = when {
|
||||||
challenge is ImageTanChallenge -> TanChallengeType.Image
|
challenge is ImageTanChallenge -> TanChallengeType.Image
|
||||||
challenge is FlickerCodeTanChallenge -> TanChallengeType.Flickercode
|
challenge is FlickerCodeTanChallenge -> TanChallengeType.Flickercode
|
||||||
else -> TanChallengeType.EnterTan
|
else -> TanChallengeType.EnterTan
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun mapActionRequiringTan(action: net.codinux.banking.fints.model.ActionRequiringTan): ActionRequiringTan =
|
private fun mapActionRequiringTan(action: net.dankito.banking.fints.model.ActionRequiringTan): ActionRequiringTan =
|
||||||
ActionRequiringTan.valueOf(action.name)
|
ActionRequiringTan.valueOf(action.name)
|
||||||
|
|
||||||
protected open fun mapTanMethod(method: net.codinux.banking.fints.model.TanMethod): TanMethod = TanMethod(
|
private fun mapTanMethod(method: net.dankito.banking.fints.model.TanMethod): TanMethod = TanMethod(
|
||||||
method.displayName, mapTanMethodType(method.type), method.securityFunction.code, method.maxTanInputLength, mapAllowedTanFormat(method.allowedTanFormat)
|
method.displayName, mapTanMethodType(method.type), method.securityFunction.code, method.maxTanInputLength, mapAllowedTanFormat(method.allowedTanFormat)
|
||||||
)
|
)
|
||||||
|
|
||||||
protected open fun mapTanMethodType(type: net.codinux.banking.fints.model.TanMethodType): TanMethodType =
|
private fun mapTanMethodType(type: net.dankito.banking.fints.model.TanMethodType): TanMethodType =
|
||||||
TanMethodType.valueOf(type.name)
|
TanMethodType.valueOf(type.name)
|
||||||
|
|
||||||
protected open fun mapAllowedTanFormat(allowedTanFormat: net.codinux.banking.fints.messages.datenelemente.implementierte.tan.AllowedTanFormat?): AllowedTanFormat =
|
private fun mapAllowedTanFormat(allowedTanFormat: net.dankito.banking.fints.messages.datenelemente.implementierte.tan.AllowedTanFormat?): AllowedTanFormat =
|
||||||
allowedTanFormat?.let { AllowedTanFormat.valueOf(it.name) } ?: AllowedTanFormat.Alphanumeric
|
allowedTanFormat?.let { AllowedTanFormat.valueOf(it.name) } ?: AllowedTanFormat.Alphanumeric
|
||||||
|
|
||||||
protected open fun mapTanImage(image: net.codinux.banking.fints.tan.TanImage): TanImage =
|
private fun mapTanImage(image: net.dankito.banking.fints.tan.TanImage): TanImage =
|
||||||
TanImage(image.mimeType, mapToBase64(image.imageBytes), mapException(image.decodingError))
|
TanImage(image.mimeType, mapToBase64(image.imageBytes), mapException(image.decodingError))
|
||||||
|
|
||||||
@OptIn(ExperimentalEncodingApi::class)
|
@OptIn(ExperimentalEncodingApi::class)
|
||||||
protected open fun mapToBase64(bytes: ByteArray): String {
|
private fun mapToBase64(bytes: ByteArray): String {
|
||||||
return Base64.Default.encode(bytes)
|
return Base64.Default.encode(bytes)
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun mapTanMedium(tanMedium: TanMedium) = net.codinux.banking.client.model.tan.TanMedium(
|
private fun mapFlickerCode(flickerCode: net.dankito.banking.fints.tan.FlickerCode): FlickerCode =
|
||||||
mapTanMediumType(tanMedium), tanMedium.mediumName, mapTanMediumStatus(tanMedium.status),
|
|
||||||
(tanMedium as? TanGeneratorTanMedium)?.let { mapTanGeneratorTanMedium(it) },
|
|
||||||
(tanMedium as? MobilePhoneTanMedium)?.let { mapMobilePhoneTanMedium(it) }
|
|
||||||
)
|
|
||||||
|
|
||||||
protected open fun mapTanMediumStatus(status: TanMediumStatus): net.codinux.banking.client.model.tan.TanMediumStatus = when (status) {
|
|
||||||
TanMediumStatus.Aktiv -> net.codinux.banking.client.model.tan.TanMediumStatus.Used
|
|
||||||
TanMediumStatus.Verfuegbar -> net.codinux.banking.client.model.tan.TanMediumStatus.Available
|
|
||||||
TanMediumStatus.AktivFolgekarte -> net.codinux.banking.client.model.tan.TanMediumStatus.ActiveFollowUpCard
|
|
||||||
TanMediumStatus.VerfuegbarFolgekarte -> net.codinux.banking.client.model.tan.TanMediumStatus.AvailableFollowUpCard
|
|
||||||
}
|
|
||||||
|
|
||||||
protected open fun mapMobilePhoneTanMedium(tanMedium: MobilePhoneTanMedium) = net.codinux.banking.client.model.tan.MobilePhoneTanMedium(
|
|
||||||
tanMedium.phoneNumber, tanMedium.concealedPhoneNumber
|
|
||||||
)
|
|
||||||
|
|
||||||
protected open fun mapTanGeneratorTanMedium(tanMedium: TanGeneratorTanMedium) = net.codinux.banking.client.model.tan.TanGeneratorTanMedium(
|
|
||||||
tanMedium.cardNumber, tanMedium.cardSequenceNumber, tanMedium.cardType,
|
|
||||||
tanMedium.validFrom, tanMedium.validTo
|
|
||||||
)
|
|
||||||
|
|
||||||
protected open fun mapTanMediumType(tanMedium: TanMedium): TanMediumType = when {
|
|
||||||
tanMedium is MobilePhoneTanMedium -> TanMediumType.MobilePhone
|
|
||||||
tanMedium is TanGeneratorTanMedium -> TanMediumType.TanGenerator
|
|
||||||
else -> TanMediumType.Generic
|
|
||||||
}
|
|
||||||
|
|
||||||
protected open fun mapFlickerCode(flickerCode: net.codinux.banking.fints.tan.FlickerCode): FlickerCode =
|
|
||||||
FlickerCode(flickerCode.challengeHHD_UC, flickerCode.parsedDataSet, mapException(flickerCode.decodingError))
|
FlickerCode(flickerCode.challengeHHD_UC, flickerCode.parsedDataSet, mapException(flickerCode.decodingError))
|
||||||
|
|
||||||
|
|
||||||
protected open fun <T> mapError(response: net.dankito.banking.client.model.response.GetAccountDataResponse): Response<T> {
|
private fun <T> mapError(response: net.dankito.banking.client.model.response.GetAccountDataResponse): Response<T> {
|
||||||
return if (response.error != null) {
|
return if (response.error != null) {
|
||||||
Response.error(ErrorType.valueOf(response.error!!.name), if (response.error == ErrorCode.BankReturnedError) null else response.errorMessage,
|
Response.error(ErrorType.valueOf(response.error!!.name), if (response.error == ErrorCode.BankReturnedError) null else response.errorMessage,
|
||||||
if (response.error == ErrorCode.BankReturnedError && response.errorMessage !== null) listOf(response.errorMessage!!) else emptyList())
|
if (response.error == ErrorCode.BankReturnedError && response.errorMessage !== null) listOf(response.errorMessage!!) else emptyList())
|
||||||
|
@ -223,7 +158,7 @@ open class FinTs4kMapper {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected open fun mapException(exception: Exception?): String? =
|
private fun mapException(exception: Exception?): String? =
|
||||||
exception?.stackTraceToString()
|
exception?.stackTraceToString()
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,72 +0,0 @@
|
||||||
package net.dankito.banking.banklistcreator.prettifier
|
|
||||||
|
|
||||||
import net.codinux.banking.client.model.BankingGroup
|
|
||||||
|
|
||||||
// TODO: class has been duplicated from BankListCreator, find a better place for it
|
|
||||||
class BankingGroupMapper {
|
|
||||||
|
|
||||||
fun getBankingGroup(bankName: String, bic: String): BankingGroup? {
|
|
||||||
val lowercase = bankName.lowercase()
|
|
||||||
|
|
||||||
return when {
|
|
||||||
bankName.contains("Sparda") -> BankingGroup.Sparda
|
|
||||||
bankName.contains("PSD") -> BankingGroup.PSD
|
|
||||||
bankName.contains("GLS") -> BankingGroup.GLS
|
|
||||||
// see https://de.wikipedia.org/wiki/Liste_der_Genossenschaftsbanken_in_Deutschland
|
|
||||||
bankName.contains("BBBank") || bankName.contains("Evangelische Bank") || bankName.contains("LIGA Bank")
|
|
||||||
|| bankName.contains("Pax") || bankName.contains("Bank für Kirche und Diakonie") || bankName.contains("Bank im Bistum Essen")
|
|
||||||
|| bankName.contains("Bank für Schiffahrt") || bankName.contains("Bank für Kirche")
|
|
||||||
-> BankingGroup.SonstigeGenossenschaftsbank
|
|
||||||
lowercase.contains("deutsche kreditbank") -> BankingGroup.DKB
|
|
||||||
// may check against https://de.wikipedia.org/wiki/Liste_der_Sparkassen_in_Deutschland
|
|
||||||
lowercase.contains("sparkasse") -> BankingGroup.Sparkasse
|
|
||||||
lowercase.contains("comdirect") -> BankingGroup.Comdirect
|
|
||||||
lowercase.contains("commerzbank") -> BankingGroup.Commerzbank
|
|
||||||
lowercase.contains("targo") -> BankingGroup.Targobank
|
|
||||||
lowercase.contains("santander") -> BankingGroup.Santander
|
|
||||||
bankName.contains("KfW") -> BankingGroup.KfW
|
|
||||||
bankName.contains("N26") -> BankingGroup.N26
|
|
||||||
else -> getBankingGroupByBic(bic)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun getBankingGroupByBic(bic: String): BankingGroup? {
|
|
||||||
if (bic.length < 4) {
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bic.startsWith("CMCIDEDD")) {
|
|
||||||
return BankingGroup.Targobank
|
|
||||||
}
|
|
||||||
|
|
||||||
val bankCodeOfBic = bic.substring(0, 4)
|
|
||||||
|
|
||||||
return when (bankCodeOfBic) {
|
|
||||||
"GENO", "VBMH", "VOHA", "VBRS", "DBPB", "VBGT", "FFVB", "WIBA", "VRBU", "MVBM", "VOBA", "ULMV", "VBRT", "VBRA", "VBPF", "VOLO" -> BankingGroup.VolksUndRaiffeisenbanken
|
|
||||||
"BFSW", // Bank fuer Sozialwirtschaft
|
|
||||||
"BEVO", // Berliner Volksbank
|
|
||||||
"DAAE", // apoBank
|
|
||||||
"MHYP", // Münchener Hypothekenbank
|
|
||||||
"DZBM", // DZB Bank
|
|
||||||
"EDEK" // Edekabank
|
|
||||||
-> BankingGroup.SonstigeGenossenschaftsbank
|
|
||||||
"BYLA", "SOLA", "NOLA", "WELA", "HELA", "MALA", "BRLA", "NASS", "TRIS", "OSDD", "ESSL", "GOPS", "SBCR", "BRUS" -> BankingGroup.Sparkasse // filter out DBK, (Bayr.) Landesbank, ...
|
|
||||||
"OLBO" -> BankingGroup.OldenburgischeLandesbank
|
|
||||||
"DEUT" -> BankingGroup.DeutscheBank
|
|
||||||
"PBNK" -> BankingGroup.Postbank
|
|
||||||
"COBA", "DRES" -> BankingGroup.Commerzbank // COBA could also be comdirect, but we cannot differentiate this at this level, this has to do getBankingGroup()
|
|
||||||
"HYVE" -> BankingGroup.Unicredit
|
|
||||||
"INGB" -> BankingGroup.ING
|
|
||||||
"SCFB" -> BankingGroup.Santander
|
|
||||||
"NORS" -> BankingGroup.Norisbank
|
|
||||||
"DEGU" -> BankingGroup.Degussa
|
|
||||||
"OBKL" -> BankingGroup.Oberbank
|
|
||||||
"MARK" -> BankingGroup.Bundesbank
|
|
||||||
"KFWI", "DTAB" -> BankingGroup.KfW
|
|
||||||
"NTSB" -> BankingGroup.N26
|
|
||||||
"CSDB" -> BankingGroup.Consors
|
|
||||||
else -> null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -19,7 +19,7 @@ class FinTs4kBankingClientTest {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private val underTest = FinTs4KBankingClientForUser(bankCode, loginName, password, SimpleBankingClientCallback { tanChallenge, callback ->
|
private val underTest = FinTs4kBankingClientForCustomer(bankCode, loginName, password, SimpleBankingClientCallback { customer, tanChallenge ->
|
||||||
|
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
10
README.md
10
README.md
|
@ -51,7 +51,7 @@ class ShowUsage {
|
||||||
|
|
||||||
|
|
||||||
fun getAccountData() {
|
fun getAccountData() {
|
||||||
val client = FinTs4kBankingClientForUser(bankCode, loginName, password, SimpleBankingClientCallback())
|
val client = FinTs4kBankingClientForCustomer(bankCode, loginName, password, SimpleBankingClientCallback())
|
||||||
|
|
||||||
val response = client.getAccountData()
|
val response = client.getAccountData()
|
||||||
|
|
||||||
|
@ -61,12 +61,12 @@ class ShowUsage {
|
||||||
|
|
||||||
private fun printReceivedData(response: Response<GetAccountDataResponse>) {
|
private fun printReceivedData(response: Response<GetAccountDataResponse>) {
|
||||||
response.data?.let { data ->
|
response.data?.let { data ->
|
||||||
val user = data.user
|
val customer = data.customer
|
||||||
println("Kunde: ${user.customerName} ${user.accounts.size} Konten @ ${user.bic} ${user.bankName}")
|
println("Kunde: ${customer.customerName} ${customer.accounts.size} Konten @ ${customer.bic} ${customer.bankName}")
|
||||||
|
|
||||||
println()
|
println()
|
||||||
println("Konten:")
|
println("Konten:")
|
||||||
user.accounts.sortedBy { it.type }.forEach { account ->
|
customer.accounts.sortedBy { it.type }.forEach { account ->
|
||||||
println("${account.identifier} ${account.productName} ${account.balance} ${account.currency}")
|
println("${account.identifier} ${account.productName} ${account.balance} ${account.currency}")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -85,7 +85,7 @@ This fetches the booked account transactions of the last 90 days. In most cases
|
||||||
In case there is, add TAN handling in Client Callback:
|
In case there is, add TAN handling in Client Callback:
|
||||||
|
|
||||||
```kotlin
|
```kotlin
|
||||||
val client = FinTs4kBankingClientForUser(bankCode, loginName, password, SimpleBankingClientCallback { tanChallenge, callback ->
|
val client = FinTs4kBankingClientForCustomer(bankCode, loginName, password, SimpleBankingClientCallback { tanChallenge, callback ->
|
||||||
val tan: String? = null // if a TAN is required, add a UI or ...
|
val tan: String? = null // if a TAN is required, add a UI or ...
|
||||||
callback.invoke(EnterTanResult(tan)) // ... set a break point here, get TAN e.g. from your TAN app, set tan variable in debugger view and resume debugger
|
callback.invoke(EnterTanResult(tan)) // ... set a break point here, get TAN e.g. from your TAN app, set tan variable in debugger view and resume debugger
|
||||||
})
|
})
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
plugins {
|
plugins {
|
||||||
kotlin("jvm") // or kotlin("multiplatform"), depending on your requirements
|
kotlin("jvm")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -2,7 +2,7 @@ package net.codinux.banking.client.fints4k.example
|
||||||
|
|
||||||
import kotlinx.datetime.LocalDate
|
import kotlinx.datetime.LocalDate
|
||||||
import net.codinux.banking.client.SimpleBankingClientCallback
|
import net.codinux.banking.client.SimpleBankingClientCallback
|
||||||
import net.codinux.banking.client.fints4k.FinTs4kBankingClientForUser
|
import net.codinux.banking.client.fints4k.FinTs4kBankingClientForCustomer
|
||||||
import net.codinux.banking.client.getAccountData
|
import net.codinux.banking.client.getAccountData
|
||||||
import net.codinux.banking.client.model.options.GetAccountDataOptions
|
import net.codinux.banking.client.model.options.GetAccountDataOptions
|
||||||
import net.codinux.banking.client.model.options.RetrieveTransactions
|
import net.codinux.banking.client.model.options.RetrieveTransactions
|
||||||
|
@ -27,7 +27,7 @@ class ShowUsage {
|
||||||
|
|
||||||
|
|
||||||
fun getAccountDataSimpleExample() {
|
fun getAccountDataSimpleExample() {
|
||||||
val client = FinTs4kBankingClientForUser(bankCode, loginName, password, SimpleBankingClientCallback())
|
val client = FinTs4kBankingClientForCustomer(bankCode, loginName, password, SimpleBankingClientCallback())
|
||||||
|
|
||||||
val response = client.getAccountData()
|
val response = client.getAccountData()
|
||||||
|
|
||||||
|
@ -35,7 +35,7 @@ class ShowUsage {
|
||||||
}
|
}
|
||||||
|
|
||||||
fun getAccountDataFullExample() {
|
fun getAccountDataFullExample() {
|
||||||
val client = FinTs4kBankingClientForUser(bankCode, loginName, password, SimpleBankingClientCallback { tanChallenge, callback ->
|
val client = FinTs4kBankingClientForCustomer(bankCode, loginName, password, SimpleBankingClientCallback { tanChallenge, callback ->
|
||||||
val tan: String? = null // if a TAN is required, add a UI or ...
|
val tan: String? = null // if a TAN is required, add a UI or ...
|
||||||
callback.invoke(EnterTanResult(tan)) // ... set a break point here, get TAN e.g. from your TAN app, set tan variable in debugger view and resume debugger
|
callback.invoke(EnterTanResult(tan)) // ... set a break point here, get TAN e.g. from your TAN app, set tan variable in debugger view and resume debugger
|
||||||
})
|
})
|
||||||
|
@ -59,12 +59,12 @@ class ShowUsage {
|
||||||
|
|
||||||
private fun printReceivedData(response: Response<GetAccountDataResponse>) {
|
private fun printReceivedData(response: Response<GetAccountDataResponse>) {
|
||||||
response.data?.let { data ->
|
response.data?.let { data ->
|
||||||
val user = data.user
|
val customer = data.customer
|
||||||
println("Kunde: ${user.customerName} ${user.accounts.size} Konten @ ${user.bic} ${user.bankName}")
|
println("Kunde: ${customer.customerName} ${customer.accounts.size} Konten @ ${customer.bic} ${customer.bankName}")
|
||||||
|
|
||||||
println()
|
println()
|
||||||
println("Konten:")
|
println("Konten:")
|
||||||
user.accounts.sortedBy { it.type }.forEach { account ->
|
customer.accounts.sortedBy { it.type }.forEach { account ->
|
||||||
println("${account.identifier} ${account.productName} ${account.balance} ${account.currency}")
|
println("${account.identifier} ${account.productName} ${account.balance} ${account.currency}")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,11 +12,10 @@ buildscript {
|
||||||
|
|
||||||
allprojects {
|
allprojects {
|
||||||
group = "net.codinux.banking.client"
|
group = "net.codinux.banking.client"
|
||||||
version = "0.5.2-SNAPSHOT"
|
version = "0.5.1-SNAPSHOT"
|
||||||
|
|
||||||
repositories {
|
repositories {
|
||||||
mavenCentral()
|
mavenCentral()
|
||||||
mavenLocal()
|
|
||||||
maven {
|
maven {
|
||||||
setUrl("https://maven.dankito.net/api/packages/codinux/maven")
|
setUrl("https://maven.dankito.net/api/packages/codinux/maven")
|
||||||
}
|
}
|
||||||
|
@ -34,7 +33,7 @@ tasks.register("publishAllToMavenLocal") {
|
||||||
":BankingClientModel:publishToMavenLocal",
|
":BankingClientModel:publishToMavenLocal",
|
||||||
":BankingClient:publishToMavenLocal",
|
":BankingClient:publishToMavenLocal",
|
||||||
|
|
||||||
":fints4k-banking-client:publishToMavenLocal"
|
":FinTs4jBankingClient:publishToMavenLocal"
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -43,6 +42,6 @@ tasks.register("publishAll") {
|
||||||
":BankingClientModel:publish",
|
":BankingClientModel:publish",
|
||||||
":BankingClient:publish",
|
":BankingClient:publish",
|
||||||
|
|
||||||
":fints4k-banking-client:publish"
|
":FinTs4jBankingClient:publish"
|
||||||
)
|
)
|
||||||
}
|
}
|
Loading…
Reference in New Issue