mirror of
				https://github.com/zulip/zulip.git
				synced 2025-10-31 03:53:50 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			114 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			114 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import ErrorStackParser from "error-stack-parser";
 | |
| import $ from "jquery";
 | |
| import StackFrame from "stackframe";
 | |
| import StackTraceGPS from "stacktrace-gps";
 | |
| 
 | |
| import render_blueslip_stacktrace from "../templates/blueslip_stacktrace.hbs";
 | |
| 
 | |
| type FunctionName = {
 | |
|     scope: string;
 | |
|     name: string;
 | |
| };
 | |
| 
 | |
| type NumberedLine = {
 | |
|     line_number: number;
 | |
|     line: string;
 | |
|     focus: boolean;
 | |
| };
 | |
| 
 | |
| type CleanStackFrame = {
 | |
|     full_path?: string;
 | |
|     show_path?: string;
 | |
|     function_name?: FunctionName;
 | |
|     line_number?: number;
 | |
|     context?: NumberedLine[];
 | |
| };
 | |
| 
 | |
| export function clean_path(full_path?: string): string | undefined {
 | |
|     // If the file is local, just show the filename.
 | |
|     // Otherwise, show the full path starting from node_modules.
 | |
|     if (full_path === undefined) {
 | |
|         return undefined;
 | |
|     }
 | |
|     const idx = full_path.indexOf("/node_modules/");
 | |
|     if (idx !== -1) {
 | |
|         return full_path.slice(idx + "/node_modules/".length);
 | |
|     }
 | |
|     if (full_path.startsWith("webpack://")) {
 | |
|         return full_path.slice("webpack://".length);
 | |
|     }
 | |
|     return full_path;
 | |
| }
 | |
| 
 | |
| export function clean_function_name(
 | |
|     function_name: string | undefined,
 | |
| ): {scope: string; name: string} | undefined {
 | |
|     if (function_name === undefined) {
 | |
|         return undefined;
 | |
|     }
 | |
|     const idx = function_name.lastIndexOf(".");
 | |
|     return {
 | |
|         scope: function_name.slice(0, idx + 1),
 | |
|         name: function_name.slice(idx + 1),
 | |
|     };
 | |
| }
 | |
| 
 | |
| const sourceCache: {[source: string]: string | Promise<string>} = {};
 | |
| 
 | |
| const stack_trace_gps = new StackTraceGPS({sourceCache});
 | |
| 
 | |
| async function get_context(location: StackFrame): Promise<NumberedLine[] | undefined> {
 | |
|     const {fileName, lineNumber} = location;
 | |
|     if (fileName === undefined || lineNumber === undefined) {
 | |
|         return undefined;
 | |
|     }
 | |
|     let sourceContent;
 | |
|     try {
 | |
|         sourceContent = await sourceCache[fileName];
 | |
|     } catch {
 | |
|         return undefined;
 | |
|     }
 | |
|     if (sourceContent === undefined) {
 | |
|         return undefined;
 | |
|     }
 | |
|     const lines = sourceContent.split("\n");
 | |
|     const lo_line_num = Math.max(lineNumber - 5, 0);
 | |
|     const hi_line_num = Math.min(lineNumber + 4, lines.length);
 | |
|     return lines.slice(lo_line_num, hi_line_num).map((line: string, i: number) => ({
 | |
|         line_number: lo_line_num + i + 1,
 | |
|         line,
 | |
|         focus: lo_line_num + i + 1 === lineNumber,
 | |
|     }));
 | |
| }
 | |
| 
 | |
| export async function display_stacktrace(error: string, stack: string): Promise<void> {
 | |
|     const ex = new Error();
 | |
|     ex.stack = stack;
 | |
| 
 | |
|     const stackframes: CleanStackFrame[] = await Promise.all(
 | |
|         ErrorStackParser.parse(ex).map(async (stack_frame: ErrorStackParser.StackFrame) => {
 | |
|             // Work around mistake in ErrorStackParser.StackFrame definition
 | |
|             // https://github.com/stacktracejs/error-stack-parser/pull/54
 | |
|             let location = (stack_frame as unknown) as StackFrame;
 | |
|             try {
 | |
|                 location = await stack_trace_gps.getMappedLocation(location);
 | |
|             } catch {
 | |
|                 // Use unmapped location
 | |
|             }
 | |
|             return {
 | |
|                 full_path: location.getFileName(),
 | |
|                 show_path: clean_path(location.getFileName()),
 | |
|                 line_number: location.getLineNumber(),
 | |
|                 function_name: clean_function_name(location.getFunctionName()),
 | |
|                 context: await get_context(location),
 | |
|             };
 | |
|         }),
 | |
|     );
 | |
| 
 | |
|     const $alert = $("<div class='stacktrace'>").html(
 | |
|         render_blueslip_stacktrace({error, stackframes}),
 | |
|     );
 | |
|     $(".alert-box").append($alert);
 | |
|     $alert.addClass("show");
 | |
| }
 |