Merge pull request #1291 from azotcsit/AMQ-8122-main

AMQ-8122 - Fix DataByteArrayInputStreamTest
This commit is contained in:
JB Onofré 2024-09-13 15:06:51 +02:00 committed by GitHub
commit dc6960de91
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
3 changed files with 99 additions and 46 deletions

View File

@ -210,7 +210,6 @@ public final class DataByteArrayOutputStream extends OutputStream implements Dat
ensureEnoughBuffer((int)(pos + encodedsize + 2)); ensureEnoughBuffer((int)(pos + encodedsize + 2));
writeShort((int)encodedsize); writeShort((int)encodedsize);
byte[] buffer = new byte[(int)encodedsize];
MarshallingSupport.writeUTFBytesToBuffer(text, (int) encodedsize, buf, pos); MarshallingSupport.writeUTFBytesToBuffer(text, (int) encodedsize, buf, pos);
pos += encodedsize; pos += encodedsize;
} }

View File

@ -17,66 +17,121 @@
package org.apache.activemq.util; package org.apache.activemq.util;
import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertThrows;
import java.io.UTFDataFormatException;
import org.junit.Test; import org.junit.Test;
/**
* https://issues.apache.org/jira/browse/AMQ-1911
* https://issues.apache.org/jira/browse/AMQ-8122
*/
public class DataByteArrayInputStreamTest { public class DataByteArrayInputStreamTest {
/**
* https://issues.apache.org/activemq/browse/AMQ-1911
*/
@Test @Test
public void testNonAscii() throws Exception { public void testOneByteCharacters() throws Exception {
doMarshallUnMarshallValidation("mei\u00DFen"); testCodePointRange(0x000000, 0x00007F);
String accumulator = new String();
int test = 0; // int to get Supplementary chars
while(Character.isDefined(test)) {
String toTest = String.valueOf((char)test);
accumulator += toTest;
doMarshallUnMarshallValidation(toTest);
test++;
} }
int massiveThreeByteCharValue = 0x0FFF; @Test
String toTest = String.valueOf((char)massiveThreeByteCharValue); public void testTwoBytesCharacters() throws Exception {
accumulator += toTest; testCodePointRange(0x000080, 0x0007FF);
doMarshallUnMarshallValidation(String.valueOf((char)massiveThreeByteCharValue)); }
// Altogether @Test
doMarshallUnMarshallValidation(accumulator); public void testThreeBytesCharacters() throws Exception {
testCodePointRange(0x000800, 0x00FFFF);
}
// the three byte values @Test
char t = '\u0800'; public void testFourBytesCharacters() throws Exception {
final char max = '\uffff'; testCodePointRange(0x010000, 0X10FFFF);
accumulator = String.valueOf(t); }
while (t < max) {
String val = String.valueOf(t); @Test
accumulator += val; public void testFourBytesCharacterEncodedAsBytes() throws Exception {
// Currently ActiveMQ does not properly support 4-bytes UTF characters.
// Ideally, this test should be failing. The current logic was kept as is
// intentionally. See https://issues.apache.org/jira/browse/AMQ-8398.
// 0xF0 0x80 0x80 0x80 (first valid 4-bytes character)
testInvalidCharacterBytes(new byte[]{-16, -128, -128, -128}, 4);
// 0xF7 0xBF 0xBF 0xBF (last valid 4-bytes character)
testInvalidCharacterBytes(new byte[]{-9, -65, -65, -65}, 4);
}
private void testCodePointRange(int from, int to) throws Exception {
StringBuilder accumulator = new StringBuilder();
for (int codePoint = from; codePoint <= to; codePoint++) {
String val = String.valueOf(Character.toChars(codePoint));
accumulator.append(val);
doMarshallUnMarshallValidation(val); doMarshallUnMarshallValidation(val);
t++;
} }
// Altogether so long as it is not too big // truncate string to last 20k characters
while (accumulator.length() > 20000) { if (accumulator.length() > 20_000) {
accumulator = accumulator.substring(20000); doMarshallUnMarshallValidation(accumulator.substring(
accumulator.length() - 20_000));
} else {
doMarshallUnMarshallValidation(accumulator.toString());
} }
doMarshallUnMarshallValidation(accumulator);
} }
void doMarshallUnMarshallValidation(String value) throws Exception { private void doMarshallUnMarshallValidation(String value) throws Exception {
DataByteArrayOutputStream out = new DataByteArrayOutputStream(); DataByteArrayOutputStream out = new DataByteArrayOutputStream();
out.writeBoolean(true);
out.writeUTF(value); out.writeUTF(value);
out.close(); out.close();
DataByteArrayInputStream in = new DataByteArrayInputStream(out.getData()); DataByteArrayInputStream in = new DataByteArrayInputStream(out.getData());
in.readBoolean();
String readBack = in.readUTF(); String readBack = in.readUTF();
assertEquals(value, readBack); assertEquals(value, readBack);
} }
@Test
public void testTwoBytesOutOfRangeCharacter() throws Exception {
// 0xC0 0x7F
testInvalidCharacterBytes(new byte[]{-64, 127}, 2);
// 0xDF 0xC0
testInvalidCharacterBytes(new byte[]{-33, -64}, 2);
}
@Test
public void testThreeBytesOutOfRangeCharacter() throws Exception {
// 0xE0 0x80 0x7F
testInvalidCharacterBytes(new byte[]{-32, -128, 127}, 3);
// 0xEF 0xBF 0xC0
testInvalidCharacterBytes(new byte[]{-17, -65, -64}, 3);
}
@Test
public void testFourBytesOutOfRangeCharacter() throws Exception {
// 0xF0 0x80 0x80 0x7F
testInvalidCharacterBytes(new byte[]{-16, -128, -128, 127}, 4);
// 0xF7 0xBF 0xBF 0xC0
testInvalidCharacterBytes(new byte[]{-9, -65, -65, -64}, 4);
}
private void testInvalidCharacterBytes(byte[] bytes, int encodedSize) throws Exception {
// Java guarantees that strings are always UTF-8 compliant and valid,
// any invalid sequence of bytes is either replaced or removed.
// This test demonstrates that Java takes care about and does not allow
// anything to break.
String val = new String(bytes);
doMarshallUnMarshallValidation(val);
// However, a non-java client can send an invalid sequence of bytes.
// Such data causes exceptions while unmarshalling.
DataByteArrayOutputStream out = new DataByteArrayOutputStream();
out.writeShort(encodedSize);
out.write(bytes);
out.close();
DataByteArrayInputStream in = new DataByteArrayInputStream(out.getData());
assertThrows(UTFDataFormatException.class, () -> in.readUTF());
}
@Test @Test
public void testReadLong() throws Exception { public void testReadLong() throws Exception {
DataByteArrayOutputStream out = new DataByteArrayOutputStream(8); DataByteArrayOutputStream out = new DataByteArrayOutputStream(8);

View File

@ -237,7 +237,6 @@ public class DataByteArrayOutputStream extends OutputStream implements DataOutpu
ensureEnoughBuffer((int)(pos + encodedsize + 2)); ensureEnoughBuffer((int)(pos + encodedsize + 2));
writeShort((int)encodedsize); writeShort((int)encodedsize);
byte[] buffer = new byte[(int)encodedsize];
MarshallingSupport.writeUTFBytesToBuffer(text, (int) encodedsize, buf, pos); MarshallingSupport.writeUTFBytesToBuffer(text, (int) encodedsize, buf, pos);
pos += encodedsize; pos += encodedsize;
onWrite(); onWrite();