2019-01-21 14:03:30 -05:00
|
|
|
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
|
2018-07-23 14:46:30 -04:00
|
|
|
|
2018-06-11 11:01:35 -04:00
|
|
|
// A simple runtime that doesn't involve typescript or protobufs to test
|
2018-08-10 15:09:28 -04:00
|
|
|
// libdeno. Invoked by libdeno_test.cc
|
2018-06-11 11:01:35 -04:00
|
|
|
|
2018-06-19 11:45:58 -04:00
|
|
|
const global = this;
|
|
|
|
|
2018-06-11 12:17:28 -04:00
|
|
|
function assert(cond) {
|
2018-08-10 15:09:28 -04:00
|
|
|
if (!cond) throw Error("libdeno_test.js assert failed");
|
2018-06-11 12:17:28 -04:00
|
|
|
}
|
|
|
|
|
2018-06-19 11:45:58 -04:00
|
|
|
global.CanCallFunction = () => {
|
2019-03-26 08:22:07 -04:00
|
|
|
Deno.core.print("Hello world from foo");
|
2018-06-11 15:30:58 -04:00
|
|
|
return "foo";
|
2018-06-19 11:45:58 -04:00
|
|
|
};
|
2018-06-11 15:30:58 -04:00
|
|
|
|
2018-06-12 00:36:01 -04:00
|
|
|
// This object is created to test snapshotting.
|
|
|
|
// See DeserializeInternalFieldsCallback and SerializeInternalFieldsCallback.
|
|
|
|
const snapshotted = new Uint8Array([1, 3, 3, 7]);
|
|
|
|
|
2018-06-19 11:45:58 -04:00
|
|
|
global.TypedArraySnapshots = () => {
|
2018-06-12 00:36:01 -04:00
|
|
|
assert(snapshotted[0] === 1);
|
|
|
|
assert(snapshotted[1] === 3);
|
|
|
|
assert(snapshotted[2] === 3);
|
|
|
|
assert(snapshotted[3] === 7);
|
2018-06-19 11:45:58 -04:00
|
|
|
};
|
2018-06-12 00:36:01 -04:00
|
|
|
|
2018-07-01 12:07:12 -04:00
|
|
|
global.RecvReturnEmpty = () => {
|
2018-07-08 21:35:34 -04:00
|
|
|
const m1 = new Uint8Array("abc".split("").map(c => c.charCodeAt(0)));
|
|
|
|
const m2 = m1.slice();
|
2019-03-26 08:22:07 -04:00
|
|
|
const r1 = Deno.core.send(m1);
|
2018-07-08 21:35:34 -04:00
|
|
|
assert(r1 == null);
|
2019-03-26 08:22:07 -04:00
|
|
|
const r2 = Deno.core.send(m2);
|
2018-07-08 21:35:34 -04:00
|
|
|
assert(r2 == null);
|
2018-06-19 11:45:58 -04:00
|
|
|
};
|
2018-06-11 14:18:56 -04:00
|
|
|
|
2018-07-01 12:07:12 -04:00
|
|
|
global.RecvReturnBar = () => {
|
2018-07-08 21:35:34 -04:00
|
|
|
const m = new Uint8Array("abc".split("").map(c => c.charCodeAt(0)));
|
2019-03-26 08:22:07 -04:00
|
|
|
const r = Deno.core.send(m);
|
2018-07-08 21:35:34 -04:00
|
|
|
assert(r instanceof Uint8Array);
|
2018-06-11 14:18:56 -04:00
|
|
|
assert(r.byteLength === 3);
|
2018-07-08 21:35:34 -04:00
|
|
|
const rstr = String.fromCharCode(...r);
|
2018-06-11 14:18:56 -04:00
|
|
|
assert(rstr === "bar");
|
2018-06-19 11:45:58 -04:00
|
|
|
};
|
2018-06-11 16:51:11 -04:00
|
|
|
|
2018-07-01 12:07:12 -04:00
|
|
|
global.DoubleRecvFails = () => {
|
2019-03-26 08:22:07 -04:00
|
|
|
// Deno.core.recv is an internal function and should only be called once from the
|
2018-06-11 16:51:11 -04:00
|
|
|
// runtime.
|
2019-03-30 14:45:36 -04:00
|
|
|
Deno.core.recv((_channel, _msg) => assert(false));
|
|
|
|
Deno.core.recv((_channel, _msg) => assert(false));
|
2018-06-19 11:45:58 -04:00
|
|
|
};
|
2018-06-18 09:55:36 -04:00
|
|
|
|
2018-07-08 21:35:34 -04:00
|
|
|
global.SendRecvSlice = () => {
|
|
|
|
const abLen = 1024;
|
|
|
|
let buf = new Uint8Array(abLen);
|
|
|
|
for (let i = 0; i < 5; i++) {
|
|
|
|
// Set first and last byte, for verification by the native side.
|
|
|
|
buf[0] = 100 + i;
|
|
|
|
buf[buf.length - 1] = 100 - i;
|
|
|
|
// On the native side, the slice is shortened by 19 bytes.
|
2019-03-26 08:22:07 -04:00
|
|
|
buf = Deno.core.send(buf);
|
2018-07-08 21:35:34 -04:00
|
|
|
assert(buf.byteOffset === i * 11);
|
|
|
|
assert(buf.byteLength === abLen - i * 30 - 19);
|
|
|
|
assert(buf.buffer.byteLength == abLen);
|
|
|
|
// Look for values written by the backend.
|
|
|
|
assert(buf[0] === 200 + i);
|
|
|
|
assert(buf[buf.length - 1] === 200 - i);
|
|
|
|
// On the JS side, the start of the slice is moved up by 11 bytes.
|
|
|
|
buf = buf.subarray(11);
|
|
|
|
assert(buf.byteOffset === (i + 1) * 11);
|
|
|
|
assert(buf.byteLength === abLen - (i + 1) * 30);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
global.JSSendArrayBufferViewTypes = () => {
|
|
|
|
// Test that ArrayBufferView slices are transferred correctly.
|
|
|
|
// Send Uint8Array.
|
|
|
|
const ab1 = new ArrayBuffer(4321);
|
|
|
|
const u8 = new Uint8Array(ab1, 2468, 1000);
|
|
|
|
u8[0] = 1;
|
2019-03-26 08:22:07 -04:00
|
|
|
Deno.core.send(u8);
|
2018-07-08 21:35:34 -04:00
|
|
|
// Send Uint32Array.
|
|
|
|
const ab2 = new ArrayBuffer(4321);
|
|
|
|
const u32 = new Uint32Array(ab2, 2468, 1000 / Uint32Array.BYTES_PER_ELEMENT);
|
|
|
|
u32[0] = 0x02020202;
|
2019-03-26 08:22:07 -04:00
|
|
|
Deno.core.send(u32);
|
2018-07-08 21:35:34 -04:00
|
|
|
// Send DataView.
|
|
|
|
const ab3 = new ArrayBuffer(4321);
|
|
|
|
const dv = new DataView(ab3, 2468, 1000);
|
|
|
|
dv.setUint8(0, 3);
|
2019-03-26 08:22:07 -04:00
|
|
|
Deno.core.send(dv);
|
2018-07-08 21:35:34 -04:00
|
|
|
};
|
|
|
|
|
2018-06-18 09:55:36 -04:00
|
|
|
// The following join has caused SnapshotBug to segfault when using kKeep.
|
|
|
|
[].join("");
|
|
|
|
|
2018-06-19 11:45:58 -04:00
|
|
|
global.SnapshotBug = () => {
|
2018-06-18 09:55:36 -04:00
|
|
|
assert("1,2,3" === String([1, 2, 3]));
|
2018-06-19 11:45:58 -04:00
|
|
|
};
|
2018-06-22 08:57:49 -04:00
|
|
|
|
2018-08-26 03:57:16 -04:00
|
|
|
global.GlobalErrorHandling = () => {
|
2018-06-22 08:57:49 -04:00
|
|
|
eval("\n\n notdefined()\n//# sourceURL=helloworld.js");
|
|
|
|
};
|
2018-07-23 14:11:41 -04:00
|
|
|
|
2018-09-27 17:33:10 -04:00
|
|
|
// Allocate this buf at the top level to avoid GC.
|
2019-02-26 17:36:05 -05:00
|
|
|
const zeroCopyBuf = new Uint8Array([3, 4]);
|
2018-09-27 17:33:10 -04:00
|
|
|
|
2019-02-26 17:36:05 -05:00
|
|
|
global.ZeroCopyBuf = () => {
|
2018-09-27 17:33:10 -04:00
|
|
|
const a = new Uint8Array([1, 2]);
|
2019-02-26 17:36:05 -05:00
|
|
|
const b = zeroCopyBuf;
|
2018-09-27 17:33:10 -04:00
|
|
|
// The second parameter of send should modified by the
|
|
|
|
// privileged side.
|
2019-03-26 08:22:07 -04:00
|
|
|
const r = Deno.core.send(a, b);
|
2018-09-27 17:33:10 -04:00
|
|
|
assert(r == null);
|
|
|
|
// b is different.
|
|
|
|
assert(b[0] === 4);
|
|
|
|
assert(b[1] === 2);
|
|
|
|
// Now we modify it again.
|
|
|
|
b[0] = 9;
|
|
|
|
b[1] = 8;
|
2018-07-23 14:11:41 -04:00
|
|
|
};
|
2018-10-12 14:22:52 -04:00
|
|
|
|
2018-12-06 23:05:36 -05:00
|
|
|
global.CheckPromiseErrors = () => {
|
2018-10-12 14:22:52 -04:00
|
|
|
async function fn() {
|
|
|
|
throw new Error("message");
|
|
|
|
}
|
|
|
|
|
|
|
|
(async () => {
|
|
|
|
try {
|
|
|
|
await fn();
|
|
|
|
} catch (e) {
|
2019-03-26 08:22:07 -04:00
|
|
|
Deno.core.send(new Uint8Array([42]));
|
2018-10-12 14:22:52 -04:00
|
|
|
}
|
|
|
|
})();
|
2018-12-06 23:05:36 -05:00
|
|
|
};
|
2018-10-24 02:17:10 -04:00
|
|
|
|
|
|
|
global.Shared = () => {
|
2019-03-26 08:22:07 -04:00
|
|
|
const ab = Deno.core.shared;
|
2019-02-08 23:32:17 -05:00
|
|
|
assert(ab instanceof SharedArrayBuffer);
|
2019-03-26 08:22:07 -04:00
|
|
|
assert(Deno.core.shared != undefined);
|
2018-10-24 02:17:10 -04:00
|
|
|
assert(ab.byteLength === 3);
|
|
|
|
const ui8 = new Uint8Array(ab);
|
|
|
|
assert(ui8[0] === 0);
|
|
|
|
assert(ui8[1] === 1);
|
|
|
|
assert(ui8[2] === 2);
|
|
|
|
ui8[0] = 42;
|
|
|
|
ui8[1] = 43;
|
|
|
|
ui8[2] = 44;
|
2018-12-06 23:05:36 -05:00
|
|
|
};
|
2019-02-09 16:55:40 -05:00
|
|
|
|
|
|
|
global.LibDenoEvalContext = () => {
|
2019-03-26 08:22:07 -04:00
|
|
|
const [result, errInfo] = Deno.core.evalContext("let a = 1; a");
|
2019-02-09 16:55:40 -05:00
|
|
|
assert(result === 1);
|
|
|
|
assert(!errInfo);
|
2019-03-26 08:22:07 -04:00
|
|
|
const [result2, errInfo2] = Deno.core.evalContext("a = a + 1; a");
|
2019-02-09 16:55:40 -05:00
|
|
|
assert(result2 === 2);
|
|
|
|
assert(!errInfo2);
|
|
|
|
};
|
|
|
|
|
|
|
|
global.LibDenoEvalContextError = () => {
|
2019-03-26 08:22:07 -04:00
|
|
|
const [result, errInfo] = Deno.core.evalContext("not_a_variable");
|
2019-02-09 16:55:40 -05:00
|
|
|
assert(!result);
|
|
|
|
assert(!!errInfo);
|
|
|
|
assert(errInfo.isNativeError); // is a native error (ReferenceError)
|
|
|
|
assert(!errInfo.isCompileError); // is NOT a compilation error
|
|
|
|
assert(errInfo.thrown.message === "not_a_variable is not defined");
|
|
|
|
|
2019-03-26 08:22:07 -04:00
|
|
|
const [result2, errInfo2] = Deno.core.evalContext("throw 1");
|
2019-02-09 16:55:40 -05:00
|
|
|
assert(!result2);
|
|
|
|
assert(!!errInfo2);
|
|
|
|
assert(!errInfo2.isNativeError); // is NOT a native error
|
|
|
|
assert(!errInfo2.isCompileError); // is NOT a compilation error
|
|
|
|
assert(errInfo2.thrown === 1);
|
|
|
|
|
2019-03-26 08:22:07 -04:00
|
|
|
const [result3, errInfo3] = Deno.core.evalContext(
|
|
|
|
"class AError extends Error {}; throw new AError('e')"
|
|
|
|
);
|
2019-02-09 16:55:40 -05:00
|
|
|
assert(!result3);
|
|
|
|
assert(!!errInfo3);
|
|
|
|
assert(errInfo3.isNativeError); // extend from native error, still native error
|
|
|
|
assert(!errInfo3.isCompileError); // is NOT a compilation error
|
|
|
|
assert(errInfo3.thrown.message === "e");
|
|
|
|
|
2019-03-26 08:22:07 -04:00
|
|
|
const [result4, errInfo4] = Deno.core.evalContext("{");
|
2019-02-09 16:55:40 -05:00
|
|
|
assert(!result4);
|
|
|
|
assert(!!errInfo4);
|
|
|
|
assert(errInfo4.isNativeError); // is a native error (SyntaxError)
|
|
|
|
assert(errInfo4.isCompileError); // is a compilation error! (braces not closed)
|
|
|
|
assert(errInfo4.thrown.message === "Unexpected end of input");
|
|
|
|
|
2019-03-26 08:22:07 -04:00
|
|
|
const [result5, errInfo5] = Deno.core.evalContext("eval('{')");
|
2019-02-09 16:55:40 -05:00
|
|
|
assert(!result5);
|
|
|
|
assert(!!errInfo5);
|
|
|
|
assert(errInfo5.isNativeError); // is a native error (SyntaxError)
|
|
|
|
assert(!errInfo5.isCompileError); // is NOT a compilation error! (just eval)
|
|
|
|
assert(errInfo5.thrown.message === "Unexpected end of input");
|
|
|
|
};
|
2019-06-06 21:51:04 -04:00
|
|
|
|
|
|
|
global.LibDenoEvalContextInvalidArgument = () => {
|
|
|
|
try {
|
|
|
|
Deno.core.evalContext();
|
|
|
|
} catch (e) {
|
|
|
|
assert(e instanceof TypeError);
|
|
|
|
assert(e.message === "Invalid Argument");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
global.LibDenoPrintInvalidArgument = () => {
|
|
|
|
try {
|
|
|
|
Deno.core.print();
|
|
|
|
} catch (e) {
|
|
|
|
assert(e instanceof TypeError);
|
|
|
|
assert(e.message === "Invalid Argument");
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
Deno.core.print(2, 3, 4);
|
|
|
|
} catch (e) {
|
|
|
|
assert(e instanceof TypeError);
|
|
|
|
assert(e.message === "Invalid Argument");
|
|
|
|
}
|
|
|
|
};
|
2019-06-20 12:04:34 -04:00
|
|
|
|
|
|
|
global.WasmInstantiate = () => {
|
|
|
|
// The following blob can be created by taking the following s-expr and pass
|
|
|
|
// it through wat2wasm.
|
|
|
|
// (module
|
|
|
|
// (func $add (param $a i32) (param $b i32) (result i32)
|
|
|
|
// local.get $a
|
|
|
|
// local.get $b
|
|
|
|
// i32.add)
|
|
|
|
// (export "add" (func $add))
|
|
|
|
// )
|
|
|
|
// prettier-ignore
|
|
|
|
const bytes = new Uint8Array([
|
|
|
|
0x00, 0x61, 0x73, 0x6d, 0x01, 0x00, 0x00, 0x00, 0x01, 0x07, 0x01, 0x60,
|
|
|
|
0x02, 0x7f, 0x7f, 0x01, 0x7f, 0x03, 0x02, 0x01, 0x00, 0x07, 0x07, 0x01,
|
|
|
|
0x03, 0x61, 0x64, 0x64, 0x00, 0x00, 0x0a, 0x09, 0x01, 0x07, 0x00, 0x20,
|
|
|
|
0x00, 0x20, 0x01, 0x6a, 0x0b
|
|
|
|
]);
|
|
|
|
|
|
|
|
(async () => {
|
|
|
|
Deno.core.send(new Uint8Array([42]));
|
|
|
|
|
|
|
|
const wasm = await WebAssembly.instantiate(bytes);
|
|
|
|
|
|
|
|
Deno.core.send(new Uint8Array([42]));
|
|
|
|
|
|
|
|
const result = wasm.instance.exports.add(1, 3);
|
|
|
|
if (result != 4) {
|
|
|
|
throw Error("bad");
|
|
|
|
}
|
|
|
|
// To signal success, we send back a fixed buffer.
|
|
|
|
Deno.core.send(new Uint8Array([42]));
|
|
|
|
})();
|
|
|
|
};
|