Skip to content

Commit 34d93e0

Browse files
committed
Avoid barback for precompilation if possible.
Previously we were always loading an AssetEnvironment when we precompiled any executable. Now we only load that environment if some package dependended on by some executable actually uses a transformer. Otherwise, we precompile from disk. The new approach is substantially faster and produces better error messages. Closes #1473
1 parent d0a5744 commit 34d93e0

File tree

3 files changed

+86
-38
lines changed

3 files changed

+86
-38
lines changed

lib/src/barback/asset_environment.dart

+3-20
Original file line numberDiff line numberDiff line change
@@ -10,8 +10,8 @@ import 'package:path/path.dart' as path;
1010
import 'package:watcher/watcher.dart';
1111

1212
import '../cached_package.dart';
13+
import '../dart.dart' as dart;
1314
import '../entrypoint.dart';
14-
import '../exceptions.dart';
1515
import '../io.dart';
1616
import '../log.dart' as log;
1717
import '../package.dart';
@@ -272,18 +272,8 @@ class AssetEnvironment {
272272
await waitAndPrintErrors(executableIds.map((id) async {
273273
var basename = path.url.basename(id.path);
274274
var snapshotPath = path.join(directory, "$basename.snapshot");
275-
var result = await runProcess(Platform.executable, [
276-
'--snapshot=$snapshotPath',
277-
server.url.resolve(basename).toString()
278-
]);
279-
if (result.success) {
280-
log.message("Precompiled ${_formatExecutable(id)}.");
281-
precompiled[path.withoutExtension(basename)] = snapshotPath;
282-
} else {
283-
throw new ApplicationException(
284-
log.yellow("Failed to precompile ${_formatExecutable(id)}:\n") +
285-
result.stderr.join('\n'));
286-
}
275+
await dart.snapshot(server.url.resolve(basename), snapshotPath, id: id);
276+
precompiled[path.withoutExtension(basename)] = snapshotPath;
287277
}));
288278

289279
return precompiled;
@@ -294,13 +284,6 @@ class AssetEnvironment {
294284
}
295285
}
296286

297-
/// Returns the executable name for [id].
298-
///
299-
/// [id] is assumed to be an executable in a bin directory. The return value
300-
/// is intended for log output and may contain formatting.
301-
String _formatExecutable(AssetId id) =>
302-
log.bold("${id.package}:${path.basenameWithoutExtension(id.path)}");
303-
304287
/// Stops the server bound to [rootDirectory].
305288
///
306289
/// Also removes any source files within that directory from barback. Returns

lib/src/dart.dart

+33
Original file line numberDiff line numberDiff line change
@@ -8,12 +8,14 @@ import 'dart:io';
88
import 'dart:isolate';
99

1010
import 'package:analyzer/analyzer.dart';
11+
import 'package:barback/barback.dart';
1112
import 'package:compiler_unsupported/compiler.dart' as compiler;
1213
import 'package:compiler_unsupported/src/filenames.dart'
1314
show appendSlash;
1415
import 'package:path/path.dart' as p;
1516

1617
import 'asset/dart/serialize.dart';
18+
import 'exceptions.dart';
1719
import 'io.dart';
1820
import 'log.dart' as log;
1921

@@ -226,3 +228,34 @@ void _isolateBuffer(message) {
226228
});
227229
});
228230
}
231+
232+
/// Snapshots the Dart executable at [executableUrl] to a snapshot at
233+
/// [snapshotPath].
234+
///
235+
/// If [packagesFile] is passed, it's used to resolve `package:` URIs in the
236+
/// executable. Otherwise, a `packages/` directory or a package spec is inferred
237+
/// from the executable's location.
238+
///
239+
/// If [id] is passed, it's used to describe the executable in logs and error
240+
/// messages.
241+
Future snapshot(Uri executableUrl, String snapshotPath, {Uri packagesFile,
242+
AssetId id}) async {
243+
var name = log.bold(id == null
244+
? executableUrl.toString()
245+
: "${id.package}:${p.url.basenameWithoutExtension(id.path)}");
246+
247+
var args = [
248+
'--snapshot=$snapshotPath',
249+
executableUrl.toString()
250+
];
251+
if (packagesFile != null) args.insert(0, "--packages=$packagesFile");
252+
var result = await runProcess(Platform.executable, args);
253+
254+
if (result.success) {
255+
log.message("Precompiled $name.");
256+
} else {
257+
throw new ApplicationException(
258+
log.yellow("Failed to precompile $name:\n") +
259+
result.stderr.join('\n'));
260+
}
261+
}

lib/src/entrypoint.dart

+50-18
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ import 'package:path/path.dart' as p;
1111
import 'package:pub_semver/pub_semver.dart';
1212

1313
import 'barback/asset_environment.dart';
14+
import 'dart.dart' as dart;
1415
import 'exceptions.dart';
1516
import 'flutter.dart' as flutter;
1617
import 'io.dart';
@@ -214,6 +215,8 @@ class Entrypoint {
214215
_packageGraph = new PackageGraph.fromSolveResult(this, result);
215216
packageGraph.loadTransformerCache().clearIfOutdated(result.changedPackages);
216217

218+
writeTextFile(packagesFile, lockFile.packagesFile(cache, root.name));
219+
217220
try {
218221
if (precompile) {
219222
await _precompileDependencies(changed: result.changedPackages);
@@ -230,8 +233,6 @@ class Entrypoint {
230233
// dependencies, it shouldn't fail unless that fails.
231234
log.exception(error, stackTrace);
232235
}
233-
234-
writeTextFile(packagesFile, lockFile.packagesFile(cache, root.name));
235236
}
236237

237238
/// Precompile any transformed dependencies of the entrypoint.
@@ -335,7 +336,8 @@ class Entrypoint {
335336
Future precompileExecutables({Iterable<String> changed}) async {
336337
_deleteExecutableSnapshots(changed: changed);
337338

338-
var executables = new Map.fromIterable(root.immediateDependencies,
339+
var executables = new Map<String, List<AssetId>>.fromIterable(
340+
root.immediateDependencies,
339341
key: (dep) => dep.name,
340342
value: (dep) => _executablesForPackage(dep.name));
341343

@@ -354,25 +356,55 @@ class Entrypoint {
354356

355357
var packagesToLoad =
356358
unionAll(executables.keys.map(packageGraph.transitiveDependencies))
357-
.map((package) => package.name).toSet();
358-
var executableIds = unionAll(
359-
executables.values.map((ids) => ids.toSet()));
360-
var environment = await AssetEnvironment.create(this, BarbackMode.RELEASE,
361-
packages: packagesToLoad,
362-
entrypoints: executableIds,
363-
useDart2JS: false);
364-
environment.barback.errors.listen((error) {
365-
log.error(log.red("Build error:\n$error"));
366-
});
359+
.toSet();
367360

368-
await waitAndPrintErrors(executables.keys.map((package) async {
369-
var dir = p.join(_snapshotPath, package);
370-
cleanDir(dir);
371-
await environment.precompileExecutables(package, dir,
372-
executableIds: executables[package]);
361+
// Try to avoid starting up an asset server to precompile packages if
362+
// possible. This is faster and produces better error messages.
363+
if (packagesToLoad.every(
364+
(package) => package.pubspec.transformers.isEmpty)) {
365+
await _precompileExecutablesWithoutBarback(executables);
366+
} else {
367+
await _precompileExecutablesWithBarback(executables, packagesToLoad);
368+
}
369+
});
370+
}
371+
372+
Future _precompileExecutablesWithoutBarback(
373+
Map<String, List<AssetId>> executables) {
374+
return waitAndPrintErrors(executables.keys.map((package) {
375+
var dir = p.join(_snapshotPath, package);
376+
cleanDir(dir);
377+
return waitAndPrintErrors(executables[package].map((id) {
378+
var url = p.toUri(packageGraph.packages[id.package].dir);
379+
url = url.replace(path: p.url.join(url.path, id.path));
380+
return dart.snapshot(
381+
url, p.join(dir, p.url.basename(id.path) + '.snapshot'),
382+
packagesFile: p.toUri(packagesFile),
383+
id: id);
373384
}));
385+
}));
386+
}
387+
388+
Future _precompileExecutablesWithBarback(
389+
Map<String, List<AssetId>> executables, Set<Package> packagesToLoad) async {
390+
var executableIds = unionAll(
391+
executables.values.map((ids) => ids.toSet()));
392+
var environment = await AssetEnvironment.create(this, BarbackMode.RELEASE,
393+
packages: packagesToLoad.map((package) => package.name),
394+
entrypoints: executableIds,
395+
useDart2JS: false);
396+
environment.barback.errors.listen((error) {
397+
log.error(log.red("Build error:\n$error"));
374398
});
399+
400+
await waitAndPrintErrors(executables.keys.map((package) async {
401+
var dir = p.join(_snapshotPath, package);
402+
cleanDir(dir);
403+
await environment.precompileExecutables(package, dir,
404+
executableIds: executables[package]);
405+
}));
375406
}
407+
376408

377409
/// Deletes outdated cached executable snapshots.
378410
///

0 commit comments

Comments
 (0)