TestByteArrayBuffer.java
/*
* ====================================================================
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
* ====================================================================
*
* This software consists of voluntary contributions made by many
* individuals on behalf of the Apache Software Foundation. For more
* information on the Apache Software Foundation, please see
* <http://www.apache.org/>.
*
*/
package org.apache.hc.core5.util;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.nio.ByteBuffer;
import java.nio.charset.StandardCharsets;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Test;
/**
* Unit tests for {@link ByteArrayBuffer}.
*
*/
class TestByteArrayBuffer {
@Test
void testConstructor() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(16);
Assertions.assertEquals(16, buffer.capacity());
Assertions.assertEquals(0, buffer.length());
Assertions.assertNotNull(buffer.array());
Assertions.assertEquals(16, buffer.array().length);
Assertions.assertThrows(IllegalArgumentException.class, () -> new ByteArrayBuffer(-1));
}
@Test
void testSimpleAppend() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(16);
Assertions.assertEquals(16, buffer.capacity());
Assertions.assertEquals(0, buffer.length());
final byte[] b1 = buffer.toByteArray();
Assertions.assertNotNull(b1);
Assertions.assertEquals(0, b1.length);
Assertions.assertTrue(buffer.isEmpty());
Assertions.assertFalse(buffer.isFull());
final byte[] tmp = new byte[] { 1, 2, 3, 4};
buffer.append(tmp, 0, tmp.length);
Assertions.assertEquals(16, buffer.capacity());
Assertions.assertEquals(4, buffer.length());
Assertions.assertFalse(buffer.isEmpty());
Assertions.assertFalse(buffer.isFull());
final byte[] b2 = buffer.toByteArray();
Assertions.assertNotNull(b2);
Assertions.assertEquals(4, b2.length);
for (int i = 0; i < tmp.length; i++) {
Assertions.assertEquals(tmp[i], b2[i]);
Assertions.assertEquals(tmp[i], buffer.byteAt(i));
}
buffer.clear();
Assertions.assertEquals(16, buffer.capacity());
Assertions.assertEquals(0, buffer.length());
Assertions.assertTrue(buffer.isEmpty());
Assertions.assertFalse(buffer.isFull());
}
@Test
void testExpandAppend() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(4);
Assertions.assertEquals(4, buffer.capacity());
final byte[] tmp = new byte[] { 1, 2, 3, 4};
buffer.append(tmp, 0, 2);
buffer.append(tmp, 0, 4);
buffer.append(tmp, 0, 0);
Assertions.assertEquals(8, buffer.capacity());
Assertions.assertEquals(6, buffer.length());
buffer.append(tmp, 0, 4);
Assertions.assertEquals(16, buffer.capacity());
Assertions.assertEquals(10, buffer.length());
}
@Test
void testAppendHeapByteBuffer() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(4);
Assertions.assertEquals(4, buffer.capacity());
final ByteBuffer tmp = ByteBuffer.wrap(new byte[] { 1, 2, 3, 4, 5, 6});
buffer.append(tmp);
Assertions.assertFalse(tmp.hasRemaining(), "The input buffer should be drained");
Assertions.assertEquals(8, buffer.capacity());
Assertions.assertEquals(6, buffer.length());
tmp.clear();
buffer.append(tmp);
Assertions.assertFalse(tmp.hasRemaining(), "The input buffer should be drained");
Assertions.assertEquals(16, buffer.capacity());
Assertions.assertEquals(12, buffer.length());
Assertions.assertArrayEquals(new byte[] {1, 2, 3, 4, 5, 6, 1, 2, 3, 4, 5, 6}, buffer.toByteArray());
}
@Test
void testAppendHeapByteBufferWithOffset() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(4);
Assertions.assertEquals(4, buffer.capacity());
final ByteBuffer tmp = ByteBuffer.wrap(new byte[] { 7, 7, 1, 2, 3, 4, 5, 6, 7, 7}, 2, 6).slice();
Assertions.assertTrue(tmp.arrayOffset() > 0, "Validate this is testing a buffer with an array offset");
buffer.append(tmp);
Assertions.assertFalse(tmp.hasRemaining(), "The input buffer should be drained");
Assertions.assertEquals(8, buffer.capacity());
Assertions.assertEquals(6, buffer.length());
tmp.clear();
Assertions.assertEquals(6, tmp.remaining());
buffer.append(tmp);
Assertions.assertFalse(tmp.hasRemaining(), "The input buffer should be drained");
Assertions.assertEquals(16, buffer.capacity());
Assertions.assertEquals(12, buffer.length());
Assertions.assertArrayEquals(new byte[] {1, 2, 3, 4, 5, 6, 1, 2, 3, 4, 5, 6}, buffer.toByteArray());
}
@Test
void testAppendDirectByteBuffer() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(4);
Assertions.assertEquals(4, buffer.capacity());
final ByteBuffer tmp = ByteBuffer.allocateDirect(6);
tmp.put(new byte[] { 1, 2, 3, 4, 5, 6}).flip();
buffer.append(tmp);
Assertions.assertFalse(tmp.hasRemaining(), "The input buffer should be drained");
Assertions.assertEquals(8, buffer.capacity());
Assertions.assertEquals(6, buffer.length());
tmp.clear();
buffer.append(tmp);
Assertions.assertFalse(tmp.hasRemaining(), "The input buffer should be drained");
Assertions.assertEquals(16, buffer.capacity());
Assertions.assertEquals(12, buffer.length());
Assertions.assertArrayEquals(new byte[] {1, 2, 3, 4, 5, 6, 1, 2, 3, 4, 5, 6}, buffer.toByteArray());
}
@Test
void testInvalidAppend() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(4);
buffer.append((byte[])null, 0, 0);
final byte[] tmp = new byte[] { 1, 2, 3, 4};
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.append(tmp, -1, 0));
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.append(tmp, 0, -1));
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.append(tmp, 0, 8));
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.append(tmp, 10, Integer.MAX_VALUE));
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.append(tmp, 2, 4));
}
@Test
void testAppendOneByte() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(4);
Assertions.assertEquals(4, buffer.capacity());
final byte[] tmp = new byte[] { 1, 127, -1, -128, 1, -2};
for (final byte element : tmp) {
buffer.append(element);
}
Assertions.assertEquals(8, buffer.capacity());
Assertions.assertEquals(6, buffer.length());
for (int i = 0; i < tmp.length; i++) {
Assertions.assertEquals(tmp[i], buffer.byteAt(i));
}
}
@Test
void testSetLength() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(4);
buffer.setLength(2);
Assertions.assertEquals(2, buffer.length());
}
@Test
void testSetInvalidLength() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(4);
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.setLength(-2));
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.setLength(200));
}
@Test
void testEnsureCapacity() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(4);
buffer.ensureCapacity(2);
Assertions.assertEquals(4, buffer.capacity());
buffer.ensureCapacity(8);
Assertions.assertEquals(8, buffer.capacity());
}
@Test
void testIndexOf() {
final byte COLON = (byte) ':';
final byte COMMA = (byte) ',';
final byte[] bytes = "name1: value1; name2: value2".getBytes(StandardCharsets.US_ASCII);
final int index1 = 5;
final int index2 = 20;
final ByteArrayBuffer buffer = new ByteArrayBuffer(16);
buffer.append(bytes, 0, bytes.length);
Assertions.assertEquals(index1, buffer.indexOf(COLON));
Assertions.assertEquals(-1, buffer.indexOf(COMMA));
Assertions.assertEquals(index1, buffer.indexOf(COLON, -1, 11));
Assertions.assertEquals(index1, buffer.indexOf(COLON, 0, 1000));
Assertions.assertEquals(-1, buffer.indexOf(COLON, 2, 1));
Assertions.assertEquals(index2, buffer.indexOf(COLON, index1 + 1, buffer.length()));
}
@Test
void testAppendCharArrayAsAscii() {
final String s1 = "stuff";
final String s2 = " and more stuff";
final char[] b1 = s1.toCharArray();
final char[] b2 = s2.toCharArray();
final ByteArrayBuffer buffer = new ByteArrayBuffer(8);
buffer.append(b1, 0, b1.length);
buffer.append(b2, 0, b2.length);
Assertions.assertEquals(s1 + s2, new String(buffer.toByteArray(), StandardCharsets.US_ASCII));
}
@Test
void testAppendNullCharArray() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(8);
buffer.append((char[])null, 0, 0);
Assertions.assertEquals(0, buffer.length());
}
@Test
void testAppendEmptyCharArray() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(8);
buffer.append(new char[] {}, 0, 0);
Assertions.assertEquals(0, buffer.length());
}
@Test
void testAppendNullCharArrayBuffer() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(8);
buffer.append((CharArrayBuffer)null, 0, 0);
Assertions.assertEquals(0, buffer.length());
}
@Test
void testAppendNullByteBuffer() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(8);
final ByteBuffer nullBuffer = null;
buffer.append(nullBuffer);
Assertions.assertEquals(0, buffer.length());
}
@Test
void testInvalidAppendCharArrayAsAscii() {
final ByteArrayBuffer buffer = new ByteArrayBuffer(4);
buffer.append((char[])null, 0, 0);
final char[] tmp = new char[] { '1', '2', '3', '4'};
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.append(tmp, -1, 0));
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.append(tmp, 0, -1));
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.append(tmp, 0, 8));
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.append(tmp, 10, Integer.MAX_VALUE));
Assertions.assertThrows(IndexOutOfBoundsException.class, () -> buffer.append(tmp, 2, 4));
}
@Test
void testSerialization() throws Exception {
final ByteArrayBuffer orig = new ByteArrayBuffer(32);
orig.append(1);
orig.append(2);
orig.append(3);
final ByteArrayOutputStream outbuffer = new ByteArrayOutputStream();
try (final ObjectOutputStream outStream = new ObjectOutputStream(outbuffer)) {
outStream.writeObject(orig);
}
final byte[] raw = outbuffer.toByteArray();
final ByteArrayInputStream inBuffer = new ByteArrayInputStream(raw);
final ObjectInputStream inStream = new ObjectInputStream(inBuffer);
final ByteArrayBuffer clone = (ByteArrayBuffer) inStream.readObject();
Assertions.assertEquals(orig.capacity(), clone.capacity());
Assertions.assertEquals(orig.length(), clone.length());
final byte[] data = clone.toByteArray();
Assertions.assertNotNull(data);
Assertions.assertEquals(3, data.length);
Assertions.assertEquals(1, data[0]);
Assertions.assertEquals(2, data[1]);
Assertions.assertEquals(3, data[2]);
}
@Test
void testControlCharFiltering() {
final char[] chars = new char[256];
for (char i = 0; i < 256; i++) {
chars[i] = i;
}
final byte[] bytes = asByteArray(chars);
Assertions.assertEquals(
"?????????\t??????????????????????"
+ " !\"#$%&'()*+,-./0123456789:;<=>?"
+ "@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_"
+ "`abcdefghijklmnopqrstuvwxyz"
+ "{|}~???????????????????????"
+ "??????????\u00A0������������������������\u00AD����"
+ "����������������������������������������������������������������"
+ "����������������������������������������������������������������"
+ "��������������������������������",
new String(bytes, StandardCharsets.ISO_8859_1));
}
@Test
void testUnicodeFiltering() {
// Various languages
Assertions.assertEquals("?????", new String(asByteArray("����������".toCharArray()), StandardCharsets.ISO_8859_1));
Assertions.assertEquals("????", new String(asByteArray("������������".toCharArray()), StandardCharsets.ISO_8859_1));
// Unicode snowman
Assertions.assertEquals("?", new String(asByteArray("���".toCharArray()), StandardCharsets.ISO_8859_1));
// Emoji (surrogate pair)
Assertions.assertEquals("??", new String(asByteArray("\uD83D\uDE00".toCharArray()), StandardCharsets.ISO_8859_1));
}
private static byte[] asByteArray(final char[] chars) {
final ByteArrayBuffer byteArrayBuffer = new ByteArrayBuffer(chars.length);
byteArrayBuffer.append(chars, 0, chars.length);
return byteArrayBuffer.toByteArray();
}
}