Implemented extracting segment that is embedded in encrypted data segment HNVSD

This commit is contained in:
dankito 2020-05-16 14:45:59 +02:00
parent 6935ce2a59
commit 998c6d0c01
4 changed files with 44 additions and 4 deletions

View File

@ -10,6 +10,9 @@ class Separators {
const val DataElementsSeparator = ":" const val DataElementsSeparator = ":"
const val BinaryDataSeparatorChar = '@'
const val BinaryDataSeparator = BinaryDataSeparatorChar.toString()
const val MaskingCharacter = "?" const val MaskingCharacter = "?"
val AllSeparators = listOf(DataElementsSeparator, DataElementGroupsSeparator, SegmentSeparator) val AllSeparators = listOf(DataElementsSeparator, DataElementGroupsSeparator, SegmentSeparator)

View File

@ -43,7 +43,9 @@ open class ResponseParser @JvmOverloads constructor(
open fun parse(response: String): Response { open fun parse(response: String): Response {
try { try {
val segments = splitIntoPartsAndUnmask(response, Separators.SegmentSeparator) val segments = splitIntoPartsAndUnmask(response, Separators.SegmentSeparator).toMutableList()
extractSegmentEmbeddedInEncryptedData(segments)
val parsedSegments = segments.mapNotNull { parseSegment(it) } val parsedSegments = segments.mapNotNull { parseSegment(it) }
@ -55,6 +57,21 @@ open class ResponseParser @JvmOverloads constructor(
} }
} }
protected open fun extractSegmentEmbeddedInEncryptedData(elements: MutableList<String>) {
ArrayList(elements).forEachIndexed { index, element ->
if (element?.startsWith(MessageSegmentId.EncryptionData.id) == true) {
val embeddedSegmentBinaryDataStartIndex = element.indexOf(Separators.BinaryDataSeparatorChar)
if (embeddedSegmentBinaryDataStartIndex > 0) {
val inEncryptedDataSegmentEmbeddedSegment = extractBinaryData(element.substring(embeddedSegmentBinaryDataStartIndex))
elements.add(index + 1, inEncryptedDataSegmentEmbeddedSegment)
elements[index] = element.substring(0, embeddedSegmentBinaryDataStartIndex)
}
}
}
}
protected open fun parseSegment(segment: String): ReceivedSegment? { protected open fun parseSegment(segment: String): ReceivedSegment? {
try { try {
@ -814,8 +831,8 @@ open class ResponseParser @JvmOverloads constructor(
} }
protected open fun extractBinaryData(binaryData: String): String { protected open fun extractBinaryData(binaryData: String): String {
if (binaryData.startsWith('@')) { if (binaryData.startsWith(Separators.BinaryDataSeparatorChar)) {
val headerEndIndex = binaryData.indexOf('@', 2) val headerEndIndex = binaryData.indexOf(Separators.BinaryDataSeparatorChar, 2)
if (headerEndIndex > -1) { if (headerEndIndex > -1) {
return binaryData.substring(headerEndIndex + 1) return binaryData.substring(headerEndIndex + 1)

View File

@ -31,7 +31,7 @@ open class MessageUtils {
while (binaryDataMatcher.find()) { while (binaryDataMatcher.find()) {
if (isEncryptionDataSegment(dataString, binaryDataMatcher) == false) { if (isEncryptionDataSegment(dataString, binaryDataMatcher) == false) {
val startIndex = binaryDataMatcher.end() val startIndex = binaryDataMatcher.end()
val length = binaryDataMatcher.group().replace("@", "").toInt() val length = binaryDataMatcher.group().replace(Separators.BinaryDataSeparator, "").toInt()
binaryDataRanges.add(IntRange(startIndex, startIndex + length - 1)) binaryDataRanges.add(IntRange(startIndex, startIndex + length - 1))
} }

View File

@ -83,6 +83,26 @@ class ResponseParserTest : FinTsTestBase() {
} }
@Test
fun extractEmbeddedSegment() {
// when
val result = underTest.parse("HNHBK:1:3+000000000249+300+0+1+0:1'" +
"HNVSK:998:3+PIN:1+998+1+2::0+1:20200512:153303+2:2:13:@8@\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000:5:1+280:10070000:9999999999:V:0:0+0'" +
"HNVSD:999:1+@83@HIRMG:2:2+9120::Nachricht nicht erwartet.+9800::Dialoginitialisierung abgebrochen.''" +
"HNHBS:3:1+1'")
// then
assertThat(result.receivedSegments).hasSize(5)
assertCouldParseSegment(result, InstituteSegmentId.MessageFeedback, 2, 2)
assertThat(result.messageFeedback).isNotNull
assertThat(result.messageFeedback?.isError).isTrue()
assertThat(result.messageFeedback?.feedbacks?.map { it.message }).containsExactly("Nachricht nicht erwartet.", "Dialoginitialisierung abgebrochen.")
}
@Test @Test
fun parseMessageHeader() { fun parseMessageHeader() {