faastjs
Version:
Serverless batch computing made simple.
236 lines • 29.7 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const tslib_1 = require("tslib");
const ava_1 = tslib_1.__importDefault(require("ava"));
const fs_extra_1 = require("fs-extra");
const url_1 = require("url");
const util_1 = require("util");
const index_1 = require("../index");
const funcs = tslib_1.__importStar(require("./fixtures/functions"));
const util_2 = require("./fixtures/util");
const error_1 = require("../src/error");
async function testCleanup(t, options) {
const m = await (0, index_1.faastLocal)(funcs, {
gc: "off",
...options
});
let done = 0;
m.functions
.hello("there")
.then(_ => done++)
.catch(_ => { });
m.functions
.sleep(1000)
.then(_ => done++)
.catch(_ => { });
await m.cleanup();
t.is(done, 0);
}
async function testOrder(t, options) {
const faastModule = await (0, index_1.faastLocal)(funcs, {
gc: "off",
...options
});
t.plan(2);
const a = faastModule.functions.emptyReject();
const b = faastModule.functions.sleep(0);
t.is(await b, undefined);
try {
await a;
}
catch (err) {
t.is(err, undefined);
}
finally {
await faastModule.cleanup();
}
}
async function testConcurrency(t, { options, maxConcurrency, expectedConcurrency }) {
const faastModule = await (0, index_1.faastLocal)(funcs, {
...options,
gc: "off",
concurrency: maxConcurrency
});
try {
const N = maxConcurrency * 2;
const promises = [];
for (let i = 0; i < N; i++) {
promises.push(faastModule.functions.spin(2000));
}
const timings = await Promise.all(promises);
t.is((0, util_2.measureConcurrency)(timings), expectedConcurrency);
}
finally {
await faastModule.cleanup();
}
}
(0, ava_1.default)("local provider cleanup stops executions", testCleanup, {});
(0, ava_1.default)("local provider cleanup stops executions with child process", testCleanup, {
childProcess: true
});
const orderConfigs = [
{ childProcess: false, concurrency: 1, maxRetries: 0 },
{ childProcess: true, concurrency: 1, maxRetries: 0 },
{ childProcess: false, concurrency: 2, maxRetries: 0 },
{ childProcess: true, concurrency: 2, maxRetries: 0 },
{ childProcess: false, concurrency: 2, maxRetries: 2 },
{ childProcess: true, concurrency: 2, maxRetries: 2 }
];
for (const config of orderConfigs) {
(0, ava_1.default)(`out of order await (async catch) with ${(0, util_1.inspect)(config)}`, testOrder, config);
}
async function readFirstLogfile(logDirectoryUrl) {
const url = new url_1.URL(logDirectoryUrl);
const buf = await (0, fs_extra_1.readFile)(url.pathname + "/0.log");
return buf
.toString()
.split("\n")
.map(m => m.replace(/^\[(\d+)\]/, "[$pid]"));
}
(0, ava_1.default)("local provider console.log, console.warn, and console.error with child process", async (t) => {
const faastModule = await (0, index_1.faastLocal)(funcs, {
childProcess: true,
concurrency: 1,
gc: "off"
});
try {
await faastModule.functions.consoleLog("Remote console.log output");
await faastModule.functions.consoleWarn("Remote console.warn output");
await faastModule.functions.consoleError("Remote console.error output");
await (0, util_2.sleep)(1000);
await faastModule.cleanup({ deleteResources: false });
const messages = await readFirstLogfile(faastModule.logUrl());
t.truthy(messages.find(s => s === "[$pid]: Remote console.log output"));
t.truthy(messages.find(s => s === "[$pid]: Remote console.warn output"));
t.truthy(messages.find(s => s === "[$pid]: Remote console.error output"));
}
finally {
await faastModule.cleanup({ deleteResources: false });
}
});
(0, ava_1.default)("local provider log files should be appended, not truncated, after child process crash", async (t) => {
const faastModule = await (0, index_1.faastLocal)(funcs, {
childProcess: true,
concurrency: 1,
maxRetries: 1,
gc: "off"
});
try {
await faastModule.functions.consoleLog("output 1");
try {
await faastModule.functions.processExit();
}
catch (err) { }
await faastModule.functions.consoleWarn("output 2");
// Wait for flush
await (0, util_2.sleep)(500);
const messages = await readFirstLogfile(faastModule.logUrl());
t.truthy(messages.find(s => s === "[$pid]: output 1"));
t.truthy(messages.find(s => s === "[$pid]: output 2"));
}
finally {
await faastModule.cleanup({ deleteResources: false });
}
});
(0, ava_1.default)("local provider child process exceptions should result in errors with logUrl", async (t) => {
const faastModule = await (0, index_1.faastLocal)(funcs, {
childProcess: true,
concurrency: 1,
maxRetries: 1,
gc: "off"
});
t.plan(1);
try {
await faastModule.functions.error("synthetic error");
}
catch (err) {
const info = error_1.FaastError.info(err);
t.true(typeof info.logUrl === "string" && info.logUrl.startsWith(" file:///"), (0, util_1.inspect)(err));
}
finally {
await faastModule.cleanup();
}
});
(0, ava_1.default)("local provider child process crashes should result in errors with logUrl", async (t) => {
const faastModule = await (0, index_1.faastLocal)(funcs, {
childProcess: true,
concurrency: 1,
maxRetries: 1,
gc: "off"
});
t.plan(1);
try {
await faastModule.functions.processExit(-1);
}
catch (err) {
const info = error_1.FaastError.info(err);
t.true(typeof info.logUrl === "string" && info.logUrl.startsWith(" file:///"), (0, util_1.inspect)(err));
}
finally {
await faastModule.cleanup();
}
});
(0, ava_1.default)("local provider concurrent executions with child processes", async (t) => {
await testConcurrency(t, {
options: {
childProcess: true
},
maxConcurrency: 5,
expectedConcurrency: 5
});
});
(0, ava_1.default)("local provider no concurrency for cpu bound work without child processes", async (t) => {
await testConcurrency(t, {
options: {
childProcess: false
},
maxConcurrency: 5,
expectedConcurrency: 1
});
});
(0, ava_1.default)("local provider cleanup waits for all child processes to exit", async (t) => {
const faastModule = await (0, index_1.faastLocal)(funcs, {
childProcess: true,
gc: "off"
});
faastModule.functions.spin(5000).catch(_ => { });
while (true) {
await (0, util_2.sleep)(100);
if (faastModule.state.executors.length > 0) {
break;
}
}
t.is(faastModule.state.executors.length, 1, "executor is not running");
await faastModule.cleanup({ gcTimeout: 60 });
t.is(faastModule.state.executors.length, 0, "executors are running after cleanup");
});
(0, ava_1.default)("local unresolved module", async (t) => {
t.plan(1);
try {
await (0, index_1.faastLocal)({});
}
catch (err) {
t.regex(err.message, /Could not find file/);
}
});
(0, ava_1.default)("local issue #37", async (t) => {
// Previously this code caused an exception about module wrapper not being
// re-entrant. The problem was a race condition between wrapper selection
// and execution in local provider. Solved by making wrapper selector a
// regular function instead of an async function.
const m = await (0, index_1.faastLocal)(funcs);
try {
const { identityString: identity } = m.functions;
await identity("a");
const b = identity("b");
const c = identity("c");
await b;
await c;
// Test succeeds if no exceptions are thrown.
t.true(true);
}
finally {
await m.cleanup();
}
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"unit-local.test.js","sourceRoot":"","sources":["../../test/unit-local.test.ts"],"names":[],"mappings":"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","sourcesContent":["import test, { ExecutionContext } from \"ava\";\nimport { readFile } from \"fs-extra\";\nimport { URL } from \"url\";\nimport { inspect } from \"util\";\nimport { faastLocal, LocalOptions } from \"../index\";\nimport * as funcs from \"./fixtures/functions\";\nimport { measureConcurrency, sleep } from \"./fixtures/util\";\nimport { FaastError } from \"../src/error\";\n\nasync function testCleanup(t: ExecutionContext, options: LocalOptions) {\n    const m = await faastLocal(funcs, {\n        gc: \"off\",\n        ...options\n    });\n    let done = 0;\n\n    m.functions\n        .hello(\"there\")\n        .then(_ => done++)\n        .catch(_ => {});\n\n    m.functions\n        .sleep(1000)\n        .then(_ => done++)\n        .catch(_ => {});\n\n    await m.cleanup();\n    t.is(done, 0);\n}\n\nasync function testOrder(t: ExecutionContext, options: LocalOptions) {\n    const faastModule = await faastLocal(funcs, {\n        gc: \"off\",\n        ...options\n    });\n    t.plan(2);\n\n    const a = faastModule.functions.emptyReject();\n    const b = faastModule.functions.sleep(0);\n    t.is(await b, undefined);\n    try {\n        await a;\n    } catch (err: any) {\n        t.is(err, undefined);\n    } finally {\n        await faastModule.cleanup();\n    }\n}\n\nasync function testConcurrency(\n    t: ExecutionContext,\n    {\n        options,\n        maxConcurrency,\n        expectedConcurrency\n    }: {\n        options: LocalOptions;\n        maxConcurrency: number;\n        expectedConcurrency: number;\n    }\n) {\n    const faastModule = await faastLocal(funcs, {\n        ...options,\n        gc: \"off\",\n        concurrency: maxConcurrency\n    });\n\n    try {\n        const N = maxConcurrency * 2;\n        const promises = [];\n        for (let i = 0; i < N; i++) {\n            promises.push(faastModule.functions.spin(2000));\n        }\n\n        const timings = await Promise.all(promises);\n        t.is(measureConcurrency(timings), expectedConcurrency);\n    } finally {\n        await faastModule.cleanup();\n    }\n}\n\ntest(\"local provider cleanup stops executions\", testCleanup, {});\ntest(\"local provider cleanup stops executions with child process\", testCleanup, {\n    childProcess: true\n});\n\nconst orderConfigs = [\n    { childProcess: false, concurrency: 1, maxRetries: 0 },\n    { childProcess: true, concurrency: 1, maxRetries: 0 },\n    { childProcess: false, concurrency: 2, maxRetries: 0 },\n    { childProcess: true, concurrency: 2, maxRetries: 0 },\n    { childProcess: false, concurrency: 2, maxRetries: 2 },\n    { childProcess: true, concurrency: 2, maxRetries: 2 }\n];\n\nfor (const config of orderConfigs) {\n    test(`out of order await (async catch) with ${inspect(config)}`, testOrder, config);\n}\n\nasync function readFirstLogfile(logDirectoryUrl: string) {\n    const url = new URL(logDirectoryUrl);\n    const buf = await readFile(url.pathname + \"/0.log\");\n    return buf\n        .toString()\n        .split(\"\\n\")\n        .map(m => m.replace(/^\\[(\\d+)\\]/, \"[$pid]\"));\n}\n\ntest(\"local provider console.log, console.warn, and console.error with child process\", async t => {\n    const faastModule = await faastLocal(funcs, {\n        childProcess: true,\n        concurrency: 1,\n        gc: \"off\"\n    });\n    try {\n        await faastModule.functions.consoleLog(\"Remote console.log output\");\n        await faastModule.functions.consoleWarn(\"Remote console.warn output\");\n        await faastModule.functions.consoleError(\"Remote console.error output\");\n        await sleep(1000);\n        await faastModule.cleanup({ deleteResources: false });\n        const messages = await readFirstLogfile(faastModule.logUrl());\n        t.truthy(messages.find(s => s === \"[$pid]: Remote console.log output\"));\n        t.truthy(messages.find(s => s === \"[$pid]: Remote console.warn output\"));\n        t.truthy(messages.find(s => s === \"[$pid]: Remote console.error output\"));\n    } finally {\n        await faastModule.cleanup({ deleteResources: false });\n    }\n});\n\ntest(\"local provider log files should be appended, not truncated, after child process crash\", async t => {\n    const faastModule = await faastLocal(funcs, {\n        childProcess: true,\n        concurrency: 1,\n        maxRetries: 1,\n        gc: \"off\"\n    });\n    try {\n        await faastModule.functions.consoleLog(\"output 1\");\n        try {\n            await faastModule.functions.processExit();\n        } catch (err: any) {}\n        await faastModule.functions.consoleWarn(\"output 2\");\n\n        // Wait for flush\n        await sleep(500);\n        const messages = await readFirstLogfile(faastModule.logUrl());\n\n        t.truthy(messages.find(s => s === \"[$pid]: output 1\"));\n        t.truthy(messages.find(s => s === \"[$pid]: output 2\"));\n    } finally {\n        await faastModule.cleanup({ deleteResources: false });\n    }\n});\n\ntest(\"local provider child process exceptions should result in errors with logUrl\", async t => {\n    const faastModule = await faastLocal(funcs, {\n        childProcess: true,\n        concurrency: 1,\n        maxRetries: 1,\n        gc: \"off\"\n    });\n    t.plan(1);\n    try {\n        await faastModule.functions.error(\"synthetic error\");\n    } catch (err: any) {\n        const info = FaastError.info(err);\n        t.true(\n            typeof info.logUrl === \"string\" && info.logUrl.startsWith(\" file:///\"),\n            inspect(err)\n        );\n    } finally {\n        await faastModule.cleanup();\n    }\n});\n\ntest(\"local provider child process crashes should result in errors with logUrl\", async t => {\n    const faastModule = await faastLocal(funcs, {\n        childProcess: true,\n        concurrency: 1,\n        maxRetries: 1,\n        gc: \"off\"\n    });\n    t.plan(1);\n    try {\n        await faastModule.functions.processExit(-1);\n    } catch (err: any) {\n        const info = FaastError.info(err);\n        t.true(\n            typeof info.logUrl === \"string\" && info.logUrl.startsWith(\" file:///\"),\n            inspect(err)\n        );\n    } finally {\n        await faastModule.cleanup();\n    }\n});\n\ntest(\"local provider concurrent executions with child processes\", async t => {\n    await testConcurrency(t, {\n        options: {\n            childProcess: true\n        },\n        maxConcurrency: 5,\n        expectedConcurrency: 5\n    });\n});\n\ntest(\"local provider no concurrency for cpu bound work without child processes\", async t => {\n    await testConcurrency(t, {\n        options: {\n            childProcess: false\n        },\n        maxConcurrency: 5,\n        expectedConcurrency: 1\n    });\n});\n\ntest(\"local provider cleanup waits for all child processes to exit\", async t => {\n    const faastModule = await faastLocal(funcs, {\n        childProcess: true,\n        gc: \"off\"\n    });\n    faastModule.functions.spin(5000).catch(_ => {});\n    while (true) {\n        await sleep(100);\n        if (faastModule.state.executors.length > 0) {\n            break;\n        }\n    }\n    t.is(faastModule.state.executors.length, 1, \"executor is not running\");\n    await faastModule.cleanup({ gcTimeout: 60 });\n    t.is(faastModule.state.executors.length, 0, \"executors are running after cleanup\");\n});\n\ntest(\"local unresolved module\", async t => {\n    t.plan(1);\n    try {\n        await faastLocal({});\n    } catch (err: any) {\n        t.regex(err.message, /Could not find file/);\n    }\n});\n\ntest(\"local issue #37\", async t => {\n    // Previously this code caused an exception about module wrapper not being\n    // re-entrant. The problem was a race condition between wrapper selection\n    // and execution in local provider. Solved by making wrapper selector a\n    // regular function instead of an async function.\n    const m = await faastLocal(funcs);\n    try {\n        const { identityString: identity } = m.functions;\n        await identity(\"a\");\n        const b = identity(\"b\");\n        const c = identity(\"c\");\n        await b;\n        await c;\n        // Test succeeds if no exceptions are thrown.\n        t.true(true);\n    } finally {\n        await m.cleanup();\n    }\n});\n"]}