Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add local source context via devserver #741

Merged
merged 3 commits into from
Jan 16, 2020
Merged
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
93 changes: 71 additions & 22 deletions src/js/integrations/debugsymbolicator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,7 @@ import { Event, EventHint, Integration, StackFrame } from "@sentry/types";
import { logger } from "@sentry/utils";

const INTERNAL_CALLSITES_REGEX = new RegExp(
[
"/Libraries/Renderer/oss/ReactNativeRenderer-dev\\.js$",
"/Libraries/BatchedBridge/MessageQueue\\.js$"
].join("|")
["ReactNativeRenderer-dev\\.js$", "MessageQueue\\.js$"].join("|")
);

/**
Expand Down Expand Up @@ -70,7 +67,7 @@ export class DebugSymbolicator implements Integration {
await self._symbolicate(event, stack);
}
if (reactError.jsEngine === "hermes") {
const convertedFrames = this._convertReactNativeFramesToSentryFrames(
const convertedFrames = await this._convertReactNativeFramesToSentryFrames(
stack
);
this._replaceFramesInEvent(event, convertedFrames);
Expand Down Expand Up @@ -103,7 +100,7 @@ export class DebugSymbolicator implements Integration {
frame.file && frame.file.match(INTERNAL_CALLSITES_REGEX) === null
);

const symbolicatedFrames = this._convertReactNativeFramesToSentryFrames(
const symbolicatedFrames = await this._convertReactNativeFramesToSentryFrames(
stackWithoutInternalCallsites
);
this._replaceFramesInEvent(event, symbolicatedFrames);
Expand All @@ -121,25 +118,47 @@ export class DebugSymbolicator implements Integration {
* Converts ReactNativeFrames to frames in the Sentry format
* @param frames ReactNativeFrame[]
*/
private _convertReactNativeFramesToSentryFrames(
private async _convertReactNativeFramesToSentryFrames(
frames: ReactNativeFrame[]
): StackFrame[] {
): Promise<StackFrame[]> {
let getDevServer: any;
try {
if (__DEV__) {
getDevServer = require("react-native/Libraries/Core/Devtools/getDevServer");
}
} catch (_oO) {
// We can't load devserver URL
}
// Below you will find lines marked with :HACK to prevent showing errors in the sentry ui
// But since this is a debug only feature: This is Fine (TM)
return frames.map(
(frame: ReactNativeFrame): StackFrame => {
const inApp =
(frame.file && !frame.file.includes("node_modules")) ||
(!!frame.column && !!frame.lineNumber);
return {
colno: frame.column,
filename: frame.file,
function: frame.methodName,
in_app: inApp,
lineno: inApp ? frame.lineNumber : undefined, // :HACK
platform: inApp ? "javascript" : "node" // :HACK
};
}
return Promise.all(
frames.map(
async (frame: ReactNativeFrame): Promise<StackFrame> => {
let inApp = !!frame.column && !!frame.lineNumber;
inApp =
inApp &&
// tslint:disable-next-line: strict-type-predicates
frame.file !== undefined &&
!frame.file.includes("node_modules") &&
!frame.file.includes("native code");

const newFrame: StackFrame = {
colno: frame.column,
filename: frame.file,
function: frame.methodName,
in_app: inApp,
lineno: inApp ? frame.lineNumber : undefined, // :HACK
platform: inApp ? "javascript" : "node" // :HACK
Copy link
Member

@lobsterkatie lobsterkatie Jan 9, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Given that we're now filtering out both node modules and native code, would it be better to make the default undefined? Or to test for frame.file.includes("native code") also, and if it fails both the inApp test and the native code test, then have it default to node?

};

if (inApp && __DEV__) {
// tslint:disable-next-line: no-unsafe-any
await this._addSourceContext(newFrame, getDevServer);
}

return newFrame;
}
)
);
}

Expand All @@ -158,4 +177,34 @@ export class DebugSymbolicator implements Integration {
event.exception.values[0].stacktrace.frames = frames.reverse();
}
}

/**
* This tries to add source context for in_app Frames
*
* @param frame StackFrame
* @param getDevServer function from RN to get DevServer URL
*/
private async _addSourceContext(
frame: StackFrame,
getDevServer?: any
): Promise<void> {
// tslint:disable: no-unsafe-any no-non-null-assertion
const response = await fetch(
`${getDevServer().url}${frame
.filename!.replace(/\/+$/, "")
.replace(/.*\//, "")}`,
{
method: "GET"
}
);

const content = await response.text();
const lines = content.split("\n");

const line = Math.max(frame.lineno! - 1, 0);
frame.pre_context = lines.slice(line - 6, line);
frame.post_context = lines.slice(line, line + 5);
lobsterkatie marked this conversation as resolved.
Show resolved Hide resolved
frame.context_line = lines[line];
// tslint:enable: no-unsafe-any no-non-null-assertion
}
}