Migrated http_detach_socket_test from SDK. (#56)

diff --git a/test/http_detach_socket_test.dart b/test/http_detach_socket_test.dart
new file mode 100644
index 0000000..6e712ab
--- /dev/null
+++ b/test/http_detach_socket_test.dart
@@ -0,0 +1,203 @@
+// Copyright (c) 2018, 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.
+
+import 'dart:async';
+import 'dart:io' show ServerSocket, Socket;
+
+import 'package:http_io/http_io.dart';
+import 'package:test/test.dart';
+
+Future<Null> testServerDetachSocket() {
+  final completer = new Completer<Null>();
+  HttpServer.bind("127.0.0.1", 0).then((server) {
+    server.defaultResponseHeaders.clear();
+    server.serverHeader = null;
+    server.listen((request) {
+      var response = request.response;
+      response.contentLength = 0;
+      response.detachSocket().then((socket) {
+        expect(socket, isNotNull);
+        var body = new StringBuffer();
+        socket.listen((data) => body.write(new String.fromCharCodes(data)),
+            onDone: () => expect("Some data", body.toString()));
+        socket.write("Test!");
+        socket.close();
+      });
+      server.close();
+      completer.complete();
+    });
+
+    Socket.connect("127.0.0.1", server.port).then((socket) {
+      socket.write("GET / HTTP/1.1\r\n"
+          "content-length: 0\r\n\r\n"
+          "Some data");
+      var body = new StringBuffer();
+      socket.listen((data) => body.write(new String.fromCharCodes(data)),
+          onDone: () {
+        expect(
+            "HTTP/1.1 200 OK\r\n"
+            "content-length: 0\r\n"
+            "\r\n"
+            "Test!",
+            body.toString());
+        socket.close();
+      });
+    });
+  });
+  return completer.future;
+}
+
+Future<Null> testServerDetachSocketNoWriteHeaders() {
+  final completer = new Completer<Null>();
+  HttpServer.bind("127.0.0.1", 0).then((server) {
+    server.listen((request) {
+      var response = request.response;
+      response.contentLength = 0;
+      response.detachSocket(writeHeaders: false).then((socket) {
+        expect(socket, isNotNull);
+        var body = new StringBuffer();
+        socket.listen((data) => body.write(new String.fromCharCodes(data)),
+            onDone: () => expect("Some data", body.toString()));
+        socket.write("Test!");
+        socket.close();
+      });
+      server.close();
+      completer.complete();
+    });
+
+    Socket.connect("127.0.0.1", server.port).then((socket) {
+      socket.write("GET / HTTP/1.1\r\n"
+          "content-length: 0\r\n\r\n"
+          "Some data");
+      var body = new StringBuffer();
+      socket.listen((data) => body.write(new String.fromCharCodes(data)),
+          onDone: () {
+        expect("Test!", body.toString());
+        socket.close();
+      });
+    });
+  });
+  return completer.future;
+}
+
+Future<Null> testBadServerDetachSocket() {
+  final completer = new Completer<Null>();
+  HttpServer.bind("127.0.0.1", 0).then((server) {
+    server.listen((request) {
+      var response = request.response;
+      response.contentLength = 0;
+      response.close();
+      expect(response.detachSocket, throwsA(new isInstanceOf<StateError>()));
+      server.close();
+      completer.complete();
+    });
+
+    Socket.connect("127.0.0.1", server.port).then((socket) {
+      socket.write("GET / HTTP/1.1\r\n"
+          "content-length: 0\r\n\r\n");
+      socket.listen((_) {}, onDone: () {
+        socket.close();
+      });
+    });
+  });
+  return completer.future;
+}
+
+Future<Null> testClientDetachSocket() {
+  final completer = new Completer<Null>();
+  ServerSocket.bind("127.0.0.1", 0).then((server) {
+    server.listen((socket) {
+      int port = server.port;
+      socket.write("HTTP/1.1 200 OK\r\n"
+          "\r\n"
+          "Test!");
+      var body = new StringBuffer();
+      socket.listen((data) => body.write(new String.fromCharCodes(data)),
+          onDone: () {
+        List<String> lines = body.toString().split("\r\n");
+        expect(6, lines.length);
+        expect("GET / HTTP/1.1", lines[0]);
+        expect("", lines[4]);
+        expect("Some data", lines[5]);
+        lines.sort(); // Lines 1-3 becomes 3-5 in a fixed order.
+        expect("accept-encoding: gzip", lines[3]);
+        expect("content-length: 0", lines[4]);
+        expect("host: 127.0.0.1:${port}", lines[5]);
+        socket.close();
+      });
+      server.close();
+      completer.complete();
+    });
+
+    var client = new HttpClient();
+    client.userAgent = null;
+    client
+        .get("127.0.0.1", server.port, "/")
+        .then((request) => request.close())
+        .then((response) {
+      response.detachSocket().then((socket) {
+        var body = new StringBuffer();
+        socket.listen((data) => body.write(new String.fromCharCodes(data)),
+            onDone: () {
+          expect("Test!", body.toString());
+          client.close();
+        });
+        socket.write("Some data");
+        socket.close();
+      });
+    });
+  });
+  return completer.future;
+}
+
+Future<Null> testUpgradedConnection() {
+  final completer = new Completer<Null>();
+  HttpServer.bind("127.0.0.1", 0).then((server) {
+    server.listen((request) {
+      request.response.headers.set('connection', 'upgrade');
+      if (request.headers.value('upgrade') == 'mine') {
+        request.response.detachSocket().then((socket) {
+          socket.pipe(socket).then((_) {});
+        });
+      } else {
+        request.response.close();
+      }
+    });
+
+    var client = new HttpClient();
+    client.userAgent = null;
+    client.get("127.0.0.1", server.port, "/").then((request) {
+      request.headers.set('upgrade', 'mine');
+      return request.close();
+    }).then((response) {
+      client.get("127.0.0.1", server.port, "/").then((request) {
+        response.detachSocket().then((socket) {
+          // We are testing that we can detach the socket, even though
+          // we made a new connection (testing it was not reused).
+          request.close().then((response) {
+            response.listen(null, onDone: () {
+              server.close();
+              completer.complete();
+            });
+            socket.add([0]);
+            socket.close();
+            socket.fold([], (l, d) => l..addAll(d)).then((data) {
+              expect([0], data);
+            });
+          });
+        });
+      });
+    });
+  });
+  return completer.future;
+}
+
+void main() {
+  test('serverDetachSocket', testServerDetachSocket);
+  test(
+      'serverDetachSocketNoWriteHeaders', testServerDetachSocketNoWriteHeaders);
+  test('badServerDetachSocket', testBadServerDetachSocket);
+  test('clientDetachSocket', testClientDetachSocket);
+  test('upgradedConnection', testUpgradedConnection);
+}