Merge branch 'master' into ES6
This commit is contained in:
BIN
src/main/resources/debugger/favicon.png
Normal file
BIN
src/main/resources/debugger/favicon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.7 KiB |
30
src/main/resources/debugger/index.html
Normal file
30
src/main/resources/debugger/index.html
Normal file
@@ -0,0 +1,30 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>JScript Debugger</title>
|
||||
</head>
|
||||
<body>
|
||||
<p>
|
||||
This is the debugger of JScript. It implement the <a href="https://chromedevtools.github.io/devtools-protocol/1-2/">V8 Debugging protocol</a>,
|
||||
so you can use the devtools in chrome. <br>
|
||||
The debugger is still in early development, so please report any issues to
|
||||
<a href="https://github.com/TopchetoEU/java-jscript/issues">the github repo</a>.
|
||||
</p>
|
||||
|
||||
<p>
|
||||
Here are the available entrypoints:
|
||||
<ul>
|
||||
<li><a href="json/version">/json/version</a> - version and other stuff about the JScript engine</li>
|
||||
<li><a href="json/list">/json/list</a> - a list of all entrypoints</li>
|
||||
<li><a href="json/protocol">/json/protocol</a> - documentation of the implemented V8 protocol</li>
|
||||
<li>/(any target) - websocket entrypoints for debugging</li>
|
||||
</ul>
|
||||
</p>
|
||||
|
||||
<p>
|
||||
Running ${NAME} v${VERSION} by ${AUTHOR}
|
||||
</p>
|
||||
</body>
|
||||
</html>
|
||||
1004
src/main/resources/debugger/protocol.json
Normal file
1004
src/main/resources/debugger/protocol.json
Normal file
File diff suppressed because it is too large
Load Diff
313
src/main/resources/lib/index.js
Normal file
313
src/main/resources/lib/index.js
Normal file
@@ -0,0 +1,313 @@
|
||||
const target = arguments[0];
|
||||
const primordials = arguments[1];
|
||||
|
||||
const makeSymbol = primordials.symbol.makeSymbol;
|
||||
const getSymbol = primordials.symbol.getSymbol;
|
||||
const getSymbolKey = primordials.symbol.getSymbolKey;
|
||||
const getSymbolDescription = primordials.symbol.getSymbolDescription;
|
||||
|
||||
const parseInt = primordials.number.parseInt;
|
||||
const parseFloat = primordials.number.parseFloat;
|
||||
const isNaN = primordials.number.isNaN;
|
||||
const NaN = primordials.number.NaN;
|
||||
const Infinity = primordials.number.Infinity;
|
||||
|
||||
const fromCharCode = primordials.string.fromCharCode;
|
||||
const fromCodePoint = primordials.string.fromCodePoint;
|
||||
const stringBuild = primordials.string.stringBuild;
|
||||
|
||||
const defineProperty = primordials.object.defineProperty;
|
||||
const defineField = primordials.object.defineField;
|
||||
const getOwnMember = primordials.object.getMember;
|
||||
const getOwnSymbolMember = primordials.object.getOwnSymbolMember;
|
||||
const getOwnMembers = primordials.object.getOwnMembers;
|
||||
const getOwnSymbolMembers = primordials.object.getOwnSymbolMembers;
|
||||
const getPrototype = primordials.object.getPrototype;
|
||||
const setPrototype = primordials.object.setPrototype;
|
||||
|
||||
const invokeType = primordials.function.invokeType;
|
||||
const setConstructable = primordials.function.setConstructable;
|
||||
const setCallable = primordials.function.setCallable;
|
||||
const invoke = primordials.function.invoke;
|
||||
|
||||
const setGlobalPrototype = primordials.setGlobalPrototype;
|
||||
const compile = primordials.compile;
|
||||
|
||||
const json = primordials.json;
|
||||
|
||||
const valueKey = makeSymbol("Primitive.value");
|
||||
const undefined = ({}).definitelyDefined;
|
||||
|
||||
target.undefined = undefined;
|
||||
|
||||
const unwrapThis = (self, type, constr, name, arg, defaultVal) => {
|
||||
if (arg == null) arg = "this";
|
||||
if (typeof self === type) return self;
|
||||
if (self instanceof constr && valueKey in self) self = self[valueKey];
|
||||
if (typeof self === type) return self;
|
||||
if (arguments.length > 5) return defaultVal;
|
||||
|
||||
throw new TypeError(name + " requires that '" + arg + "' be a " + constr.name);
|
||||
}
|
||||
|
||||
const wrapIndex = (i, len) => {};
|
||||
|
||||
const Symbol = (name = "") => makeSymbol(name);
|
||||
|
||||
defineField(Symbol, "for", true, false, true, function(name) {
|
||||
return getSymbol(name + "");
|
||||
});
|
||||
defineField(Symbol, "keyFor", true, false, true, function(symbol) {
|
||||
return getSymbolKey(unwrapThis(symbol, "symbol", Symbol, "Symbol.keyFor"));
|
||||
});
|
||||
|
||||
defineField(Symbol, "asyncIterator", false, false, false, Symbol("Symbol.asyncIterator"));
|
||||
defineField(Symbol, "iterator", false, false, false, Symbol("Symbol.iterator"));
|
||||
defineField(Symbol, "match", false, false, false, Symbol("Symbol.match"));
|
||||
defineField(Symbol, "matchAll", false, false, false, Symbol("Symbol.matchAll"));
|
||||
defineField(Symbol, "replace", false, false, false, Symbol("Symbol.replace"));
|
||||
defineField(Symbol, "search", false, false, false, Symbol("Symbol.search"));
|
||||
defineField(Symbol, "split", false, false, false, Symbol("Symbol.split"));
|
||||
defineField(Symbol, "toStringTag", false, false, false, Symbol("Symbol.toStringTag"));
|
||||
defineField(Symbol, "prototype", false, false, false, {});
|
||||
|
||||
defineProperty(Symbol.prototype, "description", false, true, function () {
|
||||
return getSymbolDescription(unwrapThis(this, "symbol", Symbol, "Symbol.prototype.description"));
|
||||
}, undefined);
|
||||
defineField(Symbol.prototype, "toString", true, false, true, function() {
|
||||
return "Symbol(" + unwrapThis(this, "symbol", Symbol, "Symbol.prototype.toString").description + ")";
|
||||
});
|
||||
defineField(Symbol.prototype, "valueOf", true, false, true, function() {
|
||||
return unwrapThis(this, "symbol", Symbol, "Symbol.prototype.valueOf");
|
||||
});
|
||||
|
||||
target.Symbol = Symbol;
|
||||
|
||||
const Number = function(value) {
|
||||
if (invokeType(arguments) === "call") {
|
||||
if (arguments.length === 0) return 0;
|
||||
else return +value;
|
||||
}
|
||||
|
||||
this[valueKey] = target.Number(value);
|
||||
};
|
||||
|
||||
defineField(Number, "isFinite", true, false, true, function(value) {
|
||||
value = unwrapThis(value, "number", Number, "Number.isFinite", "value", undefined);
|
||||
|
||||
if (value === undefined || isNaN(value)) return false;
|
||||
if (value === Infinity || value === -Infinity) return false;
|
||||
|
||||
return true;
|
||||
});
|
||||
defineField(Number, "isInteger", true, false, true, function(value) {
|
||||
value = unwrapThis(value, "number", Number, "Number.isInteger", "value", undefined);
|
||||
if (value === undefined) return false;
|
||||
return parseInt(value) === value;
|
||||
});
|
||||
defineField(Number, "isNaN", true, false, true, function(value) {
|
||||
return isNaN(value);
|
||||
});
|
||||
defineField(Number, "isSafeInteger", true, false, true, function(value) {
|
||||
value = unwrapThis(value, "number", Number, "Number.isSafeInteger", "value", undefined);
|
||||
if (value === undefined || parseInt(value) !== value) return false;
|
||||
return value >= -9007199254740991 && value <= 9007199254740991;
|
||||
});
|
||||
defineField(Number, "parseFloat", true, false, true, function(value) {
|
||||
value = 0 + value;
|
||||
return parseFloat(value);
|
||||
});
|
||||
defineField(Number, "parseInt", true, false, true, function(value, radix) {
|
||||
value = 0 + value;
|
||||
radix = +radix;
|
||||
if (isNaN(radix)) radix = 10;
|
||||
|
||||
return parseInt(value, radix);
|
||||
});
|
||||
|
||||
defineField(Number, "EPSILON", false, false, false, 2.220446049250313e-16);
|
||||
defineField(Number, "MIN_SAFE_INTEGER", false, false, false, -9007199254740991);
|
||||
defineField(Number, "MAX_SAFE_INTEGER", false, false, false, 9007199254740991);
|
||||
defineField(Number, "POSITIVE_INFINITY", false, false, false, +Infinity);
|
||||
defineField(Number, "NEGATIVE_INFINITY", false, false, false, -Infinity);
|
||||
defineField(Number, "NaN", false, false, false, NaN);
|
||||
defineField(Number, "MAX_VALUE", false, false, false, 1.7976931348623157e+308);
|
||||
defineField(Number, "MIN_VALUE", false, false, false, 5e-324);
|
||||
defineField(Number, "prototype", false, false, false, {});
|
||||
|
||||
defineField(Number.prototype, "toString", true, false, true);
|
||||
defineField(Number.prototype, "toString", true, false, true, function() {
|
||||
return "" + unwrapThis(this, "number", Number, "Number.prototype.toString");
|
||||
});
|
||||
defineField(Number.prototype, "valueOf", true, false, true, function() {
|
||||
return unwrapThis(this, "number", Number, "Number.prototype.toString");
|
||||
});
|
||||
|
||||
target.Number = Number;
|
||||
|
||||
const String = function(value) {
|
||||
if (invokeType(arguments) === "call") {
|
||||
if (arguments.length === 0) return "";
|
||||
else return value + "";
|
||||
}
|
||||
|
||||
this[valueKey] = String(value);
|
||||
};
|
||||
|
||||
defineField(String, "fromCharCode", true, false, true, function() {
|
||||
const res = [];
|
||||
res[arguments.length] = 0;
|
||||
|
||||
for (let i = 0; i < arguments.length; i++) {
|
||||
res[i] = fromCharCode(+arguments[i]);
|
||||
}
|
||||
|
||||
return stringBuild(res);
|
||||
});
|
||||
defineField(String, "fromCodePoint", true, false, true, function() {
|
||||
const res = [];
|
||||
res[arguments.length] = 0;
|
||||
|
||||
for (let i = 0; i < arguments.length; i++) {
|
||||
res[i] = fromCodePoint(+arguments[i]);
|
||||
}
|
||||
|
||||
return stringBuild(res);
|
||||
});
|
||||
|
||||
defineField(String, "prototype", false, false, false, {});
|
||||
|
||||
defineField(String.prototype, "at", true, false, true, function(index) {
|
||||
throw "Not implemented :/";
|
||||
return unwrapThis(this, "string", String, "String.prototype.at")[index];
|
||||
});
|
||||
defineField(String.prototype, "toString", true, false, true, function() {
|
||||
return unwrapThis(this, "string", String, "String.prototype.toString");
|
||||
});
|
||||
defineField(String.prototype, "valueOf", true, false, true, function() {
|
||||
return unwrapThis(this, "string", String, "String.prototype.valueOf");
|
||||
});
|
||||
|
||||
target.String = String;
|
||||
|
||||
const Boolean = function(value) {
|
||||
if (invokeType(arguments) === "call") {
|
||||
if (arguments.length === 0) return false;
|
||||
else return !!value;
|
||||
}
|
||||
|
||||
this[valueKey] = Boolean(value);
|
||||
};
|
||||
|
||||
defineField(Boolean, "prototype", false, false, false, {});
|
||||
|
||||
defineField(Boolean.prototype, "toString", true, false, true, function() {
|
||||
return "" + unwrapThis(this, "boolean", Boolean, "Boolean.prototype.toString");
|
||||
});
|
||||
defineField(Boolean.prototype, "valueOf", true, false, true, function() {
|
||||
return unwrapThis(this, "boolean", Boolean, "Boolean.prototype.valueOf");
|
||||
});
|
||||
|
||||
target.Boolean = Boolean;
|
||||
|
||||
const Object = function(value) {
|
||||
if (typeof value === 'object' && value !== null) return value;
|
||||
|
||||
if (typeof value === 'string') return new String(value);
|
||||
if (typeof value === 'number') return new Number(value);
|
||||
if (typeof value === 'boolean') return new Boolean(value);
|
||||
if (typeof value === 'symbol') {
|
||||
const res = {};
|
||||
setPrototype(res, Symbol.prototype);
|
||||
res[valueKey] = value;
|
||||
return res;
|
||||
}
|
||||
|
||||
const target = this;
|
||||
if (target == null || typeof target !== 'object') target = {};
|
||||
|
||||
this[valueKey] = Object(value);
|
||||
};
|
||||
|
||||
defineField(Object, "prototype", false, false, false, setPrototype({}, null));
|
||||
|
||||
defineField(Object.prototype, "toString", true, false, true, function() {
|
||||
if (this !== null && this !== undefined && (Symbol.toStringTag in this)) return "[object " + this[Symbol.toStringTag] + "]";
|
||||
else if (typeof this === "number" || this instanceof Number) return "[object Number]";
|
||||
else if (typeof this === "symbol" || this instanceof Symbol) return "[object Symbol]";
|
||||
else if (typeof this === "string" || this instanceof String) return "[object String]";
|
||||
else if (typeof this === "boolean" || this instanceof Boolean) return "[object Boolean]";
|
||||
else if (typeof this === "function") return "[object Function]";
|
||||
else return "[object Object]";
|
||||
});
|
||||
defineField(Object.prototype, "valueOf", true, false, true, function() {
|
||||
return this;
|
||||
});
|
||||
|
||||
target.Boolean = Boolean;
|
||||
|
||||
const Function = function() {
|
||||
const parts = ["return function annonymous("];
|
||||
|
||||
for (let i = 0; i < arguments.length - 1; i++) {
|
||||
if (i > 0) parts[parts.length] = ",";
|
||||
parts[parts.length] = arguments[i];
|
||||
}
|
||||
parts[parts.length] = "){\n";
|
||||
parts[parts.length] = String(arguments[arguments.length - 1]);
|
||||
parts[parts.length] = "\n}";
|
||||
|
||||
print(parts);
|
||||
|
||||
const res = compile(stringBuild(parts))();
|
||||
return res;
|
||||
};
|
||||
|
||||
defineField(Function, "compile", true, false, true, (src = "", options = {}) => {
|
||||
if (options.globals == null) options.globals = [];
|
||||
if (options.wrap == null) options.wrap = true;
|
||||
|
||||
const parts = [];
|
||||
|
||||
if (options.wrap) parts[parts.length] = "return (function() {\n";
|
||||
if (options.globals.length > 0) {
|
||||
parts[parts.length] = "var ";
|
||||
|
||||
for (let i = 0; i < options.globals.length; i++) {
|
||||
if (i > 0) parts[parts.length] = ",";
|
||||
parts[parts.length] = options.globals[i];
|
||||
}
|
||||
|
||||
parts[parts.length] = ";((g=arguments[0])=>{";
|
||||
|
||||
for (let i = 0; i < options.globals.length; i++) {
|
||||
const name = options.globals[i];
|
||||
parts[parts.length] = name + "=g[" + json.stringify(name) + "];";
|
||||
}
|
||||
|
||||
parts[parts.length] = "})()\n";
|
||||
}
|
||||
|
||||
parts[parts.length] = src;
|
||||
if (options.wrap) parts[parts.length] = "\n})(arguments[0])";
|
||||
|
||||
const res = compile(stringBuild(parts));
|
||||
return res;
|
||||
});
|
||||
defineField(Function, "prototype", false, false, false, setPrototype({}, null));
|
||||
|
||||
defineField(Function.prototype, "toString", true, false, true, function() {
|
||||
if (this.name !== "") return "function " + this.name + "(...) { ... }";
|
||||
else return "function (...) { ... }";
|
||||
});
|
||||
defineField(Function.prototype, "valueOf", true, false, true, function() {
|
||||
return this;
|
||||
});
|
||||
|
||||
target.Function = Function;
|
||||
|
||||
setGlobalPrototype("string", String.prototype);
|
||||
setGlobalPrototype("number", Number.prototype);
|
||||
setGlobalPrototype("boolean", Boolean.prototype);
|
||||
setGlobalPrototype("symbol", Symbol.prototype);
|
||||
setGlobalPrototype("object", Object.prototype);
|
||||
5
src/main/resources/metadata.json
Normal file
5
src/main/resources/metadata.json
Normal file
@@ -0,0 +1,5 @@
|
||||
{
|
||||
"version": "${version}",
|
||||
"name": "${name}",
|
||||
"author": "TopchetoEU"
|
||||
}
|
||||
Reference in New Issue
Block a user