UNPKG

chrome-devtools-frontend

Version:
316 lines (284 loc) • 9 kB
// Copyright 2016 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. import * as Bindings from '../bindings/bindings.js'; import * as Common from '../common/common.js'; import * as Host from '../host/host.js'; import * as i18n from '../i18n/i18n.js'; import * as SDK from '../sdk/sdk.js'; import * as TextUtils from '../text_utils/text_utils.js'; import * as TimelineModel from '../timeline_model/timeline_model.js'; export const UIStrings = { /** *@description Text in Timeline Loader of the Performance panel */ malformedTimelineDataUnknownJson: 'Malformed timeline data: Unknown JSON format', /** *@description Text in Timeline Loader of the Performance panel */ malformedTimelineInputWrongJson: 'Malformed timeline input, wrong JSON brackets balance', /** *@description Text in Timeline Loader of the Performance panel *@example {Unknown JSON format} PH1 */ malformedTimelineDataS: 'Malformed timeline data: {PH1}', /** *@description Text in Timeline Loader of the Performance panel */ legacyTimelineFormatIsNot: 'Legacy Timeline format is not supported.', /** *@description Text in Timeline Loader of the Performance panel */ malformedCpuProfileFormat: 'Malformed CPU profile format', }; const str_ = i18n.i18n.registerUIStrings('timeline/TimelineLoader.js', UIStrings); const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_); /** * @implements {Common.StringOutputStream.OutputStream} */ export class TimelineLoader { /** * @param {!Client} client */ constructor(client) { /** @type {?Client} */ this._client = client; this._backingStorage = new Bindings.TempFile.TempFileBackingStorage(); /** @type {?SDK.TracingModel.TracingModel} */ this._tracingModel = new SDK.TracingModel.TracingModel(this._backingStorage); /** @type {?function():void} */ this._canceledCallback = null; this._state = State.Initial; this._buffer = ''; this._firstRawChunk = true; this._firstChunk = true; this._loadedBytes = 0; /** @type {number} */ this._totalSize; this._jsonTokenizer = new TextUtils.TextUtils.BalancedJSONTokenizer(this._writeBalancedJSON.bind(this), true); } /** * @param {!File} file * @param {!Client} client * @return {!TimelineLoader} */ static loadFromFile(file, client) { const loader = new TimelineLoader(client); const fileReader = new Bindings.FileUtils.ChunkedFileReader(file, TransferChunkLengthBytes); loader._canceledCallback = fileReader.cancel.bind(fileReader); loader._totalSize = file.size; fileReader.read(loader).then(success => { if (!success && fileReader.error()) { loader._reportErrorAndCancelLoading(/** @type {*} */ (fileReader.error()).message); } }); return loader; } /** * @param {!Array.<!SDK.TracingManager.EventPayload>} events * @param {!Client} client * @return {!TimelineLoader} */ static loadFromEvents(events, client) { const loader = new TimelineLoader(client); setTimeout(async () => { const eventsPerChunk = 5000; client.loadingStarted(); for (let i = 0; i < events.length; i += eventsPerChunk) { const chunk = events.slice(i, i + eventsPerChunk); /** @type {!SDK.TracingModel.TracingModel} */ (loader._tracingModel).addEvents(chunk); client.loadingProgress((i + chunk.length) / events.length); await new Promise(r => setTimeout(r)); // Yield event loop to paint. } loader.close(); }); return loader; } /** * @param {string} url * @param {!Client} client * @return {!TimelineLoader} */ static loadFromURL(url, client) { const loader = new TimelineLoader(client); Host.ResourceLoader.loadAsStream(url, null, loader); return loader; } cancel() { this._tracingModel = null; this._backingStorage.reset(); if (this._client) { this._client.loadingComplete(null); this._client = null; } if (this._canceledCallback) { this._canceledCallback(); } } /** * @override * @param {string} chunk * @return {!Promise<void>} */ write(chunk) { if (!this._client) { return Promise.resolve(); } this._loadedBytes += chunk.length; if (this._firstRawChunk) { this._client.loadingStarted(); } else { this._client.loadingProgress(this._totalSize ? this._loadedBytes / this._totalSize : undefined); } this._firstRawChunk = false; if (this._state === State.Initial) { if (chunk.startsWith('{"nodes":[')) { this._state = State.LoadingCPUProfileFormat; } else if (chunk[0] === '{') { this._state = State.LookingForEvents; } else if (chunk[0] === '[') { this._state = State.ReadingEvents; } else { this._reportErrorAndCancelLoading(i18nString(UIStrings.malformedTimelineDataUnknownJson)); return Promise.resolve(); } } if (this._state === State.LoadingCPUProfileFormat) { this._buffer += chunk; return Promise.resolve(); } if (this._state === State.LookingForEvents) { const objectName = '"traceEvents":'; const startPos = this._buffer.length - objectName.length; this._buffer += chunk; const pos = this._buffer.indexOf(objectName, startPos); if (pos === -1) { return Promise.resolve(); } chunk = this._buffer.slice(pos + objectName.length); this._state = State.ReadingEvents; } if (this._state !== State.ReadingEvents) { return Promise.resolve(); } if (this._jsonTokenizer.write(chunk)) { return Promise.resolve(); } this._state = State.SkippingTail; if (this._firstChunk) { this._reportErrorAndCancelLoading(i18nString(UIStrings.malformedTimelineInputWrongJson)); } return Promise.resolve(); } /** * @param {string} data */ _writeBalancedJSON(data) { let json = data + ']'; if (!this._firstChunk) { const commaIndex = json.indexOf(','); if (commaIndex !== -1) { json = json.slice(commaIndex + 1); } json = '[' + json; } let items; try { items = /** @type {!Array.<!SDK.TracingManager.EventPayload>} */ (JSON.parse(json)); } catch (e) { this._reportErrorAndCancelLoading(i18nString(UIStrings.malformedTimelineDataS, {PH1: e.toString()})); return; } if (this._firstChunk) { this._firstChunk = false; if (this._looksLikeAppVersion(items[0])) { this._reportErrorAndCancelLoading(i18nString(UIStrings.legacyTimelineFormatIsNot)); return; } } try { /** @type {!SDK.TracingModel.TracingModel} */ (this._tracingModel).addEvents(items); } catch (e) { this._reportErrorAndCancelLoading(i18nString(UIStrings.malformedTimelineDataS, {PH1: e.toString()})); } } /** * @param {string=} message */ _reportErrorAndCancelLoading(message) { if (message) { Common.Console.Console.instance().error(message); } this.cancel(); } /** * @param {*} item * @return {boolean} */ _looksLikeAppVersion(item) { return typeof item === 'string' && item.indexOf('Chrome') !== -1; } /** * @override */ async close() { if (!this._client) { return; } this._client.processingStarted(); setTimeout(() => this._finalizeTrace(), 0); } _finalizeTrace() { if (this._state === State.LoadingCPUProfileFormat) { this._parseCPUProfileFormat(this._buffer); this._buffer = ''; } /** @type {!SDK.TracingModel.TracingModel} */ (this._tracingModel).tracingComplete(); /** @type {!Client} */ (this._client).loadingComplete(this._tracingModel); } /** * @param {string} text */ _parseCPUProfileFormat(text) { let traceEvents; try { const profile = JSON.parse(text); traceEvents = TimelineModel.TimelineJSProfile.TimelineJSProfileProcessor.buildTraceProfileFromCpuProfile( profile, /* tid */ 1, /* injectPageEvent */ true); } catch (e) { this._reportErrorAndCancelLoading(i18nString(UIStrings.malformedCpuProfileFormat)); return; } /** @type {!SDK.TracingModel.TracingModel} */ (this._tracingModel).addEvents(traceEvents); } } export const TransferChunkLengthBytes = 5000000; /** * @interface */ export class Client { loadingStarted() { } /** * @param {number=} progress */ loadingProgress(progress) { } processingStarted() { } /** * @param {?SDK.TracingModel.TracingModel} tracingModel */ loadingComplete(tracingModel) { } } /** * @enum {symbol} */ export const State = { Initial: Symbol('Initial'), LookingForEvents: Symbol('LookingForEvents'), ReadingEvents: Symbol('ReadingEvents'), SkippingTail: Symbol('SkippingTail'), LoadingCPUProfileFormat: Symbol('LoadingCPUProfileFormat') };