#include <cstdio>
#include <iostream>
#include <cmath>
#include <emscripten/bind.h>
#include <emscripten/emscripten.h>
#include <emscripten/val.h>
using namespace emscripten;
using namespace std;
void test(string message) {
cout << "test: " << message << "...\n";
}
#define ensure(X) assert(X)
#define ensure_not(X) assert(!(X))
#define ensure_js(js_code) ensure(emscripten_run_script_int(js_code))
#define ensure_js_not(js_code) ensure_js((string("!( ") + js_code + " )").c_str())
void throw_js_error(val js_error) {
js_error.throw_();
}
struct Dummy {};
Dummy * makeDummy() {
return new Dummy();
}
EMSCRIPTEN_BINDINGS(test_bindings) {
emscripten::class_<Dummy>("Dummy");
emscripten::function("throw_js_error", &throw_js_error);
emscripten::function("makeDummy", &makeDummy, emscripten::allow_raw_pointers());
}
int main() {
printf("start\n");
test("val array()");
val::global().set("a", val::array());
ensure_js("a instanceof Array");
ensure_js_not("a instanceof Boolean");
ensure_js_not("a instanceof Number");
test("template<typename T> val array(const std::vector<T> vec)");
vector<val> vec1;
vec1.push_back(val(11));
vec1.push_back(val("a"));
vec1.push_back(val::array());
val::global().set("a", val::array(vec1));
ensure_js("a instanceof Array");
ensure_js_not("a instanceof Boolean");
ensure_js_not("a instanceof Number");
ensure_js("a[0] == 11");
ensure_js_not("a[0] == 12");
ensure_js("a[1] == 'a'");
ensure_js("a[2] instanceof Array");
vector<int> vec2;
vec2.push_back(0);
vec2.push_back(1);
vec2.push_back(3);
val::global().set("a", val::array(vec2));
ensure_js("a instanceof Array");
ensure_js_not("a instanceof Number");
ensure_js("a[0] == 0");
ensure_js_not("a[0] == 1");
ensure_js("a[1] == 1");
ensure_js("a[2] == 3");
ensure_js_not("a[2] == 2");
vector<int> vec2_from_iter;
for (val&& v : val::global("a")) {
vec2_from_iter.push_back(v.as<int>());
}
ensure(vec2 == vec2_from_iter);
test("template<typename Iter> val array(Iter begin, Iter end)");
val::global().set("a", val::array(vec1.begin(), vec1.end()));
ensure_js("a instanceof Array");
ensure_js_not("a instanceof Boolean");
ensure_js_not("a instanceof Number");
ensure_js("a[0] == 11");
ensure_js_not("a[0] == 12");
ensure_js("a[1] == 'a'");
ensure_js("a[2] instanceof Array");
val::global().set("a", val::array(vec2.begin(), vec2.end()));
ensure_js("a instanceof Array");
ensure_js_not("a instanceof Number");
ensure_js("a[0] == 0");
ensure_js_not("a[0] == 1");
ensure_js("a[1] == 1");
ensure_js("a[2] == 3");
ensure_js_not("a[2] == 2");
int arr[] = {1, 2, 3};
val::global().set("a", val::array(arr, arr + 3));
ensure_js("a instanceof Array");
ensure_js_not("a instanceof Number");
ensure_js("a[0] == 1");
ensure_js("a[1] == 2");
ensure_js("a[2] == 3");
test("val object()");
val::global().set("a", val::object());
ensure_js("a instanceof Object");
{
val a = val::global("a");
val a_roundtrip = val::take_ownership(EM_VAL(EM_ASM_PTR({
return Emval.toHandle(Emval.toValue($0));
}, a.as_handle())));
ensure(a == a_roundtrip);
}
test("val undefined()");
val::global().set("a", val::undefined());
ensure_js("a == undefined");
ensure_js("a != false");
test("val null()");
val::global().set("a", val::null());
ensure_js("a == null");
ensure_js("a != false");
test("val global(const char* name = 0)");
val::global();
val::global("addEventListener");
test("template<typename T> explicit val(T&& value)");
val::global().set("a", val(false));
ensure_js("a == false");
val::global().set("a", val(true));
ensure_js("a == true");
val::global().set("a", val(1));
ensure_js("a == 1");
val::global().set("a", val(string("1")));
ensure_js("a == '1'");
test("val(const char* v)");
val::global().set("a", val("1"));
ensure_js("a == '1'");
test("val()");
val a;
val::global().set("a", a);
ensure_js("a === undefined");
a = val(1);
val::global().set("a", a);
ensure_js("a == 1");
val ar[2];
ar[0] = val(1);
ar[1] = val(2);
val::global().set("a", val::array(ar, ar + 2));
ensure_js("a instanceof Array");
ensure_js_not("a instanceof Number");
ensure_js("a[0] == 1");
ensure_js("a[1] == 2");
test("bool isNull()");
EM_ASM(
globalThis.a = null;
globalThis.b = false;
globalThis.c = null;
globalThis.d = null;
globalThis.e = null;
globalThis.f = null;
globalThis.g = null;
globalThis.h = null;
globalThis.i = null;
);
ensure(val::global("a").isNull());
ensure_not(val::global("b").isNull());
test("bool isUndefined()");
EM_ASM(
a = undefined;
b = false;
);
ensure(val::global("a").isUndefined());
ensure_not(val::global("b").isUndefined());
test("bool isTrue()");
EM_ASM(
a = true;
b = false;
c = null;
);
ensure(val::global("a").isTrue());
ensure_not(val::global("b").isTrue());
ensure_not(val::global("c").isTrue());
test("bool isFalse()");
EM_ASM(
a = false;
b = true;
c = null;
);
ensure(val::global("a").isFalse());
ensure_not(val::global("b").isFalse());
ensure_not(val::global("c").isFalse());
test("bool isNumber()");
EM_ASM(
a = 1;
b = 1.1;
c = true;
);
ensure(val::global("a").isNumber());
ensure(val::global("b").isNumber());
ensure_not(val::global("c").isNumber());
test("bool isString()");
EM_ASM(
a = '1';
b = 1;
c = true;
);
ensure(val::global("a").isString());
ensure_not(val::global("b").isString());
ensure_not(val::global("c").isString());
test("bool isArray()");
EM_ASM(
a = [];
b = [1];
c = true;
d = {};
);
ensure(val::global("a").isArray());
ensure(val::global("b").isArray());
ensure_not(val::global("c").isArray());
ensure_not(val::global("d").isArray());
test("val& operator=(val&& v)");
val source(val::object());
val target(val::object());
source.set("val", 1);
target = std::move(source);
ensure(target["val"].as<int>() == 1);
target.set("val", 2);
target = std::move(target);
ensure(target["val"].as<int>() == 2);
test("bool equals(const val& v)");
EM_ASM(
a = 1;
b = 1;
c = 2;
d = true;
e = '1';
);
ensure(val::global("a").equals(val::global("a")));
ensure(val::global("a").equals(val::global("b")));
ensure_not(val::global("a").equals(val::global("c")));
ensure(val::global("a").equals(val::global("d")));
ensure(val::global("a").equals(val::global("e")));
ensure_not(val::global("c").equals(val::global("d")));
test("bool operator==(const val& v)");
EM_ASM(
a = 1;
b = 1;
c = 2;
d = true;
e = '1';
);
ensure(val::global("a") == val::global("a"));
ensure(val::global("a") == val::global("b"));
ensure_not(val::global("a") == val::global("c"));
ensure(val::global("a") == val::global("d"));
ensure(val::global("a") == val::global("e"));
ensure_not(val::global("c") == val::global("d"));
test("bool operator!=(const val& v)");
EM_ASM(
a = 1;
b = 1;
c = 2;
d = true;
e = '1';
);
ensure_not(val::global("a") != val::global("a"));
ensure_not(val::global("a") != val::global("b"));
ensure(val::global("a") != val::global("c"));
ensure_not(val::global("a") != val::global("d"));
ensure_not(val::global("a") != val::global("e"));
ensure(val::global("c") != val::global("d"));
test("bool strictlyEquals(const val& v)");
EM_ASM(
a = 1;
b = 1;
c = 2;
d = true;
e = '1';
);
ensure(val::global("a").strictlyEquals(val::global("a")));
ensure(val::global("a").strictlyEquals(val::global("b")));
ensure_not(val::global("a").strictlyEquals(val::global("c")));
ensure_not(val::global("a").strictlyEquals(val::global("d")));
ensure_not(val::global("a").strictlyEquals(val::global("e")));
ensure_not(val::global("c").strictlyEquals(val::global("d")));
test("bool operator>(const val& v)");
EM_ASM(
a = 1;
b = 1;
c = 2;
d = '2';
);
ensure_not(val::global("a") > val::global("a"));
ensure_not(val::global("a") > val::global("b"));
ensure_not(val::global("a") > val::global("c"));
ensure_not(val::global("a") > val::global("d"));
ensure(val::global("c") > val::global("a"));
ensure(val::global("d") > val::global("a"));
test("bool operator>= (const val& v)");
EM_ASM(
a = 1;
b = 1;
c = 2;
d = '2';
);
ensure(val::global("a") >= val::global("a"));
ensure(val::global("a") >= val::global("b"));
ensure_not(val::global("a") >= val::global("c"));
ensure_not(val::global("a") >= val::global("d"));
ensure(val::global("c") >= val::global("a"));
ensure(val::global("d") >= val::global("a"));
test("bool operator<(const val& v)");
EM_ASM(
a = 1;
b = 1;
c = 2;
d = '2';
);
ensure_not(val::global("a") < val::global("a"));
ensure_not(val::global("a") < val::global("b"));
ensure(val::global("a") < val::global("c"));
ensure(val::global("a") < val::global("d"));
ensure_not(val::global("c") < val::global("a"));
ensure_not(val::global("d") < val::global("a"));
test("bool operator<= (const val& v)");
EM_ASM(
a = 1;
b = 1;
c = 2;
d = '2';
);
ensure(val::global("a") <= val::global("a"));
ensure(val::global("a") <= val::global("b"));
ensure(val::global("a") <= val::global("c"));
ensure(val::global("a") <= val::global("d"));
ensure_not(val::global("c") <= val::global("a"));
ensure_not(val::global("d") <= val::global("a"));
test("bool operator!()");
EM_ASM(
a = true;
b = false;
c = null;
d = undefined;
e = 0;
f = 1;
g = "";
h = '0';
i = 'false';
);
ensure(!val::global("a") == false);
ensure(!val::global("b") == true);
ensure(!val::global("c") == true);
ensure(!val::global("d") == true);
ensure(!val::global("e") == true);
ensure(!val::global("f") == false);
ensure(!val::global("g") == true);
ensure(!val::global("h") == false);
ensure(!val::global("i") == false);
ensure(!!val::global("a") == true);
ensure(!!val::global("b") == false);
test("template<typename... Args> val new_(Args&&... args)");
EM_ASM(
globalThis.A = function() {
this.value = 2;
}
);
val::global().set("a", val::global("A").new_());
ensure_js("a instanceof A");
ensure_js("a.value == 2");
EM_ASM(
globalThis.A = function(arg1, arg2) {
this.arg1 = arg1;
this.arg2 = arg2;
}
);
val::global().set("a", val::global("A").new_(val(2), val("b")));
ensure_js("a instanceof A");
ensure_js("a.arg1 == 2");
ensure_js("a.arg2 == 'b'");
test("template<typename T> val operator[](const T& key)");
EM_ASM(
a = 2;
);
ensure(val::global()["a"].as<int>() == 2);
ensure_not(val::global()["a"].as<int>() == 3);
val k("a");
ensure(val::global()[k].as<int>() == 2);
ensure_not(val::global()[k].as<int>() == 3);
test("template<typename K, typename V> void set(const K& key, const V& value)");
val::global().set("a", val(2));
ensure_js("a == 2");
val::global().set("a", val(3));
ensure_js("a == 3");
val::global().set("a", 0);
ensure_js("a == 0");
val::global().set("a", false);
ensure_js("a == false");
val::global().set("a", 2);
ensure_js("a == 2");
val::global().set("a", "b");
ensure_js("a == 'b'");
val::global().set(k, 1);
ensure_js("a == 1");
val v(3);
val::global().set(k, v);
ensure("a == 3");
ensure(val::global()[k].as<int>() == 3);
test("template<typename... Args> val operator()(Args&&... args)");
EM_ASM(
f = function() {
return 2;
};
);
ensure(val::global("f")().as<int>() == 2);
ensure_not(val::global("f")().as<int>() == 3);
EM_ASM(
globalThis.f1 = function(arg1, arg2) {
return arg1;
};
globalThis.f2 = function(arg1, arg2) {
return arg2;
};
);
ensure(val::global("f1")(val(2),val("3")).as<int>() == 2);
ensure(val::global("f2")(val(2),val("3")).as<string>() == "3");
test("template<typename ReturnValue, typename... Args> ReturnValue call(const char* name, Args&&... args)");
EM_ASM(
globalThis.C = function() {
this.method = function() { return this; };
};
c = new C;
);
ensure(val::global("c").call<val>("method") == val::global("c"));
EM_ASM(
globalThis.C = function() {
this.method = function(arg) { return arg; };
};
c = new C;
);
ensure(val::global("c").call<int>("method", val(2)) == 2);
test("template<typename T, typename ...Policies> T as(Policies...)");
EM_ASM(
a = 1;
b = 'b';
);
ensure(val::global("a").as<int>() == (int)1);
ensure(val::global("a").as<double>() == (double)1.0);
ensure_not(val::global("a").as<double>() == (double)1.1);
ensure(val::global("b").as<string>() == "b");
#if __STRICT_ANSI__
EM_ASM(
a = undefined;
b = null;
c = true;
d = 2;
e = '2';
f = Symbol();
g = function () {};
h = {};
);
if (
val::global("a").typeOf().as<string>() != "undefined" ||
val::global("a").typeOf().as<string>() == "" ||
val::global("b").typeOf().as<string>() != "object" ||
val::global("c").typeOf().as<string>() != "boolean" ||
val::global("d").typeOf().as<string>() != "number" ||
val::global("e").typeOf().as<string>() != "string" ||
val::global("f").typeOf().as<string>() != "symbol" ||
val::global("g").typeOf().as<string>() != "function" ||
val::global("h").typeOf().as<string>() != "object"
) {
printf("test:\nval typeof()\nfail\n");
assert(false);
}
#endif
test("val typeOf()");
EM_ASM(
a = undefined;
b = null;
c = true;
d = 2;
e = '2';
f = Symbol();
g = function () {};
h = {};
);
ensure(val::global("a").typeOf().as<string>() == "undefined");
ensure_not(val::global("a").typeOf().as<string>() == "");
ensure(val::global("b").typeOf().as<string>() == "object");
ensure(val::global("c").typeOf().as<string>() == "boolean");
ensure(val::global("d").typeOf().as<string>() == "number");
ensure(val::global("e").typeOf().as<string>() == "string");
ensure(val::global("f").typeOf().as<string>() == "symbol");
ensure(val::global("g").typeOf().as<string>() == "function");
ensure(val::global("h").typeOf().as<string>() == "object");
ensure(val::undefined().typeOf().as<string>() == "undefined");
ensure(val::null().typeOf().as<string>() == "object");
ensure(val(true).typeOf().as<string>() == "boolean");
ensure(val(2).typeOf().as<string>() == "number");
ensure(val("2").typeOf().as<string>() == "string");
ensure(val::global().call<val>("Symbol").typeOf().as<string>() == "symbol");
ensure(val::object().typeOf().as<string>() == "object");
test("bool instanceof(const val& v)");
EM_ASM(
globalThis.A = function() {};
globalThis.B = function() {};
a = new A;
);
ensure(val::global("a").instanceof(val::global("A")));
ensure_not(val::global("a").instanceof(val::global("B")));
test("bool in(const val& v)");
EM_ASM(
a = {};
a.b = 'bb';
a.c = 'cc';
);
ensure(val("c").in(val::global("a")));
ensure_not(val("d").in(val::global("a")));
test("template<typename T> bool delete_(const T& property)");
EM_ASM(
a = {};
a.b = undefined;
a[0] = null;
a[1] = 2;
a.c = 'c';
);
ensure_js("'b' in a");
ensure_js("0 in a");
ensure_js("1 in a");
ensure_js("'c' in a");
ensure(val::global("a").delete_("b") == true);
ensure_js_not("'b' in a");
ensure_js("0 in a");
ensure_js("1 in a");
ensure_js("'c' in a");
ensure(val::global("a").delete_(0) == true);
ensure(val::global("a").delete_(val(1)) == true);
ensure(val::global("a").delete_(val("c")) == true);
ensure_js_not("'b' in a");
ensure_js_not("0 in a");
ensure_js_not("1 in a");
ensure_js_not("'c' in a");
test("void throw_() const");
EM_ASM(
globalThis.test_val_throw_ = function(error) {
try {
Module.throw_js_error(error);
return false;
} catch(error_thrown) {
if (error_thrown != error)
throw error_thrown;
}
return true;
}
);
ensure_js("test_val_throw_(new Error)");
ensure_js("test_val_throw_(Error)");
ensure_js("test_val_throw_(2)");
ensure_js("test_val_throw_('message')");
ensure_js("test_val_throw_(new TypeError('message'))");
test("template<typename T> std::vector<T> vecFromJSArray(const val& v)");
EM_ASM(
a = [];
a[0] = 1;
a[1] = '42';
a[2] = 'b';
a[3] = new Date(100000);
);
const std::vector<val>& aAsArray = vecFromJSArray<val>(val::global("a"));
ensure(aAsArray.at(0).as<int>() == 1);
ensure(aAsArray.at(1).as<string>() == "42");
ensure(aAsArray.at(2).as<string>() == "b");
ensure(aAsArray.size() == 4);
test("template<typename T> std::vector<T *> vecFromJSArray(const val& v)");
EM_ASM(
b = [];
b[0] = Module.makeDummy();
b[1] = Module.makeDummy();
);
const std::vector<Dummy *>& bAsArray = vecFromJSArray<Dummy *>(val::global("b"), allow_raw_pointers());
ensure(bAsArray.size() == 2);
for (auto *dummy : bAsArray) {
delete dummy;
}
test("template<typename T> std::vector<T> convertJSArrayToNumberVector(const val& v)");
const std::vector<float>& aAsNumberVectorFloat = convertJSArrayToNumberVector<float>(val::global("a"));
ensure(aAsNumberVectorFloat.size() == 4);
ensure(aAsNumberVectorFloat.at(0) == 1.f);
ensure(aAsNumberVectorFloat.at(1) == 42.f);
ensure(std::isnan(aAsNumberVectorFloat.at(2)));
ensure(aAsNumberVectorFloat.at(3) == 100000.f);
const std::vector<uint32_t>& aAsNumberVectorUint32_t = convertJSArrayToNumberVector<uint32_t>(val::global("a"));
ensure(aAsNumberVectorUint32_t.size() == 4);
ensure(aAsNumberVectorUint32_t.at(0) == 1);
ensure(aAsNumberVectorUint32_t.at(1) == 42);
ensure(aAsNumberVectorUint32_t.at(2) == 0);
ensure(aAsNumberVectorUint32_t.at(3) == 100000);
test("val u8string(const char* s)");
val::global().set("a", val::u8string(u8"abc"));
ensure_js("a == 'abc'");
val::global().set("a", val::u8string(u8"你好"));
ensure_js_not("a == 'abc'");
ensure_js("a == '你好'");
auto u8_str = val::global()["a"].as<std::string>();
ensure(u8_str == u8"你好");
test("val u16string(const char16_t* s)");
val::global().set("a", val::u16string(u"hello"));
ensure_js("a == 'hello'");
val::global().set("a", val::u16string(u"世界"));
ensure_js_not("a == 'hello'");
ensure_js("a == '世界'");
const char16_t* s = u"😃 = \U0001F603 is :-D";
val::global().set("a", val::u16string(s));
ensure_js("a == '😃 = \U0001F603 is :-D'");
test("val set() with policy");
Dummy *dummy = new Dummy();
val::global().set("a", dummy, allow_raw_pointers());
ensure_js("a instanceof Module.Dummy");
val::global().set("a", val::null());
delete dummy;
printf("end\n");
return 0;
}