mirror of
https://github.com/status-im/react-native.git
synced 2025-01-24 16:29:01 +00:00
d0a26a75fb
Summary:025281230de2e316f2770a2db71f8bec6fe43a07 changed `WebSocket` to fire both `onerror` and `onclose`. However, `setupDevtools` assumed that only one of the two handlers would ever be invoked. When the handler is invoked, it re-invokes itself to keep a socket open. But since both handelrs are invoked, a series of closed or failed sockets can cause an exponential increase in sockets opened. Symptoms of this bug include eventually reaching the maximum number of file descriptors allowed by the operating system, or a non-responsive system. Within React Native applications, symptoms include a failure to load resources from the React Native server or application crashes. Reviewed By: frantic Differential Revision: D3022697 fb-gh-sync-id: 8131ecbd6d8ba30281253340d30370ff511a5efd shipit-source-id: 8131ecbd6d8ba30281253340d30370ff511a5efd
111 lines
3.0 KiB
JavaScript
111 lines
3.0 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 setupDevtools
|
|
* @flow
|
|
*/
|
|
'use strict';
|
|
|
|
function setupDevtools() {
|
|
var messageListeners = [];
|
|
var closeListeners = [];
|
|
var ws = new window.WebSocket('ws://localhost:8097/devtools');
|
|
// this is accessed by the eval'd backend code
|
|
var FOR_BACKEND = { // eslint-disable-line no-unused-vars
|
|
resolveRNStyle: require('flattenStyle'),
|
|
wall: {
|
|
listen(fn) {
|
|
messageListeners.push(fn);
|
|
},
|
|
onClose(fn) {
|
|
closeListeners.push(fn);
|
|
},
|
|
send(data) {
|
|
ws.send(JSON.stringify(data));
|
|
},
|
|
},
|
|
};
|
|
ws.onclose = handleClose;
|
|
ws.onerror = handleClose;
|
|
ws.onopen = function () {
|
|
tryToConnect();
|
|
};
|
|
|
|
var hasClosed = false;
|
|
function handleClose() {
|
|
if (!hasClosed) {
|
|
hasClosed = true;
|
|
setTimeout(setupDevtools, 200);
|
|
closeListeners.forEach(fn => fn());
|
|
}
|
|
}
|
|
|
|
function tryToConnect() {
|
|
ws.send('attach:agent');
|
|
var _interval = setInterval(() => ws.send('attach:agent'), 500);
|
|
ws.onmessage = evt => {
|
|
if (evt.data.indexOf('eval:') === 0) {
|
|
clearInterval(_interval);
|
|
initialize(evt.data.slice('eval:'.length));
|
|
}
|
|
};
|
|
}
|
|
|
|
function initialize(text) {
|
|
try {
|
|
// FOR_BACKEND is used by the eval'd code
|
|
eval(text); // eslint-disable-line no-eval
|
|
} catch (e) {
|
|
console.error('Failed to eval: ' + e.message);
|
|
return;
|
|
}
|
|
window.__REACT_DEVTOOLS_GLOBAL_HOOK__.inject({
|
|
CurrentOwner: require('ReactCurrentOwner'),
|
|
InstanceHandles: require('ReactInstanceHandles'),
|
|
Mount: require('ReactNativeMount'),
|
|
Reconciler: require('ReactReconciler'),
|
|
TextComponent: require('ReactNativeTextComponent'),
|
|
});
|
|
ws.onmessage = handleMessage;
|
|
}
|
|
|
|
function handleMessage(evt) {
|
|
// It's hard to handle JSON in a safe manner without inspecting it at
|
|
// runtime, hence the any
|
|
var data: any;
|
|
try {
|
|
data = JSON.parse(evt.data);
|
|
} catch (e) {
|
|
return console.error('failed to parse json: ' + evt.data);
|
|
}
|
|
// the devtools closed
|
|
if (data.$close || data.$error) {
|
|
closeListeners.forEach(fn => fn());
|
|
window.__REACT_DEVTOOLS_GLOBAL_HOOK__.emit('shutdown');
|
|
tryToConnect();
|
|
return;
|
|
}
|
|
if (data.$open) {
|
|
return; // ignore
|
|
}
|
|
messageListeners.forEach(fn => {
|
|
try {
|
|
fn(data);
|
|
} catch (e) {
|
|
// jsc doesn't play so well with tracebacks that go into eval'd code,
|
|
// so the stack trace here will stop at the `eval()` call. Getting the
|
|
// message that caused the error is the best we can do for now.
|
|
console.log(data);
|
|
throw e;
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
module.exports = setupDevtools;
|