summaryrefslogtreecommitdiff
path: root/std/http/file_server_test.ts
diff options
context:
space:
mode:
authorCasper Beyer <caspervonb@pm.me>2021-02-02 19:05:46 +0800
committerGitHub <noreply@github.com>2021-02-02 12:05:46 +0100
commit6abf126c2a7a451cded8c6b5e6ddf1b69c84055d (patch)
treefd94c013a19fcb38954844085821ec1601c20e18 /std/http/file_server_test.ts
parenta2b5d44f1aa9d64f448a2a3cc2001272e2f60b98 (diff)
chore: remove std directory (#9361)
This removes the std folder from the tree. Various parts of the tests are pretty tightly dependent on std (47 direct imports and 75 indirect imports, not counting the cli tests that use them as fixtures) so I've added std as a submodule for now.
Diffstat (limited to 'std/http/file_server_test.ts')
-rw-r--r--std/http/file_server_test.ts465
1 files changed, 0 insertions, 465 deletions
diff --git a/std/http/file_server_test.ts b/std/http/file_server_test.ts
deleted file mode 100644
index 638121b45..000000000
--- a/std/http/file_server_test.ts
+++ /dev/null
@@ -1,465 +0,0 @@
-// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
-import {
- assert,
- assertEquals,
- assertNotEquals,
- assertStringIncludes,
-} from "../testing/asserts.ts";
-import { BufReader } from "../io/bufio.ts";
-import { TextProtoReader } from "../textproto/mod.ts";
-import { Response, ServerRequest } from "./server.ts";
-import { FileServerArgs, serveFile } from "./file_server.ts";
-import { dirname, fromFileUrl, join, resolve } from "../path/mod.ts";
-let fileServer: Deno.Process<Deno.RunOptions & { stdout: "piped" }>;
-
-type FileServerCfg = Omit<FileServerArgs, "_"> & { target?: string };
-
-const moduleDir = dirname(fromFileUrl(import.meta.url));
-const testdataDir = resolve(moduleDir, "testdata");
-
-async function startFileServer({
- target = ".",
- port = 4507,
- "dir-listing": dirListing = true,
-}: FileServerCfg = {}): Promise<void> {
- fileServer = Deno.run({
- cmd: [
- Deno.execPath(),
- "run",
- "--quiet",
- "--allow-read",
- "--allow-net",
- "file_server.ts",
- target,
- "--cors",
- "-p",
- `${port}`,
- `${dirListing ? "" : "--no-dir-listing"}`,
- ],
- cwd: moduleDir,
- stdout: "piped",
- stderr: "null",
- });
- // Once fileServer is ready it will write to its stdout.
- assert(fileServer.stdout != null);
- const r = new TextProtoReader(new BufReader(fileServer.stdout));
- const s = await r.readLine();
- assert(s !== null && s.includes("server listening"));
-}
-
-async function startFileServerAsLibrary({}: FileServerCfg = {}): Promise<void> {
- fileServer = await Deno.run({
- cmd: [
- Deno.execPath(),
- "run",
- "--quiet",
- "--allow-read",
- "--allow-net",
- "testdata/file_server_as_library.ts",
- ],
- cwd: moduleDir,
- stdout: "piped",
- stderr: "null",
- });
- assert(fileServer.stdout != null);
- const r = new TextProtoReader(new BufReader(fileServer.stdout));
- const s = await r.readLine();
- assert(s !== null && s.includes("Server running..."));
-}
-
-async function killFileServer(): Promise<void> {
- fileServer.close();
- // Process.close() kills the file server process. However this termination
- // happens asynchronously, and since we've just closed the process resource,
- // we can't use `await fileServer.status()` to wait for the process to have
- // exited. As a workaround, wait for its stdout to close instead.
- // TODO(piscisaureus): when `Process.kill()` is stable and works on Windows,
- // switch to calling `kill()` followed by `await fileServer.status()`.
- await Deno.readAll(fileServer.stdout!);
- fileServer.stdout!.close();
-}
-
-interface StringResponse extends Response {
- body: string;
-}
-
-/* HTTP GET request allowing arbitrary paths */
-async function fetchExactPath(
- hostname: string,
- port: number,
- path: string,
-): Promise<StringResponse> {
- const encoder = new TextEncoder();
- const decoder = new TextDecoder();
- const request = encoder.encode("GET " + path + " HTTP/1.1\r\n\r\n");
- let conn: void | Deno.Conn;
- try {
- conn = await Deno.connect(
- { hostname: hostname, port: port, transport: "tcp" },
- );
- await Deno.writeAll(conn, request);
- let currentResult = "";
- let contentLength = -1;
- let startOfBody = -1;
- for await (const chunk of Deno.iter(conn)) {
- currentResult += decoder.decode(chunk);
- if (contentLength === -1) {
- const match = /^content-length: (.*)$/m.exec(currentResult);
- if (match && match[1]) {
- contentLength = Number(match[1]);
- }
- }
- if (startOfBody === -1) {
- const ind = currentResult.indexOf("\r\n\r\n");
- if (ind !== -1) {
- startOfBody = ind + 4;
- }
- }
- if (startOfBody !== -1 && contentLength !== -1) {
- const byteLen = encoder.encode(currentResult).length;
- if (byteLen >= contentLength + startOfBody) {
- break;
- }
- }
- }
- const status = /^HTTP\/1.1 (...)/.exec(currentResult);
- let statusCode = 0;
- if (status && status[1]) {
- statusCode = Number(status[1]);
- }
-
- const body = currentResult.slice(startOfBody);
- const headersStr = currentResult.slice(0, startOfBody);
- const headersReg = /^(.*): (.*)$/mg;
- const headersObj: { [i: string]: string } = {};
- let match = headersReg.exec(headersStr);
- while (match !== null) {
- if (match[1] && match[2]) {
- headersObj[match[1]] = match[2];
- }
- match = headersReg.exec(headersStr);
- }
- return {
- status: statusCode,
- headers: new Headers(headersObj),
- body: body,
- };
- } finally {
- if (conn) {
- Deno.close(conn.rid);
- }
- }
-}
-
-Deno.test(
- "file_server serveFile",
- async (): Promise<void> => {
- await startFileServer();
- try {
- const res = await fetch("http://localhost:4507/README.md");
- assert(res.headers.has("access-control-allow-origin"));
- assert(res.headers.has("access-control-allow-headers"));
- assertEquals(res.headers.get("content-type"), "text/markdown");
- const downloadedFile = await res.text();
- const localFile = new TextDecoder().decode(
- await Deno.readFile(join(moduleDir, "README.md")),
- );
- assertEquals(downloadedFile, localFile);
- } finally {
- await killFileServer();
- }
- },
-);
-
-Deno.test(
- "file_server serveFile in testdata",
- async (): Promise<void> => {
- await startFileServer({ target: "./testdata" });
- try {
- const res = await fetch("http://localhost:4507/hello.html");
- assert(res.headers.has("access-control-allow-origin"));
- assert(res.headers.has("access-control-allow-headers"));
- assertEquals(res.headers.get("content-type"), "text/html");
- const downloadedFile = await res.text();
- const localFile = new TextDecoder().decode(
- await Deno.readFile(join(testdataDir, "hello.html")),
- );
- assertEquals(downloadedFile, localFile);
- } finally {
- await killFileServer();
- }
- },
-);
-
-Deno.test("serveDirectory", async function (): Promise<void> {
- await startFileServer();
- try {
- const res = await fetch("http://localhost:4507/");
- assert(res.headers.has("access-control-allow-origin"));
- assert(res.headers.has("access-control-allow-headers"));
- const page = await res.text();
- assert(page.includes("README.md"));
-
- // `Deno.FileInfo` is not completely compatible with Windows yet
- // TODO(bartlomieju): `mode` should work correctly in the future.
- // Correct this test case accordingly.
- Deno.build.os !== "windows" &&
- assert(/<td class="mode">(\s)*\([a-zA-Z-]{10}\)(\s)*<\/td>/.test(page));
- Deno.build.os === "windows" &&
- assert(/<td class="mode">(\s)*\(unknown mode\)(\s)*<\/td>/.test(page));
- assert(page.includes(`<a href="/README.md">README.md</a>`));
- } finally {
- await killFileServer();
- }
-});
-
-Deno.test("serveFallback", async function (): Promise<void> {
- await startFileServer();
- try {
- const res = await fetch("http://localhost:4507/badfile.txt");
- assert(res.headers.has("access-control-allow-origin"));
- assert(res.headers.has("access-control-allow-headers"));
- assertEquals(res.status, 404);
- const _ = await res.text();
- } finally {
- await killFileServer();
- }
-});
-
-Deno.test("checkPathTraversal", async function (): Promise<void> {
- await startFileServer();
- try {
- const res = await fetch(
- "http://localhost:4507/../../../../../../../..",
- );
- assert(res.headers.has("access-control-allow-origin"));
- assert(res.headers.has("access-control-allow-headers"));
- assertEquals(res.status, 200);
- const listing = await res.text();
- assertStringIncludes(listing, "README.md");
- } finally {
- await killFileServer();
- }
-});
-
-Deno.test("checkPathTraversalNoLeadingSlash", async function (): Promise<void> {
- await startFileServer();
- try {
- const res = await fetchExactPath("127.0.0.1", 4507, "../../../..");
- assertEquals(res.status, 400);
- } finally {
- await killFileServer();
- }
-});
-
-Deno.test("checkPathTraversalAbsoluteURI", async function (): Promise<void> {
- await startFileServer();
- try {
- //allowed per https://www.w3.org/Protocols/rfc2616/rfc2616-sec5.html
- const res = await fetchExactPath(
- "127.0.0.1",
- 4507,
- "http://localhost/../../../..",
- );
- assertEquals(res.status, 200);
- assertStringIncludes(res.body, "README.md");
- } finally {
- await killFileServer();
- }
-});
-
-Deno.test("checkURIEncodedPathTraversal", async function (): Promise<void> {
- await startFileServer();
- try {
- const res = await fetch(
- "http://localhost:4507/%2F..%2F..%2F..%2F..%2F..%2F..%2F..%2F..",
- );
- assert(res.headers.has("access-control-allow-origin"));
- assert(res.headers.has("access-control-allow-headers"));
- assertEquals(res.status, 404);
- const _ = await res.text();
- } finally {
- await killFileServer();
- }
-});
-
-Deno.test("serveWithUnorthodoxFilename", async function (): Promise<void> {
- await startFileServer();
- try {
- let res = await fetch("http://localhost:4507/testdata/%");
- assert(res.headers.has("access-control-allow-origin"));
- assert(res.headers.has("access-control-allow-headers"));
- assertEquals(res.status, 200);
- let _ = await res.text();
- res = await fetch("http://localhost:4507/testdata/test%20file.txt");
- assert(res.headers.has("access-control-allow-origin"));
- assert(res.headers.has("access-control-allow-headers"));
- assertEquals(res.status, 200);
- _ = await res.text();
- } finally {
- await killFileServer();
- }
-});
-
-Deno.test("printHelp", async function (): Promise<void> {
- const helpProcess = Deno.run({
- cmd: [
- Deno.execPath(),
- "run",
- "--quiet",
- // TODO(ry) It ought to be possible to get the help output without
- // --allow-read.
- "--allow-read",
- "file_server.ts",
- "--help",
- ],
- cwd: moduleDir,
- stdout: "piped",
- });
- assert(helpProcess.stdout != null);
- const r = new TextProtoReader(new BufReader(helpProcess.stdout));
- const s = await r.readLine();
- assert(s !== null && s.includes("Deno File Server"));
- helpProcess.close();
- helpProcess.stdout.close();
-});
-
-Deno.test("contentType", async () => {
- const request = new ServerRequest();
- const response = await serveFile(request, join(testdataDir, "hello.html"));
- const contentType = response.headers!.get("content-type");
- assertEquals(contentType, "text/html");
- (response.body as Deno.File).close();
-});
-
-Deno.test("file_server running as library", async function (): Promise<void> {
- await startFileServerAsLibrary();
- try {
- const res = await fetch("http://localhost:8000");
- assertEquals(res.status, 200);
- const _ = await res.text();
- } finally {
- await killFileServer();
- }
-});
-
-Deno.test("file_server should ignore query params", async () => {
- await startFileServer();
- try {
- const res = await fetch("http://localhost:4507/README.md?key=value");
- assertEquals(res.status, 200);
- const downloadedFile = await res.text();
- const localFile = new TextDecoder().decode(
- await Deno.readFile(join(moduleDir, "README.md")),
- );
- assertEquals(downloadedFile, localFile);
- } finally {
- await killFileServer();
- }
-});
-
-async function startTlsFileServer({
- target = ".",
- port = 4577,
-}: FileServerCfg = {}): Promise<void> {
- fileServer = Deno.run({
- cmd: [
- Deno.execPath(),
- "run",
- "--quiet",
- "--allow-read",
- "--allow-net",
- "file_server.ts",
- target,
- "--host",
- "localhost",
- "--cert",
- "./testdata/tls/localhost.crt",
- "--key",
- "./testdata/tls/localhost.key",
- "--cors",
- "-p",
- `${port}`,
- ],
- cwd: moduleDir,
- stdout: "piped",
- stderr: "null",
- });
- // Once fileServer is ready it will write to its stdout.
- assert(fileServer.stdout != null);
- const r = new TextProtoReader(new BufReader(fileServer.stdout));
- const s = await r.readLine();
- assert(s !== null && s.includes("server listening"));
-}
-
-Deno.test("serveDirectory TLS", async function (): Promise<void> {
- await startTlsFileServer();
- try {
- // Valid request after invalid
- const conn = await Deno.connectTls({
- hostname: "localhost",
- port: 4577,
- certFile: join(testdataDir, "tls/RootCA.pem"),
- });
-
- await Deno.writeAll(
- conn,
- new TextEncoder().encode("GET / HTTP/1.0\r\n\r\n"),
- );
- const res = new Uint8Array(128 * 1024);
- const nread = await conn.read(res);
- assert(nread !== null);
- conn.close();
- const page = new TextDecoder().decode(res.subarray(0, nread));
- assert(page.includes("<title>Deno File Server</title>"));
- } finally {
- await killFileServer();
- }
-});
-
-Deno.test("partial TLS arguments fail", async function (): Promise<void> {
- fileServer = Deno.run({
- cmd: [
- Deno.execPath(),
- "run",
- "--quiet",
- "--allow-read",
- "--allow-net",
- "file_server.ts",
- ".",
- "--host",
- "localhost",
- "--cert",
- "./testdata/tls/localhost.crt",
- "-p",
- `4578`,
- ],
- cwd: moduleDir,
- stdout: "piped",
- stderr: "null",
- });
- try {
- // Once fileServer is ready it will write to its stdout.
- assert(fileServer.stdout != null);
- const r = new TextProtoReader(new BufReader(fileServer.stdout));
- const s = await r.readLine();
- assert(
- s !== null && s.includes("--key and --cert are required for TLS"),
- );
- } finally {
- await killFileServer();
- }
-});
-
-Deno.test("file_server disable dir listings", async function (): Promise<void> {
- await startFileServer({ "dir-listing": false });
- try {
- const res = await fetch("http://localhost:4507/");
- assert(res.headers.has("access-control-allow-origin"));
- assert(res.headers.has("access-control-allow-headers"));
- assertEquals(res.status, 404);
- const _ = await res.text();
- } finally {
- await killFileServer();
- }
-});