Newer
Older
// Copyright (c) 2014, 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.
library pub.executable;
import 'dart:async';
import 'dart:io';
import 'package:barback/barback.dart';
import 'package:path/path.dart' as p;
import 'package:stack_trace/stack_trace.dart';
import 'barback/asset_environment.dart';
import 'entrypoint.dart';
import 'exit_codes.dart' as exit_codes;
import 'io.dart';
import 'sdk.dart' as sdk;
import 'utils.dart';
/// Runs [executable] from [package] reachable from [entrypoint].
///
/// The executable string is a relative Dart file path using native path
/// separators without a trailing ".dart" extension. It is contained within
/// [package], which should either be the entrypoint package or an immediate
/// dependency of it.
///
/// Arguments from [args] will be passed to the spawned Dart application.
///
/// Returns the exit code of the spawned app.
Future<int> runExecutable(Entrypoint entrypoint, String package,
String executable, Iterable<String> args, {bool isGlobal: false}) {
// Unless the user overrides the verbosity, we want to filter out the
// normal pub output shown while loading the environment.
if (log.verbosity == log.Verbosity.NORMAL) {
log.verbosity = log.Verbosity.WARNING;
}
var localSnapshotPath = p.join(".pub", "bin", package,
"$executable.dart.snapshot");
if (!isGlobal && fileExists(localSnapshotPath)) {
return _runCachedExecutable(entrypoint, localSnapshotPath, args);
// If the command has a path separator, then it's a path relative to the
// root of the package. Otherwise, it's implicitly understood to be in
// "bin".
var rootDir = "bin";
var parts = p.split(executable);
if (parts.length > 1) {
assert(!isGlobal && package == entrypoint.root.name);
rootDir = parts.first;
} else {
executable = p.join("bin", executable);
}
var environment;
// TODO(nweiz): Use [packages] to only load assets from packages that the
// executable might load.
return AssetEnvironment.create(entrypoint, BarbackMode.RELEASE,
useDart2JS: false).then((_environment) {
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
environment = _environment;
environment.barback.errors.listen((error) {
log.error(log.red("Build error:\n$error"));
});
if (package == entrypoint.root.name) {
// Serve the entire root-most directory containing the entrypoint. That
// ensures that, for example, things like `import '../../utils.dart';`
// will work from within some deeply nested script.
return environment.serveDirectory(rootDir);
}
// Make sure the dependency exists.
var dep = entrypoint.root.immediateDependencies.firstWhere(
(dep) => dep.name == package, orElse: () => null);
if (dep == null) {
if (environment.graph.packages.containsKey(package)) {
dataError('Package "$package" is not an immediate dependency.\n'
'Cannot run executables in transitive dependencies.');
} else {
dataError('Could not find package "$package". Did you forget to '
'add a dependency?');
}
}
// For other packages, always use the "bin" directory.
return environment.servePackageBinDirectory(package);
}).then((server) {
// Try to make sure the entrypoint script exists (or is generated) before
// we spawn the process to run it.
var assetPath = "${p.url.joinAll(p.split(executable))}.dart";
var id = new AssetId(server.package, assetPath);
return environment.barback.getAssetById(id).then((_) {
var vmArgs = [];
// Run in checked mode.
// TODO(rnystrom): Make this configurable.
vmArgs.add("--checked");
// Get the URL of the executable, relative to the server's root directory.
var relativePath = p.url.relative(assetPath,
from: p.url.joinAll(p.split(server.rootDirectory)));
vmArgs.add(server.url.resolve(relativePath).toString());
vmArgs.addAll(args);
return Process.start(Platform.executable, vmArgs).then((process) {
// Note: we're not using process.std___.pipe(std___) here because
// that prevents pub from also writing to the output streams.
process.stderr.listen(stderr.add);
process.stdout.listen(stdout.add);
stdin.listen(process.stdin.add);
return process.exitCode;
});
}).catchError((error, stackTrace) {
if (error is! AssetNotFoundException) throw error;
var message = "Could not find ${log.bold(executable + ".dart")}";
if (package != entrypoint.root.name) {
message += " in package ${log.bold(server.package)}";
}
log.error("$message.");
log.fine(new Chain.forTrace(stackTrace));
return exit_codes.NO_INPUT;
});
});
}
/// Runs the snapshot at [path] with [args] and hooks its stdout, stderr, and
/// sdtin to this process's.
///
/// Returns the snapshot's exit code.
///
/// This doesn't do any validation of the snapshot's SDK version.
Future<int> runSnapshot(String path, Iterable<String> args) {
var vmArgs = [path]..addAll(args);
return Process.start(Platform.executable, vmArgs).then((process) {
// Note: we're not using process.std___.pipe(std___) here because
// that prevents pub from also writing to the output streams.
process.stderr.listen(stderr.add);
process.stdout.listen(stdout.add);
stdin.listen(process.stdin.add);
return process.exitCode;
});
}
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
/// Runs the executable snapshot at [snapshotPath].
Future _runCachedExecutable(Entrypoint entrypoint, String snapshotPath,
List<String> args) {
return syncFuture(() {
// If the snapshot was compiled with a different SDK version, we need to
// recompile it.
var sdkVersionPath = p.join(".pub", "bin", "sdk-version");
if (fileExists(sdkVersionPath) &&
readTextFile(sdkVersionPath) == "${sdk.version}\n") {
return null;
}
log.fine("Precompiled executables are out of date.");
return entrypoint.precompileExecutables();
}).then((_) {
var vmArgs = ["--checked", snapshotPath]..addAll(args);
return Process.start(Platform.executable, vmArgs).then((process) {
// Note: we're not using process.std___.pipe(std___) here because
// that prevents pub from also writing to the output streams.
process.stderr.listen(stderr.add);
process.stdout.listen(stdout.add);
stdin.listen(process.stdin.add);
return process.exitCode;
});
});
}