| /* |
| * Copyright (C) 2012 Research In Motion Limited. All rights reserved. |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
| */ |
| |
| /** |
| * @unrestricted |
| */ |
| export class ResourceWebSocketFrameView extends UI.VBox { |
| /** |
| * @param {!SDK.NetworkRequest} request |
| */ |
| constructor(request) { |
| super(); |
| this.registerRequiredCSS('network/webSocketFrameView.css'); |
| this.element.classList.add('websocket-frame-view'); |
| this._request = request; |
| |
| this._splitWidget = new UI.SplitWidget(false, true, 'resourceWebSocketFrameSplitViewState'); |
| this._splitWidget.show(this.element); |
| |
| const columns = /** @type {!Array<!DataGrid.DataGrid.ColumnDescriptor>} */ ([ |
| {id: 'data', title: Common.UIString('Data'), sortable: false, weight: 88}, { |
| id: 'length', |
| title: Common.UIString('Length'), |
| sortable: false, |
| align: DataGrid.DataGrid.Align.Right, |
| weight: 5 |
| }, |
| {id: 'time', title: Common.UIString('Time'), sortable: true, weight: 7} |
| ]); |
| |
| this._dataGrid = new DataGrid.SortableDataGrid(columns); |
| this._dataGrid.setRowContextMenuCallback(onRowContextMenu.bind(this)); |
| this._dataGrid.setStickToBottom(true); |
| this._dataGrid.setCellClass('websocket-frame-view-td'); |
| this._timeComparator = |
| /** @type {function(!ResourceWebSocketFrameNode, !ResourceWebSocketFrameNode):number} */ ( |
| Network.ResourceWebSocketFrameNodeTimeComparator); |
| this._dataGrid.sortNodes(this._timeComparator, false); |
| this._dataGrid.markColumnAsSortedBy('time', DataGrid.DataGrid.Order.Ascending); |
| this._dataGrid.addEventListener(DataGrid.DataGrid.Events.SortingChanged, this._sortItems, this); |
| |
| this._dataGrid.setName('ResourceWebSocketFrameView'); |
| this._dataGrid.addEventListener(DataGrid.DataGrid.Events.SelectedNode, this._onFrameSelected, this); |
| this._dataGrid.addEventListener(DataGrid.DataGrid.Events.DeselectedNode, this._onFrameDeselected, this); |
| |
| this._mainToolbar = new UI.Toolbar(''); |
| |
| this._clearAllButton = new UI.ToolbarButton(Common.UIString('Clear All'), 'largeicon-clear'); |
| this._clearAllButton.addEventListener(UI.ToolbarButton.Events.Click, this._clearFrames, this); |
| this._mainToolbar.appendToolbarItem(this._clearAllButton); |
| |
| this._filterTypeCombobox = new UI.ToolbarComboBox(this._updateFilterSetting.bind(this), ls`Filter`); |
| for (const filterItem of _filterTypes) { |
| const option = this._filterTypeCombobox.createOption(filterItem.label, filterItem.name); |
| this._filterTypeCombobox.addOption(option); |
| } |
| this._mainToolbar.appendToolbarItem(this._filterTypeCombobox); |
| this._filterType = null; |
| |
| const placeholder = 'Enter regex, for example: (web)?socket'; |
| this._filterTextInput = new UI.ToolbarInput(Common.UIString(placeholder), '', 0.4); |
| this._filterTextInput.addEventListener(UI.ToolbarInput.Event.TextChanged, this._updateFilterSetting, this); |
| this._mainToolbar.appendToolbarItem(this._filterTextInput); |
| this._filterRegex = null; |
| |
| const mainContainer = new UI.VBox(); |
| mainContainer.element.appendChild(this._mainToolbar.element); |
| this._dataGrid.asWidget().show(mainContainer.element); |
| mainContainer.setMinimumSize(0, 72); |
| this._splitWidget.setMainWidget(mainContainer); |
| |
| this._frameEmptyWidget = new UI.EmptyWidget(Common.UIString('Select message to browse its content.')); |
| this._splitWidget.setSidebarWidget(this._frameEmptyWidget); |
| |
| /** @type {?ResourceWebSocketFrameNode} */ |
| this._selectedNode = null; |
| |
| /** |
| * @param {!UI.ContextMenu} contextMenu |
| * @param {!DataGrid.DataGridNode} genericNode |
| * @this {ResourceWebSocketFrameView} |
| */ |
| function onRowContextMenu(contextMenu, genericNode) { |
| const node = /** @type {!ResourceWebSocketFrameNode} */ (genericNode); |
| const binaryView = node.binaryView(); |
| if (binaryView) { |
| binaryView.addCopyToContextMenu(contextMenu, ls`Copy message...`); |
| } else { |
| contextMenu.clipboardSection().appendItem( |
| Common.UIString('Copy message'), |
| Host.InspectorFrontendHost.copyText.bind(Host.InspectorFrontendHost, node.data.data)); |
| } |
| contextMenu.footerSection().appendItem(Common.UIString('Clear all'), this._clearFrames.bind(this)); |
| } |
| } |
| |
| /** |
| * @param {number} opCode |
| * @param {boolean} mask |
| * @return {string} |
| */ |
| static opCodeDescription(opCode, mask) { |
| const localizedDescription = opCodeDescriptions[opCode] || ''; |
| if (mask) { |
| return ls`${localizedDescription} (Opcode ${opCode}, mask)`; |
| } |
| return ls`${localizedDescription} (Opcode ${opCode})`; |
| } |
| |
| /** |
| * @override |
| */ |
| wasShown() { |
| this.refresh(); |
| this._request.addEventListener(SDK.NetworkRequest.Events.WebsocketFrameAdded, this._frameAdded, this); |
| } |
| |
| /** |
| * @override |
| */ |
| willHide() { |
| this._request.removeEventListener(SDK.NetworkRequest.Events.WebsocketFrameAdded, this._frameAdded, this); |
| } |
| |
| /** |
| * @param {!Common.Event} event |
| */ |
| _frameAdded(event) { |
| const frame = /** @type {!SDK.NetworkRequest.WebSocketFrame} */ (event.data); |
| if (!this._frameFilter(frame)) { |
| return; |
| } |
| this._dataGrid.insertChild(new ResourceWebSocketFrameNode(this._request.url(), frame)); |
| } |
| |
| /** |
| * @param {!SDK.NetworkRequest.WebSocketFrame} frame |
| * @return {boolean} |
| */ |
| _frameFilter(frame) { |
| if (this._filterType && frame.type !== this._filterType) { |
| return false; |
| } |
| return !this._filterRegex || this._filterRegex.test(frame.text); |
| } |
| |
| _clearFrames() { |
| // TODO(allada): actially remove frames from request. |
| this._request[_clearFrameOffsetSymbol] = this._request.frames().length; |
| this.refresh(); |
| } |
| |
| _updateFilterSetting() { |
| const text = this._filterTextInput.value(); |
| const type = this._filterTypeCombobox.selectedOption().value; |
| this._filterRegex = text ? new RegExp(text, 'i') : null; |
| this._filterType = type === 'all' ? null : type; |
| this.refresh(); |
| } |
| |
| /** |
| * @param {!Common.Event} event |
| */ |
| async _onFrameSelected(event) { |
| this._currentSelectedNode = /** @type {!ResourceWebSocketFrameNode} */ (event.data); |
| const content = this._currentSelectedNode.dataText(); |
| |
| const binaryView = this._currentSelectedNode.binaryView(); |
| if (binaryView) { |
| this._splitWidget.setSidebarWidget(binaryView); |
| return; |
| } |
| |
| const jsonView = await SourceFrame.JSONView.createView(content); |
| if (jsonView) { |
| this._splitWidget.setSidebarWidget(jsonView); |
| return; |
| } |
| |
| this._splitWidget.setSidebarWidget(new SourceFrame.ResourceSourceFrame( |
| Common.StaticContentProvider.fromString(this._request.url(), Common.resourceTypes.WebSocket, content))); |
| } |
| |
| /** |
| * @param {!Common.Event} event |
| */ |
| _onFrameDeselected(event) { |
| this._currentSelectedNode = null; |
| this._splitWidget.setSidebarWidget(this._frameEmptyWidget); |
| } |
| |
| refresh() { |
| this._dataGrid.rootNode().removeChildren(); |
| |
| const url = this._request.url(); |
| let frames = this._request.frames(); |
| const offset = this._request[_clearFrameOffsetSymbol] || 0; |
| frames = frames.slice(offset); |
| frames = frames.filter(this._frameFilter.bind(this)); |
| frames.forEach(frame => this._dataGrid.insertChild(new ResourceWebSocketFrameNode(url, frame))); |
| } |
| |
| _sortItems() { |
| this._dataGrid.sortNodes(this._timeComparator, !this._dataGrid.isSortOrderAscending()); |
| } |
| } |
| |
| /** @enum {number} */ |
| export const OpCodes = { |
| ContinuationFrame: 0, |
| TextFrame: 1, |
| BinaryFrame: 2, |
| ConnectionCloseFrame: 8, |
| PingFrame: 9, |
| PongFrame: 10 |
| }; |
| |
| /** @type {!Array.<string> } */ |
| export const opCodeDescriptions = (function() { |
| const opCodes = OpCodes; |
| const map = []; |
| map[opCodes.ContinuationFrame] = ls`Continuation Frame`; |
| map[opCodes.TextFrame] = ls`Text Message`; |
| map[opCodes.BinaryFrame] = ls`Binary Message`; |
| map[opCodes.ContinuationFrame] = ls`Connection Close Message`; |
| map[opCodes.PingFrame] = ls`Ping Message`; |
| map[opCodes.PongFrame] = ls`Pong Message`; |
| return map; |
| })(); |
| |
| /** @type {!Array<!UI.NamedBitSetFilterUI.Item>} */ |
| export const _filterTypes = [ |
| {name: 'all', label: Common.UIString('All')}, |
| {name: 'send', label: Common.UIString('Send')}, |
| {name: 'receive', label: Common.UIString('Receive')}, |
| ]; |
| |
| /** |
| * @unrestricted |
| */ |
| export class ResourceWebSocketFrameNode extends DataGrid.SortableDataGridNode { |
| /** |
| * @param {string} url |
| * @param {!SDK.NetworkRequest.WebSocketFrame} frame |
| */ |
| constructor(url, frame) { |
| let length = frame.text.length; |
| const time = new Date(frame.time * 1000); |
| const timeText = ('0' + time.getHours()).substr(-2) + ':' + ('0' + time.getMinutes()).substr(-2) + ':' + |
| ('0' + time.getSeconds()).substr(-2) + '.' + ('00' + time.getMilliseconds()).substr(-3); |
| const timeNode = createElement('div'); |
| timeNode.createTextChild(timeText); |
| timeNode.title = time.toLocaleString(); |
| |
| let dataText = frame.text; |
| let description = ResourceWebSocketFrameView.opCodeDescription(frame.opCode, frame.mask); |
| const isTextFrame = frame.opCode === OpCodes.TextFrame; |
| |
| if (frame.type === SDK.NetworkRequest.WebSocketFrameType.Error) { |
| description = dataText; |
| length = ls`N/A`; |
| |
| } else if (isTextFrame) { |
| description = dataText; |
| |
| } else if (frame.opCode === OpCodes.BinaryFrame) { |
| length = Number.bytesToString(base64ToSize(frame.text)); |
| description = opCodeDescriptions[frame.opCode]; |
| |
| } else { |
| dataText = description; |
| } |
| |
| super({data: description, length: length, time: timeNode}); |
| |
| this._url = url; |
| this._frame = frame; |
| this._isTextFrame = isTextFrame; |
| this._dataText = dataText; |
| } |
| |
| /** |
| * @override |
| * @param {!Element} element |
| */ |
| createCells(element) { |
| element.classList.toggle( |
| 'websocket-frame-view-row-error', this._frame.type === SDK.NetworkRequest.WebSocketFrameType.Error); |
| element.classList.toggle( |
| 'websocket-frame-view-row-send', this._frame.type === SDK.NetworkRequest.WebSocketFrameType.Send); |
| element.classList.toggle( |
| 'websocket-frame-view-row-receive', this._frame.type === SDK.NetworkRequest.WebSocketFrameType.Receive); |
| super.createCells(element); |
| } |
| |
| /** |
| * @override |
| * @return {number} |
| */ |
| nodeSelfHeight() { |
| return 21; |
| } |
| |
| /** |
| * @return {string} |
| */ |
| dataText() { |
| return this._dataText; |
| } |
| |
| /** |
| * @return {!OpCodes} |
| */ |
| opCode() { |
| return /** @type {!OpCodes} */ (this._frame.opCode); |
| } |
| |
| /** |
| * @return {?Network.BinaryResourceView} |
| */ |
| binaryView() { |
| if (this._isTextFrame || this._frame.type === SDK.NetworkRequest.WebSocketFrameType.Error) { |
| return null; |
| } |
| |
| if (!this._binaryView) { |
| this._binaryView = new Network.BinaryResourceView(this._dataText, /* url */ '', Common.resourceTypes.WebSocket); |
| } |
| return this._binaryView; |
| } |
| } |
| |
| /** |
| * @param {!ResourceWebSocketFrameNode} a |
| * @param {!ResourceWebSocketFrameNode} b |
| * @return {number} |
| */ |
| export function ResourceWebSocketFrameNodeTimeComparator(a, b) { |
| return a._frame.time - b._frame.time; |
| } |
| |
| export const _clearFrameOffsetSymbol = Symbol('ClearFrameOffset'); |
| |
| /* Legacy exported object */ |
| self.Network = self.Network || {}; |
| |
| /* Legacy exported object */ |
| Network = Network || {}; |
| |
| /** |
| * @constructor |
| */ |
| Network.ResourceWebSocketFrameView = ResourceWebSocketFrameView; |
| |
| /** @enum {number} */ |
| Network.ResourceWebSocketFrameView.OpCodes = OpCodes; |
| |
| /** @type {!Array.<string> } */ |
| Network.ResourceWebSocketFrameView.opCodeDescriptions = opCodeDescriptions; |
| |
| /** @type {!Array<!UI.NamedBitSetFilterUI.Item>} */ |
| Network.ResourceWebSocketFrameView._filterTypes = _filterTypes; |
| Network.ResourceWebSocketFrameView._clearFrameOffsetSymbol = _clearFrameOffsetSymbol; |
| Network.ResourceWebSocketFrameNode = ResourceWebSocketFrameNode; |