Files
bun.sh/test/bun.js/buffer.test.js
Colin McDonnell f7f1b60444 Add bun-types, add typechecking, add child_process types (#1475)
* Add bun-types to packages

* Improve typing

* Fix types in tests

* Fix dts tests

* Run formatter

* Fix all type errors

* Add strict mode, fix type errors

* Add ffi changes

* Move workflows to root

* Add workflows

* Remove labeler

* Add child_process types

* Fix synthetic defaults issue

* Remove docs

* Move scripts

* Run prettier

* Include examples in typechecking

* captureStackTrace types

* moved captureStackTrace types to globals

* Address reviews

Co-authored-by: Colin McDonnell <colinmcd@alum.mit.edu>
Co-authored-by: Dylan Conway <dylan.conway567@gmail.com>
2022-11-09 15:40:40 -08:00

615 lines
18 KiB
JavaScript

import { describe, it, expect, beforeEach, afterEach } from "bun:test";
import { gc } from "./gc";
beforeEach(() => gc());
afterEach(() => gc());
it("Buffer.toJSON()", () => {
expect(JSON.stringify(Buffer.from("hello"))).toBe(
JSON.stringify({
type: "Buffer",
data: [104, 101, 108, 108, 111],
}),
);
});
it("buffer", () => {
var buf = new Buffer(20);
gc();
// if this fails or infinitely loops, it means there is a memory issue with the JSC::Structure object
expect(Object.keys(buf).length > 0).toBe(true);
gc();
expect(buf.write("hello world ")).toBe(12);
expect(buf.write("hello world ", "utf8")).toBe(12);
gc();
expect(buf.toString("utf8", 0, "hello world ".length)).toBe("hello world ");
gc();
expect(buf.toString("base64url", 0, "hello world ".length)).toBe(
btoa("hello world "),
);
gc();
expect(buf instanceof Uint8Array).toBe(true);
gc();
expect(buf instanceof Buffer).toBe(true);
gc();
expect(buf.slice() instanceof Uint8Array).toBe(true);
gc();
expect(buf.slice(0, 1) instanceof Buffer).toBe(true);
gc();
expect(buf.slice(0, 1) instanceof Uint8Array).toBe(true);
gc();
expect(buf.slice(0, 1) instanceof Buffer).toBe(true);
gc();
expect(buf.slice(0, 0).length).toBe(0);
});
it("Buffer", () => {
var inputs = [
"hello world",
"hello world".repeat(100),
`😋 Get Emoji — All Emojis to ✂️ Copy and 📋 Paste 👌`,
];
var good = inputs.map((a) => new TextEncoder().encode(a));
for (let i = 0; i < inputs.length; i++) {
var input = inputs[i];
expect(new Buffer(input).toString("utf8")).toBe(inputs[i]);
gc();
expect(Array.from(new Buffer(input)).join(",")).toBe(good[i].join(","));
gc();
expect(Buffer.byteLength(input)).toBe(good[i].length);
gc();
expect(Buffer.from(input).byteLength).toBe(Buffer.byteLength(input));
}
});
it("Buffer.byteLength", () => {
expect(Buffer.byteLength("😀😃😄😁😆😅😂🤣☺️😊😊😇")).toBe(
new TextEncoder().encode("😀😃😄😁😆😅😂🤣☺️😊😊😇").byteLength,
);
});
it("Buffer.isBuffer", () => {
expect(Buffer.isBuffer(new Buffer(1))).toBe(true);
gc();
expect(Buffer.isBuffer(new Buffer(0))).toBe(true);
gc();
expect(Buffer.isBuffer(new Uint8Array(0))).toBe(false);
gc();
expect(Buffer.isBuffer(new Uint8Array(1))).toBe(false);
gc();
var a = new Uint8Array(1);
gc();
expect(Buffer.isBuffer(a)).toBe(false);
gc();
Buffer.toBuffer(a);
gc();
expect(Buffer.isBuffer(a)).toBe(true);
gc();
});
it("Buffer.toBuffer throws", () => {
const checks = [
[],
{},
"foo",
new Uint16Array(),
new DataView(new Uint8Array(14).buffer),
];
for (let i = 0; i < checks.length; i++) {
try {
Buffer.toBuffer(checks[i]);
expect(false).toBe(true);
} catch (exception) {
expect(exception.message).toBe("Expected Uint8Array");
}
}
expect(true).toBe(true);
});
it("Buffer.toBuffer works", () => {
var array = new Uint8Array(20);
expect(array instanceof Buffer).toBe(false);
var buf = Buffer.toBuffer(array);
expect(array instanceof Buffer).toBe(true);
// if this fails or infinitely loops, it means there is a memory issue with the JSC::Structure object
expect(Object.keys(buf).length > 0).toBe(true);
expect(buf.write("hello world ")).toBe(12);
gc();
expect(buf.toString("utf8", 0, "hello world ".length)).toBe("hello world ");
gc();
expect(buf.toString("base64url", 0, "hello world ".length)).toBe(
btoa("hello world "),
);
gc();
expect(buf instanceof Uint8Array).toBe(true);
expect(buf instanceof Buffer).toBe(true);
expect(buf.slice() instanceof Uint8Array).toBe(true);
expect(buf.slice(0, 1) instanceof Buffer).toBe(true);
expect(buf.slice(0, 1) instanceof Uint8Array).toBe(true);
expect(buf.slice(0, 1) instanceof Buffer).toBe(true);
expect(new Buffer(buf) instanceof Buffer).toBe(true);
expect(new Buffer(buf.buffer) instanceof Buffer).toBe(true);
});
it("writeInt", () => {
var buf = new Buffer(1024);
var data = new DataView(buf.buffer);
buf.writeInt32BE(100);
expect(data.getInt32(0, false)).toBe(100);
buf.writeInt32BE(100);
expect(data.getInt32(0, false)).toBe(100);
var childBuf = buf.subarray(0, 4);
expect(data.getInt32(0, false)).toBe(100);
expect(childBuf.readInt32BE(0, false)).toBe(100);
});
it("Buffer.from", () => {
expect(Buffer.from("hello world").toString("utf8")).toBe("hello world");
expect(Buffer.from("hello world", "ascii").toString("utf8")).toBe(
"hello world",
);
expect(Buffer.from("hello world", "latin1").toString("utf8")).toBe(
"hello world",
);
gc();
expect(Buffer.from([254]).join(",")).toBe("254");
expect(Buffer.from([254], "utf8").join(",")).toBe("254");
expect(Buffer.from([254], "utf-8").join(",")).toBe("254");
expect(Buffer.from([254], "latin").join(",")).toBe("254");
expect(Buffer.from([254], "uc2").join(",")).toBe("254");
expect(Buffer.from([254], "utf16").join(",")).toBe("254");
expect(Buffer.isBuffer(Buffer.from([254], "utf16"))).toBe(true);
expect(Buffer.from(123).join(",")).toBe(Uint8Array.from(123).join(","));
expect(Buffer.from({ length: 124 }).join(",")).toBe(
Uint8Array.from({ length: 124 }).join(","),
);
expect(Buffer.from(new ArrayBuffer(1024), 0, 512).join(",")).toBe(
new Uint8Array(512).join(","),
);
expect(Buffer.from(new Buffer(new ArrayBuffer(1024), 0, 512)).join(",")).toBe(
new Uint8Array(512).join(","),
);
gc();
});
it("Buffer.from latin1 vs ascii", () => {
const simpleBuffer = Buffer.from("\xa4", "binary");
expect(simpleBuffer.toString("latin1")).toBe("¤");
expect(simpleBuffer.toString("ascii")).toBe("$");
gc();
const asciiBuffer = Buffer.from("\xa4", "ascii");
expect(asciiBuffer.toString("latin1")).toBe("¤");
expect(asciiBuffer.toString("ascii")).toBe("$");
gc();
});
it("Buffer.equals", () => {
var a = new Uint8Array(10);
a[2] = 1;
var b = new Uint8Array(10);
b[2] = 1;
Buffer.toBuffer(a);
Buffer.toBuffer(b);
expect(a.equals(b)).toBe(true);
b[2] = 0;
expect(a.equals(b)).toBe(false);
});
it("Buffer.compare", () => {
var a = new Uint8Array(10);
a[2] = 1;
var b = new Uint8Array(10);
b[2] = 1;
Buffer.toBuffer(a);
Buffer.toBuffer(b);
expect(a.compare(b)).toBe(0);
b[2] = 0;
expect(a.compare(b)).toBe(1);
expect(b.compare(a)).toBe(-1);
const buf = Buffer.from("0123456789", "utf8");
const expectedSameBufs = [
[buf.slice(-10, 10), Buffer.from("0123456789", "utf8")],
[buf.slice(-20, 10), Buffer.from("0123456789", "utf8")],
[buf.slice(-20, -10), Buffer.from("", "utf8")],
[buf.slice(), Buffer.from("0123456789", "utf8")],
[buf.slice(0), Buffer.from("0123456789", "utf8")],
[buf.slice(0, 0), Buffer.from("", "utf8")],
[buf.slice(undefined), Buffer.from("0123456789", "utf8")],
[buf.slice("foobar"), Buffer.from("0123456789", "utf8")],
[buf.slice(undefined, undefined), Buffer.from("0123456789", "utf8")],
[buf.slice(2), Buffer.from("23456789", "utf8")],
[buf.slice(5), Buffer.from("56789", "utf8")],
[buf.slice(10), Buffer.from("", "utf8")],
[buf.slice(5, 8), Buffer.from("567", "utf8")],
[buf.slice(8, -1), Buffer.from("8", "utf8")],
[buf.slice(-10), Buffer.from("0123456789", "utf8")],
[buf.slice(0, -9), Buffer.from("0", "utf8")],
[buf.slice(0, -10), Buffer.from("", "utf8")],
[buf.slice(0, -1), Buffer.from("012345678", "utf8")],
[buf.slice(2, -2), Buffer.from("234567", "utf8")],
[buf.slice(0, 65536), Buffer.from("0123456789", "utf8")],
[buf.slice(65536, 0), Buffer.from("", "utf8")],
[buf.slice(-5, -8), Buffer.from("", "utf8")],
[buf.slice(-5, -3), Buffer.from("56", "utf8")],
[buf.slice(-10, 10), Buffer.from("0123456789", "utf8")],
[buf.slice("0", "1"), Buffer.from("0", "utf8")],
[buf.slice("-5", "10"), Buffer.from("56789", "utf8")],
[buf.slice("-10", "10"), Buffer.from("0123456789", "utf8")],
[buf.slice("-10", "-5"), Buffer.from("01234", "utf8")],
[buf.slice("-10", "-0"), Buffer.from("", "utf8")],
[buf.slice("111"), Buffer.from("", "utf8")],
[buf.slice("0", "-111"), Buffer.from("", "utf8")],
];
for (let i = 0, s = buf.toString(); i < buf.length; ++i) {
expectedSameBufs.push(
[buf.slice(i), Buffer.from(s.slice(i))],
[buf.slice(0, i), Buffer.from(s.slice(0, i))],
[buf.slice(-i), Buffer.from(s.slice(-i))],
[buf.slice(0, -i), Buffer.from(s.slice(0, -i))],
);
}
expectedSameBufs.forEach(([buf1, buf2]) => {
expect(Buffer.compare(buf1, buf2)).toBe(0);
});
{
const buf = Buffer.from([
1, 29, 0, 0, 1, 143, 216, 162, 92, 254, 248, 63, 0, 0, 0, 18, 184, 6, 0,
175, 29, 0, 8, 11, 1, 0, 0,
]);
const chunk1 = Buffer.from([
1, 29, 0, 0, 1, 143, 216, 162, 92, 254, 248, 63, 0,
]);
const chunk2 = Buffer.from([
0, 0, 18, 184, 6, 0, 175, 29, 0, 8, 11, 1, 0, 0,
]);
const middle = buf.length / 2;
expect(JSON.stringify(buf.slice(0, middle))).toBe(JSON.stringify(chunk1));
expect(JSON.stringify(buf.slice(middle))).toBe(JSON.stringify(chunk2));
}
});
it("Buffer.copy", () => {
var array1 = new Uint8Array(128);
array1.fill(100);
Buffer.toBuffer(array1);
var array2 = new Uint8Array(128);
array2.fill(200);
Buffer.toBuffer(array2);
var array3 = new Uint8Array(128);
Buffer.toBuffer(array3);
gc();
expect(array1.copy(array2)).toBe(128);
expect(array1.join("")).toBe(array2.join(""));
{
// Create two `Buffer` instances.
const buf1 = Buffer.allocUnsafe(26);
const buf2 = Buffer.allocUnsafe(26).fill("!");
for (let i = 0; i < 26; i++) {
// 97 is the decimal ASCII value for 'a'.
buf1[i] = i + 97;
}
// Copy `buf1` bytes 16 through 19 into `buf2` starting at byte 8 of `buf2`.
buf1.copy(buf2, 8, 16, 20);
expect(buf2.toString("ascii", 0, 25)).toBe("!!!!!!!!qrst!!!!!!!!!!!!!");
}
{
const buf = Buffer.allocUnsafe(26);
for (let i = 0; i < 26; i++) {
// 97 is the decimal ASCII value for 'a'.
buf[i] = i + 97;
}
buf.copy(buf, 0, 4, 10);
expect(buf.toString()).toBe("efghijghijklmnopqrstuvwxyz");
}
});
export function fillRepeating(dstBuffer, start, end) {
let len = dstBuffer.length, // important: use indices length, not byte-length
sLen = end - start,
p = sLen; // set initial position = source sequence length
// step 2: copy existing data doubling segment length per iteration
while (p < len) {
if (p + sLen > len) sLen = len - p; // if not power of 2, truncate last segment
dstBuffer.copyWithin(p, start, sLen); // internal copy
p += sLen; // add current length to offset
sLen <<= 1; // double length for next segment
}
}
describe("Buffer.fill string", () => {
for (let text of [
"hello world",
"1234567890",
"\uD83D\uDE00",
"😀😃😄😁😆😅😂🤣☺️😊😊😇",
]) {
it(text, () => {
var input = new Buffer(1024);
input.fill(text);
var demo = new Uint8Array(1024);
var encoded = new TextEncoder().encode(text);
demo.set(encoded);
fillRepeating(demo, 0, encoded.length);
expect(input.join("")).toBe(demo.join(""));
});
}
});
it("Buffer.fill 1 char string", () => {
var input = new Buffer(1024);
input.fill("h");
var demo = new Uint8Array(1024);
var encoded = new TextEncoder().encode("h");
demo.set(encoded);
fillRepeating(demo, 0, encoded.length);
expect(input.join("")).toBe(demo.join(""));
});
it("Buffer.concat", () => {
var array1 = new Uint8Array(128);
array1.fill(100);
var array2 = new Uint8Array(128);
array2.fill(200);
var array3 = new Uint8Array(128);
array3.fill(300);
gc();
expect(Buffer.concat([array1, array2, array3]).join("")).toBe(
array1.join("") + array2.join("") + array3.join(""),
);
expect(Buffer.concat([array1, array2, array3], 222).length).toBe(222);
expect(
Buffer.concat([array1, array2, array3], 222).subarray(0, 128).join(""),
).toBe("100".repeat(128));
expect(
Buffer.concat([array1, array2, array3], 222).subarray(129, 222).join(""),
).toBe("200".repeat(222 - 129));
});
it("read", () => {
var buf = new Buffer(1024);
var data = new DataView(buf.buffer);
function reset() {
new Uint8Array(buf.buffer).fill(0);
}
data.setBigInt64(0, BigInt(1000), false);
expect(buf.readBigInt64BE(0)).toBe(BigInt(1000));
reset();
data.setBigInt64(0, BigInt(1000), true);
expect(buf.readBigInt64LE(0)).toBe(BigInt(1000));
reset();
data.setBigUint64(0, BigInt(1000), false);
expect(buf.readBigUInt64BE(0)).toBe(BigInt(1000));
reset();
data.setBigUint64(0, BigInt(1000), true);
expect(buf.readBigUInt64LE(0)).toBe(BigInt(1000));
reset();
data.setFloat64(0, 1000, false);
expect(buf.readDoubleBE(0)).toBe(1000);
reset();
data.setFloat64(0, 1000, true);
expect(buf.readDoubleLE(0)).toBe(1000);
reset();
data.setFloat32(0, 1000, false);
expect(buf.readFloatBE(0)).toBe(1000);
reset();
data.setFloat32(0, 1000, true);
expect(buf.readFloatLE(0)).toBe(1000);
reset();
data.setInt16(0, 1000, false);
expect(buf.readInt16BE(0)).toBe(1000);
reset();
data.setInt16(0, 1000, true);
expect(buf.readInt16LE(0)).toBe(1000);
reset();
data.setInt32(0, 1000, false);
expect(buf.readInt32BE(0)).toBe(1000);
reset();
data.setInt32(0, 1000, true);
expect(buf.readInt32LE(0)).toBe(1000);
reset();
data.setInt8(0, 100, false);
expect(buf.readInt8(0)).toBe(100);
reset();
data.setUint16(0, 1000, false);
expect(buf.readUInt16BE(0)).toBe(1000);
reset();
data.setUint16(0, 1000, true);
expect(buf.readUInt16LE(0)).toBe(1000);
reset();
data.setUint32(0, 1000, false);
expect(buf.readUInt32BE(0)).toBe(1000);
reset();
data.setUint32(0, 1000, true);
expect(buf.readUInt32LE(0)).toBe(1000);
reset();
data.setUint8(0, 255, false);
expect(buf.readUInt8(0)).toBe(255);
reset();
data.setUint8(0, 255, false);
expect(buf.readUInt8(0)).toBe(255);
reset();
});
it("write", () => {
let buf = Buffer.alloc(16);
function reset() {
new Uint8Array(buf.buffer).fill(0);
}
expect(buf.write("hello", 8, 8)).toBe(5);
reset();
expect(buf.write("hello!", 3, 8)).toBe(6);
reset();
expect(buf.write("Foo Bar!", 4, 4)).toBe(4);
reset();
expect(buf.write("foo", 0, 1)).toBe(1);
reset();
expect(buf.write("foo", 0, 2)).toBe(2);
reset();
expect(buf.write("foo", 0)).toBe(3);
reset();
expect(buf.write("Foo Bar!", 4, 6)).toBe(6);
reset();
expect(buf.write("Foo Bar!", 4, 7)).toBe(7);
reset();
});
it("includes", () => {
const buf = Buffer.from("this is a buffer");
expect(buf.includes("this")).toBe(true);
expect(buf.includes("is")).toBe(true);
expect(buf.includes(Buffer.from("a buffer"))).toBe(true);
expect(buf.includes(97)).toBe(true);
expect(buf.includes(Buffer.from("a buffer example"))).toBe(false);
expect(buf.includes(Buffer.from("a buffer example").slice(0, 8))).toBe(true);
expect(buf.includes("this", 4)).toBe(false);
});
it("indexOf", () => {
const buf = Buffer.from("this is a buffer");
expect(buf.indexOf("this")).toBe(0);
expect(buf.indexOf("is")).toBe(2);
expect(buf.indexOf(Buffer.from("a buffer"))).toBe(8);
expect(buf.indexOf(97)).toBe(8);
expect(buf.indexOf(Buffer.from("a buffer example"))).toBe(-1);
expect(buf.indexOf(Buffer.from("a buffer example").slice(0, 8))).toBe(8);
const utf16Buffer = Buffer.from("\u039a\u0391\u03a3\u03a3\u0395", "utf16le");
expect(utf16Buffer.indexOf("\u03a3", 0, "utf16le")).toBe(4);
expect(utf16Buffer.indexOf("\u03a3", -4, "utf16le")).toBe(6);
const b = Buffer.from("abcdef");
// Passing a value that's a number, but not a valid byte.
// Prints: 2, equivalent to searching for 99 or 'c'.
expect(b.indexOf(99.9)).toBe(2);
expect(b.indexOf(256 + 99)).toBe(2);
// Passing a byteOffset that coerces to NaN or 0.
// Prints: 1, searching the whole buffer.
expect(b.indexOf("b", undefined)).toBe(1);
expect(b.indexOf("b", {})).toBe(1);
expect(b.indexOf("b", null)).toBe(1);
expect(b.indexOf("b", [])).toBe(1);
});
it("lastIndexOf", () => {
const buf = Buffer.from("this buffer is a buffer");
expect(buf.lastIndexOf("this")).toBe(0);
expect(buf.lastIndexOf("this", 0)).toBe(0);
expect(buf.lastIndexOf("this", -1000)).toBe(-1);
expect(buf.lastIndexOf("buffer")).toBe(17);
expect(buf.lastIndexOf(Buffer.from("buffer"))).toBe(17);
expect(buf.lastIndexOf(97)).toBe(15);
expect(buf.lastIndexOf(Buffer.from("yolo"))).toBe(-1);
expect(buf.lastIndexOf("buffer", 5)).toBe(5);
expect(buf.lastIndexOf("buffer", 4)).toBe(-1);
const utf16Buffer = Buffer.from("\u039a\u0391\u03a3\u03a3\u0395", "utf16le");
expect(utf16Buffer.lastIndexOf("\u03a3", undefined, "utf16le")).toBe(6);
expect(utf16Buffer.lastIndexOf("\u03a3", -5, "utf16le")).toBe(4);
const b = Buffer.from("abcdef");
// Passing a value that's a number, but not a valid byte.
// Prints: 2, equivalent to searching for 99 or 'c'.
expect(b.lastIndexOf(99.9)).toBe(2);
expect(b.lastIndexOf(256 + 99)).toBe(2);
// Passing a byteOffset that coerces to NaN or 0.
// Prints: 1, searching the whole buffer.
expect(b.lastIndexOf("b", undefined)).toBe(1);
expect(b.lastIndexOf("b", {})).toBe(1);
// Passing a byteOffset that coerces to 0.
// Prints: -1, equivalent to passing 0.
expect(b.lastIndexOf("b", null)).toBe(-1);
expect(b.lastIndexOf("b", [])).toBe(-1);
});
for (let fn of [Buffer.prototype.slice, Buffer.prototype.subarray]) {
it(`Buffer.${fn.name}`, () => {
const buf = new Buffer("buffer");
const slice = fn.call(buf, 1, 3);
expect(slice.toString()).toBe("uf");
const slice2 = fn.call(slice, 100);
expect(slice2.toString()).toBe("");
const slice3 = fn.call(slice, -1);
expect(slice3.toString()).toBe("f");
});
}
it("Buffer.from(base64)", () => {
const buf = Buffer.from("aGVsbG8gd29ybGQ=", "base64");
expect(buf.toString()).toBe("hello world");
expect(
Buffer.from(btoa('console.log("hello world")\n'), "base64").toString()
).toBe('console.log("hello world")\n');
});
it("Buffer.toString(base64)", () => {
{
const buf = Buffer.from("hello world");
expect(buf.toString("base64")).toBe("aGVsbG8gd29ybGQ=");
}
{
expect(Buffer.from(`console.log("hello world")\n`).toString("base64")).toBe(
btoa('console.log("hello world")\n')
);
}
});