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; |
} |
} |