From 90564f39a364028fdd8010e7953b370c6e1c43b8 Mon Sep 17 00:00:00 2001 From: Collin Smith Date: Mon, 10 Aug 2020 18:43:01 -0700 Subject: [PATCH] Committing JUnit tests for BitOutput and interoperability --- core/test/com/riiablo/io/BitOutputTest.java | 171 ++++++++++++++++++++ core/test/com/riiablo/io/IOTest.java | 116 +++++++++++++ 2 files changed, 287 insertions(+) create mode 100644 core/test/com/riiablo/io/BitOutputTest.java create mode 100644 core/test/com/riiablo/io/IOTest.java diff --git a/core/test/com/riiablo/io/BitOutputTest.java b/core/test/com/riiablo/io/BitOutputTest.java new file mode 100644 index 00000000..ddcc3a9e --- /dev/null +++ b/core/test/com/riiablo/io/BitOutputTest.java @@ -0,0 +1,171 @@ +package com.riiablo.io; + +import io.netty.buffer.ByteBuf; +import io.netty.buffer.ByteBufUtil; +import io.netty.buffer.Unpooled; +import org.junit.Assert; +import org.junit.Test; + +public class BitOutputTest { + private static BitOutput newInstance() { + return new ByteOutput(Unpooled.buffer(16, 16)).unalign(); + } + + public static void assertBuffer(BitOutput b, String expected) { + ByteBuf buffer = b.byteOutput().buffer(); + Assert.assertEquals(expected, ByteBufUtil.hexDump(buffer, 0, 16)); + } + + @Test + public void write_u8_aligned() { + BitOutput b = newInstance(); + b._writeUnsigned(0xFF, Byte.SIZE); + assertBuffer(b, "ff000000000000000000000000000000"); + } + + @Test + public void write_u4_aligned() { + BitOutput b = newInstance(); + b._writeUnsigned(0xF, 4); + assertBuffer(b, "00000000000000000000000000000000"); + Assert.assertEquals(4, b.bitsCached()); + Assert.assertEquals(0xF, b.cache()); + b.flush(); + assertBuffer(b, "0f000000000000000000000000000000"); + } + + @Test + public void write_u12_aligned() { + BitOutput b = newInstance(); + b._writeUnsigned(0xFFF, 12); + assertBuffer(b, "ff000000000000000000000000000000"); + Assert.assertEquals(4, b.bitsCached()); + Assert.assertEquals(0xF, b.cache()); + b.flush(); + assertBuffer(b, "ff0f0000000000000000000000000000"); + } + + @Test + public void write_u12_u4_aligned() { + BitOutput b = newInstance(); + b._writeUnsigned(0xFFF, 12); + assertBuffer(b, "ff000000000000000000000000000000"); + Assert.assertEquals(4, b.bitsCached()); + Assert.assertEquals(0xF, b.cache()); + b._writeUnsigned(0xF, 4); + assertBuffer(b, "ffff0000000000000000000000000000"); + Assert.assertEquals(0, b.bitsCached()); + Assert.assertEquals(0, b.cache()); + } + + @Test + public void write_u12_u12_aligned() { + BitOutput b = newInstance(); + b._writeUnsigned(0xFFF, 12); + assertBuffer(b, "ff000000000000000000000000000000"); + Assert.assertEquals(4, b.bitsCached()); + Assert.assertEquals(0xF, b.cache()); + b._writeUnsigned(0xFF, 12); + assertBuffer(b, "ffff0f00000000000000000000000000"); + Assert.assertEquals(0, b.bitsCached()); + Assert.assertEquals(0, b.cache()); + } + + @Test + public void write_u7_u63_aligned() { + BitOutput b = newInstance(); + b._writeUnsigned(0x7F, 7); + assertBuffer(b, "00000000000000000000000000000000"); + Assert.assertEquals(7, b.bitsCached()); + Assert.assertEquals(0x7F, b.cache()); + b._writeUnsigned(0x7FFFFFFF_FFFFFFFFL, 63); + assertBuffer(b, "ffffffffffffffff0000000000000000"); + Assert.assertEquals(6, b.bitsCached()); + Assert.assertEquals(0x1f, b.cache()); + b.flush(); + assertBuffer(b, "ffffffffffffffff1f00000000000000"); + } + + @Test + public void write_u8_u63_aligned() { + BitOutput b = newInstance(); + b._writeUnsigned(0x7F, 8); + assertBuffer(b, "7f000000000000000000000000000000"); + Assert.assertEquals(0, b.bitsCached()); + Assert.assertEquals(0, b.cache()); + b._writeUnsigned(0x7FFFFFFF_FFFFFFFFL, 63); + assertBuffer(b, "7fffffffffffffff0000000000000000"); + Assert.assertEquals(7, b.bitsCached()); + Assert.assertEquals(0x7f, b.cache()); + b.flush(); + assertBuffer(b, "7fffffffffffffff7f00000000000000"); + } + +// @Test +// public void empty_bit_input_is_empty() { +// BitInput b = BitInput.emptyBitInput(); +// Assert.assertEquals(0, b.bitsRemaining()); +// Assert.assertEquals(0, b.bytesRemaining()); +// } +// +// @Test(expected = IllegalArgumentException.class) +// public void align_neg_bytes_throws_IllegalArgumentException() { +// BitInput b = newInstance(); +// try { +// assert b.bitsRead() == 0; +// b.align(-1); +// } finally { +// Assert.assertEquals(0L, b.bitsRead()); +// } +// } +// +// @Test +// public void align_0_bytes_aligned() { +// BitInput b = newInstance(); +// assert b.isAligned(); +// assert b.bitsCached() == 0; +// b.align(0); +// Assert.assertTrue(b.isAligned()); +// Assert.assertEquals(0, b.bitsRead()); +// Assert.assertEquals(0, b.bitsCached()); +// Assert.assertEquals(0, b.cache()); +// } +// +// @Test +// public void align_0_bytes_unaligned() { +// BitInput b = newInstance(); +// b.skip(4); +// assert !b.isAligned(); +// assert b.bitsCached() > 0; +// b.align(0); +// Assert.assertTrue(b.isAligned()); +// Assert.assertEquals(Byte.SIZE, b.bitsRead()); +// Assert.assertEquals(0, b.bitsCached()); +// Assert.assertEquals(0, b.cache()); +// } +// +// @Test +// public void align_1_byte_aligned() { +// BitInput b = newInstance(); +// assert b.isAligned(); +// assert b.bitsCached() == 0; +// b.align(1); +// Assert.assertTrue(b.isAligned()); +// Assert.assertEquals(Byte.SIZE, b.bitsRead()); +// Assert.assertEquals(0, b.bitsCached()); +// Assert.assertEquals(0, b.cache()); +// } +// +// @Test +// public void align_1_byte_unaligned() { +// BitInput b = newInstance(); +// b.skip(4); +// assert !b.isAligned(); +// assert b.bitsCached() > 0; +// b.align(1); +// Assert.assertTrue(b.isAligned()); +// Assert.assertEquals(Byte.SIZE, b.bitsRead()); +// Assert.assertEquals(0, b.bitsCached()); +// Assert.assertEquals(0, b.cache()); +// } +} \ No newline at end of file diff --git a/core/test/com/riiablo/io/IOTest.java b/core/test/com/riiablo/io/IOTest.java new file mode 100644 index 00000000..da44286b --- /dev/null +++ b/core/test/com/riiablo/io/IOTest.java @@ -0,0 +1,116 @@ +package com.riiablo.io; + +import io.netty.buffer.ByteBufUtil; +import io.netty.buffer.Unpooled; +import org.junit.Assert; +import org.junit.Test; + +public class IOTest { + private static ByteInput newInstance() { + return ByteInput.wrap(new byte[] { + (byte) 0xEF, (byte) 0xBE, (byte) 0xAD, (byte) 0xDE, + (byte) 0x00, (byte) 0xAD, (byte) 0xBB, (byte) 0xDA + }); + } + + @Test + public void readBytes_aligned() { + final byte[] signature = new byte[]{0x4A, 0x4D}; + ByteInput b = ByteInput.wrap(new byte[]{ + signature[0], signature[1], 0x10, 0x00, (byte) 0x80, 0x00, 0x65, 0x00, 0x04, + (byte) 0x82, 0x26, 0x76, 0x07, (byte) 0x82, 0x09, (byte) 0xD4, + (byte) 0xAA, 0x12, 0x03, 0x01, (byte) 0x80, 0x70, 0x01, 0x01, + (byte) 0x91, 0x03, 0x01, 0x04, 0x64, (byte) 0xFC, 0x07}); + BitInput bits = b.bitInput(); // creates bit input instance + assert b.aligned(); + final byte[] bytesRead = b.readBytes(signature.length); + Assert.assertTrue(b.aligned()); + Assert.assertArrayEquals(signature, bytesRead); // signature + Assert.assertEquals(signature.length * Byte.SIZE, bits.bitsRead()); + Assert.assertEquals(0, bits.bitsCached()); + Assert.assertEquals(0, bits.cache()); + } + + @Test + public void read_hunters_bow_of_blight() { + ByteInput bytes = ByteInput.wrap(new byte[] { + 0x4A, 0x4D, 0x10, 0x00, (byte) 0x80, 0x00, 0x65, 0x00, 0x04, + (byte) 0x82, 0x26, 0x76, 0x07, (byte) 0x82, 0x09, (byte) 0xD4, + (byte) 0xAA, 0x12, 0x03, 0x01, (byte) 0x80, 0x70, 0x01, 0x01, + (byte) 0x91, 0x03, 0x01, 0x04, 0x64, (byte) 0xFC, 0x07}); + Assert.assertArrayEquals(new byte[] {0x4A, 0x4D}, bytes.readBytes(2)); // signature + Assert.assertEquals(0x00800010, bytes.read32()); // flags + Assert.assertEquals(101, bytes.read8u()); // version + BitInput bits = bytes.unalign(); + bits.skipBits(2); // unknown + Assert.assertEquals(0, bits.read7u(3)); // location + Assert.assertEquals(0, bits.read7u(4)); // body location + Assert.assertEquals(2, bits.read7u(4)); // grid x + Assert.assertEquals(0, bits.read7u(4)); // grid y + Assert.assertEquals(1, bits.read7u(3)); // store location + Assert.assertEquals("hbw ", bits.readString(4)); // code + Assert.assertEquals(0, bits.read7u(3)); // sockets filled + Assert.assertEquals(0x2555A813, bits.readRaw(32)); // id + Assert.assertEquals(6, bits.read7u(7)); // ilvl + Assert.assertEquals(4, bits.read7u(4)); // quality + Assert.assertEquals(false, bits.readBoolean()); // picture id + Assert.assertEquals(false, bits.readBoolean()); // class only + Assert.assertEquals(0, bits.read15u(11)); // magic prefix + Assert.assertEquals(737, bits.read15u(11)); // magic suffix + bits.skipBits(1); // unknown + Assert.assertEquals(32, bits.read15u(8)); // max durability + Assert.assertEquals(32, bits.read15u(9)); // durability + Assert.assertEquals(57, bits.read15u(9)); // poisonmindam + Assert.assertEquals(0, bits.read31u(0)); // poisonmindam param bits + Assert.assertEquals(8, bits.read31u(10)); // poisonmindam value + Assert.assertEquals(0, bits.read31u(0)); // poisonmaxdam param bits + Assert.assertEquals(8, bits.read31u(10)); // poisonmaxdam value + Assert.assertEquals(0, bits.read31u(0)); // poisonlength param bits + Assert.assertEquals(50, bits.read31u(9)); // poisonlength value + Assert.assertEquals(0x1ff, bits.read15u(9)); // stat list finished + Assert.assertEquals(5, bits.bitsRemaining()); // tail end of stream + } + + @Test + public void write_hunters_bow_of_blight() { + ByteInput bytes = ByteInput.wrap(new byte[] { + 0x4A, 0x4D, 0x10, 0x00, (byte) 0x80, 0x00, 0x65, 0x00, 0x04, + (byte) 0x82, 0x26, 0x76, 0x07, (byte) 0x82, 0x09, (byte) 0xD4, + (byte) 0xAA, 0x12, 0x03, 0x01, (byte) 0x80, 0x70, 0x01, 0x01, + (byte) 0x91, 0x03, 0x01, 0x04, 0x64, (byte) 0xFC, 0x07}); + ByteOutput b = ByteOutput.wrap(Unpooled.buffer(256)); + b.writeBytes(new byte[]{0x4A, 0x4D}); // signature + b.write32(0x00800010); // flags + b.write8(101); // version + BitOutput bits = b.unalign(); + bits.skipBits(2); // unknown + bits.write7u((byte) 0, 3); // location + bits.write7u((byte) 0, 4); // body location + bits.write7u((byte) 2, 4); // grid x + bits.write7u((byte) 0, 4); // grid y + bits.write7u((byte) 1, 3); // store location + bits.writeString("hbw ", Byte.SIZE); // code + bits.write7u((byte) 0, 3); // sockets filled + bits.writeRaw(0x2555A813, 32); // id + bits.write7u((byte) 6, 7); // ilvl + bits.write7u((byte) 4, 4); // quality + bits.writeBoolean(false); // picture id + bits.writeBoolean(false); // class only + bits.write15u((short) 0, 11); // magic prefix + bits.write15u((short) 737, 11); // magic suffix + bits.skipBits(1); // unknown + bits.write15u((short) 32, 8); // max durability + bits.write15u((short) 32, 9); // durability + bits.write15u((short) 57, 9); // poisonmindam + bits.write31u(0, 0); // poisonmindam param bits + bits.write31u(8, 10); // poisonmindam value + bits.write31u(0, 0); // poisonmaxdam param bits + bits.write31u(8, 10); // poisonmaxdam value + bits.write31u(0, 0); // poisonlength param bits + bits.write31u(50, 9); // poisonlength value + bits.write15u((short) 0x1ff, 9); // stat list finished + bits.flush(); + System.out.println(ByteBufUtil.prettyHexDump(b.buffer())); + Assert.assertTrue(ByteBufUtil.equals(bytes.buffer(), b.buffer())); + } +} \ No newline at end of file