blob: dda7d9c9e9ae9cee6ea2ea1bbc93bd086c464d44 [file] [log] [blame]
// Copyright 2014 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.
import '../base/file_system.dart';
import '../base/logger.dart';
import '../convert.dart';
/// Represents a configured deferred component as defined in
/// the app's pubspec.yaml.
class DeferredComponent {
DeferredComponent({
required this.name,
this.libraries = const <String>[],
this.assets = const <Uri>[],
}) : _assigned = false;
/// The name of the deferred component. There should be a matching
/// android dynamic feature module with the same name.
final String name;
/// The dart libraries this component includes as listed in pubspec.yaml.
///
/// This list is only of dart libraries manually configured to be in this component.
/// Valid libraries that are listed here will always be guaranteed to be
/// packaged in this component. However, libraries that are not listed here
/// may also be included if the loading units that are needed also contain
/// libraries that are not listed here.
final List<String> libraries;
/// Assets that are part of this component as a Uri relative to the project directory.
final List<Uri> assets;
/// The minimal set of [LoadingUnit]s needed that contain all of the dart libraries in
/// [libraries].
///
/// Each [LoadingUnit] contains the compiled code for a set of dart libraries. Each
/// [DeferredComponent] contains a list of dart libraries that must be included in the
/// component. The set [loadingUnits] is all of the [LoadingUnit]s needed such that
/// all required dart libs in [libraries] are in the union of the [LoadingUnit.libraries]
/// included by the loading units in [loadingUnits].
///
/// When [loadingUnits] is non-null, then the component is considered [assigned] and the
/// field [assigned] will be true. When [loadingUnits] is null, then the component is
/// unassigned and should not be used for any tasks that require loading unit information.
/// When using [loadingUnits], [assigned] should be checked first. Loading units can be
/// assigned with [assignLoadingUnits].
Set<LoadingUnit>? get loadingUnits => _loadingUnits;
Set<LoadingUnit>? _loadingUnits;
/// Indicates if the component has loading units assigned.
///
/// Unassigned components simply reflect the pubspec.yaml configuration directly,
/// contain no loading unit data, and [loadingUnits] is null. Once assigned, the component
/// will contain a set of [loadingUnits] which contains the [LoadingUnit]s that the
/// component needs to include. Loading units can be assigned with the [assignLoadingUnits]
/// call.
bool get assigned => _assigned;
bool _assigned;
/// Selects the [LoadingUnit]s that contain this component's dart libraries.
///
/// After calling this method, this [DeferredComponent] will be considered [assigned],
/// and [loadingUnits] will return a non-null result.
///
/// [LoadingUnit]s in `allLoadingUnits` that contain libraries that are in [libraries]
/// are added to the set [loadingUnits].
///
/// Providing null or empty list of `allLoadingUnits` will still change the assigned
/// status, but will result in [loadingUnits] returning an empty set.
void assignLoadingUnits(List<LoadingUnit> allLoadingUnits) {
_assigned = true;
_loadingUnits = <LoadingUnit>{};
if (allLoadingUnits == null) {
return;
}
for (final String lib in libraries) {
for (final LoadingUnit loadingUnit in allLoadingUnits) {
if (loadingUnit.libraries.contains(lib)) {
_loadingUnits!.add(loadingUnit);
}
}
}
}
/// Provides a human readable string representation of the
/// configuration.
@override
String toString() {
final StringBuffer out = StringBuffer('\nDeferredComponent: $name\n Libraries:');
for (final String lib in libraries) {
out.write('\n - $lib');
}
if (loadingUnits != null && _assigned) {
out.write('\n LoadingUnits:');
for (final LoadingUnit loadingUnit in loadingUnits!) {
out.write('\n - ${loadingUnit.id}');
}
}
out.write('\n Assets:');
for (final Uri asset in assets) {
out.write('\n - ${asset.path}');
}
return out.toString();
}
}
/// Represents a single loading unit and holds information regarding it's id,
/// shared library path, and dart libraries in it.
class LoadingUnit {
/// Constructs a [LoadingUnit].
///
/// Loading units must include an [id] and [libraries]. The [path] is only present when
/// parsing the loading unit from a loading unit manifest produced by gen_snapshot.
LoadingUnit({
required this.id,
required this.libraries,
this.path,
});
/// The unique loading unit id that is used to identify the loading unit within dart.
final int id;
/// A list of dart libraries that the loading unit contains.
final List<String> libraries;
/// The output path of the shared library .so file created by gen_snapshot.
///
/// This value may be null when the loading unit is parsed from a
/// `deferred_components_golden.yaml` file, which does not store the path.
final String? path;
/// Returns a human readable string representation of this LoadingUnit, ignoring
/// the [path] field. The [path] is not included as it is not relevant when the
@override
String toString() {
final StringBuffer out = StringBuffer('\nLoadingUnit $id\n Libraries:');
for (final String lib in libraries) {
out.write('\n - $lib');
}
return out.toString();
}
/// Returns true if the other loading unit has the same [id] and the same set of [libraries],
/// ignoring order.
bool equalsIgnoringPath(LoadingUnit other) {
return other.id == id && other.libraries.toSet().containsAll(libraries);
}
/// Parses the loading unit manifests from the [outputDir] of the latest
/// gen_snapshot/assemble run.
///
/// This will read all existing loading units for every provided abi. If no abis are
/// provided, loading units for all abis will be parsed.
static List<LoadingUnit> parseGeneratedLoadingUnits(Directory outputDir, Logger logger, {List<String>? abis}) {
final List<LoadingUnit> loadingUnits = <LoadingUnit>[];
final List<FileSystemEntity> files = outputDir.listSync(recursive: true);
for (final FileSystemEntity fileEntity in files) {
if (fileEntity is File) {
final File file = fileEntity;
// Determine if the abi is one we build.
bool matchingAbi = abis == null;
if (abis != null) {
for (final String abi in abis) {
if (file.parent.path.endsWith(abi)) {
matchingAbi = true;
break;
}
}
}
if (!file.path.endsWith('manifest.json') || !matchingAbi) {
continue;
}
loadingUnits.addAll(parseLoadingUnitManifest(file, logger));
}
}
return loadingUnits;
}
/// Parses loading units from a single loading unit manifest json file.
///
/// Returns an empty list if the manifestFile does not exist or is invalid.
static List<LoadingUnit> parseLoadingUnitManifest(File manifestFile, Logger logger) {
if (!manifestFile.existsSync()) {
return <LoadingUnit>[];
}
// Read gen_snapshot manifest
final String fileString = manifestFile.readAsStringSync();
Map<String, dynamic>? manifest;
try {
manifest = jsonDecode(fileString) as Map<String, dynamic>;
} on FormatException catch (e) {
logger.printError('Loading unit manifest at `${manifestFile.path}` was invalid JSON:\n$e');
}
final List<LoadingUnit> loadingUnits = <LoadingUnit>[];
// Setup android source directory
if (manifest != null) {
for (final dynamic loadingUnitMetadata in manifest['loadingUnits'] as List<dynamic>) {
final Map<String, dynamic> loadingUnitMap = loadingUnitMetadata as Map<String, dynamic>;
if (loadingUnitMap['id'] == 1) {
continue; // Skip base unit
}
loadingUnits.add(LoadingUnit(
id: loadingUnitMap['id'] as int,
path: loadingUnitMap['path'] as String,
libraries: List<String>.from(loadingUnitMap['libraries'] as List<dynamic>)),
);
}
}
return loadingUnits;
}
}