2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2019-02-11 08:49:48 +09:00
|
|
|
|
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-10-16 19:39:33 +01:00
|
|
|
import * as path from "../path/mod.ts";
|
2019-02-11 08:49:48 +09:00
|
|
|
import {
|
2020-01-10 19:29:09 +00:00
|
|
|
FormFile,
|
2019-02-11 08:49:48 +09:00
|
|
|
MultipartReader,
|
|
|
|
MultipartWriter,
|
2020-01-10 19:29:09 +00:00
|
|
|
isFormFile,
|
|
|
|
matchAfterPrefix,
|
2019-02-11 08:49:48 +09:00
|
|
|
scanUntilBoundary
|
|
|
|
} from "./multipart.ts";
|
|
|
|
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-07-08 04:20:41 +09:00
|
|
|
assertEquals(n, Deno.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");
|
2020-01-10 19:29:09 +00:00
|
|
|
const f = await open(path.resolve("./mime/testdata/sample.txt"), "r");
|
2019-02-11 08:49:48 +09:00
|
|
|
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,
|
2019-06-19 12:22:01 +08:00
|
|
|
"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" +
|
|
|
|
"aaaaaaaa"
|
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 => 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-05-30 14:59:30 +02:00
|
|
|
// @ts-ignore
|
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
|
2020-01-10 19:29:09 +00:00
|
|
|
const o = await open(path.resolve("./mime/testdata/sample.txt"));
|
2019-02-11 08:49:48 +09:00
|
|
|
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> {
|
2020-01-10 19:29:09 +00:00
|
|
|
const o = await open(path.resolve("./mime/testdata/sample.txt"));
|
2019-02-11 08:49:48 +09:00
|
|
|
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-05-30 14:59:30 +02:00
|
|
|
const f = await open(file.tempfile!);
|
2019-02-11 08:49:48 +09:00
|
|
|
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;
|
2019-05-30 14:59:30 +02:00
|
|
|
await remove(file.tempfile!);
|
2019-02-11 08:49:48 +09:00
|
|
|
}
|
|
|
|
});
|
2019-05-23 19:04:06 -07:00
|
|
|
|
|
|
|
runIfMain(import.meta);
|