Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1071)

Unified Diff: sdk/lib/_internal/pub/lib/src/command/serve.dart

Issue 23625002: Support loading transformer plugins from pub. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Code review changes. Created 7 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « sdk/lib/_internal/pub/lib/src/command.dart ('k') | sdk/lib/_internal/pub/lib/src/dart.dart » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: sdk/lib/_internal/pub/lib/src/command/serve.dart
diff --git a/sdk/lib/_internal/pub/lib/src/command/serve.dart b/sdk/lib/_internal/pub/lib/src/command/serve.dart
index 63934280d8204a3e76bad095c709369b6819fc52..d1149bc17fb88956c8e78e1711f125dca6b6a223 100644
--- a/sdk/lib/_internal/pub/lib/src/command/serve.dart
+++ b/sdk/lib/_internal/pub/lib/src/command/serve.dart
@@ -7,21 +7,17 @@ library pub.command.serve;
import 'dart:async';
import 'dart:io';
-import 'package:barback/barback.dart';
-import 'package:path/path.dart' as path;
-import 'package:watcher/watcher.dart';
-
+import '../barback.dart' as barback;
import '../command.dart';
import '../entrypoint.dart';
import '../exit_codes.dart' as exit_codes;
-import '../io.dart';
import '../log.dart' as log;
-import '../pub_package_provider.dart';
import '../utils.dart';
-final _green = getPlatformString('\u001b[32m');
-final _red = getPlatformString('\u001b[31m');
-final _none = getPlatformString('\u001b[0m');
+final _green = getSpecial('\u001b[32m');
+final _red = getSpecial('\u001b[31m');
+final _none = getSpecial('\u001b[0m');
+final _arrow = getSpecial('\u2192', '=>');
/// Handles the `serve` pub command.
class ServeCommand extends PubCommand {
@@ -29,13 +25,6 @@ class ServeCommand extends PubCommand {
String get usage => 'pub serve';
PubPackageProvider _provider;
- Barback _barback;
-
- /// The completer for the top-level future returned by the command.
- ///
- /// Only used to keep pub running (by not completing) and to pipe fatal
- /// errors to pub's top-level error-handling machinery.
- final _commandCompleter = new Completer();
ServeCommand() {
commandParser.addOption('port', defaultsTo: '8080',
@@ -45,35 +34,53 @@ class ServeCommand extends PubCommand {
Future onRun() {
var port = parsePort();
- return ensureLockFileIsUpToDate().then((_) {
- return PubPackageProvider.create(entrypoint);
- }).then((provider) {
- _provider = provider;
+ return ensureLockFileIsUpToDate()
+ .then((_) => entrypoint.loadPackageGraph())
+ .then((graph) => barback.createServer("localhost", port, graph))
+ .then((server) {
+ /// This completer is used to keep pub running (by not completing) and
+ /// to pipe fatal errors to pub's top-level error-handling machinery.
+ var completer = new Completer();
- initBarback();
+ server.barback.errors.listen((error) {
+ log.error("${_red}Build error:\n$error$_none");
+ });
- HttpServer.bind("localhost", port).then((server) {
- watchSources();
+ server.barback.results.listen((result) {
+ if (result.succeeded) {
+ // TODO(rnystrom): Report using growl/inotify-send where available.
+ log.message("Build completed ${_green}successfully$_none");
+ } else {
+ log.message("Build completed with "
+ "${_red}${result.errors.length}$_none errors.");
+ }
+ }, onError: (error) {
+ if (!completer.isCompleted) completer.completeError(error);
+ });
- log.message("Serving ${entrypoint.root.name} "
- "on http://localhost:${server.port}");
+ server.results.listen((result) {
+ if (result.isSuccess) {
+ log.message("${_green}GET$_none ${result.url.path} $_arrow "
+ "${result.id}");
+ return;
+ }
- server.listen(handleRequest);
+ var msg = "${_red}GET$_none ${result.url.path} $_arrow";
+ var error = result.error.toString();
+ if (error.contains("\n")) {
+ log.message("$msg\n${prefixLines(error)}");
+ } else {
+ log.message("$msg $error");
+ }
+ }, onError: (error) {
+ if (!completer.isCompleted) completer.completeError(error);
});
- return _commandCompleter.future;
- });
- }
+ log.message("Serving ${entrypoint.root.name} "
+ "on http://localhost:${server.port}");
- /// Parses the `--port` command-line argument and exits if it isn't valid.
- int parsePort() {
- try {
- return int.parse(commandOptions['port']);
- } on FormatException catch(_) {
- log.error('Could not parse port "${commandOptions['port']}"');
- this.printUsage();
- exit(exit_codes.USAGE);
- }
+ return completer.future;
+ });
}
/// Installs dependencies is the lockfile is out of date with respect to the
@@ -91,185 +98,14 @@ class ServeCommand extends PubCommand {
});
}
- void handleRequest(HttpRequest request) {
- var id = getIdFromUri(request.uri);
- if (id == null) {
- notFound(request, "Path ${request.uri.path} is not valid.");
- return;
- }
-
- _barback.getAssetById(id).then((asset) {
- return validateStream(asset.read()).then((stream) {
- log.message(
- "$_green${request.method}$_none ${request.uri} -> $asset");
- // TODO(rnystrom): Set content-type based on asset type.
- return request.response.addStream(stream).then((_) {
- request.response.close();
- });
- }).catchError((error) {
- log.error("$_red${request.method}$_none "
- "${request.uri} -> $error");
-
- // If we couldn't read the asset, handle the error gracefully.
- if (error is FileException) {
- // Assume this means the asset was a file-backed source asset
- // and we couldn't read it, so treat it like a missing asset.
- notFound(request, error);
- return;
- }
-
- // Otherwise, it's some internal error.
- request.response.statusCode = 500;
- request.response.reasonPhrase = "Internal Error";
- request.response.write(error);
- request.response.close();
- });
- }).catchError((error) {
- log.error("$_red${request.method}$_none ${request.uri} -> $error");
- if (error is! AssetNotFoundException) {
- _commandCompleter.completeError(error);
- return;
- }
-
- notFound(request, error);
- });
- }
-
- /// Responds to [request] with a 404 response and closes it.
- void notFound(HttpRequest request, message) {
- request.response.statusCode = 404;
- request.response.reasonPhrase = "Not Found";
- request.response.write(message);
- request.response.close();
- }
-
- AssetId getIdFromUri(Uri uri) {
- var parts = path.url.split(uri.path);
-
- // Strip the leading "/" from the URL.
- parts.removeAt(0);
-
- var isSpecial = false;
-
- // Checks to see if [uri]'s path contains a special directory [name] that
- // identifies an asset within some package. If so, maps the package name
- // and path following that to be within [dir] inside that package.
- AssetId _trySpecialUrl(String name, String dir) {
- // Find the package name and the relative path in the package.
- var index = parts.indexOf(name);
- if (index == -1) return null;
-
- // If we got here, the path *did* contain the special directory, which
- // means we should not interpret it as a regular path, even if it's
- // missing the package name after it, which makes it invalid here.
- isSpecial = true;
- if (index + 1 >= parts.length) return null;
-
- var package = parts[index + 1];
- var assetPath = path.url.join(dir,
- path.url.joinAll(parts.skip(index + 2)));
- return new AssetId(package, assetPath);
- }
-
- // See if it's "packages" URL.
- var id = _trySpecialUrl("packages", "lib");
- if (id != null) return id;
-
- // See if it's an "assets" URL.
- id = _trySpecialUrl("assets", "asset");
- if (id != null) return id;
-
- // If we got here, we had a path like "/packages" which is a special
- // directory, but not a valid path since it lacks a following package name.
- if (isSpecial) return null;
-
- // Otherwise, it's a path in current package's web directory.
- return new AssetId(entrypoint.root.name,
- path.url.join("web", path.url.joinAll(parts)));
- }
-
- /// Creates the [Barback] instance and listens to its outputs.
- void initBarback() {
- assert(_provider != null);
-
- _barback = new Barback(_provider);
-
- _barback.results.listen((result) {
- if (result.succeeded) {
- // TODO(rnystrom): Report using growl/inotify-send where available.
- log.message("Build completed ${_green}successfully$_none");
- } else {
- log.message("Build completed with "
- "${_red}${result.errors.length}$_none errors.");
- }
- });
-
- _barback.errors.listen((error) {
- log.error("${_red}Build error:\n$error$_none");
- });
- }
-
- /// Adds all of the source assets in the provided packages to barback and
- /// then watches the public directories for changes.
- void watchSources() {
- assert(_provider != null);
- assert(_barback != null);
-
- for (var package in _provider.packages) {
- // Add the initial sources.
- _barback.updateSources(listAssets(package));
-
- // Watch the visible package directories for changes.
- var packageDir = _provider.getPackageDir(package);
-
- for (var name in getPublicDirectories(package)) {
- var subdirectory = path.join(packageDir, name);
- var watcher = new DirectoryWatcher(subdirectory);
- watcher.events.listen((event) {
- var id = new AssetId(package,
- path.relative(event.path, from: packageDir));
- if (event.type == ChangeType.REMOVE) {
- _barback.removeSources([id]);
- } else {
- _barback.updateSources([id]);
- }
- });
- }
- }
- }
-
- /// Lists all of the visible files in [package].
- ///
- /// This is the recursive contents of the "asset" and "lib" directories (if
- /// present). If [package] is the entrypoint package, it also includes the
- /// contents of "web".
- List<AssetId> listAssets(String package) {
- var files = <AssetId>[];
-
- for (var dirPath in getPublicDirectories(package)) {
- var packageDir = _provider.getPackageDir(package);
- var dir = path.join(packageDir, dirPath);
- if (!dirExists(dir)) continue;
- for (var entry in listDir(dir, recursive: true)) {
- // Ignore "packages" symlinks if there.
- if (path.split(entry).contains("packages")) continue;
-
- // Skip directories.
- if (!fileExists(entry)) continue;
-
- var id = new AssetId(package, path.relative(entry, from: packageDir));
- files.add(id);
- }
+ /// Parses the `--port` command-line argument and exits if it isn't valid.
+ int parsePort() {
+ try {
+ return int.parse(commandOptions['port']);
+ } on FormatException catch(_) {
+ log.error('Could not parse port "${commandOptions['port']}"');
+ this.printUsage();
+ exit(exit_codes.USAGE);
}
-
- return files;
- }
-
- /// Gets the names of the top-level directories in [package] whose contents
- /// should be provided as source assets.
- Iterable<String> getPublicDirectories(String package) {
- var directories = ["asset", "lib"];
- if (package == entrypoint.root.name) directories.add("web");
- return directories;
}
}
« no previous file with comments | « sdk/lib/_internal/pub/lib/src/command.dart ('k') | sdk/lib/_internal/pub/lib/src/dart.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698