blob: aa23f8f7e2e6cc6672c6c758c1938cc4998a65c9 [file] [log] [blame]
// Copyright (C) 2019 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import * as rpc from 'noice-json-rpc';
// To really understand how this works it is useful to see the implementation
// of noice-json-rpc.
export class DevToolsSocket implements rpc.LikeSocket {
private messageCallback: Function = (_: string) => {};
private openCallback: Function = () => {};
private closeCallback: Function = () => {};
private target: chrome.debugger.Debuggee|undefined;
constructor() {
chrome.debugger.onDetach.addListener(this.onDetach.bind(this));
chrome.debugger.onEvent.addListener((_source, method, params) => {
if (this.messageCallback) {
const msg: rpc.JsonRpc2.Notification = {method, params};
this.messageCallback(JSON.stringify(msg));
}
});
}
send(message: string): void {
if (this.target === undefined) return;
const msg: rpc.JsonRpc2.Request = JSON.parse(message);
chrome.debugger.sendCommand(
this.target, msg.method, msg.params, (result) => {
if (result === undefined) result = {};
const response: rpc.JsonRpc2.Response = {id: msg.id, result};
this.messageCallback(JSON.stringify(response));
});
}
// This method will be called once for each event soon after the creation of
// this object. To understand better what happens, checking the implementation
// of noice-json-rpc is very useful.
// While the events "message" and "open" are for implementing the LikeSocket,
// "close" is a callback set from ChromeTracingController, to reset the state
// after a detach.
on(event: string, cb: Function) {
if (event === 'message') {
this.messageCallback = cb;
} else if (event === 'open') {
this.openCallback = cb;
} else if (event === 'close') {
this.closeCallback = cb;
}
}
removeListener(_event: string, _cb: Function) {
throw new Error('Call unexpected');
}
findTarget(then: (target: chrome.debugger.Debuggee) => void) {
chrome.debugger.getTargets(targets => {
const perfettoTab =
targets.find(target => target.title.includes('Perfetto'));
if (perfettoTab === undefined) {
console.log('No perfetto tab found');
return;
}
this.target = {targetId: perfettoTab.id};
then(this.target);
});
}
findAndAttachTarget(then: (target: chrome.debugger.Debuggee) => void) {
this.findTarget(t => {
chrome.debugger.attach(t, /*requiredVersion=*/ '1.3', () => {
this.openCallback();
then(t);
});
});
}
detach() {
if (this.target === undefined) return;
chrome.debugger.detach(this.target, () => {
this.target = undefined;
});
}
onDetach(_source: chrome.debugger.Debuggee, _reason: string) {
if (_source === this.target) {
this.target = undefined;
this.closeCallback();
}
}
isAttached(): boolean {
return this.target !== undefined;
}
}