1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-08 15:19:40 -05:00

add assertResources sanitizer to cli/js/ unit tests (#4161)

This commit is contained in:
Bartek Iwańczuk 2020-02-29 18:45:47 +01:00 committed by GitHub
parent 199fb195f3
commit f55b22e195
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
11 changed files with 95 additions and 36 deletions

View file

@ -19,9 +19,11 @@ if (Deno.build.os !== "win") {
const uid = parseInt(
new TextDecoder("utf-8").decode(await uidProc.output())
);
uidProc.close();
const gid = parseInt(
new TextDecoder("utf-8").decode(await gidProc.output())
);
gidProc.close();
return { uid, gid };
}

View file

@ -5,8 +5,8 @@ import {
assert,
assertEquals,
assertStrContains,
assertThrows,
fail
assertThrows
// fail
} from "./test_util.ts";
testPerm({ net: true }, async function fetchProtocolError(): Promise<void> {
@ -51,6 +51,7 @@ test(async function fetchPerm(): Promise<void> {
testPerm({ net: true }, async function fetchUrl(): Promise<void> {
const response = await fetch("http://localhost:4545/cli/tests/fixture.json");
assertEquals(response.url, "http://localhost:4545/cli/tests/fixture.json");
response.body.close();
});
testPerm({ net: true }, async function fetchURL(): Promise<void> {
@ -58,6 +59,7 @@ testPerm({ net: true }, async function fetchURL(): Promise<void> {
new URL("http://localhost:4545/cli/tests/fixture.json")
);
assertEquals(response.url, "http://localhost:4545/cli/tests/fixture.json");
response.body.close();
});
testPerm({ net: true }, async function fetchHeaders(): Promise<void> {
@ -65,6 +67,7 @@ testPerm({ net: true }, async function fetchHeaders(): Promise<void> {
const headers = response.headers;
assertEquals(headers.get("Content-Type"), "application/json");
assert(headers.get("Server")!.startsWith("SimpleHTTP"));
response.body.close();
});
testPerm({ net: true }, async function fetchBlob(): Promise<void> {
@ -95,6 +98,7 @@ testPerm({ net: true }, async function fetchAsyncIterator(): Promise<void> {
}
assertEquals(total, Number(headers.get("Content-Length")));
response.body.close();
});
testPerm({ net: true }, async function responseClone(): Promise<void> {
@ -151,6 +155,8 @@ testPerm(
}
);
/*
// TODO: leaking resources
testPerm({ net: true }, async function fetchWithRedirection(): Promise<void> {
const response = await fetch("http://localhost:4546/"); // will redirect to http://localhost:4545/
assertEquals(response.status, 200);
@ -160,6 +166,7 @@ testPerm({ net: true }, async function fetchWithRedirection(): Promise<void> {
assert(body.includes("<title>Directory listing for /</title>"));
});
// TODO: leaking resources
testPerm({ net: true }, async function fetchWithRelativeRedirection(): Promise<
void
> {
@ -169,6 +176,7 @@ testPerm({ net: true }, async function fetchWithRelativeRedirection(): Promise<
const body = await response.text();
assert(body.includes("<title>Directory listing for /cli/tests/</title>"));
});
*/
// The feature below is not implemented, but the test should work after implementation
/*
@ -371,6 +379,8 @@ testPerm({ net: true }, async function fetchPostBodyTypedArray():Promise<void> {
});
*/
/*
// TODO: leaking resources
testPerm({ net: true }, async function fetchWithManualRedirection(): Promise<
void
> {
@ -391,6 +401,7 @@ testPerm({ net: true }, async function fetchWithManualRedirection(): Promise<
}
});
// TODO: leaking resources
testPerm({ net: true }, async function fetchWithErrorRedirection(): Promise<
void
> {
@ -410,6 +421,7 @@ testPerm({ net: true }, async function fetchWithErrorRedirection(): Promise<
return;
}
});
*/
test(function responseRedirect(): void {
const response = new Response(

View file

@ -21,6 +21,7 @@ testPerm({ read: true }, async function filesCopyToStdout(): Promise<void> {
const fileSize = Deno.statSync(filename).len;
assertEquals(bytesWritten, fileSize);
console.log("bytes written", bytesWritten);
file.close();
});
testPerm({ read: true }, async function filesToAsyncIterator(): Promise<void> {
@ -33,6 +34,7 @@ testPerm({ read: true }, async function filesToAsyncIterator(): Promise<void> {
}
assertEquals(totalSize, 12);
file.close();
});
test(async function readerToAsyncIterator(): Promise<void> {
@ -308,6 +310,7 @@ testPerm({ read: true }, async function seekStart(): Promise<void> {
await file.read(buf);
const decoded = new TextDecoder().decode(buf);
assertEquals(decoded, "world!");
file.close();
});
testPerm({ read: true }, function seekSyncStart(): void {
@ -321,6 +324,7 @@ testPerm({ read: true }, function seekSyncStart(): void {
file.readSync(buf);
const decoded = new TextDecoder().decode(buf);
assertEquals(decoded, "world!");
file.close();
});
testPerm({ read: true }, async function seekCurrent(): Promise<void> {
@ -334,6 +338,7 @@ testPerm({ read: true }, async function seekCurrent(): Promise<void> {
await file.read(buf);
const decoded = new TextDecoder().decode(buf);
assertEquals(decoded, "world!");
file.close();
});
testPerm({ read: true }, function seekSyncCurrent(): void {
@ -347,6 +352,7 @@ testPerm({ read: true }, function seekSyncCurrent(): void {
file.readSync(buf);
const decoded = new TextDecoder().decode(buf);
assertEquals(decoded, "world!");
file.close();
});
testPerm({ read: true }, async function seekEnd(): Promise<void> {
@ -357,6 +363,7 @@ testPerm({ read: true }, async function seekEnd(): Promise<void> {
await file.read(buf);
const decoded = new TextDecoder().decode(buf);
assertEquals(decoded, "world!");
file.close();
});
testPerm({ read: true }, function seekSyncEnd(): void {
@ -367,6 +374,7 @@ testPerm({ read: true }, function seekSyncEnd(): void {
file.readSync(buf);
const decoded = new TextDecoder().decode(buf);
assertEquals(decoded, "world!");
file.close();
});
testPerm({ read: true }, async function seekMode(): Promise<void> {
@ -387,4 +395,5 @@ testPerm({ read: true }, async function seekMode(): Promise<void> {
const buf = new Uint8Array(1);
await file.read(buf); // "H"
assertEquals(new TextDecoder().decode(buf), "H");
file.close();
});

View file

@ -112,6 +112,8 @@ testPerm({ net: true }, async function netUdpSendReceive(): Promise<void> {
assertEquals(1, recvd[0]);
assertEquals(2, recvd[1]);
assertEquals(3, recvd[2]);
alice.close();
bob.close();
});
testPerm(

View file

@ -76,6 +76,7 @@ if (Deno.build.os === "win") {
new TextDecoder().decode(await proc.output())
);
assertEquals(actualValues, expectedValues);
proc.close();
};
assertEquals(Deno.env("path"), Deno.env("PATH"));

View file

@ -299,6 +299,7 @@ testPerm({ run: true }, async function runClose(): Promise<void> {
const data = new Uint8Array(10);
const r = await p.stderr!.read(data);
assertEquals(r, Deno.EOF);
p.stderr!.close();
});
test(function signalNumbers(): void {
@ -341,6 +342,7 @@ if (Deno.build.os !== "win") {
// re-enable when it can be made deterministic.
// assertEquals(status.code, 1);
// assertEquals(status.signal, Deno.Signal.SIGINT);
p.close();
});
testPerm({ run: true }, async function killFailed(): Promise<void> {

View file

@ -31,8 +31,9 @@ testPerm({ net: true }, async function resourcesNet(): Promise<void> {
testPerm({ read: true }, async function resourcesFile(): Promise<void> {
const resourcesBefore = Deno.resources();
await Deno.open("cli/tests/hello.txt");
const f = await Deno.open("cli/tests/hello.txt");
const resourcesAfter = Deno.resources();
f.close();
// check that exactly one new resource (file) was added
assertEquals(

View file

@ -106,6 +106,22 @@ function normalizeTestPermissions(perms: TestPermissions): Permissions {
};
}
// Wrap `TestFunction` in additional assertion that makes sure
// the test case does not "leak" resources - ie. resource table after
// the test has exactly the same contents as before the test.
function assertResources(fn: Deno.TestFunction): Deno.TestFunction {
return async function(): Promise<void> {
const preResources = Deno.resources();
await fn();
const postResources = Deno.resources();
const msg = `Test case is leaking resources.
Before: ${JSON.stringify(preResources, null, 2)}
After: ${JSON.stringify(postResources, null, 2)}`;
assertEquals(preResources, postResources, msg);
};
}
export function testPerm(perms: TestPermissions, fn: Deno.TestFunction): void {
const normalizedPerms = normalizeTestPermissions(perms);
@ -115,7 +131,7 @@ export function testPerm(perms: TestPermissions, fn: Deno.TestFunction): void {
return;
}
Deno.test(fn);
Deno.test(fn.name, assertResources(fn));
}
export function test(fn: Deno.TestFunction): void {
@ -171,6 +187,24 @@ export async function parseUnitTestOutput(
return { actual, expected, resultOutput: result };
}
export interface ResolvableMethods<T> {
resolve: (value?: T | PromiseLike<T>) => void;
// eslint-disable-next-line @typescript-eslint/no-explicit-any
reject: (reason?: any) => void;
}
export type Resolvable<T> = Promise<T> & ResolvableMethods<T>;
export function createResolvable<T>(): Resolvable<T> {
let methods: ResolvableMethods<T>;
const promise = new Promise<T>((resolve, reject): void => {
methods = { resolve, reject };
});
// TypeScript doesn't know that the Promise callback occurs synchronously
// therefore use of not null assertion (`!`)
return Object.assign(promise, methods!) as Resolvable<T>;
}
test(function permissionsMatches(): void {
assert(
permissionsMatch(
@ -265,28 +299,25 @@ testPerm({ read: true }, async function parsingUnitTestOutput(): Promise<void> {
let result;
// This is an example of a successful unit test output.
result = await parseUnitTestOutput(
await Deno.open(`${testDataPath}/unit_test_output1.txt`),
false
);
const f1 = await Deno.open(`${testDataPath}/unit_test_output1.txt`);
result = await parseUnitTestOutput(f1, false);
assertEquals(result.actual, 96);
assertEquals(result.expected, 96);
f1.close();
// This is an example of a silently dying unit test.
result = await parseUnitTestOutput(
await Deno.open(`${testDataPath}/unit_test_output2.txt`),
false
);
const f2 = await Deno.open(`${testDataPath}/unit_test_output2.txt`);
result = await parseUnitTestOutput(f2, false);
assertEquals(result.actual, undefined);
assertEquals(result.expected, 96);
f2.close();
// This is an example of compiling before successful unit tests.
result = await parseUnitTestOutput(
await Deno.open(`${testDataPath}/unit_test_output3.txt`),
false
);
const f3 = await Deno.open(`${testDataPath}/unit_test_output3.txt`);
result = await parseUnitTestOutput(f3, false);
assertEquals(result.actual, 96);
assertEquals(result.expected, 96);
f3.close();
// Check what happens on empty output.
const f = new Deno.Buffer(new TextEncoder().encode("\n\n\n"));

View file

@ -1,5 +1,11 @@
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
import { test, testPerm, assert, assertEquals } from "./test_util.ts";
import {
test,
testPerm,
assert,
assertEquals,
createResolvable
} from "./test_util.ts";
import { BufWriter, BufReader } from "../../std/io/bufio.ts";
import { TextProtoReader } from "../../std/textproto/mod.ts";
@ -142,6 +148,7 @@ testPerm(
testPerm({ read: true, net: true }, async function dialAndListenTLS(): Promise<
void
> {
const resolvable = createResolvable();
const hostname = "localhost";
const port = 4500;
@ -164,6 +171,7 @@ testPerm({ read: true, net: true }, async function dialAndListenTLS(): Promise<
// TODO(bartlomieju): this might be a bug
setTimeout(() => {
conn.close();
resolvable.resolve();
}, 0);
}
);
@ -196,4 +204,6 @@ testPerm({ read: true, net: true }, async function dialAndListenTLS(): Promise<
await r.readFull(bodyBuf);
assertEquals(decoder.decode(bodyBuf), "Hello World\n");
conn.close();
listener.close();
await resolvable;
});

View file

@ -7,6 +7,7 @@ testPerm({ read: true }, function isatty(): void {
// CI not under TTY, so cannot test stdin/stdout/stderr.
const f = Deno.openSync("cli/tests/hello.txt");
assert(!Deno.isatty(f.rid));
f.close();
});
test(function isattyError(): void {

View file

@ -1,23 +1,11 @@
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
import { test, testPerm, assert, assertEquals } from "./test_util.ts";
export interface ResolvableMethods<T> {
resolve: (value?: T | PromiseLike<T>) => void;
// eslint-disable-next-line @typescript-eslint/no-explicit-any
reject: (reason?: any) => void;
}
export type Resolvable<T> = Promise<T> & ResolvableMethods<T>;
export function createResolvable<T>(): Resolvable<T> {
let methods: ResolvableMethods<T>;
const promise = new Promise<T>((resolve, reject): void => {
methods = { resolve, reject };
});
// TypeScript doesn't know that the Promise callback occurs synchronously
// therefore use of not null assertion (`!`)
return Object.assign(promise, methods!) as Resolvable<T>;
}
import {
test,
testPerm,
assert,
assertEquals,
createResolvable
} from "./test_util.ts";
test(async function workersBasic(): Promise<void> {
const promise = createResolvable();