source.dart 10.6 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 '../artifacts.dart';
import '../base/file_system.dart';
import '../build_info.dart';
import 'build_system.dart';
import 'exceptions.dart';

11 12 13 14 15 16 17 18 19 20 21 22
/// A set of source files.
abstract class ResolvedFiles {
  /// Whether any of the sources we evaluated contained a missing depfile.
  ///
  /// If so, the build system needs to rerun the visitor after executing the
  /// build to ensure all hashes are up to date.
  bool get containsNewDepfile;

  /// The resolved source files.
  List<File> get sources;
}

23
/// Collects sources for a [Target] into a single list of [FileSystemEntities].
24
class SourceVisitor implements ResolvedFiles {
25
  /// Create a new [SourceVisitor] from an [Environment].
26
  SourceVisitor(this.environment, [ this.inputs = true ]);
27 28 29 30 31 32 33 34 35

  /// The current environment.
  final Environment environment;

  /// Whether we are visiting inputs or outputs.
  ///
  /// Defaults to `true`.
  final bool inputs;

36
  @override
37 38
  final List<File> sources = <File>[];

39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
  @override
  bool get containsNewDepfile => _containsNewDepfile;
  bool _containsNewDepfile = false;

  /// Visit a depfile which contains both input and output files.
  ///
  /// If the file is missing, this visitor is marked as [containsNewDepfile].
  /// This is used by the [Node] class to tell the [BuildSystem] to
  /// defer hash computation until after executing the target.
  // depfile logic adopted from https://github.com/flutter/flutter/blob/7065e4330624a5a216c8ffbace0a462617dc1bf5/dev/devicelab/lib/framework/apk_utils.dart#L390
  void visitDepfile(String name) {
    final File depfile = environment.buildDir.childFile(name);
    if (!depfile.existsSync()) {
      _containsNewDepfile = true;
      return;
    }
    final String contents = depfile.readAsStringSync();
    final List<String> colonSeparated = contents.split(': ');
    if (colonSeparated.length != 2) {
58
      environment.logger.printError('Invalid depfile: ${depfile.path}');
59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76
      return;
    }
    if (inputs) {
      sources.addAll(_processList(colonSeparated[1].trim()));
    } else {
      sources.addAll(_processList(colonSeparated[0].trim()));
    }
  }

  final RegExp _separatorExpr = RegExp(r'([^\\]) ');
  final RegExp _escapeExpr = RegExp(r'\\(.)');

  Iterable<File> _processList(String rawText) {
    return rawText
    // Put every file on right-hand side on the separate line
        .replaceAllMapped(_separatorExpr, (Match match) => '${match.group(1)}\n')
        .split('\n')
    // Expand escape sequences, so that '\ ', for example,ß becomes ' '
77
        .map<String>((String path) => path.replaceAllMapped(_escapeExpr, (Match match) => match.group(1)!).trim())
78 79
        .where((String path) => path.isNotEmpty)
        .toSet()
80
        .map(environment.fileSystem.file);
81 82
  }

83
  /// Visit a [Source] which contains a file URL.
84
  ///
85
  /// The URL may include constants defined in an [Environment]. If
86 87 88
  /// [optional] is true, the file is not required to exist. In this case, it
  /// is never resolved as an input.
  void visitPattern(String pattern, bool optional) {
89 90 91 92 93
    // perform substitution of the environmental values and then
    // of the local values.
    final List<String> segments = <String>[];
    final List<String> rawParts = pattern.split('/');
    final bool hasWildcard = rawParts.last.contains('*');
94
    String? wildcardFile;
95 96 97 98 99 100 101 102
    if (hasWildcard) {
      wildcardFile = rawParts.removeLast();
    }
    // If the pattern does not start with an env variable, then we have nothing
    // to resolve it to, error out.
    switch (rawParts.first) {
      case Environment.kProjectDirectory:
        segments.addAll(
103
          environment.fileSystem.path.split(environment.projectDir.resolveSymbolicLinksSync()));
104 105
        break;
      case Environment.kBuildDirectory:
106 107
        segments.addAll(environment.fileSystem.path.split(
          environment.buildDir.resolveSymbolicLinksSync()));
108 109 110
        break;
      case Environment.kCacheDirectory:
        segments.addAll(
111
          environment.fileSystem.path.split(environment.cacheDir.resolveSymbolicLinksSync()));
112 113
        break;
      case Environment.kFlutterRootDirectory:
114
        // flutter root will not contain a symbolic link.
115
        segments.addAll(
116
          environment.fileSystem.path.split(environment.flutterRootDir.absolute.path));
117
        break;
118 119
      case Environment.kOutputDirectory:
        segments.addAll(
120
          environment.fileSystem.path.split(environment.outputDir.resolveSymbolicLinksSync()));
121
        break;
122 123 124 125
      default:
        throw InvalidPatternException(pattern);
    }
    rawParts.skip(1).forEach(segments.add);
126
    final String filePath = environment.fileSystem.path.joinAll(segments);
127
    if (!hasWildcard) {
128
      if (optional && !environment.fileSystem.isFileSync(filePath)) {
129
        return;
130
      }
131 132
      sources.add(environment.fileSystem.file(
        environment.fileSystem.path.normalize(filePath)));
133 134 135 136 137 138 139 140 141
      return;
    }
    // Perform a simple match by splitting the wildcard containing file one
    // the `*`. For example, for `/*.dart`, we get [.dart]. We then check
    // that part of the file matches. If there are values before and after
    // the `*` we need to check that both match without overlapping. For
    // example, `foo_*_.dart`. We want to match `foo_b_.dart` but not
    // `foo_.dart`. To do so, we first subtract the first section from the
    // string if the first segment matches.
142
    final List<String> wildcardSegments = wildcardFile?.split('*') ?? <String>[];
143 144 145
    if (wildcardSegments.length > 2) {
      throw InvalidPatternException(pattern);
    }
146
    if (!environment.fileSystem.directory(filePath).existsSync()) {
147
      environment.fileSystem.directory(filePath).createSync(recursive: true);
148
    }
149 150
    for (final FileSystemEntity entity in environment.fileSystem.directory(filePath).listSync()) {
      final String filename = environment.fileSystem.path.basename(entity.path);
151
      if (wildcardSegments.isEmpty) {
152
        sources.add(environment.fileSystem.file(entity.absolute));
153 154 155
      } else if (wildcardSegments.length == 1) {
        if (filename.startsWith(wildcardSegments[0]) ||
            filename.endsWith(wildcardSegments[0])) {
156
          sources.add(environment.fileSystem.file(entity.absolute));
157 158 159
        }
      } else if (filename.startsWith(wildcardSegments[0])) {
        if (filename.substring(wildcardSegments[0].length).endsWith(wildcardSegments[1])) {
160
          sources.add(environment.fileSystem.file(entity.absolute));
161 162 163 164 165 166 167 168
        }
      }
    }
  }

  /// Visit a [Source] which is defined by an [Artifact] from the flutter cache.
  ///
  /// If the [Artifact] points to a directory then all child files are included.
169 170 171
  /// To increase the performance of builds that use a known revision of Flutter,
  /// these are updated to point towards the engine.version file instead of
  /// the artifact itself.
172
  void visitArtifact(Artifact artifact, TargetPlatform? platform, BuildMode? mode) {
173 174 175 176 177 178 179 180 181
    // This is not a local engine.
    if (environment.engineVersion != null) {
      sources.add(environment.flutterRootDir
        .childDirectory('bin')
        .childDirectory('internal')
        .childFile('engine.version'),
      );
      return;
    }
182 183 184
    final String path = environment.artifacts
      .getArtifactPath(artifact, platform: platform, mode: mode);
    if (environment.fileSystem.isDirectorySync(path)) {
185
      sources.addAll(<File>[
186
        for (FileSystemEntity entity in environment.fileSystem.directory(path).listSync(recursive: true))
187
          if (entity is File)
188
            entity,
189
      ]);
190
      return;
191
    }
192
    sources.add(environment.fileSystem.file(path));
193
  }
194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221

  /// Visit a [Source] which is defined by an [HostArtifact] from the flutter cache.
  ///
  /// If the [Artifact] points to a directory then all child files are included.
  /// To increase the performance of builds that use a known revision of Flutter,
  /// these are updated to point towards the engine.version file instead of
  /// the artifact itself.
  void visitHostArtifact(HostArtifact artifact) {
    // This is not a local engine.
    if (environment.engineVersion != null) {
      sources.add(environment.flutterRootDir
        .childDirectory('bin')
        .childDirectory('internal')
        .childFile('engine.version'),
      );
      return;
    }
    final FileSystemEntity entity = environment.artifacts.getHostArtifact(artifact);
    if (entity is Directory) {
      sources.addAll(<File>[
        for (FileSystemEntity entity in entity.listSync(recursive: true))
          if (entity is File)
            entity,
      ]);
      return;
    }
    sources.add(entity as File);
  }
222 223 224 225
}

/// A description of an input or output of a [Target].
abstract class Source {
226
  /// This source is a file URL which contains some references to magic
227
  /// environment variables.
228
  const factory Source.pattern(String pattern, { bool optional }) = _PatternSource;
229

230 231 232
  /// The source is provided by an [Artifact].
  ///
  /// If [artifact] points to a directory then all child files are included.
233
  const factory Source.artifact(Artifact artifact, {TargetPlatform? platform, BuildMode? mode}) = _ArtifactSource;
234

235 236 237 238 239
  /// The source is provided by an [HostArtifact].
  ///
  /// If [artifact] points to a directory then all child files are included.
  const factory Source.hostArtifact(HostArtifact artifact) = _HostArtifactSource;

240 241 242 243 244 245 246 247 248
  /// Visit the particular source type.
  void accept(SourceVisitor visitor);

  /// Whether the output source provided can be known before executing the rule.
  ///
  /// This does not apply to inputs, which are always explicit and must be
  /// evaluated before the build.
  ///
  /// For example, [Source.pattern] and [Source.version] are not implicit
249
  /// provided they do not use any wildcards.
250 251 252 253
  bool get implicit;
}

class _PatternSource implements Source {
254
  const _PatternSource(this.value, { this.optional = false });
255 256

  final String value;
257
  final bool optional;
258 259

  @override
260
  void accept(SourceVisitor visitor) => visitor.visitPattern(value, optional);
261 262 263 264 265 266 267 268 269

  @override
  bool get implicit => value.contains('*');
}

class _ArtifactSource implements Source {
  const _ArtifactSource(this.artifact, { this.platform, this.mode });

  final Artifact artifact;
270 271
  final TargetPlatform? platform;
  final BuildMode? mode;
272 273 274 275 276 277 278

  @override
  void accept(SourceVisitor visitor) => visitor.visitArtifact(artifact, platform, mode);

  @override
  bool get implicit => false;
}
279 280 281 282 283 284 285 286 287 288 289 290

class _HostArtifactSource implements Source {
  const _HostArtifactSource(this.artifact);

  final HostArtifact artifact;

  @override
  void accept(SourceVisitor visitor) => visitor.visitHostArtifact(artifact);

  @override
  bool get implicit => false;
}