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

Unified Diff: sdk/lib/_internal/pub/lib/src/barback/server.dart

Issue 26634003: Web Socket API for talking to Editor. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Created 7 years, 2 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
Index: sdk/lib/_internal/pub/lib/src/barback/server.dart
diff --git a/sdk/lib/_internal/pub/lib/src/barback/server.dart b/sdk/lib/_internal/pub/lib/src/barback/server.dart
index 36faf4a8369e591b5c61fea1be022d7247a804d6..22b45e9269321f36ddc84df7b4addb4c5cd2fc50 100644
--- a/sdk/lib/_internal/pub/lib/src/barback/server.dart
+++ b/sdk/lib/_internal/pub/lib/src/barback/server.dart
@@ -5,6 +5,7 @@
library pub.barback.server;
import 'dart:async';
+import 'dart:convert';
import 'dart:io';
import 'package:barback/barback.dart';
@@ -70,6 +71,11 @@ class BarbackServer {
/// Handles an HTTP request.
void _handleRequest(HttpRequest request) {
+ if (WebSocketTransformer.isUpgradeRequest(request)) {
+ _handleWebSocket(request);
+ return;
+ }
+
if (request.method != "GET" && request.method != "HEAD") {
_methodNotAllowed(request);
return;
@@ -77,7 +83,7 @@ class BarbackServer {
var id;
try {
- id = _uriToId(_rootPackage, request.uri);
+ id = _uriToId(request.uri);
} on FormatException catch (ex) {
// 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.
@@ -135,9 +141,65 @@ class BarbackServer {
});
}
+ /// Creates a web socket for [request] which is should be an upgrade request.
nweiz 2013/10/09 19:16:37 "which is should be" -> "which should be"
Bob Nystrom 2013/10/11 20:50:10 Done.
+ void _handleWebSocket(HttpRequest request) {
+ WebSocketTransformer.upgrade(request).then((socket) {
+ socket.listen((data) {
+ var command;
+ try {
+ command = JSON.decode(data);
+ } on FormatException catch (ex) {
+ socket.add(JSON.encode({"error": '"$data" is not valid JSON.'}));
nweiz 2013/10/09 19:16:37 I think we should include a stack trace in the err
Bob Nystrom 2013/10/11 20:50:10 I really don't. In these cases, these are meaningf
+ return;
+ }
+
+ if (command is! Map) {
+ socket.add(JSON.encode({"error": "Command must be a JSON map."}));
nweiz 2013/10/09 19:16:37 Include the actual value. In fact, every error mes
Bob Nystrom 2013/10/11 20:50:10 Done.
Bob Nystrom 2013/10/11 20:50:10 Added to some of them. In general, I don't think t
+ return;
+ }
+
+ if (!command.containsKey("command")) {
+ socket.add(JSON.encode({"error": "Missing command name."}));
+ return;
+ }
+
+ switch (command["command"]) {
+ case "uriToAsset":
nweiz 2013/10/09 19:16:37 Using "URI" in the command names seems weird, sinc
Bob Nystrom 2013/10/11 20:50:10 I've never learned the difference between the two.
nweiz 2013/10/15 23:04:49 My rule of thumb is that anything a browser can un
Bob Nystrom 2013/10/16 00:01:28 Done.
+ var uri = new Uri(path: command["uri"]);
nweiz 2013/10/09 19:16:37 Send an error response if command["uri"] is not a
Bob Nystrom 2013/10/11 20:50:10 Done.
+ var asset = _uriToId(uri);
nweiz 2013/10/09 19:16:37 "asset" -> "id" or "assetId"
Bob Nystrom 2013/10/11 20:50:10 Done.
+ socket.add(JSON.encode({
+ "package": asset.package,
+ "path": asset.path
+ }));
+ break;
+
+ case "assetToUri":
+ var id = new AssetId(command["package"], command["path"]);
nweiz 2013/10/09 19:16:37 Send an error response if command["package"] or co
Bob Nystrom 2013/10/11 20:50:10 Done.
+ try {
+ var uri = _idtoUri(id);
+ socket.add(JSON.encode({"uri": uri}));
+ } on FormatException catch (ex) {
+ socket.add(JSON.encode({"error": ex.message}));
+ }
+ break;
+
+ default:
+ socket.add(JSON.encode({
+ "error": 'Unknown command "${command["command"]}".'
+ }));
+ break;
+ }
+ }, onError: (error) {
+ _resultsController.addError(error);
+ });
nweiz 2013/10/09 19:16:37 This should probably have cancelOnError set to tru
Bob Nystrom 2013/10/11 20:50:10 Done.
+ }).catchError((error) {
+ _resultsController.addError(error);
+ });
+ }
+
/// Converts a [url] served by pub serve into an [AssetId] that can be
/// requested from barback.
- AssetId _uriToId(String rootPackage, Uri url) {
+ AssetId _uriToId(Uri url) {
var id = specialUrlToId(url);
if (id != null) return id;
@@ -145,10 +207,44 @@ class BarbackServer {
var parts = path.url.split(url.path);
// Strip the leading "/" from the URL.
- parts = parts.skip(1);
+ if (parts.isNotEmpty && parts.first == "/") parts = parts.skip(1);
var relativePath = path.url.join("web", path.url.joinAll(parts));
- return new AssetId(rootPackage, relativePath);
+ return new AssetId(_rootPackage, relativePath);
+ }
+
+ /// Converts [id] to at URI path that would access an asset with that ID.
nweiz 2013/10/09 19:16:37 "URI path" -> "root-relative URL"
Bob Nystrom 2013/10/11 20:50:10 This function is now in barback.dart (the pub buil
+ ///
+ /// Throws a [FormatException] if [id] is not a valid servable asset.
+ String _idtoUri(AssetId id) {
+ var parts = path.url.split(id.path);
+
+ if (parts.length < 2) {
+ throw new FormatException(
+ "Can not serve asset from top-level directory.");
nweiz 2013/10/09 19:16:37 "Can not serve asset" -> "Cannot serve assets"
Bob Nystrom 2013/10/11 20:50:10 Done.
+ }
+
+ // Each top-level directory gets handled differently.
+ var dir = parts[0];
+ parts = parts.skip(1);
+
+ switch (dir) {
+ case "asset":
+ return path.join("assets", id.package, path.joinAll(parts));
nweiz 2013/10/09 19:16:37 All of these return values should be root-relative
Bob Nystrom 2013/10/11 20:50:10 Added TODO.
nweiz 2013/10/15 23:04:49 I don't think this warrants a TODO; just add an ad
Bob Nystrom 2013/10/16 00:01:28 Done.
+
+ case "lib":
+ return path.join("packages", id.package, path.joinAll(parts));
+
+ case "web":
+ if (id.package != _rootPackage) {
+ throw new FormatException(
+ 'Can only access "web" directory of root package.');
nweiz 2013/10/09 19:16:37 This is ambiguous; it could mean "of all the asset
Bob Nystrom 2013/10/11 20:50:10 Reworded.
+ }
+ return path.join(path.joinAll(parts));
+
+ default:
+ throw new FormatException('Cannot serve assets from "$dir".');
+ }
}
/// Responds to [request] with a 405 response and closes it.
« no previous file with comments | « no previous file | sdk/lib/_internal/pub/test/serve/utils.dart » ('j') | sdk/lib/_internal/pub/test/serve/utils.dart » ('J')

Powered by Google App Engine
This is Rietveld 408576698