From b07647c8a8da71b36dc0274939aed35e73da0b3f Mon Sep 17 00:00:00 2001
From: "rnystrom@google.com" <rnystrom@google.com>
Date: Mon, 12 Aug 2013 21:32:26 +0000
Subject: [PATCH] Handle missing files more gracefully in pub serve.

BUG=
R=nweiz@google.com

Review URL: https://codereview.chromium.org//22878002

git-svn-id: https://dart.googlecode.com/svn/branches/bleeding_edge@26036 260f80e4-7a28-3924-810f-c04153c831b5
---
 lib/src/command/serve.dart         | 35 +++++++++++++++++++-------
 lib/src/utils.dart                 | 40 ++++++++++++++++++++++++++++++
 test/serve/missing_asset_test.dart | 28 +++++++++++++++++++++
 test/serve/missing_file_test.dart  | 32 ++++++++++++++++++++++--
 4 files changed, 124 insertions(+), 11 deletions(-)
 create mode 100644 test/serve/missing_asset_test.dart

diff --git a/lib/src/command/serve.dart b/lib/src/command/serve.dart
index ce8db31d..69d92a1f 100644
--- a/lib/src/command/serve.dart
+++ b/lib/src/command/serve.dart
@@ -78,14 +78,14 @@ class ServeCommand extends PubCommand {
       // are added or modified.
 
       HttpServer.bind("localhost", port).then((server) {
-        log.message("Serving ${entrypoint.root.name} "
-                    "on http://localhost:${server.port}");
-
         // Add all of the visible files.
         for (var package in provider.packages) {
           barback.updateSources(provider.listAssets(package));
         }
 
+        log.message("Serving ${entrypoint.root.name} "
+            "on http://localhost:${server.port}");
+
         server.listen((request) {
           var id = getIdFromUri(request.uri);
           if (id == null) {
@@ -93,14 +93,31 @@ class ServeCommand extends PubCommand {
           }
 
           barback.getAssetById(id).then((asset) {
-            log.message(
-                "$_green${request.method}$_none ${request.uri} -> $asset");
-            // TODO(rnystrom): Set content-type based on asset type.
-            return request.response.addStream(asset.read()).then((_) {
+            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();
             });
-            // TODO(rnystrom): Serve up a 500 if we get an error reading the
-            // asset.
           }).catchError((error) {
             log.error("$_red${request.method}$_none ${request.uri} -> $error");
             if (error is! AssetNotFoundException) {
diff --git a/lib/src/utils.dart b/lib/src/utils.dart
index 6a048cf1..da8f640e 100644
--- a/lib/src/utils.dart
+++ b/lib/src/utils.dart
@@ -168,6 +168,46 @@ void chainToCompleter(Future future, Completer completer) {
       onError: (e) => completer.completeError(e));
 }
 
+/// Ensures that [stream] can emit at least one value successfully (or close
+/// without any values).
+///
+/// For example, reading asynchronously from a non-existent file will return a
+/// stream that fails on the first chunk. In order to handle that more
+/// gracefully, you may want to check that the stream looks like it's working
+/// before you pipe the stream to something else.
+///
+/// This lets you do that. It returns a [Future] that completes to a [Stream]
+/// emitting the same values and errors as [stream], but only if at least one
+/// value can be read successfully. If an error occurs before any values are
+/// emitted, the returned Future completes to that error.
+Future<Stream> validateStream(Stream stream) {
+  var completer = new Completer<Stream>();
+  var controller = new StreamController(sync: true);
+
+  StreamSubscription subscription;
+  subscription = stream.listen((value) {
+    // We got a value, so the stream is valid.
+    if (!completer.isCompleted) completer.complete(controller.stream);
+    controller.add(value);
+  }, onError: (error) {
+    // If the error came after values, it's OK.
+    if (completer.isCompleted) controller.addError(error);
+
+    // Otherwise, the error came first and the stream is invalid.
+    completer.completeError(error);
+
+    // We don't be returning the stream at all in this case, so unsubscribe
+    // and swallow the error.
+    subscription.cancel();
+  }, onDone: () {
+    // It closed with no errors, so the stream is valid.
+    if (!completer.isCompleted) completer.complete(controller.stream);
+    controller.close();
+  });
+
+  return completer.future;
+}
+
 // TODO(nweiz): remove this when issue 7964 is fixed.
 /// Returns a [Future] that will complete to the first element of [stream].
 /// Unlike [Stream.first], this is safe to use with single-subscription streams.
diff --git a/test/serve/missing_asset_test.dart b/test/serve/missing_asset_test.dart
new file mode 100644
index 00000000..1f69b6c4
--- /dev/null
+++ b/test/serve/missing_asset_test.dart
@@ -0,0 +1,28 @@
+// Copyright (c) 2013, the Dart project authors.  Please see the AUTHORS d.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_tests;
+
+import 'dart:io';
+
+import '../descriptor.dart' as d;
+import '../test_pub.dart';
+import 'utils.dart';
+
+main() {
+  initConfig();
+  integration("responds with a 404 for missing assets", () {
+    d.dir(appPath, [
+      d.appPubspec()
+    ]).create();
+
+    startPubServe();
+    requestShould404("index.html");
+    requestShould404("packages/myapp/nope.dart");
+    requestShould404("assets/myapp/nope.png");
+    requestShould404("dir/packages/myapp/nope.dart");
+    requestShould404("dir/assets/myapp/nope.png");
+    endPubServe();
+  });
+}
diff --git a/test/serve/missing_file_test.dart b/test/serve/missing_file_test.dart
index cfd7d077..693ae553 100644
--- a/test/serve/missing_file_test.dart
+++ b/test/serve/missing_file_test.dart
@@ -4,8 +4,10 @@
 
 library pub_tests;
 
-import 'dart:io';
+import 'package:path/path.dart' as path;
+import 'package:scheduled_test/scheduled_test.dart';
 
+import '../../lib/src/io.dart';
 import '../descriptor.dart' as d;
 import '../test_pub.dart';
 import 'utils.dart';
@@ -14,10 +16,36 @@ main() {
   initConfig();
   integration("responds with a 404 for missing files", () {
     d.dir(appPath, [
-      d.appPubspec()
+      d.appPubspec(),
+      d.dir("asset", [
+        d.file("nope.png", "nope")
+      ]),
+      d.dir("lib", [
+        d.file("nope.dart", "nope")
+      ]),
+      d.dir("web", [
+        d.file("index.html", "<body>"),
+      ])
     ]).create();
 
+    // Start the server with the files present so that it creates barback
+    // assets for them.
     startPubServe();
+
+    // TODO(rnystrom): When pub serve supports file watching, we'll have to do
+    // something here to specifically disable that so that we can get barback
+    // into the inconsistent state of thinking there is an asset but where the
+    // underlying file does not exist. One option would be configure barback
+    // with an insanely long delay between polling to ensure a poll doesn't
+    // happen.
+
+    // Now delete them.
+    schedule(() {
+      deleteEntry(path.join(sandboxDir, appPath, "asset", "nope.png"));
+      deleteEntry(path.join(sandboxDir, appPath, "lib", "nope.dart"));
+      deleteEntry(path.join(sandboxDir, appPath, "web", "index.html"));
+    }, "delete files");
+
     requestShould404("index.html");
     requestShould404("packages/myapp/nope.dart");
     requestShould404("assets/myapp/nope.png");
-- 
GitLab