forked from LeenkxTeam/LNXSDK
		
	
		
			
				
	
	
		
			150 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Haxe
		
	
	
	
	
	
			
		
		
	
	
			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);
 | |
| 		}
 | |
| 	}
 | |
| } |