blob: c04c95127f373ec841b0e45954c24adc137bc484 [file] [log] [blame]
import 'dart:math' show max;
import 'package:stack_trace/stack_trace.dart';
import 'package:dart2js_tools/src/trace.dart';
import 'package:dart2js_tools/src/name_decoder.dart';
import 'package:dart2js_tools/src/util.dart';
import 'package:dart2js_tools/src/trace_decoder.dart';
/// Deobfuscates the given [obfuscatedTrace].
///
/// This method assumes a stack trace contains URIs normalized to be file URIs.
/// If for example you obtain a stack from a browser, you may need to preprocess
/// the stack to map the URI of the JavaScript files to URIs in the local file
/// system.
///
/// For example, a valid input to this method would be:
///
/// Error: no such method
/// at aB.a20 (/usr/local/foo/main.dart.js:71969:32)
/// at aNk.goV (/usr/local/foo/main.dart.js:72040:52)
/// at aNk.gfK (/usr/local/foo/main.dart.js:72038:27)
/// at FE.gtn (/usr/local/foo/main.dart.js:72640:24)
/// at aBZ.ghN (/usr/local/foo/main.dart.js:72642:24)
/// at inheritance (/usr/local/foo/main.dart.js:105334:0)
/// at FE (/usr/local/foo/main.dart.js:5037:18)
///
/// Internally this method will read those JavaScript files, search for the
/// `//# sourceMappingURL=` line at the end, and load the corresponding
/// source-map file.
String deobfuscateStackTrace(String obfuscatedTrace) {
String? error = extractErrorMessage(obfuscatedTrace);
var provider = CachingFileProvider();
StackDeobfuscationResult result = deobfuscateStack(obfuscatedTrace, provider);
Frame firstFrame = result.original.frames.first;
String translatedError = (firstFrame.uri.isScheme('error')
? null
: translate(error, provider.mappingFor(firstFrame.uri))) ??
'<no error message found>';
var sb = StringBuffer();
sb.writeln(translatedError);
maxMemberLengthHelper(int m, Frame f) => max(f.member!.length, m);
int longest = result.deobfuscated.frames.fold(0, maxMemberLengthHelper);
longest = result.original.frames.fold(longest, maxMemberLengthHelper);
for (var originalFrame in result.original.frames) {
var deobfuscatedFrames = result.frameMap[originalFrame];
if (deobfuscatedFrames == null) {
var name = originalFrame.member;
sb.writeln(' at ${name!.padRight(longest)} ${originalFrame.location}');
} else {
for (var frame in deobfuscatedFrames) {
var name = frame.member;
// TODO(sigmund): eventually when ddc stops shipping source-maps to the
// client, we can start encoding the function name and remove this
// workaround.
if (name == '<unknown>') name = originalFrame.member;
sb.writeln(' at ${name!.padRight(longest)} ${frame.location}');
}
}
}
return '$sb';
}