Commit 073ac691 authored by Devon Carew's avatar Devon Carew

Merge pull request #1329 from devoncarew/refactor_commands

refactor commands to move out re-usable functionality
parents b5a09839 5daf58ce
...@@ -302,7 +302,7 @@ class ApkCommand extends FlutterCommand { ...@@ -302,7 +302,7 @@ class ApkCommand extends FlutterCommand {
await downloadToolchain(); await downloadToolchain();
// Find the path to the main Dart file. // Find the path to the main Dart file.
String mainPath = StartCommandBase.findMainDartFile(argResults['target']); String mainPath = findMainDartFile(argResults['target']);
// Build the FLX. // Build the FLX.
int result; int result;
......
...@@ -11,10 +11,12 @@ import '../base/logging.dart'; ...@@ -11,10 +11,12 @@ import '../base/logging.dart';
import '../device.dart'; import '../device.dart';
import '../runner/flutter_command.dart'; import '../runner/flutter_command.dart';
import 'start.dart'; import 'start.dart';
import 'stop.dart'; import 'stop.dart' as stop;
const String protocolVersion = '0.0.2'; const String protocolVersion = '0.0.2';
// TODO(devoncarew): Pass logging data back to the client.
/// A server process command. This command will start up a long-lived server. /// A server process command. This command will start up a long-lived server.
/// It reads JSON-RPC based commands from stdin, executes them, and returns /// It reads JSON-RPC based commands from stdin, executes them, and returns
/// JSON-RPC based responses and events to stdout. /// JSON-RPC based responses and events to stdout.
...@@ -132,6 +134,8 @@ abstract class Domain { ...@@ -132,6 +134,8 @@ abstract class Domain {
_handlers[name] = handler; _handlers[name] = handler;
} }
FlutterCommand get command => daemon.daemonCommand;
String toString() => name; String toString() => name;
void handleCommand(String name, dynamic id, dynamic args) { void handleCommand(String name, dynamic id, dynamic args) {
...@@ -190,17 +194,23 @@ class AppDomain extends Domain { ...@@ -190,17 +194,23 @@ class AppDomain extends Domain {
registerHandler('stopAll', stopAll); registerHandler('stopAll', stopAll);
} }
Future<dynamic> start(dynamic args) { Future<dynamic> start(dynamic args) async {
// TODO: Add the ability to pass args: target, http, checked // TODO: Add the ability to pass args: target, http, checked
StartCommand startComand = new StartCommand();
startComand.inheritFromParent(daemon.daemonCommand); await Future.wait([
return startComand.runInProject().then((_) => null); command.downloadToolchain(),
command.downloadApplicationPackagesAndConnectToDevices(),
], eagerError: true);
return startApp(
command.devices,
command.applicationPackages,
command.toolchain
).then((int result) => null);
} }
Future<bool> stopAll(dynamic args) { Future<bool> stopAll(dynamic args) {
StopCommand stopCommand = new StopCommand(); return stop.stopAll(command.devices, command.applicationPackages);
stopCommand.inheritFromParent(daemon.daemonCommand);
return stopCommand.stop();
} }
} }
......
...@@ -13,17 +13,26 @@ class InstallCommand extends FlutterCommand { ...@@ -13,17 +13,26 @@ class InstallCommand extends FlutterCommand {
final String description = 'Install Flutter apps on attached devices.'; final String description = 'Install Flutter apps on attached devices.';
InstallCommand() { InstallCommand() {
argParser.addFlag('boot', argParser.addFlag('boot', help: 'Boot the iOS Simulator if it isn\'t already running.');
help: 'Boot the iOS Simulator if it isn\'t already running.');
} }
@override @override
Future<int> runInProject() async { Future<int> runInProject() async {
await downloadApplicationPackagesAndConnectToDevices(); await downloadApplicationPackagesAndConnectToDevices();
return install(boot: argResults['boot']) ? 0 : 2; bool installedAny = installApp(
devices,
applicationPackages,
boot: argResults['boot']
);
return installedAny ? 0 : 2;
} }
}
bool install({ bool boot: false }) { bool installApp(
DeviceStore devices,
ApplicationPackageStore applicationPackages, {
bool boot: false
}) {
if (boot) if (boot)
devices.iOSSimulator?.boot(); devices.iOSSimulator?.boot();
...@@ -38,5 +47,4 @@ class InstallCommand extends FlutterCommand { ...@@ -38,5 +47,4 @@ class InstallCommand extends FlutterCommand {
} }
return installedSomewhere; return installedSomewhere;
}
} }
...@@ -18,6 +18,8 @@ class ListCommand extends FlutterCommand { ...@@ -18,6 +18,8 @@ class ListCommand extends FlutterCommand {
help: 'Log additional details about attached devices.'); help: 'Log additional details about attached devices.');
} }
bool get requiresProjectRoot => false;
@override @override
Future<int> runInProject() async { Future<int> runInProject() async {
connectToDevices(); connectToDevices();
......
...@@ -34,7 +34,17 @@ class ListenCommand extends StartCommandBase { ...@@ -34,7 +34,17 @@ class ListenCommand extends StartCommandBase {
bool firstTime = true; bool firstTime = true;
do { do {
logging.info('Updating running Flutter apps...'); logging.info('Updating running Flutter apps...');
result = await startApp(install: firstTime, stop: true); result = await startApp(
devices,
applicationPackages,
toolchain,
target: argResults['target'],
install: firstTime,
stop: true,
checked: argResults['checked'],
traceStartup: argResults['trace-startup'],
route: argResults['route']
);
firstTime = false; firstTime = false;
} while (!singleRun && result == 0 && _watchDirectory(watchCommand)); } while (!singleRun && result == 0 && _watchDirectory(watchCommand));
return 0; return 0;
......
...@@ -154,7 +154,7 @@ class RunMojoCommand extends FlutterCommand { ...@@ -154,7 +154,7 @@ class RunMojoCommand extends FlutterCommand {
if (bundlePath == null) { if (bundlePath == null) {
bundlePath = _kDefaultBundlePath; bundlePath = _kDefaultBundlePath;
String mainPath = StartCommandBase.findMainDartFile(argResults['target']); String mainPath = findMainDartFile(argResults['target']);
int result = await flx.build( int result = await flx.build(
toolchain, toolchain,
......
...@@ -13,9 +13,23 @@ import '../build_configuration.dart'; ...@@ -13,9 +13,23 @@ import '../build_configuration.dart';
import '../device.dart'; import '../device.dart';
import '../flx.dart' as flx; import '../flx.dart' as flx;
import '../runner/flutter_command.dart'; import '../runner/flutter_command.dart';
import '../toolchain.dart';
import 'install.dart'; import 'install.dart';
import 'stop.dart'; import 'stop.dart';
/// Given the value of the --target option, return the path of the Dart file
/// where the app's main function should be.
String findMainDartFile([String target]) {
if (target == null)
target = '';
String targetPath = path.absolute(target);
if (FileSystemEntity.isDirectorySync(targetPath)) {
return path.join(targetPath, 'lib', 'main.dart');
} else {
return targetPath;
}
}
// We don't yet support iOS here. https://github.com/flutter/flutter/issues/1036 // We don't yet support iOS here. https://github.com/flutter/flutter/issues/1036
abstract class StartCommandBase extends FlutterCommand { abstract class StartCommandBase extends FlutterCommand {
...@@ -29,52 +43,90 @@ abstract class StartCommandBase extends FlutterCommand { ...@@ -29,52 +43,90 @@ abstract class StartCommandBase extends FlutterCommand {
defaultsTo: false, defaultsTo: false,
help: 'Start tracing during startup.'); help: 'Start tracing during startup.');
argParser.addOption('target', argParser.addOption('target',
defaultsTo: '',
abbr: 't', abbr: 't',
help: 'Target app path or filename to start.'); help: 'Target app path or filename to start.');
argParser.addOption('route', argParser.addOption('route',
help: 'Which route to load when starting the app.'); help: 'Which route to load when starting the app.');
} }
}
/// Given the value of the --target option, return the path of the Dart file class StartCommand extends StartCommandBase {
/// where the app's main function should be. final String name = 'start';
static String findMainDartFile(String target) { final String description = 'Start your Flutter app on an attached device '
String targetPath = path.absolute(target); '(defaults to checked/debug mode) (Android only).';
if (FileSystemEntity.isDirectorySync(targetPath)) {
return path.join(targetPath, 'lib', 'main.dart'); StartCommand() {
} else { argParser.addFlag('poke',
return targetPath; negatable: false,
help: 'Restart the connection to the server.');
argParser.addFlag('clear-logs',
defaultsTo: true,
help: 'Clear log history before starting the app.');
} }
@override
Future<int> runInProject() async {
logging.fine('Downloading toolchain.');
await Future.wait([
downloadToolchain(),
downloadApplicationPackagesAndConnectToDevices(),
], eagerError: true);
bool poke = argResults['poke'];
bool clearLogs = argResults['clear-logs'];
// Only stop and reinstall if the user did not specify a poke.
int result = await startApp(
devices,
applicationPackages,
toolchain,
target: argResults['target'],
install: !poke,
stop: !poke,
checked: argResults['checked'],
traceStartup: argResults['trace-startup'],
route: argResults['route'],
poke: poke,
clearLogs: clearLogs
);
logging.fine('Finished start command.');
return result;
} }
}
Future<int> startApp({ Future<int> startApp(
DeviceStore devices,
ApplicationPackageStore applicationPackages,
Toolchain toolchain, {
String target,
bool stop: true, bool stop: true,
bool install: true, bool install: true,
bool checked: true,
bool traceStartup: false,
String route,
bool poke: false, bool poke: false,
bool clearLogs: false bool clearLogs: false
}) async { }) async {
String mainPath = findMainDartFile(argResults['target']); String mainPath = findMainDartFile(target);
if (!FileSystemEntity.isFileSync(mainPath)) { if (!FileSystemEntity.isFileSync(mainPath)) {
String message = 'Tried to run $mainPath, but that file does not exist.'; String message = 'Tried to run $mainPath, but that file does not exist.';
if (!argResults.wasParsed('target')) if (target == null)
message += '\nConsider using the -t option to specify that Dart file to start.'; message += '\nConsider using the -t option to specify the Dart file to start.';
logging.severe(message); logging.severe(message);
return 1; return 1;
} }
if (stop) { if (stop) {
logging.fine('Running stop command.'); logging.fine('Running stop command.');
StopCommand stopper = new StopCommand(); stopAll(devices, applicationPackages);
stopper.inheritFromParent(this);
stopper.stop();
} }
if (install) { if (install) {
logging.fine('Running install command.'); logging.fine('Running install command.');
InstallCommand installer = new InstallCommand(); installApp(devices, applicationPackages);
installer.inheritFromParent(this);
installer.install();
} }
bool startedSomething = false; bool startedSomething = false;
...@@ -95,9 +147,9 @@ abstract class StartCommandBase extends FlutterCommand { ...@@ -95,9 +147,9 @@ abstract class StartCommandBase extends FlutterCommand {
final AndroidDevice androidDevice = device; // https://github.com/flutter/flutter/issues/1035 final AndroidDevice androidDevice = device; // https://github.com/flutter/flutter/issues/1035
if (androidDevice.startBundle(package, localBundlePath, if (androidDevice.startBundle(package, localBundlePath,
poke: poke, poke: poke,
checked: argResults['checked'], checked: checked,
traceStartup: argResults['trace-startup'], traceStartup: traceStartup,
route: argResults['route'], route: route,
clearLogs: clearLogs clearLogs: clearLogs
)) { )) {
startedSomething = true; startedSomething = true;
...@@ -123,39 +175,4 @@ abstract class StartCommandBase extends FlutterCommand { ...@@ -123,39 +175,4 @@ abstract class StartCommandBase extends FlutterCommand {
} }
return startedSomething ? 0 : 2; return startedSomething ? 0 : 2;
}
}
class StartCommand extends StartCommandBase {
final String name = 'start';
final String description = 'Start your Flutter app on an attached device '
'(defaults to checked/debug mode) (Android only).';
StartCommand() {
argParser.addFlag('poke',
negatable: false,
help: 'Restart the connection to the server.');
argParser.addFlag('clear-logs',
defaultsTo: true,
help: 'Clear log history before starting the app.');
}
@override
Future<int> runInProject() async {
logging.fine('Downloading toolchain.');
await Future.wait([
downloadToolchain(),
downloadApplicationPackagesAndConnectToDevices(),
], eagerError: true);
bool poke = argResults['poke'];
bool clearLogs = argResults['clear-logs'];
// Only stop and reinstall if the user did not specify a poke
int result = await startApp(stop: !poke, install: !poke, poke: poke, clearLogs: clearLogs);
logging.fine('Finished start command.');
return result;
}
} }
...@@ -18,7 +18,10 @@ class StopCommand extends FlutterCommand { ...@@ -18,7 +18,10 @@ class StopCommand extends FlutterCommand {
return await stop() ? 0 : 2; return await stop() ? 0 : 2;
} }
Future<bool> stop() async { Future<bool> stop() => stopAll(devices, applicationPackages);
}
Future<bool> stopAll(DeviceStore devices, ApplicationPackageStore applicationPackages) async {
bool stoppedSomething = false; bool stoppedSomething = false;
for (Device device in devices.all) { for (Device device in devices.all) {
...@@ -30,5 +33,4 @@ class StopCommand extends FlutterCommand { ...@@ -30,5 +33,4 @@ class StopCommand extends FlutterCommand {
} }
return stoppedSomething; return stoppedSomething;
}
} }
...@@ -310,7 +310,7 @@ class IOSSimulator extends Device { ...@@ -310,7 +310,7 @@ class IOSSimulator extends Device {
String output = runCheckedSync([xcrunPath, 'simctl', 'list', 'devices']); String output = runCheckedSync([xcrunPath, 'simctl', 'list', 'devices']);
Match match; Match match;
/// iPhone 6s Plus (8AC808E1-6BAE-4153-BBC5-77F83814D414) (Booted) // iPhone 6s Plus (8AC808E1-6BAE-4153-BBC5-77F83814D414) (Booted)
Iterable<Match> matches = new RegExp( Iterable<Match> matches = new RegExp(
r'[\W]*(.*) \(([^\)]+)\) \(Booted\)', r'[\W]*(.*) \(([^\)]+)\) \(Booted\)',
multiLine: true multiLine: true
......
...@@ -47,12 +47,6 @@ abstract class FlutterCommand extends Command { ...@@ -47,12 +47,6 @@ abstract class FlutterCommand extends Command {
connectToDevices(); connectToDevices();
} }
void inheritFromParent(FlutterCommand other) {
applicationPackages = other.applicationPackages;
toolchain = other.toolchain;
devices = other.devices;
}
Future<int> run() async { Future<int> run() async {
if (requiresProjectRoot && !validateProjectRoot()) if (requiresProjectRoot && !validateProjectRoot())
return 1; return 1;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment