blob: 03b520a1450b2c866d60587a788d443dafd5bf68 [file] [log] [blame]
// Copyright (c) 2018, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'dart:async';
import 'dart:convert';
// ignore: deprecated_member_use
import 'package:analyzer/analyzer.dart';
import 'package:build/build.dart';
import 'package:build_modules/build_modules.dart';
import 'package:pool/pool.dart';
import '../builders.dart';
import 'platform.dart';
/// Because we hold bytes in memory we don't want to compile to many app entry
/// points at once.
final _buildPool = Pool(16);
/// A builder which combines several [vmKernelModuleExtension] modules into a
/// single [vmKernelEntrypointExtension] file, which represents an entire
/// application.
class VmEntrypointBuilder implements Builder {
const VmEntrypointBuilder();
@override
final buildExtensions = const {
'.dart': [vmKernelEntrypointExtension],
};
@override
Future<void> build(BuildStep buildStep) async {
await _buildPool.withResource(() async {
var dartEntrypointId = buildStep.inputId;
var isAppEntrypoint = await _isAppEntryPoint(dartEntrypointId, buildStep);
if (!isAppEntrypoint) return;
var moduleId =
buildStep.inputId.changeExtension(moduleExtension(vmPlatform));
var module = Module.fromJson(
json.decode(await buildStep.readAsString(moduleId))
as Map<String, dynamic>);
List<Module> transitiveModules;
try {
transitiveModules = await module
.computeTransitiveDependencies(buildStep, throwIfUnsupported: true);
} on UnsupportedModules catch (e) {
var librariesString = (await e.exactLibraries(buildStep).toList())
.map((lib) => AssetId(lib.id.package,
lib.id.path.replaceFirst(moduleLibraryExtension, '.dart')))
.join('\n');
log.warning('''
Skipping compiling ${buildStep.inputId} for the vm because some of its
transitive libraries have sdk dependencies that not supported on this platform:
$librariesString
https://github.com/dart-lang/build/blob/master/docs/faq.md#how-can-i-resolve-skipped-compiling-warnings
''');
return;
}
var transitiveKernelModules = [
module.primarySource.changeExtension(vmKernelModuleExtension)
].followedBy(transitiveModules.reversed.map(
(m) => m.primarySource.changeExtension(vmKernelModuleExtension)));
var appContents = <int>[];
for (var dependencyId in transitiveKernelModules) {
appContents.addAll(await buildStep.readAsBytes(dependencyId));
}
await buildStep.writeAsBytes(
buildStep.inputId.changeExtension(vmKernelEntrypointExtension),
appContents);
});
}
}
/// Returns whether or not [dartId] is an app entrypoint (basically, whether
/// or not it has a `main` function).
Future<bool> _isAppEntryPoint(AssetId dartId, AssetReader reader) async {
assert(dartId.extension == '.dart');
// Skip reporting errors here, dartdevc will report them later with nicer
// formatting.
// ignore: deprecated_member_use
var parsed = parseCompilationUnit(await reader.readAsString(dartId),
suppressErrors: true);
// Allow two or fewer arguments so that entrypoints intended for use with
// [spawnUri] get counted.
//
// TODO: This misses the case where a Dart file doesn't contain main(),
// but has a part that does, or it exports a `main` from another library.
return parsed.declarations.any((node) {
return node is FunctionDeclaration &&
node.name.name == 'main' &&
node.functionExpression.parameters.parameters.length <= 2;
});
}