LNXSDK/Kha/Tools/macos/std/js/_std/haxe/NativeStackTrace.hx
2025-01-22 16:18:30 +01:00

150 lines
4.2 KiB
Haxe

package haxe;
import js.Syntax;
import js.lib.Error;
import haxe.CallStack.StackItem;
// https://v8.dev/docs/stack-trace-api
@:native("Error")
private extern class V8Error {
static var prepareStackTrace:(error:Error, structuredStackTrace:Array<V8CallSite>)->Any;
}
typedef V8CallSite = {
function getFunctionName():String;
function getFileName():String;
function getLineNumber():Int;
function getColumnNumber():Int;
}
/**
Do not use manually.
**/
@:dox(hide)
@:noCompletion
@:allow(haxe.Exception)
class NativeStackTrace {
static var lastError:Error;
// support for source-map-support module
@:noCompletion
public static var wrapCallSite:V8CallSite->V8CallSite;
@:ifFeature('haxe.NativeStackTrace.exceptionStack')
static public inline function saveStack(e:Error):Void {
lastError = e;
}
static public function callStack():Any {
var e:Null<Error> = new Error('');
var stack = tryHaxeStack(e);
//Internet Explorer provides call stack only if error was thrown
if(Syntax.typeof(stack) == "undefined") {
try throw e catch(e:Exception) {}
stack = e.stack;
}
return normalize(stack, 2);
}
static public function exceptionStack():Any {
return normalize(tryHaxeStack(lastError));
}
static public function toHaxe(s:Null<Any>, skip:Int = 0):Array<StackItem> {
if (s == null) {
return [];
} else if (Syntax.typeof(s) == "string") {
// Return the raw lines in browsers that don't support prepareStackTrace
var stack:Array<String> = (s:String).split("\n");
if (stack[0] == "Error")
stack.shift();
var m = [];
for (i in 0...stack.length) {
if(skip > i) continue;
var line = stack[i];
var matched:Null<Array<String>> = Syntax.code('{0}.match(/^ at ([A-Za-z0-9_. ]+) \\(([^)]+):([0-9]+):([0-9]+)\\)$/)', line);
if (matched != null) {
var path = matched[1].split(".");
if(path[0] == "$hxClasses") {
path.shift();
}
var meth = path.pop();
var file = matched[2];
var line = Std.parseInt(matched[3]);
var column = Std.parseInt(matched[4]);
m.push(FilePos(meth == "Anonymous function" ? LocalFunction() : meth == "Global code" ? null : Method(path.join("."), meth), file, line,
column));
} else {
m.push(Module(StringTools.trim(line))); // A little weird, but better than nothing
}
}
return m;
} else if(skip > 0 && Syntax.code('Array.isArray({0})', s)) {
return (s:Array<StackItem>).slice(skip);
} else {
return cast s;
}
}
static function tryHaxeStack(e:Null<Error>):Any {
if (e == null) {
return [];
}
// https://v8.dev/docs/stack-trace-api
var oldValue = V8Error.prepareStackTrace;
V8Error.prepareStackTrace = prepareHxStackTrace;
var stack = e.stack;
V8Error.prepareStackTrace = oldValue;
return stack;
}
static function prepareHxStackTrace(e:Error, callsites:Array<V8CallSite>):Any {
var stack = [];
for (site in callsites) {
if (wrapCallSite != null)
site = wrapCallSite(site);
var method = null;
var fullName = site.getFunctionName();
if (fullName != null) {
var idx = fullName.lastIndexOf(".");
if (idx >= 0) {
var className = fullName.substring(0, idx);
var methodName = fullName.substring(idx + 1);
method = Method(className, methodName);
} else {
method = Method(null, fullName);
}
}
var fileName = site.getFileName();
var fileAddr = fileName == null ? -1 : fileName.indexOf("file:");
if (wrapCallSite != null && fileAddr > 0)
fileName = fileName.substring(fileAddr + 6);
stack.push(FilePos(method, fileName, site.getLineNumber(), site.getColumnNumber()));
}
return stack;
}
static function normalize(stack:Any, skipItems:Int = 0):Any {
if(Syntax.code('Array.isArray({0})', stack) && skipItems > 0) {
return (stack:Array<StackItem>).slice(skipItems);
} else if(Syntax.typeof(stack) == "string") {
switch (stack:String).substring(0, 6) {
case 'Error:' | 'Error\n': skipItems += 1;
case _:
}
return skipLines(stack, skipItems);
} else {
//nothing we can do
return stack;
}
}
static function skipLines(stack:String, skip:Int, pos:Int = 0):String {
return if(skip > 0) {
pos = stack.indexOf('\n', pos);
return pos < 0 ? '' : skipLines(stack, --skip, pos + 1);
} else {
return stack.substring(pos);
}
}
}