Summary: It makes more sense for the navigation ability to be inside the Device's class instead of scattered throughout the rest of the app. I have moved the Android navigation logic inside the AndroidDevice and added a function stub to BaseDevice. I also encoded the URI as an initial safeguard to injection attacks via adb, but will remove this altogether once I enable navigation through the socket connection. Reviewed By: jknoxville Differential Revision: D16182374 fbshipit-source-id: be3c6d1cfcbe293583edada1f77c023965dfd12c
106 lines
2.6 KiB
JavaScript
106 lines
2.6 KiB
JavaScript
/**
|
|
* Copyright 2018-present Facebook.
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
* @format
|
|
*/
|
|
|
|
import type {DeviceType, DeviceShell} from './BaseDevice.js';
|
|
|
|
import {Priority} from 'adbkit-logcat-fb';
|
|
import child_process from 'child_process';
|
|
import child_process_promise from 'child-process-es6-promise';
|
|
import BaseDevice from './BaseDevice.js';
|
|
import ArchivedDevice from './ArchivedDevice.js';
|
|
|
|
type ADBClient = any;
|
|
|
|
export default class AndroidDevice extends BaseDevice {
|
|
constructor(
|
|
serial: string,
|
|
deviceType: DeviceType,
|
|
title: string,
|
|
adb: ADBClient,
|
|
) {
|
|
super(serial, deviceType, title);
|
|
this.adb = adb;
|
|
|
|
this.adb.openLogcat(this.serial).then(reader => {
|
|
reader.on('entry', entry => {
|
|
let type = 'unknown';
|
|
if (entry.priority === Priority.VERBOSE) {
|
|
type = 'verbose';
|
|
}
|
|
if (entry.priority === Priority.DEBUG) {
|
|
type = 'debug';
|
|
}
|
|
if (entry.priority === Priority.INFO) {
|
|
type = 'info';
|
|
}
|
|
if (entry.priority === Priority.WARN) {
|
|
type = 'warn';
|
|
}
|
|
if (entry.priority === Priority.ERROR) {
|
|
type = 'error';
|
|
}
|
|
if (entry.priority === Priority.FATAL) {
|
|
type = 'fatal';
|
|
}
|
|
|
|
this.addLogEntry({
|
|
tag: entry.tag,
|
|
pid: entry.pid,
|
|
tid: entry.tid,
|
|
message: entry.message,
|
|
date: entry.date,
|
|
type,
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
icon = 'icons/android.svg';
|
|
os = 'Android';
|
|
adb: ADBClient;
|
|
pidAppMapping: {[key: number]: string} = {};
|
|
logReader: any;
|
|
|
|
supportedColumns(): Array<string> {
|
|
return ['date', 'pid', 'tid', 'tag', 'message', 'type', 'time'];
|
|
}
|
|
|
|
reverse(ports: [number, number]): Promise<void> {
|
|
return Promise.all(
|
|
ports.map(port =>
|
|
this.adb.reverse(this.serial, `tcp:${port}`, `tcp:${port}`),
|
|
),
|
|
).then(() => {
|
|
return;
|
|
});
|
|
}
|
|
|
|
spawnShell(): ?DeviceShell {
|
|
return child_process.spawn('adb', ['-s', this.serial, 'shell', '-t', '-t']);
|
|
}
|
|
|
|
clearLogs(): Promise<void> {
|
|
this.logEntries = [];
|
|
return child_process_promise.spawn('adb', ['logcat', '-c']);
|
|
}
|
|
|
|
archive(): ArchivedDevice {
|
|
return new ArchivedDevice(
|
|
this.serial,
|
|
this.deviceType,
|
|
this.title,
|
|
this.os,
|
|
[...this.logEntries],
|
|
);
|
|
}
|
|
|
|
navigateToLocation(location: string) {
|
|
const shellCommand = `am start ${encodeURI(location)}`;
|
|
this.adb.shell(this.serial, shellCommand);
|
|
}
|
|
}
|