// Copyright 2015 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 'dart:async'; import 'package:logging/logging.dart'; import 'flutter_command.dart'; import '../application_package.dart'; import '../device.dart'; final Logger _logging = new Logger('sky_tools.trace'); class TraceCommand extends FlutterCommand { final String name = 'trace'; final String description = 'Start and stop tracing a running Flutter app ' '(Android only, requires root).\n' 'To start a trace, wait, and then stop the trace, don\'t set any flags ' 'except (optionally) duration.\n' 'Otherwise, specify either start or stop to manually control the trace.'; TraceCommand() { argParser.addFlag('start', negatable: false, help: 'Start tracing.'); argParser.addFlag('stop', negatable: false, help: 'Stop tracing.'); argParser.addOption('duration', defaultsTo: '10', abbr: 'd', help: 'Duration in seconds to trace.'); } @override Future<int> run() async { await downloadApplicationPackagesAndConnectToDevices(); if (!devices.android.isConnected()) { _logging.warning('No device connected, so no trace was completed.'); return 1; } ApplicationPackage androidApp = applicationPackages.android; if ((!argResults['start'] && !argResults['stop']) || (argResults['start'] && argResults['stop'])) { // Setting neither flags or both flags means do both commands and wait // duration seconds in between. devices.android.startTracing(androidApp); await new Future.delayed( new Duration(seconds: int.parse(argResults['duration'])), () => _stopTracing(devices.android, androidApp)); } else if (argResults['stop']) { _stopTracing(devices.android, androidApp); } else { devices.android.startTracing(androidApp); } return 0; } void _stopTracing(AndroidDevice android, AndroidApk androidApp) { String tracePath = android.stopTracing(androidApp); if (tracePath == null) { _logging.warning('No trace file saved.'); } else { print('Trace file saved to $tracePath'); } } }