2018-12-17 11:49:10 -05:00
|
|
|
// Copyright 2010 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
// Ported from
|
|
|
|
// https://github.com/golang/go/blob/master/src/net/http/responsewrite_test.go
|
|
|
|
|
2019-02-15 11:03:57 -05:00
|
|
|
import { Buffer, copy, Reader } from "deno";
|
2019-02-18 18:32:31 -05:00
|
|
|
import { assert, assertEqual, runTests, test } from "../testing/mod.ts";
|
2019-02-15 11:03:57 -05:00
|
|
|
import {
|
|
|
|
createResponder,
|
|
|
|
createServer,
|
2019-02-18 18:32:31 -05:00
|
|
|
findLongestAndNearestMatch,
|
2019-02-15 11:03:57 -05:00
|
|
|
readRequest,
|
|
|
|
readResponse,
|
|
|
|
ServerResponse,
|
|
|
|
writeResponse
|
|
|
|
} from "./server.ts";
|
|
|
|
import { encode } from "../strings/strings.ts";
|
|
|
|
import { StringReader } from "../io/readers.ts";
|
|
|
|
import { StringWriter } from "../io/writers.ts";
|
|
|
|
import { defer } from "../util/deferred.ts";
|
2018-12-17 11:49:10 -05:00
|
|
|
|
|
|
|
interface ResponseTest {
|
2019-02-15 11:03:57 -05:00
|
|
|
response: ServerResponse;
|
2018-12-17 11:49:10 -05:00
|
|
|
raw: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
const responseTests: ResponseTest[] = [
|
|
|
|
// Default response
|
|
|
|
{
|
|
|
|
response: {},
|
|
|
|
raw: "HTTP/1.1 200 OK\r\n" + "\r\n"
|
|
|
|
},
|
|
|
|
// HTTP/1.1, chunked coding; empty trailer; close
|
|
|
|
{
|
|
|
|
response: {
|
|
|
|
status: 200,
|
2019-02-15 11:03:57 -05:00
|
|
|
body: new Buffer(encode("abcdef"))
|
2018-12-17 11:49:10 -05:00
|
|
|
},
|
|
|
|
|
|
|
|
raw:
|
|
|
|
"HTTP/1.1 200 OK\r\n" +
|
|
|
|
"transfer-encoding: chunked\r\n\r\n" +
|
|
|
|
"6\r\nabcdef\r\n0\r\n\r\n"
|
|
|
|
}
|
|
|
|
];
|
|
|
|
|
2019-02-15 11:03:57 -05:00
|
|
|
test(async function httpWriteResponse() {
|
|
|
|
for (const { raw, response } of responseTests) {
|
2018-12-17 11:49:10 -05:00
|
|
|
const buf = new Buffer();
|
2019-02-15 11:03:57 -05:00
|
|
|
await writeResponse(buf, response);
|
|
|
|
assertEqual(buf.toString(), raw);
|
2018-12-17 11:49:10 -05:00
|
|
|
}
|
|
|
|
});
|
2018-12-18 20:48:05 -05:00
|
|
|
|
2019-02-15 11:03:57 -05:00
|
|
|
test(async function httpReadRequest() {
|
|
|
|
const body = "0123456789";
|
|
|
|
const lines = [
|
|
|
|
"GET /index.html?deno=land HTTP/1.1",
|
|
|
|
"Host: deno.land",
|
|
|
|
"Content-Type: text/plain",
|
|
|
|
`Content-Length: ${body.length}`,
|
|
|
|
"\r\n"
|
|
|
|
];
|
|
|
|
let msg = lines.join("\r\n");
|
|
|
|
msg += body;
|
|
|
|
const req = await readRequest(new StringReader(`${msg}`));
|
|
|
|
assert.equal(req.url, "/index.html?deno=land");
|
|
|
|
assert.equal(req.method, "GET");
|
|
|
|
assert.equal(req.proto, "HTTP/1.1");
|
|
|
|
assert.equal(req.headers.get("host"), "deno.land");
|
|
|
|
assert.equal(req.headers.get("content-type"), "text/plain");
|
|
|
|
assert.equal(req.headers.get("content-length"), `${body.length}`);
|
|
|
|
const w = new StringWriter();
|
|
|
|
await copy(w, req.body);
|
|
|
|
assert.equal(w.toString(), body);
|
|
|
|
});
|
2018-12-18 20:48:05 -05:00
|
|
|
|
2019-02-15 11:03:57 -05:00
|
|
|
test(async function httpReadRequestChunkedBody() {
|
|
|
|
const lines = [
|
|
|
|
"GET /index.html?deno=land HTTP/1.1",
|
|
|
|
"Host: deno.land",
|
|
|
|
"Content-Type: text/plain",
|
|
|
|
`Transfer-Encoding: chunked`,
|
|
|
|
"\r\n"
|
|
|
|
];
|
|
|
|
const hd = lines.join("\r\n");
|
|
|
|
const buf = new Buffer();
|
|
|
|
await buf.write(encode(hd));
|
|
|
|
await buf.write(encode("4\r\ndeno\r\n"));
|
|
|
|
await buf.write(encode("5\r\n.land\r\n"));
|
|
|
|
await buf.write(encode("0\r\n\r\n"));
|
|
|
|
const req = await readRequest(buf);
|
|
|
|
assert.equal(req.url, "/index.html?deno=land");
|
|
|
|
assert.equal(req.method, "GET");
|
|
|
|
assert.equal(req.proto, "HTTP/1.1");
|
|
|
|
assert.equal(req.headers.get("host"), "deno.land");
|
|
|
|
assert.equal(req.headers.get("content-type"), "text/plain");
|
|
|
|
assert.equal(req.headers.get("transfer-encoding"), `chunked`);
|
|
|
|
const dest = new Buffer();
|
|
|
|
await copy(dest, req.body);
|
|
|
|
assert.equal(dest.toString(), "deno.land");
|
2018-12-18 20:48:05 -05:00
|
|
|
});
|
|
|
|
|
2019-02-18 18:32:31 -05:00
|
|
|
test(function httpMatchNearest() {
|
|
|
|
assert.equal(
|
|
|
|
findLongestAndNearestMatch("/foo", ["/foo", "/bar", "/f"]).index,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
findLongestAndNearestMatch("/foo", ["/foo", "/foo/bar"]).index,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
findLongestAndNearestMatch("/foo/bar", [
|
|
|
|
"/",
|
|
|
|
"/foo",
|
|
|
|
"/hoo",
|
|
|
|
"/hoo/foo/bar",
|
|
|
|
"/foo/bar"
|
|
|
|
]).index,
|
|
|
|
4
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
findLongestAndNearestMatch("/foo/bar/foo", ["/foo", "/foo/bar", "/bar/foo"])
|
|
|
|
.index,
|
|
|
|
1
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
findLongestAndNearestMatch("/foo", ["/", "/hoo", "/hoo/foo"]).index,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
findLongestAndNearestMatch("/deno/land", [/d(.+?)o/, /d(.+?)d/]).index,
|
|
|
|
1
|
|
|
|
);
|
|
|
|
assert.equal(findLongestAndNearestMatch("/foo", ["/", "/a/foo"]).index, 0);
|
|
|
|
assert.equal(
|
|
|
|
findLongestAndNearestMatch("/foo", [/\/foo/, /\/bar\/foo/]).index,
|
|
|
|
0
|
|
|
|
);
|
|
|
|
assert.equal(
|
|
|
|
findLongestAndNearestMatch("/foo", [/\/a\/foo/, /\/foo/]).index,
|
|
|
|
1
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2019-02-15 11:03:57 -05:00
|
|
|
test(async function httpServer() {
|
|
|
|
const server = createServer();
|
|
|
|
server.handle("/index", async (req, res) => {
|
|
|
|
await res.respond({
|
|
|
|
status: 200,
|
|
|
|
body: encode("ok")
|
|
|
|
});
|
|
|
|
});
|
|
|
|
server.handle(new RegExp("/foo/(?<id>.+)"), async (req, res) => {
|
|
|
|
const { id } = req.match.groups;
|
|
|
|
await res.respond({
|
|
|
|
status: 200,
|
|
|
|
headers: new Headers({
|
|
|
|
"content-type": "application/json"
|
|
|
|
}),
|
|
|
|
body: encode(JSON.stringify({ id }))
|
|
|
|
});
|
|
|
|
});
|
|
|
|
server.handle("/no-response", async (req, res) => {});
|
|
|
|
const cancel = defer<void>();
|
|
|
|
try {
|
|
|
|
server.listen("127.0.0.1:8080", cancel);
|
|
|
|
{
|
|
|
|
const res1 = await fetch("http://127.0.0.1:8080/index");
|
|
|
|
const text = await res1.body.text();
|
|
|
|
assert.equal(res1.status, 200);
|
|
|
|
assert.equal(text, "ok");
|
2018-12-18 20:48:05 -05:00
|
|
|
}
|
2019-02-15 11:03:57 -05:00
|
|
|
{
|
|
|
|
const res2 = await fetch("http://127.0.0.1:8080/foo/123");
|
|
|
|
const json = await res2.body.json();
|
|
|
|
assert.equal(res2.status, 200);
|
|
|
|
assert.equal(res2.headers.get("content-type"), "application/json");
|
|
|
|
assert.equal(json["id"], "123");
|
|
|
|
}
|
|
|
|
{
|
|
|
|
const res = await fetch("http://127.0.0.1:8080/no-response");
|
|
|
|
assert.equal(res.status, 500);
|
|
|
|
const text = await res.body.text();
|
|
|
|
assert.assert(!!text.match("Not Responded"));
|
|
|
|
}
|
|
|
|
{
|
|
|
|
const res = await fetch("http://127.0.0.1:8080/not-found");
|
|
|
|
const text = await res.body.text();
|
|
|
|
assert.equal(res.status, 404);
|
|
|
|
assert.assert(!!text.match("Not Found"));
|
2018-12-18 20:48:05 -05:00
|
|
|
}
|
2019-02-15 11:03:57 -05:00
|
|
|
} finally {
|
|
|
|
cancel.resolve();
|
2018-12-18 20:48:05 -05:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-02-15 11:03:57 -05:00
|
|
|
test(async function httpServerResponder() {
|
|
|
|
const w = new Buffer();
|
|
|
|
const res = createResponder(w);
|
|
|
|
assert.assert(!res.isResponded);
|
|
|
|
await res.respond({
|
|
|
|
status: 200,
|
|
|
|
headers: new Headers({
|
|
|
|
"content-type": "text/plain"
|
|
|
|
}),
|
|
|
|
body: encode("ok")
|
|
|
|
});
|
|
|
|
assert.assert(res.isResponded);
|
|
|
|
const resp = await readResponse(w);
|
|
|
|
assert.equal(resp.status, 200);
|
|
|
|
assert.equal(resp.headers.get("content-type"), "text/plain");
|
|
|
|
const sw = new StringWriter();
|
|
|
|
await copy(sw, resp.body as Reader);
|
|
|
|
assert.equal(sw.toString(), "ok");
|
|
|
|
});
|
2018-12-18 20:48:05 -05:00
|
|
|
|
2019-02-15 11:03:57 -05:00
|
|
|
test(async function httpServerResponderRespondJson() {
|
|
|
|
const w = new Buffer();
|
|
|
|
const res = createResponder(w);
|
|
|
|
const json = {
|
|
|
|
id: 1,
|
|
|
|
deno: {
|
|
|
|
is: "land"
|
2018-12-18 20:48:05 -05:00
|
|
|
}
|
2019-02-15 11:03:57 -05:00
|
|
|
};
|
|
|
|
assert.assert(!res.isResponded);
|
|
|
|
await res.respondJson(
|
|
|
|
json,
|
|
|
|
new Headers({
|
|
|
|
deno: "land"
|
|
|
|
})
|
|
|
|
);
|
|
|
|
assert.assert(res.isResponded);
|
|
|
|
const resp = await readResponse(w);
|
|
|
|
assert.equal(resp.status, 200);
|
|
|
|
assert.equal(resp.headers.get("content-type"), "application/json");
|
|
|
|
const sw = new StringWriter();
|
|
|
|
await copy(sw, resp.body as Reader);
|
|
|
|
const resJson = JSON.parse(sw.toString());
|
|
|
|
assert.equal(resJson, json);
|
|
|
|
assert.equal(resp.headers.get("deno"), "land");
|
|
|
|
});
|
|
|
|
|
|
|
|
test(async function httpServerResponderRespondText() {
|
|
|
|
const w = new Buffer();
|
|
|
|
const res = createResponder(w);
|
|
|
|
assert.assert(!res.isResponded);
|
|
|
|
await res.respondText(
|
|
|
|
"deno.land",
|
|
|
|
new Headers({
|
|
|
|
deno: "land"
|
|
|
|
})
|
|
|
|
);
|
|
|
|
assert.assert(res.isResponded);
|
|
|
|
const resp = await readResponse(w);
|
|
|
|
assert.equal(resp.status, 200);
|
|
|
|
assert.equal(resp.headers.get("content-type"), "text/plain");
|
|
|
|
const sw = new StringWriter();
|
|
|
|
await copy(sw, resp.body as Reader);
|
|
|
|
assert.equal(sw.toString(), "deno.land");
|
|
|
|
assert.equal(resp.headers.get("deno"), "land");
|
2018-12-18 20:48:05 -05:00
|
|
|
});
|
|
|
|
|
2019-02-15 11:03:57 -05:00
|
|
|
test(async function httpServerResponderShouldThrow() {
|
|
|
|
const w = new Buffer();
|
2018-12-18 20:48:05 -05:00
|
|
|
{
|
2019-02-15 11:03:57 -05:00
|
|
|
const res = createResponder(w);
|
|
|
|
await res.respond({
|
|
|
|
body: null
|
|
|
|
});
|
|
|
|
await assert.throwsAsync(
|
|
|
|
async () => res.respond({ body: null }),
|
|
|
|
Error,
|
|
|
|
"responded"
|
|
|
|
);
|
|
|
|
await assert.throwsAsync(
|
|
|
|
async () => res.respondJson({}),
|
|
|
|
Error,
|
|
|
|
"responded"
|
|
|
|
);
|
|
|
|
await assert.throwsAsync(
|
|
|
|
async () => res.respondText(""),
|
|
|
|
Error,
|
|
|
|
"responded"
|
|
|
|
);
|
2018-12-18 20:48:05 -05:00
|
|
|
}
|
|
|
|
{
|
2019-02-15 11:03:57 -05:00
|
|
|
const res = createResponder(w);
|
|
|
|
await res.respondText("");
|
|
|
|
await assert.throwsAsync(
|
|
|
|
async () => res.respond({ body: null }),
|
|
|
|
Error,
|
|
|
|
"responded"
|
|
|
|
);
|
|
|
|
await assert.throwsAsync(
|
|
|
|
async () => res.respondJson({}),
|
|
|
|
Error,
|
|
|
|
"responded"
|
|
|
|
);
|
|
|
|
await assert.throwsAsync(
|
|
|
|
async () => res.respondText(""),
|
|
|
|
Error,
|
|
|
|
"responded"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
const res = createResponder(w);
|
|
|
|
await res.respondJson({});
|
|
|
|
await assert.throwsAsync(
|
|
|
|
async () => res.respond({ body: null }),
|
|
|
|
Error,
|
|
|
|
"responded"
|
|
|
|
);
|
|
|
|
await assert.throwsAsync(
|
|
|
|
async () => res.respondJson({}),
|
|
|
|
Error,
|
|
|
|
"responded"
|
|
|
|
);
|
|
|
|
await assert.throwsAsync(
|
|
|
|
async () => res.respondText(""),
|
|
|
|
Error,
|
|
|
|
"responded"
|
|
|
|
);
|
2018-12-18 20:48:05 -05:00
|
|
|
}
|
|
|
|
});
|