Skip to content

Exceptions from parsing malformed source maps are not caught by promise #62

Open
@3bl3gamer

Description

@3bl3gamer

If _parseJson (and internal JSON.parse) throws an error, it will not be passed to reject, so outer getMappedLocation (and StackTrace.fromError from stacktrace-js) will never resolve/reject.

https://github.com/stacktracejs/stacktrace-gps/blob/v3.0.4/stacktrace-gps.js#L240-L251

var sourceMapConsumerPromise = new Promise(function(resolve, reject) {
    return this._get(sourceMappingURL).then(function(sourceMapSource) {
        if (typeof sourceMapSource === 'string') {
            // throws from here vvv
            sourceMapSource = _parseJson(sourceMapSource.replace(/^\)\]\}'/, ''));
        }
        if (typeof sourceMapSource.sourceRoot === 'undefined') {
            sourceMapSource.sourceRoot = defaultSourceRoot;
        }

        resolve(new SourceMap.SourceMapConsumer(sourceMapSource));
    }, reject); //<- not caught in here
}.bind(this));

Expected Behavior

Invalid source map errors can be caught by StackTrace.fromError(...).catch an others.

Current Behavior

Error becomes unhandled rejection and can not be caught.

Steps to Reproduce

Try to getMappedLocation for source map with some invalid JSON content (for example <html).

Context

I'm trying to catch every possible error with addEventListener('error' and addEventListener('unhandledrejection', enhance stack trace and send it to server. Currently, server is configured to respond with index.html for non-exiting paths. If .map file is missing for some reason, stacktrace-js receives some HTML instead of JSON, fails and no error is sent at all.

What is worse, this unhandled rejection is caught by 'unhandledrejection' handler and passed it to StackTrace.fromError again, which results in an infinite loop.

Your Environment

  • stacktrace.js version: 2.0.2
  • Browser Name and version: Chromium 80.0.3987.116
  • Operating System and version: Arch Linux (desktop)
  • Link to your project: it is private

Possible Solution

- return this._get(sourceMappingURL).then(..., reject);
+ return this._get(sourceMappingURL).then(...).catch(reject);

Metadata

Metadata

Assignees

No one assigned

    Labels

    No labels
    No labels

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions