UNPKG

debug-server-next

Version:

Dev server for hippy-core.

413 lines (412 loc) 18.3 kB
// Copyright 2021 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. /* * 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 */ /* eslint-disable rulesdir/no_underscored_properties */ import * as Common from '../../core/common/common.js'; import * as Host from '../../core/host/host.js'; import * as i18n from '../../core/i18n/i18n.js'; import * as Platform from '../../core/platform/platform.js'; import * as SDK from '../../core/sdk/sdk.js'; import * as TextUtils from '../../models/text_utils/text_utils.js'; import * as DataGrid from '../../ui/legacy/components/data_grid/data_grid.js'; import * as SourceFrame from '../../ui/legacy/components/source_frame/source_frame.js'; import * as UI from '../../ui/legacy/legacy.js'; import { BinaryResourceView } from './BinaryResourceView.js'; const UIStrings = { /** *@description Text in Event Source Messages View of the Network panel */ data: 'Data', /** *@description Text in Resource Web Socket Frame View of the Network panel */ length: 'Length', /** *@description Text that refers to the time */ time: 'Time', /** *@description Data grid name for Web Socket Frame data grids */ webSocketFrame: 'Web Socket Frame', /** *@description Text to clear everything */ clearAll: 'Clear All', /** *@description Text to filter result items */ filter: 'Filter', /** *@description Text in Resource Web Socket Frame View of the Network panel */ selectMessageToBrowseItsContent: 'Select message to browse its content.', /** *@description Text in Resource Web Socket Frame View of the Network panel */ copyMessageD: 'Copy message...', /** *@description A context menu item in the Resource Web Socket Frame View of the Network panel */ copyMessage: 'Copy message', /** *@description Text to clear everything */ clearAllL: 'Clear all', /** * @description Text in Resource Web Socket Frame View of the Network panel. Displays which Opcode * is relevant to a particular operation. 'mask' indicates that the Opcode used a mask, which is a * way of modifying a value by overlaying another value on top of it, partially covering/changing * it, hence 'masking' it. * https://developer.mozilla.org/en-US/docs/Web/API/WebSockets_API/Writing_WebSocket_servers * @example {Localized name of the Opcode} PH1 * @example {0} PH2 */ sOpcodeSMask: '{PH1} (Opcode {PH2}, mask)', /** * @description Text in Resource Web Socket Frame View of the Network panel. Displays which Opcode * is relevant to a particular operation. * @example {Localized name of the Opcode} PH1 * @example {0} PH2 */ sOpcodeS: '{PH1} (Opcode {PH2})', /** *@description Op codes continuation frame of map in Resource Web Socket Frame View of the Network panel */ continuationFrame: 'Continuation Frame', /** *@description Op codes text frame of map in Resource Web Socket Frame View of the Network panel */ textMessage: 'Text Message', /** *@description Op codes binary frame of map in Resource Web Socket Frame View of the Network panel */ binaryMessage: 'Binary Message', /** *@description Op codes continuation frame of map in Resource Web Socket Frame View of the Network panel indicating that the web socket connection has been closed. */ connectionCloseMessage: 'Connection Close Message', /** *@description Op codes ping frame of map in Resource Web Socket Frame View of the Network panel */ pingMessage: 'Ping Message', /** *@description Op codes pong frame of map in Resource Web Socket Frame View of the Network panel */ pongMessage: 'Pong Message', /** *@description Text for everything */ all: 'All', /** *@description Text in Resource Web Socket Frame View of the Network panel */ send: 'Send', /** *@description Text in Resource Web Socket Frame View of the Network panel */ receive: 'Receive', /** *@description Text for something not available */ na: 'N/A', /** *@description Example for placeholder text */ enterRegex: 'Enter regex, for example: (web)?socket', }; const str_ = i18n.i18n.registerUIStrings('panels/network/ResourceWebSocketFrameView.ts', UIStrings); const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_); const i18nLazyString = i18n.i18n.getLazilyComputedLocalizedString.bind(undefined, str_); export class ResourceWebSocketFrameView extends UI.Widget.VBox { _request; _splitWidget; _dataGrid; _timeComparator; _mainToolbar; _clearAllButton; _filterTypeCombobox; _filterType; _filterTextInput; _filterRegex; _frameEmptyWidget; _selectedNode; _currentSelectedNode; messageFilterSetting = Common.Settings.Settings.instance().createSetting('networkWebSocketMessageFilter', ''); constructor(request) { super(); this.registerRequiredCSS('panels/network/webSocketFrameView.css'); this.element.classList.add('websocket-frame-view'); this._request = request; this._splitWidget = new UI.SplitWidget.SplitWidget(false, true, 'resourceWebSocketFrameSplitViewState'); this._splitWidget.show(this.element); const columns = [ { id: 'data', title: i18nString(UIStrings.data), sortable: false, weight: 88 }, { id: 'length', title: i18nString(UIStrings.length), sortable: false, align: DataGrid.DataGrid.Align.Right, weight: 5, }, { id: 'time', title: i18nString(UIStrings.time), sortable: true, weight: 7 }, ]; this._dataGrid = new DataGrid.SortableDataGrid.SortableDataGrid({ displayName: i18nString(UIStrings.webSocketFrame), columns, editCallback: undefined, deleteCallback: undefined, refreshCallback: undefined, }); this._dataGrid.setRowContextMenuCallback(onRowContextMenu.bind(this)); this._dataGrid.setStickToBottom(true); this._dataGrid.setCellClass('websocket-frame-view-td'); this._timeComparator = 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, event => { this._onFrameSelected(event); }, this); this._dataGrid.addEventListener(DataGrid.DataGrid.Events.DeselectedNode, this._onFrameDeselected, this); this._mainToolbar = new UI.Toolbar.Toolbar(''); this._clearAllButton = new UI.Toolbar.ToolbarButton(i18nString(UIStrings.clearAll), 'largeicon-clear'); this._clearAllButton.addEventListener(UI.Toolbar.ToolbarButton.Events.Click, this._clearFrames, this); this._mainToolbar.appendToolbarItem(this._clearAllButton); this._filterTypeCombobox = new UI.Toolbar.ToolbarComboBox(this._updateFilterSetting.bind(this), i18nString(UIStrings.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 = i18nString(UIStrings.enterRegex); this._filterTextInput = new UI.Toolbar.ToolbarInput(placeholder, '', 0.4); this._filterTextInput.addEventListener(UI.Toolbar.ToolbarInput.Event.TextChanged, this._updateFilterSetting, this); const filter = this.messageFilterSetting.get(); if (filter) { this._filterTextInput.setValue(filter); } this._filterRegex = null; this._mainToolbar.appendToolbarItem(this._filterTextInput); const mainContainer = new UI.Widget.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.EmptyWidget(i18nString(UIStrings.selectMessageToBrowseItsContent)); this._splitWidget.setSidebarWidget(this._frameEmptyWidget); this._selectedNode = null; if (filter) { this._applyFilter(filter); } function onRowContextMenu(contextMenu, genericNode) { const node = genericNode; const binaryView = node.binaryView(); if (binaryView) { binaryView.addCopyToContextMenu(contextMenu, i18nString(UIStrings.copyMessageD)); } else { contextMenu.clipboardSection().appendItem(i18nString(UIStrings.copyMessage), Host.InspectorFrontendHost.InspectorFrontendHostInstance.copyText.bind(Host.InspectorFrontendHost.InspectorFrontendHostInstance, node.data.data)); } contextMenu.footerSection().appendItem(i18nString(UIStrings.clearAllL), this._clearFrames.bind(this)); } } static opCodeDescription(opCode, mask) { const localizedDescription = opCodeDescriptions[opCode] || (() => ''); if (mask) { return i18nString(UIStrings.sOpcodeSMask, { PH1: localizedDescription(), PH2: opCode }); } return i18nString(UIStrings.sOpcodeS, { PH1: localizedDescription(), PH2: opCode }); } wasShown() { this.refresh(); this._request.addEventListener(SDK.NetworkRequest.Events.WebsocketFrameAdded, this._frameAdded, this); } willHide() { this._request.removeEventListener(SDK.NetworkRequest.Events.WebsocketFrameAdded, this._frameAdded, this); } _frameAdded(event) { const frame = event.data; if (!this._frameFilter(frame)) { return; } this._dataGrid.insertChild(new ResourceWebSocketFrameNode(this._request.url(), frame)); } _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. _clearFrameOffsets.set(this._request, this._request.frames().length); this.refresh(); } _updateFilterSetting() { const text = this._filterTextInput.value(); this.messageFilterSetting.set(text); this._applyFilter(text); } _applyFilter(text) { const type = this._filterTypeCombobox.selectedOption().value; this._filterRegex = text ? new RegExp(text, 'i') : null; this._filterType = type === 'all' ? null : type; this.refresh(); } async _onFrameSelected(event) { this._currentSelectedNode = event.data; const content = this._currentSelectedNode.dataText(); const binaryView = this._currentSelectedNode.binaryView(); if (binaryView) { this._splitWidget.setSidebarWidget(binaryView); return; } const jsonView = await SourceFrame.JSONView.JSONView.createView(content); if (jsonView) { this._splitWidget.setSidebarWidget(jsonView); return; } this._splitWidget.setSidebarWidget(new SourceFrame.ResourceSourceFrame.ResourceSourceFrame(TextUtils.StaticContentProvider.StaticContentProvider.fromString(this._request.url(), Common.ResourceType.resourceTypes.WebSocket, content))); } _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 = _clearFrameOffsets.get(this._request) || 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()); } } // TODO(crbug.com/1167717): Make this a const enum again // eslint-disable-next-line rulesdir/const_enum export var OpCodes; (function (OpCodes) { OpCodes[OpCodes["ContinuationFrame"] = 0] = "ContinuationFrame"; OpCodes[OpCodes["TextFrame"] = 1] = "TextFrame"; OpCodes[OpCodes["BinaryFrame"] = 2] = "BinaryFrame"; OpCodes[OpCodes["ConnectionCloseFrame"] = 8] = "ConnectionCloseFrame"; OpCodes[OpCodes["PingFrame"] = 9] = "PingFrame"; OpCodes[OpCodes["PongFrame"] = 10] = "PongFrame"; })(OpCodes || (OpCodes = {})); export const opCodeDescriptions = (function () { const opCodes = OpCodes; const map = []; map[opCodes.ContinuationFrame] = i18nLazyString(UIStrings.continuationFrame); map[opCodes.TextFrame] = i18nLazyString(UIStrings.textMessage); map[opCodes.BinaryFrame] = i18nLazyString(UIStrings.binaryMessage); map[opCodes.ConnectionCloseFrame] = i18nLazyString(UIStrings.connectionCloseMessage); map[opCodes.PingFrame] = i18nLazyString(UIStrings.pingMessage); map[opCodes.PongFrame] = i18nLazyString(UIStrings.pongMessage); return map; })(); // TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration) // eslint-disable-next-line @typescript-eslint/naming-convention export const _filterTypes = [ { name: 'all', label: i18nLazyString(UIStrings.all), title: undefined }, { name: 'send', label: i18nLazyString(UIStrings.send), title: undefined }, { name: 'receive', label: i18nLazyString(UIStrings.receive), title: undefined }, ]; export class ResourceWebSocketFrameNode extends DataGrid.SortableDataGrid.SortableDataGridNode { _url; _frame; _isTextFrame; _dataText; _binaryView; constructor(url, frame) { let length = String(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 = document.createElement('div'); UI.UIUtils.createTextChild(timeNode, timeText); UI.Tooltip.Tooltip.install(timeNode, 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 = i18nString(UIStrings.na); } else if (isTextFrame) { description = dataText; } else if (frame.opCode === OpCodes.BinaryFrame) { length = Platform.NumberUtilities.bytesToString(Platform.StringUtilities.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; this._binaryView = null; } 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); } nodeSelfHeight() { return 21; } dataText() { return this._dataText; } opCode() { return this._frame.opCode; } binaryView() { if (this._isTextFrame || this._frame.type === SDK.NetworkRequest.WebSocketFrameType.Error) { return null; } if (!this._binaryView) { if (this._dataText.length > 0) { this._binaryView = new BinaryResourceView(this._dataText, /* url */ '', Common.ResourceType.resourceTypes.WebSocket); } } return this._binaryView; } } // TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration) // eslint-disable-next-line @typescript-eslint/naming-convention export function ResourceWebSocketFrameNodeTimeComparator(a, b) { return a._frame.time - b._frame.time; } // TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration) // eslint-disable-next-line @typescript-eslint/naming-convention const _clearFrameOffsets = new WeakMap();