mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
232 lines
8.1 KiB
TypeScript
232 lines
8.1 KiB
TypeScript
import { BufferCursor } from './BufferCursor';
|
|
|
|
test('simple properties and invariants', () => {
|
|
const cursor = new BufferCursor(10, true);
|
|
|
|
expect(cursor.size).toBe(cursor.position + cursor.bytes_left);
|
|
expect(cursor.size).toBeLessThanOrEqual(cursor.capacity);
|
|
expect(cursor.size).toBe(0);
|
|
expect(cursor.capacity).toBe(10);
|
|
expect(cursor.position).toBe(0);
|
|
expect(cursor.bytes_left).toBe(0);
|
|
expect(cursor.little_endian).toBe(true);
|
|
|
|
cursor.write_u8(99).write_u8(99).write_u8(99).write_u8(99);
|
|
cursor.seek(-1);
|
|
|
|
expect(cursor.size).toBe(cursor.position + cursor.bytes_left);
|
|
expect(cursor.size).toBeLessThanOrEqual(cursor.capacity);
|
|
expect(cursor.size).toBe(4);
|
|
expect(cursor.capacity).toBe(10);
|
|
expect(cursor.position).toBe(3);
|
|
expect(cursor.bytes_left).toBe(1);
|
|
expect(cursor.little_endian).toBe(true);
|
|
});
|
|
|
|
test('correct byte order handling', () => {
|
|
const buffer = new Uint8Array([1, 2, 3, 4]).buffer;
|
|
|
|
expect(new BufferCursor(buffer, false).u32()).toBe(0x01020304);
|
|
expect(new BufferCursor(buffer, true).u32()).toBe(0x04030201);
|
|
});
|
|
|
|
test('reallocation of internal buffer when necessary', () => {
|
|
const cursor = new BufferCursor(3, true);
|
|
cursor.write_u8(99).write_u8(99).write_u8(99).write_u8(99);
|
|
|
|
expect(cursor.size).toBe(4);
|
|
expect(cursor.capacity).toBeGreaterThanOrEqual(4);
|
|
expect(cursor.buffer.byteLength).toBeGreaterThanOrEqual(4);
|
|
});
|
|
|
|
function test_integer_read(method_name: string) {
|
|
test(method_name, () => {
|
|
const bytes = parseInt(method_name.replace(/^[iu](\d+)$/, '$1'), 10) / 8;
|
|
let test_number_1 = 0;
|
|
let test_number_2 = 0;
|
|
// The "false" arrays are for big endian tests and the "true" arrays for little endian tests.
|
|
const test_arrays: { [index: string]: number[] } = { false: [], true: [] };
|
|
|
|
for (let i = 1; i <= bytes; ++i) {
|
|
// Generates numbers of the form 0x010203...
|
|
test_number_1 <<= 8;
|
|
test_number_1 |= i;
|
|
test_arrays['false'].push(i);
|
|
test_arrays['true'].unshift(i);
|
|
}
|
|
|
|
for (let i = bytes + 1; i <= 2 * bytes; ++i) {
|
|
test_number_2 <<= 8;
|
|
test_number_2 |= i;
|
|
test_arrays['false'].push(i);
|
|
test_arrays['true'].splice(bytes, 0, i);
|
|
}
|
|
|
|
for (const little_endian of [false, true]) {
|
|
const cursor = new BufferCursor(
|
|
new Uint8Array(test_arrays[String(little_endian)]).buffer, little_endian);
|
|
|
|
expect((cursor as any)[method_name]()).toBe(test_number_1);
|
|
expect(cursor.position).toBe(bytes);
|
|
|
|
expect((cursor as any)[method_name]()).toBe(test_number_2);
|
|
expect(cursor.position).toBe(2 * bytes);
|
|
}
|
|
});
|
|
}
|
|
|
|
test_integer_read('u8');
|
|
test_integer_read('u16');
|
|
test_integer_read('u32');
|
|
test_integer_read('i32');
|
|
|
|
test('u8_array', () => {
|
|
const cursor = new BufferCursor(new Uint8Array([1, 2, 3, 4, 5, 6, 7, 8]).buffer, true);
|
|
|
|
expect(cursor.u8_array(3)).toEqual([1, 2, 3]);
|
|
expect(cursor.seek_start(2).u8_array(4)).toEqual([3, 4, 5, 6]);
|
|
expect(cursor.seek_start(5).u8_array(3)).toEqual([6, 7, 8]);
|
|
});
|
|
|
|
function test_string_read(method_name: string, char_size: number) {
|
|
test(method_name, () => {
|
|
const char_array = [7, 65, 66, 0, 255, 13];
|
|
|
|
for (const little_endian of [false, true]) {
|
|
const char_array_copy = [];
|
|
|
|
for (const char of char_array) {
|
|
if (little_endian) char_array_copy.push(char);
|
|
|
|
for (let i = 0; i < char_size - 1; ++i) {
|
|
char_array_copy.push(0);
|
|
}
|
|
|
|
if (!little_endian) char_array_copy.push(char);
|
|
}
|
|
|
|
const cursor = new BufferCursor(
|
|
new Uint8Array(char_array_copy).buffer, little_endian);
|
|
|
|
cursor.seek_start(char_size);
|
|
expect((cursor as any)[method_name](4 * char_size, true, true)).toBe('AB');
|
|
expect(cursor.position).toBe(5 * char_size);
|
|
cursor.seek_start(char_size);
|
|
expect((cursor as any)[method_name](2 * char_size, true, true)).toBe('AB');
|
|
expect(cursor.position).toBe(3 * char_size);
|
|
|
|
cursor.seek_start(char_size);
|
|
expect((cursor as any)[method_name](4 * char_size, true, false)).toBe('AB');
|
|
expect(cursor.position).toBe(4 * char_size);
|
|
cursor.seek_start(char_size);
|
|
expect((cursor as any)[method_name](2 * char_size, true, false)).toBe('AB');
|
|
expect(cursor.position).toBe(3 * char_size);
|
|
|
|
cursor.seek_start(char_size);
|
|
expect((cursor as any)[method_name](4 * char_size, false, true)).toBe('AB\0ÿ');
|
|
expect(cursor.position).toBe(5 * char_size);
|
|
|
|
cursor.seek_start(char_size);
|
|
expect((cursor as any)[method_name](4 * char_size, false, false)).toBe('AB\0ÿ');
|
|
expect(cursor.position).toBe(5 * char_size);
|
|
}
|
|
});
|
|
}
|
|
|
|
test_string_read('string_ascii', 1);
|
|
test_string_read('string_utf16', 2);
|
|
|
|
function test_integer_write(method_name: string) {
|
|
test(method_name, () => {
|
|
const bytes = parseInt(method_name.replace(/^write_[iu](\d+)$/, '$1'), 10) / 8;
|
|
let test_number_1 = 0;
|
|
let test_number_2 = 0;
|
|
// The "false" arrays are for big endian tests and the "true" arrays for little endian tests.
|
|
const test_arrays_1: { [index: string]: number[] } = { false: [], true: [] };
|
|
const test_arrays_2: { [index: string]: number[] } = { false: [], true: [] };
|
|
|
|
for (let i = 1; i <= bytes; ++i) {
|
|
// Generates numbers of the form 0x010203...
|
|
test_number_1 <<= 8;
|
|
test_number_1 |= i;
|
|
test_number_2 <<= 8;
|
|
test_number_2 |= i + bytes;
|
|
test_arrays_1['false'].push(i);
|
|
test_arrays_1['true'].unshift(i);
|
|
test_arrays_2['false'].push(i + bytes);
|
|
test_arrays_2['true'].unshift(i + bytes);
|
|
}
|
|
|
|
for (const little_endian of [false, true]) {
|
|
const cursor = new BufferCursor(0, little_endian);
|
|
(cursor as any)[method_name](test_number_1);
|
|
|
|
expect(cursor.position).toBe(bytes);
|
|
expect(cursor.seek_start(0).u8_array(bytes))
|
|
.toEqual(test_arrays_1[String(little_endian)]);
|
|
expect(cursor.position).toBe(bytes);
|
|
|
|
(cursor as any)[method_name](test_number_2);
|
|
|
|
expect(cursor.position).toBe(2 * bytes);
|
|
expect(cursor.seek_start(0).u8_array(2 * bytes))
|
|
.toEqual(test_arrays_1[String(little_endian)].concat(test_arrays_2[String(little_endian)]));
|
|
}
|
|
});
|
|
}
|
|
|
|
test_integer_write('write_u8');
|
|
test_integer_write('write_u16');
|
|
test_integer_write('write_u32');
|
|
|
|
test('write_f32', () => {
|
|
for (const little_endian of [false, true]) {
|
|
const cursor = new BufferCursor(0, little_endian);
|
|
cursor.write_f32(1337.9001);
|
|
|
|
expect(cursor.position).toBe(4);
|
|
expect(cursor.seek(-4).f32()).toBeCloseTo(1337.9001, 4);
|
|
expect(cursor.position).toBe(4);
|
|
|
|
cursor.write_f32(103.502);
|
|
|
|
expect(cursor.position).toBe(8);
|
|
expect(cursor.seek(-4).f32()).toBeCloseTo(103.502, 3);
|
|
}
|
|
});
|
|
|
|
test('write_u8_array', () => {
|
|
for (const little_endian of [false, true]) {
|
|
const bytes = 10;
|
|
const cursor = new BufferCursor(2 * bytes, little_endian);
|
|
const uint8_array = new Uint8Array(cursor.buffer);
|
|
const test_array_1 = [];
|
|
const test_array_2 = [];
|
|
|
|
for (let i = 1; i <= bytes; ++i) {
|
|
test_array_1.push(i);
|
|
test_array_2.push(i + bytes);
|
|
}
|
|
|
|
cursor.write_u8_array(test_array_1);
|
|
|
|
expect(cursor.position).toBe(bytes);
|
|
|
|
for (let i = 0; i < bytes; ++i) {
|
|
expect(uint8_array[i]).toBe(test_array_1[i]);
|
|
}
|
|
|
|
cursor.write_u8_array(test_array_2);
|
|
|
|
expect(cursor.position).toBe(2 * bytes);
|
|
|
|
for (let i = 0; i < bytes; ++i) {
|
|
expect(uint8_array[i]).toBe(test_array_1[i]);
|
|
}
|
|
|
|
for (let i = 0; i < bytes; ++i) {
|
|
expect(uint8_array[i + bytes]).toBe(test_array_2[i]);
|
|
}
|
|
}
|
|
});
|