614 lines
17 KiB
JavaScript
614 lines
17 KiB
JavaScript
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @flow
|
|
*/
|
|
'use strict';
|
|
|
|
const ListView = require('ListView');
|
|
const React = require('React');
|
|
const ScrollView = require('ScrollView');
|
|
const StyleSheet = require('StyleSheet');
|
|
const Text = require('Text');
|
|
const TouchableHighlight = require('TouchableHighlight');
|
|
const View = require('View');
|
|
const WebSocketInterceptor = require('WebSocketInterceptor');
|
|
const XHRInterceptor = require('XHRInterceptor');
|
|
|
|
const LISTVIEW_CELL_HEIGHT = 15;
|
|
const SEPARATOR_THICKNESS = 2;
|
|
|
|
// Global id for the intercepted XMLHttpRequest objects.
|
|
let nextXHRId = 0;
|
|
|
|
type NetworkRequestInfo = {
|
|
type?: string,
|
|
url?: string,
|
|
method?: string,
|
|
status?: number,
|
|
dataSent?: any,
|
|
responseContentType?: string,
|
|
responseSize?: number,
|
|
requestHeaders?: Object,
|
|
responseHeaders?: string,
|
|
response?: Object | string,
|
|
responseURL?: string,
|
|
responseType?: string,
|
|
timeout?: number,
|
|
closeReason?: string,
|
|
messages?: string,
|
|
serverClose?: Object,
|
|
serverError?: Object,
|
|
};
|
|
|
|
/**
|
|
* Show all the intercepted network requests over the InspectorPanel.
|
|
*/
|
|
class NetworkOverlay extends React.Component<Object, {
|
|
dataSource: ListView.DataSource,
|
|
newDetailInfo: bool,
|
|
detailRowID: ?number,
|
|
}> {
|
|
_requests: Array<NetworkRequestInfo>;
|
|
_listViewDataSource: ListView.DataSource;
|
|
_listView: ?ListView;
|
|
_listViewHighlighted: bool;
|
|
_listViewHeight: number;
|
|
_scrollView: ?ScrollView;
|
|
_detailViewItems: Array<Array<React.Element<any>>>;
|
|
_listViewOnLayout: (event: Event) => void;
|
|
_captureRequestListView: (listRef: ?ListView) => void;
|
|
_captureDetailScrollView: (scrollRef: ?ScrollView) => void;
|
|
_renderRow: (
|
|
rowData: NetworkRequestInfo,
|
|
sectionID: number,
|
|
rowID: number,
|
|
highlightRow: (sectionID: number, rowID: number) => void,
|
|
) => React.Element<any>;
|
|
_closeButtonClicked: () => void;
|
|
// Map of `socketId` -> `index in `_requests``.
|
|
_socketIdMap: Object;
|
|
// Map of `xhr._index` -> `index in `_requests``.
|
|
_xhrIdMap: {[key: number]: number};
|
|
|
|
constructor(props: Object) {
|
|
super(props);
|
|
this._requests = [];
|
|
this._detailViewItems = [];
|
|
this._listViewDataSource =
|
|
new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2});
|
|
this.state = {
|
|
dataSource: this._listViewDataSource.cloneWithRows([]),
|
|
newDetailInfo: false,
|
|
detailRowID: null,
|
|
};
|
|
this._listViewHighlighted = false;
|
|
this._listViewHeight = 0;
|
|
this._captureRequestListView = this._captureRequestListView.bind(this);
|
|
this._captureDetailScrollView = this._captureDetailScrollView.bind(this);
|
|
this._listViewOnLayout = this._listViewOnLayout.bind(this);
|
|
this._renderRow = this._renderRow.bind(this);
|
|
this._closeButtonClicked = this._closeButtonClicked.bind(this);
|
|
this._socketIdMap = {};
|
|
this._xhrIdMap = {};
|
|
}
|
|
|
|
_enableXHRInterception(): void {
|
|
if (XHRInterceptor.isInterceptorEnabled()) {
|
|
return;
|
|
}
|
|
// Show the XHR request item in listView as soon as it was opened.
|
|
XHRInterceptor.setOpenCallback((method, url, xhr) => {
|
|
// Generate a global id for each intercepted xhr object, add this id
|
|
// to the xhr object as a private `_index` property to identify it,
|
|
// so that we can distinguish different xhr objects in callbacks.
|
|
xhr._index = nextXHRId++;
|
|
const xhrIndex = this._requests.length;
|
|
this._xhrIdMap[xhr._index] = xhrIndex;
|
|
|
|
const _xhr: NetworkRequestInfo = {
|
|
'type': 'XMLHttpRequest',
|
|
'method': method,
|
|
'url': url
|
|
};
|
|
this._requests.push(_xhr);
|
|
this._detailViewItems.push([]);
|
|
this._genDetailViewItem(xhrIndex);
|
|
this.setState(
|
|
{dataSource: this._listViewDataSource.cloneWithRows(this._requests)},
|
|
this._scrollToBottom(),
|
|
);
|
|
});
|
|
|
|
XHRInterceptor.setRequestHeaderCallback((header, value, xhr) => {
|
|
const xhrIndex = this._getRequestIndexByXHRID(xhr._index);
|
|
if (xhrIndex === -1) {
|
|
return;
|
|
}
|
|
const networkInfo = this._requests[xhrIndex];
|
|
if (!networkInfo.requestHeaders) {
|
|
networkInfo.requestHeaders = {};
|
|
}
|
|
networkInfo.requestHeaders[header] = value;
|
|
this._genDetailViewItem(xhrIndex);
|
|
});
|
|
|
|
XHRInterceptor.setSendCallback((data, xhr) => {
|
|
const xhrIndex = this._getRequestIndexByXHRID(xhr._index);
|
|
if (xhrIndex === -1) {
|
|
return;
|
|
}
|
|
this._requests[xhrIndex].dataSent = data;
|
|
this._genDetailViewItem(xhrIndex);
|
|
});
|
|
|
|
XHRInterceptor.setHeaderReceivedCallback(
|
|
(type, size, responseHeaders, xhr) => {
|
|
const xhrIndex = this._getRequestIndexByXHRID(xhr._index);
|
|
if (xhrIndex === -1) {
|
|
return;
|
|
}
|
|
const networkInfo = this._requests[xhrIndex];
|
|
networkInfo.responseContentType = type;
|
|
networkInfo.responseSize = size;
|
|
networkInfo.responseHeaders = responseHeaders;
|
|
this._genDetailViewItem(xhrIndex);
|
|
}
|
|
);
|
|
|
|
XHRInterceptor.setResponseCallback((
|
|
status,
|
|
timeout,
|
|
response,
|
|
responseURL,
|
|
responseType,
|
|
xhr,
|
|
) => {
|
|
const xhrIndex = this._getRequestIndexByXHRID(xhr._index);
|
|
if (xhrIndex === -1) {
|
|
return;
|
|
}
|
|
const networkInfo = this._requests[xhrIndex];
|
|
networkInfo.status = status;
|
|
networkInfo.timeout = timeout;
|
|
networkInfo.response = response;
|
|
networkInfo.responseURL = responseURL;
|
|
networkInfo.responseType = responseType;
|
|
this._genDetailViewItem(xhrIndex);
|
|
}
|
|
);
|
|
|
|
// Fire above callbacks.
|
|
XHRInterceptor.enableInterception();
|
|
}
|
|
|
|
_enableWebSocketInterception(): void {
|
|
if (WebSocketInterceptor.isInterceptorEnabled()) {
|
|
return;
|
|
}
|
|
// Show the WebSocket request item in listView when 'connect' is called.
|
|
WebSocketInterceptor.setConnectCallback(
|
|
(url, protocols, options, socketId) => {
|
|
const socketIndex = this._requests.length;
|
|
this._socketIdMap[socketId] = socketIndex;
|
|
const _webSocket: NetworkRequestInfo = {
|
|
'type': 'WebSocket',
|
|
'url': url,
|
|
'protocols': protocols,
|
|
};
|
|
this._requests.push(_webSocket);
|
|
this._detailViewItems.push([]);
|
|
this._genDetailViewItem(socketIndex);
|
|
this.setState(
|
|
{dataSource: this._listViewDataSource.cloneWithRows(this._requests)},
|
|
this._scrollToBottom(),
|
|
);
|
|
}
|
|
);
|
|
|
|
WebSocketInterceptor.setCloseCallback(
|
|
(statusCode, closeReason, socketId) => {
|
|
const socketIndex = this._socketIdMap[socketId];
|
|
if (socketIndex === undefined) {
|
|
return;
|
|
}
|
|
if (statusCode !== null && closeReason !== null) {
|
|
this._requests[socketIndex].status = statusCode;
|
|
this._requests[socketIndex].closeReason = closeReason;
|
|
}
|
|
this._genDetailViewItem(socketIndex);
|
|
}
|
|
);
|
|
|
|
WebSocketInterceptor.setSendCallback((data, socketId) => {
|
|
const socketIndex = this._socketIdMap[socketId];
|
|
if (socketIndex === undefined) {
|
|
return;
|
|
}
|
|
if (!this._requests[socketIndex].messages) {
|
|
this._requests[socketIndex].messages = '';
|
|
}
|
|
this._requests[socketIndex].messages +=
|
|
'Sent: ' + JSON.stringify(data) + '\n';
|
|
this._genDetailViewItem(socketIndex);
|
|
});
|
|
|
|
WebSocketInterceptor.setOnMessageCallback((socketId, message) => {
|
|
const socketIndex = this._socketIdMap[socketId];
|
|
if (socketIndex === undefined) {
|
|
return;
|
|
}
|
|
if (!this._requests[socketIndex].messages) {
|
|
this._requests[socketIndex].messages = '';
|
|
}
|
|
this._requests[socketIndex].messages +=
|
|
'Received: ' + JSON.stringify(message) + '\n';
|
|
this._genDetailViewItem(socketIndex);
|
|
});
|
|
|
|
WebSocketInterceptor.setOnCloseCallback((socketId, message) => {
|
|
const socketIndex = this._socketIdMap[socketId];
|
|
if (socketIndex === undefined) {
|
|
return;
|
|
}
|
|
this._requests[socketIndex].serverClose = message;
|
|
this._genDetailViewItem(socketIndex);
|
|
});
|
|
|
|
WebSocketInterceptor.setOnErrorCallback((socketId, message) => {
|
|
const socketIndex = this._socketIdMap[socketId];
|
|
if (socketIndex === undefined) {
|
|
return;
|
|
}
|
|
this._requests[socketIndex].serverError = message;
|
|
this._genDetailViewItem(socketIndex);
|
|
});
|
|
|
|
// Fire above callbacks.
|
|
WebSocketInterceptor.enableInterception();
|
|
}
|
|
|
|
componentDidMount() {
|
|
this._enableXHRInterception();
|
|
this._enableWebSocketInterception();
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
XHRInterceptor.disableInterception();
|
|
WebSocketInterceptor.disableInterception();
|
|
}
|
|
|
|
_renderRow(
|
|
rowData: NetworkRequestInfo,
|
|
sectionID: number,
|
|
rowID: number,
|
|
highlightRow: (sectionID: number, rowID: number) => void,
|
|
): React.Element<any> {
|
|
let urlCellViewStyle = styles.urlEvenCellView;
|
|
let methodCellViewStyle = styles.methodEvenCellView;
|
|
if (rowID % 2 === 1) {
|
|
urlCellViewStyle = styles.urlOddCellView;
|
|
methodCellViewStyle = styles.methodOddCellView;
|
|
}
|
|
return (
|
|
<TouchableHighlight onPress={() => {
|
|
this._pressRow(rowID);
|
|
highlightRow(sectionID, rowID);
|
|
}}>
|
|
<View>
|
|
<View style={styles.tableRow}>
|
|
<View style={urlCellViewStyle}>
|
|
<Text style={styles.cellText} numberOfLines={1}>
|
|
{rowData.url}
|
|
</Text>
|
|
</View>
|
|
<View style={methodCellViewStyle}>
|
|
<Text style={styles.cellText} numberOfLines={1}>
|
|
{this._getTypeShortName(rowData.type)}
|
|
</Text>
|
|
</View>
|
|
</View>
|
|
</View>
|
|
</TouchableHighlight>
|
|
);
|
|
}
|
|
|
|
_renderSeperator(
|
|
sectionID: number,
|
|
rowID: number,
|
|
adjacentRowHighlighted: bool): React.Element<any> {
|
|
return (
|
|
<View
|
|
key={`${sectionID}-${rowID}`}
|
|
style={{
|
|
height: adjacentRowHighlighted ? SEPARATOR_THICKNESS : 0,
|
|
backgroundColor: adjacentRowHighlighted ? '#3B5998' : '#CCCCCC',
|
|
}}
|
|
/>
|
|
);
|
|
}
|
|
|
|
_scrollToBottom(): void {
|
|
if (this._listView) {
|
|
const scrollResponder = this._listView.getScrollResponder();
|
|
if (scrollResponder) {
|
|
const scrollY = Math.max(
|
|
this._requests.length * LISTVIEW_CELL_HEIGHT +
|
|
(this._listViewHighlighted ? 2 * SEPARATOR_THICKNESS : 0) -
|
|
this._listViewHeight,
|
|
0,
|
|
);
|
|
scrollResponder.scrollResponderScrollTo({
|
|
x: 0,
|
|
y: scrollY,
|
|
animated: true
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
_captureRequestListView(listRef: ?ListView): void {
|
|
this._listView = listRef;
|
|
}
|
|
|
|
_listViewOnLayout(event: any): void {
|
|
const {height} = event.nativeEvent.layout;
|
|
this._listViewHeight = height;
|
|
}
|
|
|
|
/**
|
|
* Popup a scrollView to dynamically show detailed information of
|
|
* the request, when pressing a row in the network flow listView.
|
|
*/
|
|
_pressRow(rowID: number): void {
|
|
this._listViewHighlighted = true;
|
|
this.setState(
|
|
{detailRowID: rowID},
|
|
this._scrollToTop(),
|
|
);
|
|
}
|
|
|
|
_scrollToTop(): void {
|
|
if (this._scrollView) {
|
|
this._scrollView.scrollTo({
|
|
y: 0,
|
|
animated: false,
|
|
});
|
|
}
|
|
}
|
|
|
|
_captureDetailScrollView(scrollRef: ?ScrollView): void {
|
|
this._scrollView = scrollRef;
|
|
}
|
|
|
|
_closeButtonClicked() {
|
|
this.setState({detailRowID: null});
|
|
}
|
|
|
|
_getStringByValue(value: any): string {
|
|
if (value === undefined) {
|
|
return 'undefined';
|
|
}
|
|
if (typeof value === 'object') {
|
|
return JSON.stringify(value);
|
|
}
|
|
if (typeof value === 'string' && value.length > 500) {
|
|
return String(value).substr(0, 500).concat(
|
|
'\n***TRUNCATED TO 500 CHARACTERS***');
|
|
}
|
|
return value;
|
|
}
|
|
|
|
_getRequestIndexByXHRID(index: number): number {
|
|
if (index === undefined) {
|
|
return -1;
|
|
}
|
|
const xhrIndex = this._xhrIdMap[index];
|
|
if (xhrIndex === undefined) {
|
|
return -1;
|
|
} else {
|
|
return xhrIndex;
|
|
}
|
|
}
|
|
|
|
_getTypeShortName(type: any): string {
|
|
if (type === 'XMLHttpRequest') {
|
|
return 'XHR';
|
|
} else if (type === 'WebSocket') {
|
|
return 'WS';
|
|
}
|
|
|
|
return '';
|
|
}
|
|
|
|
/**
|
|
* Generate a list of views containing network request information for
|
|
* a XHR object, to be shown in the detail scrollview. This function
|
|
* should be called every time there is a new update of the XHR object,
|
|
* in order to show network request/response information in real time.
|
|
*/
|
|
_genDetailViewItem(index: number): void {
|
|
this._detailViewItems[index] = [];
|
|
const detailViewItem = this._detailViewItems[index];
|
|
const requestItem = this._requests[index];
|
|
for (let key in requestItem) {
|
|
detailViewItem.push(
|
|
<View style={styles.detailViewRow} key={key}>
|
|
<Text style={[styles.detailViewText, styles.detailKeyCellView]}>
|
|
{key}
|
|
</Text>
|
|
<Text style={[styles.detailViewText, styles.detailValueCellView]}>
|
|
{this._getStringByValue(requestItem[key])}
|
|
</Text>
|
|
</View>
|
|
);
|
|
}
|
|
// Re-render if this network request is showing in the detail view.
|
|
if (this.state.detailRowID != null &&
|
|
Number(this.state.detailRowID) === index) {
|
|
this.setState({newDetailInfo: true});
|
|
}
|
|
}
|
|
|
|
render() {
|
|
return (
|
|
<View style={styles.container}>
|
|
{this.state.detailRowID != null &&
|
|
<TouchableHighlight
|
|
style={styles.closeButton}
|
|
onPress={this._closeButtonClicked}>
|
|
<View>
|
|
<Text style={styles.clostButtonText}>v</Text>
|
|
</View>
|
|
</TouchableHighlight>}
|
|
{this.state.detailRowID != null &&
|
|
<ScrollView
|
|
style={styles.detailScrollView}
|
|
ref={this._captureDetailScrollView}>
|
|
{this._detailViewItems[this.state.detailRowID]}
|
|
</ScrollView>}
|
|
<View style={styles.listViewTitle}>
|
|
{this._requests.length > 0 &&
|
|
<View style={styles.tableRow}>
|
|
<View style={styles.urlTitleCellView}>
|
|
<Text style={styles.cellText} numberOfLines={1}>URL</Text>
|
|
</View>
|
|
<View style={styles.methodTitleCellView}>
|
|
<Text style={styles.cellText} numberOfLines={1}>Type</Text>
|
|
</View>
|
|
</View>}
|
|
</View>
|
|
<ListView
|
|
style={styles.listView}
|
|
ref={this._captureRequestListView}
|
|
dataSource={this.state.dataSource}
|
|
renderRow={this._renderRow}
|
|
enableEmptySections={true}
|
|
renderSeparator={this._renderSeperator}
|
|
onLayout={this._listViewOnLayout}
|
|
/>
|
|
</View>
|
|
);
|
|
}
|
|
}
|
|
|
|
const styles = StyleSheet.create({
|
|
container: {
|
|
paddingTop: 10,
|
|
paddingBottom: 10,
|
|
paddingLeft: 5,
|
|
paddingRight: 5,
|
|
},
|
|
listViewTitle: {
|
|
height: 20,
|
|
},
|
|
listView: {
|
|
flex: 1,
|
|
height: 60,
|
|
},
|
|
tableRow: {
|
|
flexDirection: 'row',
|
|
flex: 1,
|
|
},
|
|
cellText: {
|
|
color: 'white',
|
|
fontSize: 12,
|
|
},
|
|
methodTitleCellView: {
|
|
height: 18,
|
|
borderColor: '#DCD7CD',
|
|
borderTopWidth: 1,
|
|
borderBottomWidth: 1,
|
|
borderRightWidth: 1,
|
|
alignItems: 'center',
|
|
justifyContent: 'center',
|
|
backgroundColor: '#444',
|
|
flex: 1,
|
|
},
|
|
urlTitleCellView: {
|
|
height: 18,
|
|
borderColor: '#DCD7CD',
|
|
borderTopWidth: 1,
|
|
borderBottomWidth: 1,
|
|
borderLeftWidth: 1,
|
|
borderRightWidth: 1,
|
|
justifyContent: 'center',
|
|
backgroundColor: '#444',
|
|
flex: 5,
|
|
paddingLeft: 3,
|
|
},
|
|
methodOddCellView: {
|
|
height: 15,
|
|
borderColor: '#DCD7CD',
|
|
borderRightWidth: 1,
|
|
alignItems: 'center',
|
|
justifyContent: 'center',
|
|
backgroundColor: '#000',
|
|
flex: 1,
|
|
},
|
|
urlOddCellView: {
|
|
height: 15,
|
|
borderColor: '#DCD7CD',
|
|
borderLeftWidth: 1,
|
|
borderRightWidth: 1,
|
|
justifyContent: 'center',
|
|
backgroundColor: '#000',
|
|
flex: 5,
|
|
paddingLeft: 3,
|
|
},
|
|
methodEvenCellView: {
|
|
height: 15,
|
|
borderColor: '#DCD7CD',
|
|
borderRightWidth: 1,
|
|
alignItems: 'center',
|
|
justifyContent: 'center',
|
|
backgroundColor: '#888',
|
|
flex: 1,
|
|
},
|
|
urlEvenCellView: {
|
|
height: 15,
|
|
borderColor: '#DCD7CD',
|
|
borderLeftWidth: 1,
|
|
borderRightWidth: 1,
|
|
justifyContent: 'center',
|
|
backgroundColor: '#888',
|
|
flex: 5,
|
|
paddingLeft: 3,
|
|
},
|
|
detailScrollView: {
|
|
flex: 1,
|
|
height: 180,
|
|
marginTop: 5,
|
|
marginBottom: 5,
|
|
},
|
|
detailKeyCellView: {
|
|
flex: 1.3,
|
|
},
|
|
detailValueCellView: {
|
|
flex: 2,
|
|
},
|
|
detailViewRow: {
|
|
flexDirection: 'row',
|
|
paddingHorizontal: 3,
|
|
},
|
|
detailViewText: {
|
|
color: 'white',
|
|
fontSize: 11,
|
|
},
|
|
clostButtonText: {
|
|
color: 'white',
|
|
fontSize: 10,
|
|
},
|
|
closeButton: {
|
|
marginTop: 5,
|
|
backgroundColor: '#888',
|
|
justifyContent: 'center',
|
|
alignItems: 'center',
|
|
},
|
|
});
|
|
|
|
module.exports = NetworkOverlay;
|