Skip to content

Instantly share code, notes, and snippets.

@stympy
Last active February 25, 2025 17:14
Show Gist options
  • Save stympy/34968396b9dd2ba48a32863df5a6eb13 to your computer and use it in GitHub Desktop.
Save stympy/34968396b9dd2ba48a32863df5a6eb13 to your computer and use it in GitHub Desktop.
Override javascript error fingerprint for Honeybadger
import Honeybadger from "@honeybadger-io/js";
const honeybadger = Honeybadger.configure({
apiKey: "your-api-key",
});
// Function to generate a SHA-256 hash in pure JavaScript
async function generateFingerprint(stackFrame) {
const { file, line, method } = stackFrame;
try {
// Extract the file path without the domain
const url = new URL(file);
const filePath = url.pathname; // Removes protocol and domain
// Concatenate the relevant data
const fingerprintData = `${filePath}:${line}:${method}`;
// Convert to a Uint8Array
const encoder = new TextEncoder();
const data = encoder.encode(fingerprintData);
// Compute SHA-256 hash
const hashBuffer = await crypto.subtle.digest("SHA-256", data);
const hashArray = Array.from(new Uint8Array(hashBuffer));
// Convert to hex string
return hashArray.map((b) => b.toString(16).padStart(2, "0")).join("");
} catch (error) {
console.error("Error generating fingerprint:", error);
return null; // Fallback if hashing fails
}
}
// Modify Honeybadger before reporting
honeybadger.beforeNotify(async (notice) => {
if (notice.backtrace && notice.backtrace.length > 0) {
const firstFrame = notice.backtrace[0];
const fingerprint = await generateFingerprint(firstFrame);
if (fingerprint) {
notice.fingerprint = fingerprint;
}
}
});
export default honeybadger;
@theonlyriddle
Copy link

The code you provided, specifically line 38, does not work as intended:
const firstFrame = notice.stack[0];

When debugging, I see that notice.stack[0] is just a string. See the output of notice below.

{
    "name": "TypeError",
    "message": "Can only add numbers",
    "stack": "TypeError: Can only add numbers\n    at extended.connect (https://assets.lvh.me/dev-assets/javascript/client_templates/template0/home-0ccd96d03bf685359d28db11cb3603d585a28972f2ec7a1fb40441d967f7b26e.js:12082:13)\n    at Context.connect (https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js:18413:25)\n    at Module.connectContextForScope (https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js:18577:15)\n    at Router.scopeConnected (https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js:18949:16)\n    at ScopeObserver.elementMatchedValue (https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js:18858:23)\n    at ValueListObserver.tokenMatched (https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js:17926:23)\n    at TokenListObserver.tokenMatched (https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js:17863:21)\n    at https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js:17857:38\n    at Array.forEach (<anonymous>)\n    at TokenListObserver.tokensMatched (https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js:17857:14)",
    "context": {},
    "projectRoot": "https://enplo.lvh.me",
    "environment": "development",
    "component": null,
    "action": null,
    "revision": "git SHA/project version",
    "tags": [],
    "backtrace": [
        {
            "file": "https://assets.lvh.me/dev-assets/javascript/client_templates/template0/home-0ccd96d03bf685359d28db11cb3603d585a28972f2ec7a1fb40441d967f7b26e.js",
            "method": "extended.connect",
            "number": 12082,
            "column": 13
        },
        {
            "file": "https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js",
            "method": "Context.connect",
            "number": 18413,
            "column": 25
        },
        {
            "file": "https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js",
            "method": "Module.connectContextForScope",
            "number": 18577,
            "column": 15
        },
        {
            "file": "https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js",
            "method": "Router.scopeConnected",
            "number": 18949,
            "column": 16
        },
        {
            "file": "https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js",
            "method": "ScopeObserver.elementMatchedValue",
            "number": 18858,
            "column": 23
        },
        {
            "file": "https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js",
            "method": "ValueListObserver.tokenMatched",
            "number": 17926,
            "column": 23
        },
        {
            "file": "https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js",
            "method": "TokenListObserver.tokenMatched",
            "number": 17863,
            "column": 21
        },
        {
            "file": "https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js",
            "method": "<unknown>",
            "number": 17857,
            "column": 38
        },
        {
            "file": "<anonymous>",
            "method": "Array.forEach",
            "number": null,
            "column": null
        },
        {
            "file": "https://assets.lvh.me/dev-assets/javascript/client_templates/template0/essentials-29b99215fcfc184674db54cd6ab80ac2f4bd7bea129a6e6a7a65f69170558164.js",
            "method": "TokenListObserver.tokensMatched",
            "number": 17857,
            "column": 14
        }
    ],
    "url": "https://enplo.lvh.me/"
}

When line 9 const { file, line, method } = stackFrame; tries to run, I get a new error saying Error generating fingerprint: TypeError: Failed to construct 'URL': Invalid URL

Is there something that I'm missing or perhaps a different way for this to be done?

@stympy
Copy link
Author

stympy commented Feb 25, 2025

Sorry, I meant to use notice.backtrace instead of notice.stack. I have updated the gist.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment