build_system.dart 31 KB
Newer Older
Ian Hickson's avatar
Ian Hickson committed
1
// Copyright 2014 The Flutter Authors. All rights reserved.
2 3 4 5 6 7 8 9 10
// 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:async/async.dart';
import 'package:convert/convert.dart';
import 'package:crypto/crypto.dart';
import 'package:meta/meta.dart';
11
import 'package:platform/platform.dart';
12
import 'package:pool/pool.dart';
13
import 'package:process/process.dart';
14

15
import '../artifacts.dart';
16
import '../base/file_system.dart';
17
import '../base/logger.dart';
18
import '../base/utils.dart';
19 20 21 22 23 24 25 26
import '../cache.dart';
import '../convert.dart';
import 'exceptions.dart';
import 'file_hash_store.dart';
import 'source.dart';

export 'source.dart';

27 28 29
/// A reasonable amount of files to open at the same time.
///
/// This number is somewhat arbitrary - it is difficult to detect whether
30
/// or not we'll run out of file descriptors when using async dart:io
31 32 33
/// APIs.
const int kMaxOpenFiles = 64;

34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83
/// Configuration for the build system itself.
class BuildSystemConfig {
  /// Create a new [BuildSystemConfig].
  const BuildSystemConfig({this.resourcePoolSize});

  /// The maximum number of concurrent tasks the build system will run.
  ///
  /// If not provided, defaults to [platform.numberOfProcessors].
  final int resourcePoolSize;
}

/// A Target describes a single step during a flutter build.
///
/// The target inputs are required to be files discoverable via a combination
/// of at least one of the environment values and zero or more local values.
///
/// To determine if the action for a target needs to be executed, the
/// [BuildSystem] performs a hash of the file contents for both inputs and
/// outputs. This is tracked separately in the [FileHashStore].
///
/// A Target has both implicit and explicit inputs and outputs. Only the
/// later are safe to evaluate before invoking the [buildAction]. For example,
/// a wildcard output pattern requires the outputs to exist before it can
/// glob files correctly.
///
/// - All listed inputs are considered explicit inputs.
/// - Outputs which are provided as [Source.pattern].
///   without wildcards are considered explicit.
/// - The remaining outputs are considered implicit.
///
/// For each target, executing its action creates a corresponding stamp file
/// which records both the input and output files. This file is read by
/// subsequent builds to determine which file hashes need to be checked. If the
/// stamp file is missing, the target's action is always rerun.
///
///  file: `example_target.stamp`
///
/// {
///   "inputs": [
///      "absolute/path/foo",
///      "absolute/path/bar",
///      ...
///    ],
///    "outputs": [
///      "absolute/path/fizz"
///    ]
/// }
///
/// ## Code review
///
84
/// ### Targets should only depend on files that are provided as inputs
85 86 87 88 89 90 91 92 93
///
/// Example: gen_snapshot must be provided as an input to the aot_elf
/// build steps, even though it isn't a source file. This ensures that changes
/// to the gen_snapshot binary (during a local engine build) correctly
/// trigger a corresponding build update.
///
/// Example: aot_elf has a dependency on the dill and packages file
/// produced by the kernel_snapshot step.
///
94
/// ### Targets should declare all outputs produced
95 96 97 98 99 100 101 102 103 104 105 106 107
///
/// If a target produces an output it should be listed, even if it is not
/// intended to be consumed by another target.
///
/// ## Unit testing
///
/// Most targets will invoke an external binary which makes unit testing
/// trickier. It is recommend that for unit testing that a Fake is used and
/// provided via the dependency injection system. a [Testbed] may be used to
/// set up the environment before the test is run. Unit tests should fully
/// exercise the rule, ensuring that the existing input and output verification
/// logic can run, as well as verifying it correctly handles provided defines
/// and meets any additional contracts present in the target.
108 109
abstract class Target {
  const Target();
110 111 112 113
  /// The user-readable name of the target.
  ///
  /// This information is surfaced in the assemble commands and used as an
  /// argument to build a particular target.
114
  String get name;
115

116 117 118 119 120 121 122 123
  /// A name that measurements can be categorized under for this [Target].
  ///
  /// Unlike [name], this is not expected to be unique, so multiple targets
  /// that are conceptually the same can share an analytics name.
  ///
  /// If not provided, defaults to [name]
  String get analyticsName => name;

124
  /// The dependencies of this target.
125
  List<Target> get dependencies;
126 127

  /// The input [Source]s which are diffed to determine if a target should run.
128
  List<Source> get inputs;
129 130

  /// The output [Source]s which we attempt to verify are correctly produced.
131
  List<Source> get outputs;
132

133 134 135
  /// A list of zero or more depfiles, located directly under {BUILD_DIR}.
  List<String> get depfiles => const <String>[];

136
  /// The action which performs this build step.
137
  Future<void> build(Environment environment);
138

139 140
  /// Create a [Node] with resolved inputs and outputs.
  Node _toNode(Environment environment) {
141 142
    final ResolvedFiles inputsFiles = resolveInputs(environment);
    final ResolvedFiles outputFiles = resolveOutputs(environment);
143 144
    return Node(
      this,
145 146
      inputsFiles.sources,
      outputFiles.sources,
147
      <Node>[
148
        for (final Target target in dependencies) target._toNode(environment),
149
      ],
150
      environment,
151
      inputsFiles.containsNewDepfile,
152
    );
153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169
  }

  /// Invoke to remove the stamp file if the [buildAction] threw an exception;
  void clearStamp(Environment environment) {
    final File stamp = _findStampFile(environment);
    if (stamp.existsSync()) {
      stamp.deleteSync();
    }
  }

  void _writeStamp(
    List<File> inputs,
    List<File> outputs,
    Environment environment,
  ) {
    final File stamp = _findStampFile(environment);
    final List<String> inputPaths = <String>[];
170
    for (final File input in inputs) {
171
      inputPaths.add(input.path);
172 173
    }
    final List<String> outputPaths = <String>[];
174
    for (final File output in outputs) {
175
      outputPaths.add(output.path);
176 177 178 179 180 181 182 183 184 185 186 187 188
    }
    final Map<String, Object> result = <String, Object>{
      'inputs': inputPaths,
      'outputs': outputPaths,
    };
    if (!stamp.existsSync()) {
      stamp.createSync();
    }
    stamp.writeAsStringSync(json.encode(result));
  }

  /// Resolve the set of input patterns and functions into a concrete list of
  /// files.
189
  ResolvedFiles resolveInputs(Environment environment) {
190
    return _resolveConfiguration(inputs, depfiles, environment, implicit: true, inputs: true);
191 192 193 194 195 196
  }

  /// Find the current set of declared outputs, including wildcard directories.
  ///
  /// The [implicit] flag controls whether it is safe to evaluate [Source]s
  /// which uses functions, behaviors, or patterns.
197
  ResolvedFiles resolveOutputs(Environment environment) {
198
    return _resolveConfiguration(outputs, depfiles, environment, inputs: false);
199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215
  }

  /// Performs a fold across this target and its dependencies.
  T fold<T>(T initialValue, T combine(T previousValue, Target target)) {
    final T dependencyResult = dependencies.fold(
        initialValue, (T prev, Target t) => t.fold(prev, combine));
    return combine(dependencyResult, this);
  }

  /// Convert the target to a JSON structure appropriate for consumption by
  /// external systems.
  ///
  /// This requires constants from the [Environment] to resolve the paths of
  /// inputs and the output stamp.
  Map<String, Object> toJson(Environment environment) {
    return <String, Object>{
      'name': name,
216
      'dependencies': <String>[
217
        for (final Target target in dependencies) target.name,
218 219
      ],
      'inputs': <String>[
220
        for (final File file in resolveInputs(environment).sources) file.path,
221 222
      ],
      'outputs': <String>[
223
        for (final File file in resolveOutputs(environment).sources) file.path,
224
      ],
225 226 227 228 229 230 231 232 233 234
      'stamp': _findStampFile(environment).absolute.path,
    };
  }

  /// Locate the stamp file for a particular target name and environment.
  File _findStampFile(Environment environment) {
    final String fileName = '$name.stamp';
    return environment.buildDir.childFile(fileName);
  }

235 236
  static ResolvedFiles _resolveConfiguration(List<Source> config,
    List<String> depfiles, Environment environment, { bool implicit = true, bool inputs = true,
237
  }) {
238
    final SourceVisitor collector = SourceVisitor(environment, inputs);
239
    for (final Source source in config) {
240 241
      source.accept(collector);
    }
242
    depfiles.forEach(collector.visitDepfile);
243
    return collector;
244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264
  }
}

/// The [Environment] defines several constants for use during the build.
///
/// The environment contains configuration and file paths that are safe to
/// depend on and reference during the build.
///
/// Example (Good):
///
/// Use the environment to determine where to write an output file.
///
///    environment.buildDir.childFile('output')
///      ..createSync()
///      ..writeAsStringSync('output data');
///
/// Example (Bad):
///
/// Use a hard-coded path or directory relative to the current working
/// directory to write an output file.
///
265
///   globals.fs.file('build/linux/out')
266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288
///     ..createSync()
///     ..writeAsStringSync('output data');
///
/// Example (Good):
///
/// Using the build mode to produce different output. Note that the action
/// is still responsible for outputting a different file, as defined by the
/// corresponding output [Source].
///
///    final BuildMode buildMode = getBuildModeFromDefines(environment.defines);
///    if (buildMode == BuildMode.debug) {
///      environment.buildDir.childFile('debug.output')
///        ..createSync()
///        ..writeAsStringSync('debug');
///    } else {
///      environment.buildDir.childFile('non_debug.output')
///        ..createSync()
///        ..writeAsStringSync('non_debug');
///    }
class Environment {
  /// Create a new [Environment] object.
  factory Environment({
    @required Directory projectDir,
289
    @required Directory outputDir,
290 291
    @required Directory cacheDir,
    @required Directory flutterRootDir,
292 293 294 295
    @required FileSystem fileSystem,
    @required Logger logger,
    @required Artifacts artifacts,
    @required ProcessManager processManager,
296 297 298 299 300 301 302 303 304
    Directory buildDir,
    Map<String, String> defines = const <String, String>{},
  }) {
    // Compute a unique hash of this build's particular environment.
    // Sort the keys by key so that the result is stable. We always
    // include the engine and dart versions.
    String buildPrefix;
    final List<String> keys = defines.keys.toList()..sort();
    final StringBuffer buffer = StringBuffer();
305
    for (final String key in keys) {
306 307 308
      buffer.write(key);
      buffer.write(defines[key]);
    }
309
    buffer.write(outputDir.path);
310 311 312 313 314 315 316
    final String output = buffer.toString();
    final Digest digest = md5.convert(utf8.encode(output));
    buildPrefix = hex.encode(digest.bytes);

    final Directory rootBuildDir = buildDir ?? projectDir.childDirectory('build');
    final Directory buildDirectory = rootBuildDir.childDirectory(buildPrefix);
    return Environment._(
317
      outputDir: outputDir,
318 319 320
      projectDir: projectDir,
      buildDir: buildDirectory,
      rootBuildDir: rootBuildDir,
321 322 323
      cacheDir: cacheDir,
      defines: defines,
      flutterRootDir: flutterRootDir,
324 325 326 327
      fileSystem: fileSystem,
      logger: logger,
      artifacts: artifacts,
      processManager: processManager,
328 329 330 331 332 333 334 335 336 337 338 339 340
    );
  }

  /// Create a new [Environment] object for unit testing.
  ///
  /// Any directories not provided will fallback to a [testDirectory]
  factory Environment.test(Directory testDirectory, {
    Directory projectDir,
    Directory outputDir,
    Directory cacheDir,
    Directory flutterRootDir,
    Directory buildDir,
    Map<String, String> defines = const <String, String>{},
341 342 343 344
    @required FileSystem fileSystem,
    @required Logger logger,
    @required Artifacts artifacts,
    @required ProcessManager processManager,
345 346 347 348 349 350 351
  }) {
    return Environment(
      projectDir: projectDir ?? testDirectory,
      outputDir: outputDir ?? testDirectory,
      cacheDir: cacheDir ?? testDirectory,
      flutterRootDir: flutterRootDir ?? testDirectory,
      buildDir: buildDir,
352
      defines: defines,
353 354 355 356
      fileSystem: fileSystem,
      logger: logger,
      artifacts: artifacts,
      processManager: processManager,
357 358 359 360
    );
  }

  Environment._({
361
    @required this.outputDir,
362 363 364 365 366
    @required this.projectDir,
    @required this.buildDir,
    @required this.rootBuildDir,
    @required this.cacheDir,
    @required this.defines,
367
    @required this.flutterRootDir,
368 369 370 371
    @required this.processManager,
    @required this.logger,
    @required this.fileSystem,
    @required this.artifacts,
372 373 374 375 376 377 378 379 380 381 382 383 384 385
  });

  /// The [Source] value which is substituted with the path to [projectDir].
  static const String kProjectDirectory = '{PROJECT_DIR}';

  /// The [Source] value which is substituted with the path to [buildDir].
  static const String kBuildDirectory = '{BUILD_DIR}';

  /// The [Source] value which is substituted with the path to [cacheDir].
  static const String kCacheDirectory = '{CACHE_DIR}';

  /// The [Source] value which is substituted with a path to the flutter root.
  static const String kFlutterRootDirectory = '{FLUTTER_ROOT}';

386 387 388
  /// The [Source] value which is substituted with a path to [outputDir].
  static const String kOutputDirectory = '{OUTPUT_DIR}';

389 390 391 392 393 394 395 396
  /// The `PROJECT_DIR` environment variable.
  ///
  /// This should be root of the flutter project where a pubspec and dart files
  /// can be located.
  final Directory projectDir;

  /// The `BUILD_DIR` environment variable.
  ///
397 398
  /// Defaults to `{PROJECT_ROOT}/build`. The root of the output directory where
  /// build step intermediates and outputs are written.
399 400 401 402 403 404 405 406
  final Directory buildDir;

  /// The `CACHE_DIR` environment variable.
  ///
  /// Defaults to `{FLUTTER_ROOT}/bin/cache`. The root of the artifact cache for
  /// the flutter tool.
  final Directory cacheDir;

407 408
  /// The `FLUTTER_ROOT` environment variable.
  ///
409
  /// Defaults to the value of [Cache.flutterRoot].
410 411
  final Directory flutterRootDir;

412 413 414 415 416
  /// The `OUTPUT_DIR` environment variable.
  ///
  /// Must be provided to configure the output location for the final artifacts.
  final Directory outputDir;

417 418 419 420 421 422 423 424
  /// Additional configuration passed to the build targets.
  ///
  /// Setting values here forces a unique build directory to be chosen
  /// which prevents the config from leaking into different builds.
  final Map<String, String> defines;

  /// The root build directory shared by all builds.
  final Directory rootBuildDir;
425 426 427 428 429 430 431 432

  final ProcessManager processManager;

  final Logger logger;

  final Artifacts artifacts;

  final FileSystem fileSystem;
433 434 435 436
}

/// The result information from the build system.
class BuildResult {
437 438 439 440 441 442 443
  BuildResult({
    @required this.success,
    this.exceptions = const <String, ExceptionMeasurement>{},
    this.performance = const <String, PerformanceMeasurement>{},
    this.inputFiles = const <File>[],
    this.outputFiles = const <File>[],
  });
444 445 446 447

  final bool success;
  final Map<String, ExceptionMeasurement> exceptions;
  final Map<String, PerformanceMeasurement> performance;
448 449
  final List<File> inputFiles;
  final List<File> outputFiles;
450 451 452 453 454 455

  bool get hasException => exceptions.isNotEmpty;
}

/// The build system is responsible for invoking and ordering [Target]s.
class BuildSystem {
456 457 458 459 460 461 462 463 464 465 466
  const BuildSystem({
    @required FileSystem fileSystem,
    @required Platform platform,
    @required Logger logger,
  }) : _fileSystem = fileSystem,
       _platform = platform,
       _logger = logger;

  final FileSystem _fileSystem;
  final Platform _platform;
  final Logger _logger;
467 468

  /// Build `target` and all of its dependencies.
469
  Future<BuildResult> build(
470
    Target target,
471 472 473
    Environment environment, {
    BuildSystemConfig buildSystemConfig = const BuildSystemConfig(),
  }) async {
474
    environment.buildDir.createSync(recursive: true);
475
    environment.outputDir.createSync(recursive: true);
476 477

    // Load file hash store from previous builds.
478 479
    final FileHashStore fileCache = FileHashStore(
      environment: environment,
480 481
      fileSystem: _fileSystem,
      logger: _logger,
482
    )..initialize();
483 484 485 486

    // Perform sanity checks on build.
    checkCycles(target);

487
    final Node node = target._toNode(environment);
488 489 490 491 492 493 494 495
    final _BuildInstance buildInstance = _BuildInstance(
      environment: environment,
      fileCache: fileCache,
      buildSystemConfig: buildSystemConfig,
      logger: _logger,
      fileSystem: _fileSystem,
      platform: _platform,
    );
496 497
    bool passed = true;
    try {
498
      passed = await buildInstance.invokeTarget(node);
499 500 501 502
    } finally {
      // Always persist the file cache to disk.
      fileCache.persist();
    }
503 504
    // TODO(jonahwilliams): this is a bit of a hack, due to various parts of
    // the flutter tool writing these files unconditionally. Since Xcode uses
505
    // timestamps to track files, this leads to unnecessary rebuilds if they
506 507
    // are included. Once all the places that write these files have been
    // tracked down and moved into assemble, these checks should be removable.
508 509
    // We also remove files under .dart_tool, since these are intermediaries
    // and don't need to be tracked by external systems.
510 511
    {
      buildInstance.inputFiles.removeWhere((String path, File file) {
512 513 514
        return path.contains('.flutter-plugins') ||
                       path.contains('xcconfig') ||
                     path.contains('.dart_tool');
515 516
      });
      buildInstance.outputFiles.removeWhere((String path, File file) {
517 518 519
        return path.contains('.flutter-plugins') ||
                       path.contains('xcconfig') ||
                     path.contains('.dart_tool');
520 521
      });
    }
522
    return BuildResult(
523 524 525 526 527 528 529
      success: passed,
      exceptions: buildInstance.exceptionMeasurements,
      performance: buildInstance.stepTimings,
      inputFiles: buildInstance.inputFiles.values.toList()
          ..sort((File a, File b) => a.path.compareTo(b.path)),
      outputFiles: buildInstance.outputFiles.values.toList()
          ..sort((File a, File b) => a.path.compareTo(b.path)),
530 531 532 533
    );
  }
}

534

535 536
/// An active instance of a build.
class _BuildInstance {
537 538 539 540 541 542 543 544 545 546 547 548
  _BuildInstance({
    this.environment,
    this.fileCache,
    this.buildSystemConfig,
    this.logger,
    this.fileSystem,
    Platform platform,
  })
    : resourcePool = Pool(buildSystemConfig.resourcePoolSize ?? platform?.numberOfProcessors ?? 1);

  final Logger logger;
  final FileSystem fileSystem;
549 550
  final BuildSystemConfig buildSystemConfig;
  final Pool resourcePool;
551
  final Map<String, AsyncMemoizer<bool>> pending = <String, AsyncMemoizer<bool>>{};
552 553
  final Environment environment;
  final FileHashStore fileCache;
554 555
  final Map<String, File> inputFiles = <String, File>{};
  final Map<String, File> outputFiles = <String, File>{};
556 557 558 559 560 561 562

  // Timings collected during target invocation.
  final Map<String, PerformanceMeasurement> stepTimings = <String, PerformanceMeasurement>{};

  // Exceptions caught during the build process.
  final Map<String, ExceptionMeasurement> exceptionMeasurements = <String, ExceptionMeasurement>{};

563 564
  Future<bool> invokeTarget(Node node) async {
    final List<bool> results = await Future.wait(node.dependencies.map(invokeTarget));
565 566 567
    if (results.any((bool result) => !result)) {
      return false;
    }
568 569
    final AsyncMemoizer<bool> memoizer = pending[node.target.name] ??= AsyncMemoizer<bool>();
    return memoizer.runOnce(() => _invokeInternal(node));
570 571
  }

572
  Future<bool> _invokeInternal(Node node) async {
573 574 575 576
    final PoolResource resource = await resourcePool.request();
    final Stopwatch stopwatch = Stopwatch()..start();
    bool passed = true;
    bool skipped = false;
577 578 579 580 581 582 583 584 585 586 587 588

    // The build system produces a list of aggregate input and output
    // files for the overall build. This list is provided to a hosting build
    // system, such as Xcode, to configure logic for when to skip the
    // rule/phase which contains the flutter build.
    //
    // When looking at the inputs and outputs for the individual rules, we need
    // to be careful to remove inputs that were actually output from previous
    // build steps. This indicates that the file is an intermediary. If
    // these files are included as both inputs and outputs then it isn't
    // possible to construct a DAG describing the build.
    void updateGraph() {
589
      for (final File output in node.outputs) {
590 591
        outputFiles[output.path] = output;
      }
592
      for (final File input in node.inputs) {
593
        final String resolvedPath = input.absolute.path;
594 595 596 597 598
        if (outputFiles.containsKey(resolvedPath)) {
          continue;
        }
        inputFiles[resolvedPath] = input;
      }
599 600 601 602 603 604
    }

    try {
      // If we're missing a depfile, wait until after evaluating the target to
      // compute changes.
      final bool canSkip = !node.missingDepfile &&
605
        await node.computeChanges(environment, fileCache, fileSystem, logger);
606

607
      if (canSkip) {
608
        skipped = true;
609
        logger.printTrace('Skipping target: ${node.target.name}');
610 611 612
        updateGraph();
        return passed;
      }
613
      logger.printTrace('${node.target.name}: Starting due to ${node.invalidatedReasons}');
614
      await node.target.build(environment);
615
      logger.printTrace('${node.target.name}: Complete');
616

617 618 619 620 621 622 623 624
      node.inputs
        ..clear()
        ..addAll(node.target.resolveInputs(environment).sources);
      node.outputs
        ..clear()
        ..addAll(node.target.resolveOutputs(environment).sources);

      // If we were missing the depfile, resolve  input files after executing the
625 626 627 628 629
      // target so that all file hashes are up to date on the next run.
      if (node.missingDepfile) {
        await fileCache.hashFiles(node.inputs);
      }

630
      // Always update hashes for output files.
631 632 633 634 635 636
      await fileCache.hashFiles(node.outputs);
      node.target._writeStamp(node.inputs, node.outputs, environment);
      updateGraph();

      // Delete outputs from previous stages that are no longer a part of the
      // build.
637
      for (final String previousOutput in node.previousOutputs) {
638 639
        if (outputFiles.containsKey(previousOutput)) {
          continue;
640
        }
641
        final File previousFile = fileSystem.file(previousOutput);
642 643
        if (previousFile.existsSync()) {
          previousFile.deleteSync();
644
        }
645
      }
646
    } on Exception catch (exception, stackTrace) {
647 648
      // TODO(jonahwilliams): throw specific exception for expected errors to mark
      // as non-fatal. All others should be fatal.
649
      node.target.clearStamp(environment);
650 651
      passed = false;
      skipped = false;
652 653
      exceptionMeasurements[node.target.name] = ExceptionMeasurement(
          node.target.name, exception, stackTrace);
654 655 656
    } finally {
      resource.release();
      stopwatch.stop();
657
      stepTimings[node.target.name] = PerformanceMeasurement(
658 659 660 661 662 663
        target: node.target.name,
        elapsedMilliseconds: stopwatch.elapsedMilliseconds,
        skipped: skipped,
        passed: passed,
        analyicsName: node.target.analyticsName,
      );
664 665 666 667 668 669 670
    }
    return passed;
  }
}

/// Helper class to collect exceptions.
class ExceptionMeasurement {
671
  ExceptionMeasurement(this.target, this.exception, this.stackTrace, {this.fatal = false});
672 673 674 675

  final String target;
  final dynamic exception;
  final StackTrace stackTrace;
676

677 678 679
  /// Whether this exception was a fatal build system error.
  final bool fatal;

680 681
  @override
  String toString() => 'target: $target\nexception:$exception\n$stackTrace';
682 683 684 685
}

/// Helper class to collect measurement data.
class PerformanceMeasurement {
686 687 688 689 690 691 692 693
  PerformanceMeasurement({
    @required this.target,
    @required this.elapsedMilliseconds,
    @required this.skipped,
    @required this.passed,
    @required this.analyicsName,
  });

694 695
  final int elapsedMilliseconds;
  final String target;
696
  final bool skipped;
697
  final bool passed;
698
  final String analyicsName;
699 700 701 702 703 704 705 706 707 708 709 710 711 712 713
}

/// Check if there are any dependency cycles in the target.
///
/// Throws a [CycleException] if one is encountered.
void checkCycles(Target initial) {
  void checkInternal(Target target, Set<Target> visited, Set<Target> stack) {
    if (stack.contains(target)) {
      throw CycleException(stack..add(target));
    }
    if (visited.contains(target)) {
      return;
    }
    visited.add(target);
    stack.add(target);
714
    for (final Target dependency in target.dependencies) {
715 716 717 718 719 720 721 722 723 724 725 726
      checkInternal(dependency, visited, stack);
    }
    stack.remove(target);
  }
  checkInternal(initial, <Target>{}, <Target>{});
}

/// Verifies that all files exist and are in a subdirectory of [Environment.buildDir].
void verifyOutputDirectories(List<File> outputs, Environment environment, Target target) {
  final String buildDirectory = environment.buildDir.resolveSymbolicLinksSync();
  final String projectDirectory = environment.projectDir.resolveSymbolicLinksSync();
  final List<File> missingOutputs = <File>[];
727
  for (final File sourceFile in outputs) {
728 729 730 731
    if (!sourceFile.existsSync()) {
      missingOutputs.add(sourceFile);
      continue;
    }
732
    final String path = sourceFile.path;
733 734 735 736 737 738 739 740
    if (!path.startsWith(buildDirectory) && !path.startsWith(projectDirectory)) {
      throw MisplacedOutputException(path, target.name);
    }
  }
  if (missingOutputs.isNotEmpty) {
    throw MissingOutputException(missingOutputs, target.name);
  }
}
741 742 743

/// A node in the build graph.
class Node {
744 745 746 747 748 749 750 751
  Node(
    this.target,
    this.inputs,
    this.outputs,
    this.dependencies,
    Environment environment,
    this.missingDepfile,
  ) {
752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770
    final File stamp = target._findStampFile(environment);

    // If the stamp file doesn't exist, we haven't run this step before and
    // all inputs were added.
    if (!stamp.existsSync()) {
      // No stamp file, not safe to skip.
      _dirty = true;
      return;
    }
    final String content = stamp.readAsStringSync();
    // Something went wrong writing the stamp file.
    if (content == null || content.isEmpty) {
      stamp.deleteSync();
      // Malformed stamp file, not safe to skip.
      _dirty = true;
      return;
    }
    Map<String, Object> values;
    try {
771
      values = castStringKeyedMap(json.decode(content));
772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797
    } on FormatException {
      // The json is malformed in some way.
      _dirty = true;
      return;
    }
    final Object inputs = values['inputs'];
    final Object outputs = values['outputs'];
    if (inputs is List<Object> && outputs is List<Object>) {
      inputs?.cast<String>()?.forEach(previousInputs.add);
      outputs?.cast<String>()?.forEach(previousOutputs.add);
    } else {
      // The json is malformed in some way.
      _dirty = true;
    }
  }

  /// The resolved input files.
  ///
  /// These files may not yet exist if they are produced by previous steps.
  final List<File> inputs;

  /// The resolved output files.
  ///
  /// These files may not yet exist if the target hasn't run yet.
  final List<File> outputs;

798 799 800 801 802 803
  /// Whether this node is missing a depfile.
  ///
  /// This requires an additional pass of source resolution after the target
  /// has been executed.
  final bool missingDepfile;

804 805 806 807 808 809 810 811 812
  /// The target definition which contains the build action to invoke.
  final Target target;

  /// All of the nodes that this one depends on.
  final List<Node> dependencies;

  /// Output file paths from the previous invocation of this build node.
  final Set<String> previousOutputs = <String>{};

813
  /// Input file paths from the previous invocation of this build node.
814 815
  final Set<String> previousInputs = <String>{};

816 817 818
  /// One or more reasons why a task was invalidated.
  ///
  /// May be empty if the task was skipped.
819
  final Set<InvalidatedReason> invalidatedReasons = <InvalidatedReason>{};
820

821 822 823 824 825 826 827 828 829 830
  /// Whether this node needs an action performed.
  bool get dirty => _dirty;
  bool _dirty = false;

  /// Collect hashes for all inputs to determine if any have changed.
  ///
  /// Returns whether this target can be skipped.
  Future<bool> computeChanges(
    Environment environment,
    FileHashStore fileHashStore,
831 832
    FileSystem fileSystem,
    Logger logger,
833 834
  ) async {
    final Set<String> currentOutputPaths = <String>{
835
      for (final File file in outputs) file.path,
836 837 838 839 840
    };
    // For each input, first determine if we've already computed the hash
    // for it. Then collect it to be sent off for hashing as a group.
    final List<File> sourcesToHash = <File>[];
    final List<File> missingInputs = <File>[];
841
    for (final File file in inputs) {
842 843 844 845 846 847 848 849 850 851
      if (!file.existsSync()) {
        missingInputs.add(file);
        continue;
      }

      final String absolutePath = file.path;
      final String previousHash = fileHashStore.previousHashes[absolutePath];
      if (fileHashStore.currentHashes.containsKey(absolutePath)) {
        final String currentHash = fileHashStore.currentHashes[absolutePath];
        if (currentHash != previousHash) {
852
          invalidatedReasons.add(InvalidatedReason.inputChanged);
853 854 855 856 857 858 859 860 861
          _dirty = true;
        }
      } else {
        sourcesToHash.add(file);
      }
    }

    // For each output, first determine if we've already computed the hash
    // for it. Then collect it to be sent off for hashing as a group.
862
    for (final String previousOutput in previousOutputs) {
863 864 865
      // output paths changed.
      if (!currentOutputPaths.contains(previousOutput)) {
        _dirty = true;
866
        invalidatedReasons.add(InvalidatedReason.outputSetChanged);
867 868 869
        // if this isn't a current output file there is no reason to compute the hash.
        continue;
      }
870
      final File file = fileSystem.file(previousOutput);
871
      if (!file.existsSync()) {
872
        invalidatedReasons.add(InvalidatedReason.outputMissing);
873 874 875 876 877 878 879 880
        _dirty = true;
        continue;
      }
      final String absolutePath = file.path;
      final String previousHash = fileHashStore.previousHashes[absolutePath];
      if (fileHashStore.currentHashes.containsKey(absolutePath)) {
        final String currentHash = fileHashStore.currentHashes[absolutePath];
        if (currentHash != previousHash) {
881
          invalidatedReasons.add(InvalidatedReason.outputChanged);
882 883 884 885 886 887 888
          _dirty = true;
        }
      } else {
        sourcesToHash.add(file);
      }
    }

889 890 891
    // If we depend on a file that doesnt exist on disk, mark the build as
    // dirty. if the rule is not correctly specified, this will result in it
    // always being rerun.
892
    if (missingInputs.isNotEmpty) {
893 894
      _dirty = true;
      final String missingMessage = missingInputs.map((File file) => file.path).join(', ');
895
      logger.printTrace('invalidated build due to missing files: $missingMessage');
896
      invalidatedReasons.add(InvalidatedReason.inputMissing);
897 898 899 900 901 902 903
    }

    // If we have files to hash, compute them asynchronously and then
    // update the result.
    if (sourcesToHash.isNotEmpty) {
      final List<File> dirty = await fileHashStore.hashFiles(sourcesToHash);
      if (dirty.isNotEmpty) {
904
        invalidatedReasons.add(InvalidatedReason.inputChanged);
905 906 907 908 909 910
        _dirty = true;
      }
    }
    return !_dirty;
  }
}
911

912 913 914 915 916 917
/// A description of why a target was rerun.
enum InvalidatedReason {
  /// An input file that was expected is missing. This can occur when using
  /// depfile dependencies, or if a target is incorrectly specified.
  inputMissing,

918 919 920 921 922 923 924 925 926 927 928 929
  /// An input file has an updated hash.
  inputChanged,

  /// An output file has an updated hash.
  outputChanged,

  /// An output file that is expected is missing.
  outputMissing,

  /// The set of expected output files changed.
  outputSetChanged,
}