2019-02-11 08:49:48 +09:00
|
|
|
// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
|
|
|
|
|
2019-02-25 21:35:50 -08:00
|
|
|
const { Buffer, copy, open, remove } = Deno;
|
2019-03-06 22:39:50 +01:00
|
|
|
import {
|
|
|
|
assert,
|
2019-03-06 19:42:24 -05:00
|
|
|
assertEquals,
|
2019-03-06 22:39:50 +01:00
|
|
|
assertThrows,
|
|
|
|
assertThrowsAsync
|
|
|
|
} from "../testing/asserts.ts";
|
2019-05-23 19:04:06 -07:00
|
|
|
import { test, runIfMain } from "../testing/mod.ts";
|
2019-02-11 08:49:48 +09:00
|
|
|
import {
|
|
|
|
matchAfterPrefix,
|
|
|
|
MultipartReader,
|
|
|
|
MultipartWriter,
|
|
|
|
scanUntilBoundary
|
|
|
|
} from "./multipart.ts";
|
|
|
|
import * as path from "../fs/path.ts";
|
2019-05-21 15:36:12 +02:00
|
|
|
import { FormFile, isFormFile } from "../multipart/formfile.ts";
|
2019-05-23 19:04:06 -07:00
|
|
|
import { EOF } from "../io/bufio.ts";
|
2019-02-11 08:49:48 +09:00
|
|
|
import { StringWriter } from "../io/writers.ts";
|
|
|
|
|
|
|
|
const e = new TextEncoder();
|
|
|
|
const boundary = "--abcde";
|
|
|
|
const dashBoundary = e.encode("--" + boundary);
|
|
|
|
const nlDashBoundary = e.encode("\r\n--" + boundary);
|
|
|
|
|
2019-04-24 13:41:23 +02:00
|
|
|
test(function multipartScanUntilBoundary1(): void {
|
2019-02-11 08:49:48 +09:00
|
|
|
const data = `--${boundary}`;
|
2019-05-23 19:04:06 -07:00
|
|
|
const n = scanUntilBoundary(
|
2019-02-11 08:49:48 +09:00
|
|
|
e.encode(data),
|
|
|
|
dashBoundary,
|
|
|
|
nlDashBoundary,
|
|
|
|
0,
|
2019-05-23 19:04:06 -07:00
|
|
|
true
|
2019-02-11 08:49:48 +09:00
|
|
|
);
|
2019-05-23 19:04:06 -07:00
|
|
|
assertEquals(n, EOF);
|
2019-02-11 08:49:48 +09:00
|
|
|
});
|
|
|
|
|
2019-04-24 13:41:23 +02:00
|
|
|
test(function multipartScanUntilBoundary2(): void {
|
2019-02-11 08:49:48 +09:00
|
|
|
const data = `foo\r\n--${boundary}`;
|
2019-05-23 19:04:06 -07:00
|
|
|
const n = scanUntilBoundary(
|
2019-02-11 08:49:48 +09:00
|
|
|
e.encode(data),
|
|
|
|
dashBoundary,
|
|
|
|
nlDashBoundary,
|
|
|
|
0,
|
2019-05-23 19:04:06 -07:00
|
|
|
true
|
2019-02-11 08:49:48 +09:00
|
|
|
);
|
2019-03-06 19:42:24 -05:00
|
|
|
assertEquals(n, 3);
|
2019-02-11 08:49:48 +09:00
|
|
|
});
|
|
|
|
|
2019-05-23 19:04:06 -07:00
|
|
|
test(function multipartScanUntilBoundary3(): void {
|
|
|
|
const data = `foobar`;
|
|
|
|
const n = scanUntilBoundary(
|
2019-02-11 08:49:48 +09:00
|
|
|
e.encode(data),
|
|
|
|
dashBoundary,
|
|
|
|
nlDashBoundary,
|
|
|
|
0,
|
2019-05-23 19:04:06 -07:00
|
|
|
false
|
2019-02-11 08:49:48 +09:00
|
|
|
);
|
2019-05-23 19:04:06 -07:00
|
|
|
assertEquals(n, data.length);
|
2019-02-11 08:49:48 +09:00
|
|
|
});
|
|
|
|
|
2019-05-23 19:04:06 -07:00
|
|
|
test(function multipartScanUntilBoundary4(): void {
|
|
|
|
const data = `foo\r\n--`;
|
|
|
|
const n = scanUntilBoundary(
|
2019-02-11 08:49:48 +09:00
|
|
|
e.encode(data),
|
|
|
|
dashBoundary,
|
|
|
|
nlDashBoundary,
|
|
|
|
0,
|
2019-05-23 19:04:06 -07:00
|
|
|
false
|
2019-02-11 08:49:48 +09:00
|
|
|
);
|
2019-05-23 19:04:06 -07:00
|
|
|
assertEquals(n, 3);
|
2019-02-11 08:49:48 +09:00
|
|
|
});
|
|
|
|
|
2019-04-24 13:41:23 +02:00
|
|
|
test(function multipartMatchAfterPrefix1(): void {
|
2019-02-11 08:49:48 +09:00
|
|
|
const data = `${boundary}\r`;
|
2019-05-23 19:04:06 -07:00
|
|
|
const v = matchAfterPrefix(e.encode(data), e.encode(boundary), false);
|
2019-03-06 19:42:24 -05:00
|
|
|
assertEquals(v, 1);
|
2019-02-11 08:49:48 +09:00
|
|
|
});
|
|
|
|
|
2019-04-24 13:41:23 +02:00
|
|
|
test(function multipartMatchAfterPrefix2(): void {
|
2019-02-11 08:49:48 +09:00
|
|
|
const data = `${boundary}hoge`;
|
2019-05-23 19:04:06 -07:00
|
|
|
const v = matchAfterPrefix(e.encode(data), e.encode(boundary), false);
|
2019-03-06 19:42:24 -05:00
|
|
|
assertEquals(v, -1);
|
2019-02-11 08:49:48 +09:00
|
|
|
});
|
|
|
|
|
2019-04-24 13:41:23 +02:00
|
|
|
test(function multipartMatchAfterPrefix3(): void {
|
2019-02-11 08:49:48 +09:00
|
|
|
const data = `${boundary}`;
|
2019-05-23 19:04:06 -07:00
|
|
|
const v = matchAfterPrefix(e.encode(data), e.encode(boundary), false);
|
2019-03-06 19:42:24 -05:00
|
|
|
assertEquals(v, 0);
|
2019-02-11 08:49:48 +09:00
|
|
|
});
|
|
|
|
|
2019-04-24 13:41:23 +02:00
|
|
|
test(async function multipartMultipartWriter(): Promise<void> {
|
2019-02-11 08:49:48 +09:00
|
|
|
const buf = new Buffer();
|
|
|
|
const mw = new MultipartWriter(buf);
|
|
|
|
await mw.writeField("foo", "foo");
|
|
|
|
await mw.writeField("bar", "bar");
|
|
|
|
const f = await open(path.resolve("./multipart/fixtures/sample.txt"), "r");
|
|
|
|
await mw.writeFile("file", "sample.txt", f);
|
|
|
|
await mw.close();
|
|
|
|
});
|
|
|
|
|
2019-04-24 13:41:23 +02:00
|
|
|
test(function multipartMultipartWriter2(): void {
|
2019-02-11 08:49:48 +09:00
|
|
|
const w = new StringWriter();
|
2019-03-06 22:39:50 +01:00
|
|
|
assertThrows(
|
2019-04-24 13:41:23 +02:00
|
|
|
(): MultipartWriter => new MultipartWriter(w, ""),
|
2019-02-11 08:49:48 +09:00
|
|
|
Error,
|
|
|
|
"invalid boundary length"
|
|
|
|
);
|
2019-03-06 22:39:50 +01:00
|
|
|
assertThrows(
|
2019-04-24 13:41:23 +02:00
|
|
|
(): MultipartWriter =>
|
2019-02-11 08:49:48 +09:00
|
|
|
new MultipartWriter(
|
|
|
|
w,
|
|
|
|
"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"
|
|
|
|
),
|
|
|
|
Error,
|
|
|
|
"invalid boundary length"
|
|
|
|
);
|
2019-03-06 22:39:50 +01:00
|
|
|
assertThrows(
|
2019-04-24 13:41:23 +02:00
|
|
|
(): MultipartWriter => new MultipartWriter(w, "aaa aaa"),
|
2019-02-11 08:49:48 +09:00
|
|
|
Error,
|
|
|
|
"invalid boundary character"
|
|
|
|
);
|
2019-03-06 22:39:50 +01:00
|
|
|
assertThrows(
|
2019-04-24 13:41:23 +02:00
|
|
|
(): MultipartWriter => new MultipartWriter(w, "boundary¥¥"),
|
2019-02-11 08:49:48 +09:00
|
|
|
Error,
|
|
|
|
"invalid boundary character"
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2019-04-24 13:41:23 +02:00
|
|
|
test(async function multipartMultipartWriter3(): Promise<void> {
|
2019-02-11 08:49:48 +09:00
|
|
|
const w = new StringWriter();
|
|
|
|
const mw = new MultipartWriter(w);
|
|
|
|
await mw.writeField("foo", "foo");
|
|
|
|
await mw.close();
|
2019-03-06 22:39:50 +01:00
|
|
|
await assertThrowsAsync(
|
2019-04-24 13:41:23 +02:00
|
|
|
async (): Promise<void> => {
|
2019-02-11 08:49:48 +09:00
|
|
|
await mw.close();
|
|
|
|
},
|
|
|
|
Error,
|
|
|
|
"closed"
|
|
|
|
);
|
2019-03-06 22:39:50 +01:00
|
|
|
await assertThrowsAsync(
|
2019-04-24 13:41:23 +02:00
|
|
|
async (): Promise<void> => {
|
2019-02-11 08:49:48 +09:00
|
|
|
await mw.writeFile("bar", "file", null);
|
|
|
|
},
|
|
|
|
Error,
|
|
|
|
"closed"
|
|
|
|
);
|
2019-03-06 22:39:50 +01:00
|
|
|
await assertThrowsAsync(
|
2019-04-24 13:41:23 +02:00
|
|
|
async (): Promise<void> => {
|
2019-02-11 08:49:48 +09:00
|
|
|
await mw.writeField("bar", "bar");
|
|
|
|
},
|
|
|
|
Error,
|
|
|
|
"closed"
|
|
|
|
);
|
2019-03-06 22:39:50 +01:00
|
|
|
assertThrows(
|
2019-04-24 13:41:23 +02:00
|
|
|
(): void => {
|
2019-02-11 08:49:48 +09:00
|
|
|
mw.createFormField("bar");
|
|
|
|
},
|
|
|
|
Error,
|
|
|
|
"closed"
|
|
|
|
);
|
2019-03-06 22:39:50 +01:00
|
|
|
assertThrows(
|
2019-04-24 13:41:23 +02:00
|
|
|
(): void => {
|
2019-02-11 08:49:48 +09:00
|
|
|
mw.createFormFile("bar", "file");
|
|
|
|
},
|
|
|
|
Error,
|
|
|
|
"closed"
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2019-04-24 13:41:23 +02:00
|
|
|
test(async function multipartMultipartReader(): Promise<void> {
|
2019-02-11 08:49:48 +09:00
|
|
|
// FIXME: path resolution
|
|
|
|
const o = await open(path.resolve("./multipart/fixtures/sample.txt"));
|
|
|
|
const mr = new MultipartReader(
|
|
|
|
o,
|
|
|
|
"--------------------------434049563556637648550474"
|
|
|
|
);
|
|
|
|
const form = await mr.readForm(10 << 20);
|
2019-03-06 19:42:24 -05:00
|
|
|
assertEquals(form["foo"], "foo");
|
|
|
|
assertEquals(form["bar"], "bar");
|
2019-02-11 08:49:48 +09:00
|
|
|
const file = form["file"] as FormFile;
|
2019-03-06 19:42:24 -05:00
|
|
|
assertEquals(isFormFile(file), true);
|
2019-03-06 22:39:50 +01:00
|
|
|
assert(file.content !== void 0);
|
2019-02-11 08:49:48 +09:00
|
|
|
});
|
|
|
|
|
2019-04-24 13:41:23 +02:00
|
|
|
test(async function multipartMultipartReader2(): Promise<void> {
|
2019-02-11 08:49:48 +09:00
|
|
|
const o = await open(path.resolve("./multipart/fixtures/sample.txt"));
|
|
|
|
const mr = new MultipartReader(
|
|
|
|
o,
|
|
|
|
"--------------------------434049563556637648550474"
|
|
|
|
);
|
|
|
|
const form = await mr.readForm(20); //
|
|
|
|
try {
|
2019-03-06 19:42:24 -05:00
|
|
|
assertEquals(form["foo"], "foo");
|
|
|
|
assertEquals(form["bar"], "bar");
|
2019-02-11 08:49:48 +09:00
|
|
|
const file = form["file"] as FormFile;
|
2019-03-06 19:42:24 -05:00
|
|
|
assertEquals(file.type, "application/octet-stream");
|
2019-02-11 08:49:48 +09:00
|
|
|
const f = await open(file.tempfile);
|
|
|
|
const w = new StringWriter();
|
|
|
|
await copy(w, f);
|
|
|
|
const json = JSON.parse(w.toString());
|
2019-03-06 19:42:24 -05:00
|
|
|
assertEquals(json["compilerOptions"]["target"], "es2018");
|
2019-02-11 08:49:48 +09:00
|
|
|
f.close();
|
|
|
|
} finally {
|
|
|
|
const file = form["file"] as FormFile;
|
|
|
|
await remove(file.tempfile);
|
|
|
|
}
|
|
|
|
});
|
2019-05-23 19:04:06 -07:00
|
|
|
|
|
|
|
runIfMain(import.meta);
|