blob: 7e31a3fdb2399f3a5065185e7ad007ef418d533a [file] [log] [blame]
// Copyright 2013 The Flutter Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// @dart = 2.6
import 'dart:async';
import 'package:args/command_runner.dart';
import 'package:meta/meta.dart';
import 'package:path/path.dart' as path;
import 'package:watcher/watcher.dart';
import 'environment.dart';
import 'utils.dart';
class BuildCommand extends Command<bool> with ArgUtils {
BuildCommand() {
argParser
..addFlag(
'watch',
abbr: 'w',
help: 'Run the build in watch mode so it rebuilds whenever a change'
'is made.',
)
..addOption(
'ninja-jobs',
abbr: 'j',
help: 'Number of parallel jobs to use in the ninja build.',
);
}
@override
String get name => 'build';
@override
String get description => 'Build the Flutter web engine.';
bool get isWatchMode => boolArg('watch');
int getNinjaJobCount() => intArg('ninja-jobs');
@override
FutureOr<bool> run() async {
final int ninjaJobCount = getNinjaJobCount();
final FilePath libPath = FilePath.fromWebUi('lib');
final Pipeline buildPipeline = Pipeline(steps: <PipelineStep>[
gn,
() => ninja(ninjaJobCount),
]);
await buildPipeline.start();
if (isWatchMode) {
print('Initial build done!');
print('Watching directory: ${libPath.relativeToCwd}/');
PipelineWatcher(
dir: libPath.absolute,
pipeline: buildPipeline,
// Ignore font files that are copied whenever tests run.
ignore: (event) => event.path.endsWith('.ttf'),
).start();
// Return a never-ending future.
return Completer<bool>().future;
} else {
return true;
}
}
}
Future<void> gn() {
print('Running gn...');
return runProcess(
path.join(environment.flutterDirectory.path, 'tools', 'gn'),
<String>[
'--unopt',
'--full-dart-sdk',
],
);
}
// TODO(mdebbar): Make the ninja step interruptable in the pipeline.
Future<void> ninja(int ninjaJobs) {
if (ninjaJobs == null) {
print('Running ninja (with default ninja parallelization)...');
} else {
print('Running ninja (with $ninjaJobs parallel jobs)...');
}
return runProcess('ninja', <String>[
'-C',
environment.hostDebugUnoptDir.path,
if (ninjaJobs != null) ...['-j', '$ninjaJobs'],
]);
}
enum PipelineStatus {
idle,
started,
stopping,
stopped,
error,
done,
}
typedef PipelineStep = Future<void> Function();
class Pipeline {
Pipeline({@required this.steps});
final Iterable<PipelineStep> steps;
Future<dynamic> _currentStepFuture;
PipelineStatus status = PipelineStatus.idle;
Future<void> start() async {
status = PipelineStatus.started;
try {
for (PipelineStep step in steps) {
if (status != PipelineStatus.started) {
break;
}
_currentStepFuture = step();
await _currentStepFuture;
}
status = PipelineStatus.done;
} catch (error, stackTrace) {
status = PipelineStatus.error;
print('Error in the pipeline: $error');
print(stackTrace);
} finally {
_currentStepFuture = null;
}
}
Future<void> stop() {
status = PipelineStatus.stopping;
return (_currentStepFuture ?? Future<void>.value(null)).then((_) {
status = PipelineStatus.stopped;
});
}
}
typedef WatchEventPredicate = bool Function(WatchEvent event);
class PipelineWatcher {
PipelineWatcher({
@required this.dir,
@required this.pipeline,
this.ignore,
}) : watcher = DirectoryWatcher(dir);
/// The path of the directory to watch for changes.
final String dir;
/// The pipeline to be executed when an event is fired by the watcher.
final Pipeline pipeline;
/// Used to watch a directory for any file system changes.
final DirectoryWatcher watcher;
/// A callback that determines whether to rerun the pipeline or not for a
/// given [WatchEvent] instance.
final WatchEventPredicate ignore;
void start() {
watcher.events.listen(_onEvent);
}
int _pipelineRunCount = 0;
Timer _scheduledPipeline;
void _onEvent(WatchEvent event) {
if (ignore != null && ignore(event)) {
return;
}
final String relativePath = path.relative(event.path, from: dir);
print('- [${event.type}] ${relativePath}');
_pipelineRunCount++;
_scheduledPipeline?.cancel();
_scheduledPipeline = Timer(const Duration(milliseconds: 100), () {
_scheduledPipeline = null;
_runPipeline();
});
}
void _runPipeline() {
int runCount;
switch (pipeline.status) {
case PipelineStatus.started:
pipeline.stop().then((_) {
runCount = _pipelineRunCount;
pipeline.start().then((_) => _pipelineDone(runCount));
});
break;
case PipelineStatus.stopping:
// We are already trying to stop the pipeline. No need to do anything.
break;
default:
runCount = _pipelineRunCount;
pipeline.start().then((_) => _pipelineDone(runCount));
break;
}
}
void _pipelineDone(int pipelineRunCount) {
if (pipelineRunCount == _pipelineRunCount) {
print('*** Done! ***');
}
}
}