mirror of
https://github.com/status-im/react-native.git
synced 2025-01-16 12:34:17 +00:00
be5e300abf
Summary: This PR provides a workaround for issue #9605 which affects projects which directly or indirectly require `whatwg-fetch`. At that point, all symbolication on stack trace errors fails due to a circular require caused by RN's lazy loading of `fetch`, which results in `fetch` becoming `undefined`. The workaround is simple: we postpone requiring `fetch` until the first `symbolicateStackTrace` call, at which point we can be certain `fetch` is properly exported. It is then cached for future use. No tests have been added since, apparently, this module did not have any tests to begin with. Manual tests were performed on a regular project (not including `whatwg-fetch` externally) and a project that has direct and indirect includes for `whatwg-fetch`. Closes https://github.com/facebook/react-native/pull/10247 Differential Revision: D4175835 Pulled By: hramos fbshipit-source-id: d849a5874373e76f166ea6ab23f0f888792d1aef
76 lines
2.8 KiB
JavaScript
76 lines
2.8 KiB
JavaScript
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*
|
|
* @providesModule symbolicateStackTrace
|
|
* @flow
|
|
*/
|
|
'use strict';
|
|
|
|
const getDevServer = require('getDevServer');
|
|
|
|
const {SourceCode} = require('NativeModules');
|
|
|
|
// Avoid requiring fetch on load of this module; see symbolicateStackTrace
|
|
let fetch;
|
|
|
|
import type {StackFrame} from 'parseErrorStack';
|
|
|
|
function isSourcedFromDisk(sourcePath: string): boolean {
|
|
return !/^http/.test(sourcePath) && /[\\/]/.test(sourcePath);
|
|
}
|
|
|
|
async function symbolicateStackTrace(stack: Array<StackFrame>): Promise<Array<StackFrame>> {
|
|
// RN currently lazy loads whatwg-fetch using a custom fetch module, which,
|
|
// when called for the first time, requires and re-exports 'whatwg-fetch'.
|
|
// However, when a dependency of the project tries to require whatwg-fetch
|
|
// either directly or indirectly, whatwg-fetch is required before
|
|
// RN can lazy load whatwg-fetch. As whatwg-fetch checks
|
|
// for a fetch polyfill before loading, it will in turn try to load
|
|
// RN's fetch module, which immediately tries to import whatwg-fetch AGAIN.
|
|
// This causes a circular require which results in RN's fetch module
|
|
// exporting fetch as 'undefined'.
|
|
// The fix below postpones trying to load fetch until the first call to symbolicateStackTrace.
|
|
// At that time, we will have either global.fetch (whatwg-fetch) or RN's fetch.
|
|
if (!fetch) {
|
|
fetch = global.fetch || require('fetch').fetch;
|
|
}
|
|
|
|
const devServer = getDevServer();
|
|
if (!devServer.bundleLoadedFromServer) {
|
|
throw new Error('Bundle was not loaded from the packager');
|
|
}
|
|
|
|
let stackCopy = stack;
|
|
|
|
if (SourceCode.scriptURL) {
|
|
let foundInternalSource: boolean = false;
|
|
stackCopy = stack.map((frame: StackFrame) => {
|
|
// If the sources exist on disk rather than appearing to come from the packager,
|
|
// replace the location with the packager URL until we reach an internal source
|
|
// which does not have a path (no slashes), indicating a switch from within
|
|
// the application to a surrounding debugging environment.
|
|
if (!foundInternalSource && isSourcedFromDisk(frame.file)) {
|
|
// Copy frame into new object and replace 'file' property
|
|
return {...frame, file: SourceCode.scriptURL};
|
|
}
|
|
|
|
foundInternalSource = true;
|
|
return frame;
|
|
});
|
|
}
|
|
|
|
const response = await fetch(devServer.url + 'symbolicate', {
|
|
method: 'POST',
|
|
body: JSON.stringify({stack: stackCopy}),
|
|
});
|
|
const json = await response.json();
|
|
return json.stack;
|
|
}
|
|
|
|
module.exports = symbolicateStackTrace;
|