Implemented replacing empty data element group separators at end of segment (so that segment doesn't end with e.g. ...+DE+++')

This commit is contained in:
dankl 2019-10-06 15:37:11 +02:00 committed by dankito
parent 46a23d17d5
commit dcda52d65f
5 changed files with 70 additions and 5 deletions

View File

@ -3,13 +3,33 @@ package net.dankito.fints.messages.segmente
import net.dankito.fints.messages.Existenzstatus import net.dankito.fints.messages.Existenzstatus
import net.dankito.fints.messages.Separators import net.dankito.fints.messages.Separators
import net.dankito.fints.messages.datenelemente.DatenelementBase import net.dankito.fints.messages.datenelemente.DatenelementBase
import java.util.regex.Pattern
abstract class Segment(val dataElementsAndGroups: List<DatenelementBase>, existenzstatus: Existenzstatus) abstract class Segment(val dataElementsAndGroups: List<DatenelementBase>, existenzstatus: Existenzstatus)
: DatenelementBase(existenzstatus) { : DatenelementBase(existenzstatus) {
companion object {
val ReplaceEmptyDataElementGroupSeparatorsAtEndPattern =
Pattern.compile("\\${Separators.DataElementGroupsSeparator}*\$")
}
override fun format(): String { override fun format(): String {
return dataElementsAndGroups.joinToString(Separators.DataElementGroupsSeparator) { it.format() } val formattedSegment = dataElementsAndGroups.joinToString(Separators.DataElementGroupsSeparator) { it.format() }
return cutEmptyDataElementGroupsAtEndOfSegment(formattedSegment)
}
/**
* Auslassen von Datenelementen durch Abschneiden
* Ist für DE, die am Ende eines Segments stehen, kein Inhalt vorhanden, können sie ausgelassen werden.
* In diesem Fall wird das Segmentende-Zeichen unmittelbar nach dem letzten mit Inhalt belegten DE angegeben.
*/
protected open fun cutEmptyDataElementGroupsAtEndOfSegment(formattedSegment: String): String {
val matcher = ReplaceEmptyDataElementGroupSeparatorsAtEndPattern.matcher(formattedSegment)
return matcher.replaceFirst("")
} }
} }

View File

@ -54,6 +54,5 @@ open class Verschluesselungskopf(
VerschluesselungsalgorithmusDatenelementgruppe(mode), VerschluesselungsalgorithmusDatenelementgruppe(mode),
Schluesselname(bank.countryCode, bank.bankCode, customer.customerId, key, keyNumber, keyVersion), Schluesselname(bank.countryCode, bank.bankCode, customer.customerId, key, keyNumber, keyVersion),
KomprimierungsfunktionDatenelement(algorithm), KomprimierungsfunktionDatenelement(algorithm),
// Certificate not applicapable for PIN/TAN; it should be also fine to write nothing at all and therefore leave NotAllowedDatenelement away NotAllowedDatenelement() // Certificate not applicapable for PIN/TAN
NotAllowedDatenelement() // Zertifikat is actually a Datenelementgruppe, not a Datenelement
), Existenzstatus.Mandatory) ), Existenzstatus.Mandatory)

View File

@ -64,7 +64,7 @@ class MessageBuilderTest : FinTsTestBase() {
// then // then
assertThat(normalizeBinaryData(result)).isEqualTo(normalizeBinaryData( assertThat(normalizeBinaryData(result)).isEqualTo(normalizeBinaryData(
"HNHBK:1:3+000000000386+300+0+1'" + "HNHBK:1:3+000000000386+300+0+1'" +
"HNVSK:998:3+PIN:2+998+1+1::0+1:$Date:$Time+2:16:14:@8@ :5:1+280:$BankCode:$CustomerId:V:0:0+0+'" + "HNVSK:998:3+PIN:2+998+1+1::0+1:$Date:$Time+2:16:14:@8@ :5:1+280:$BankCode:$CustomerId:V:0:0+0'" +
"HNVSD:999:1+@221@" + "HNSHK:2:4+PIN:2+${SecurityFunction.code}+$ControlReference+1+1+1::0+1+1:$Date:$Time+1:999:1+6:10:16+280:$BankCode:$CustomerId:S:0:0'" + "HNVSD:999:1+@221@" + "HNSHK:2:4+PIN:2+${SecurityFunction.code}+$ControlReference+1+1+1::0+1+1:$Date:$Time+1:999:1+6:10:16+280:$BankCode:$CustomerId:S:0:0'" +
"HKIDN:3:2+280:$BankCode+$CustomerId+0+1'" + "HKIDN:3:2+280:$BankCode+$CustomerId+0+1'" +
"HKVVB:4:3+0+0+${Language.code}+$ProductName+$ProductVersion'" + "HKVVB:4:3+0+0+${Language.code}+$ProductName+$ProductVersion'" +

View File

@ -0,0 +1,46 @@
package net.dankito.fints.messages.segmente
import net.dankito.fints.messages.Existenzstatus
import net.dankito.fints.messages.datenelemente.DatenelementBase
import net.dankito.fints.messages.datenelemente.basisformate.TextDatenelement
import org.assertj.core.api.Assertions.assertThat
import org.junit.Test
class SegmentTest {
@Test
fun format_CutEmptyDataElementGroupsAtSegmentEnd() {
// given
val underTest = object : Segment(listOf(
createTextDataElement("DE1"),
createTextDataElement("DE2"),
createEmptyTextDataElement(),
createEmptyTextDataElement(),
createEmptyTextDataElement(),
createTextDataElement("DE6"),
createEmptyTextDataElement(),
createTextDataElement("DE8"),
createEmptyTextDataElement(),
createEmptyTextDataElement()
), Existenzstatus.Mandatory) { }
// when
val result = underTest.format()
// then
// assert that empty data elements at end get cut but that the empty ones in the middle remain
assertThat(result).isEqualTo("DE1+DE2++++DE6++DE8")
}
private fun createEmptyTextDataElement(): DatenelementBase {
return createTextDataElement("")
}
private fun createTextDataElement(text: String): DatenelementBase {
return object : TextDatenelement(text, Existenzstatus.Optional) { }
}
}

View File

@ -18,7 +18,7 @@ class VerschluesselungskopfTest : FinTsTestBase() {
val result = underTest.format() val result = underTest.format()
// then // then
assertThat(normalizeBinaryData(result)).isEqualTo("HNVSK:998:3+PIN:2+998+1+1::0+1:$Date:$Time+2:16:14:@8@ :5:1+280:$BankCode:$CustomerId:V:0:0+0+") assertThat(normalizeBinaryData(result)).isEqualTo("HNVSK:998:3+PIN:2+998+1+1::0+1:$Date:$Time+2:16:14:@8@ :5:1+280:$BankCode:$CustomerId:V:0:0+0")
} }
} }