UNPKG

29.6 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3const ava_1 = require("ava");
4const fs_extra_1 = require("fs-extra");
5const url_1 = require("url");
6const util_1 = require("util");
7const index_1 = require("../index");
8const funcs = require("./fixtures/functions");
9const util_2 = require("./fixtures/util");
10const error_1 = require("../src/error");
11async function testCleanup(t, options) {
12 const m = await (0, index_1.faastLocal)(funcs, {
13 gc: "off",
14 ...options
15 });
16 let done = 0;
17 m.functions
18 .hello("there")
19 .then(_ => done++)
20 .catch(_ => { });
21 m.functions
22 .sleep(1000)
23 .then(_ => done++)
24 .catch(_ => { });
25 await m.cleanup();
26 t.is(done, 0);
27}
28async function testOrder(t, options) {
29 const faastModule = await (0, index_1.faastLocal)(funcs, {
30 gc: "off",
31 ...options
32 });
33 t.plan(2);
34 const a = faastModule.functions.emptyReject();
35 const b = faastModule.functions.sleep(0);
36 t.is(await b, undefined);
37 try {
38 await a;
39 }
40 catch (err) {
41 t.is(err, undefined);
42 }
43 finally {
44 await faastModule.cleanup();
45 }
46}
47async function testConcurrency(t, { options, maxConcurrency, expectedConcurrency }) {
48 const faastModule = await (0, index_1.faastLocal)(funcs, {
49 ...options,
50 gc: "off",
51 concurrency: maxConcurrency
52 });
53 try {
54 const N = maxConcurrency * 2;
55 const promises = [];
56 for (let i = 0; i < N; i++) {
57 promises.push(faastModule.functions.spin(2000));
58 }
59 const timings = await Promise.all(promises);
60 t.is((0, util_2.measureConcurrency)(timings), expectedConcurrency);
61 }
62 finally {
63 await faastModule.cleanup();
64 }
65}
66(0, ava_1.default)("local provider cleanup stops executions", testCleanup, {});
67(0, ava_1.default)("local provider cleanup stops executions with child process", testCleanup, {
68 childProcess: true
69});
70const orderConfigs = [
71 { childProcess: false, concurrency: 1, maxRetries: 0 },
72 { childProcess: true, concurrency: 1, maxRetries: 0 },
73 { childProcess: false, concurrency: 2, maxRetries: 0 },
74 { childProcess: true, concurrency: 2, maxRetries: 0 },
75 { childProcess: false, concurrency: 2, maxRetries: 2 },
76 { childProcess: true, concurrency: 2, maxRetries: 2 }
77];
78for (const config of orderConfigs) {
79 (0, ava_1.default)(`out of order await (async catch) with ${(0, util_1.inspect)(config)}`, testOrder, config);
80}
81async function readFirstLogfile(logDirectoryUrl) {
82 const url = new url_1.URL(logDirectoryUrl);
83 const buf = await (0, fs_extra_1.readFile)(url.pathname + "/0.log");
84 return buf
85 .toString()
86 .split("\n")
87 .map(m => m.replace(/^\[(\d+)\]/, "[$pid]"));
88}
89(0, ava_1.default)("local provider console.log, console.warn, and console.error with child process", async (t) => {
90 const faastModule = await (0, index_1.faastLocal)(funcs, {
91 childProcess: true,
92 concurrency: 1,
93 gc: "off"
94 });
95 try {
96 await faastModule.functions.consoleLog("Remote console.log output");
97 await faastModule.functions.consoleWarn("Remote console.warn output");
98 await faastModule.functions.consoleError("Remote console.error output");
99 await (0, util_2.sleep)(1000);
100 await faastModule.cleanup({ deleteResources: false });
101 const messages = await readFirstLogfile(faastModule.logUrl());
102 t.truthy(messages.find(s => s === "[$pid]: Remote console.log output"));
103 t.truthy(messages.find(s => s === "[$pid]: Remote console.warn output"));
104 t.truthy(messages.find(s => s === "[$pid]: Remote console.error output"));
105 }
106 finally {
107 await faastModule.cleanup({ deleteResources: false });
108 }
109});
110(0, ava_1.default)("local provider log files should be appended, not truncated, after child process crash", async (t) => {
111 const faastModule = await (0, index_1.faastLocal)(funcs, {
112 childProcess: true,
113 concurrency: 1,
114 maxRetries: 1,
115 gc: "off"
116 });
117 try {
118 await faastModule.functions.consoleLog("output 1");
119 try {
120 await faastModule.functions.processExit();
121 }
122 catch (err) { }
123 await faastModule.functions.consoleWarn("output 2");
124 // Wait for flush
125 await (0, util_2.sleep)(500);
126 const messages = await readFirstLogfile(faastModule.logUrl());
127 t.truthy(messages.find(s => s === "[$pid]: output 1"));
128 t.truthy(messages.find(s => s === "[$pid]: output 2"));
129 }
130 finally {
131 await faastModule.cleanup({ deleteResources: false });
132 }
133});
134(0, ava_1.default)("local provider child process exceptions should result in errors with logUrl", async (t) => {
135 const faastModule = await (0, index_1.faastLocal)(funcs, {
136 childProcess: true,
137 concurrency: 1,
138 maxRetries: 1,
139 gc: "off"
140 });
141 t.plan(1);
142 try {
143 await faastModule.functions.error("synthetic error");
144 }
145 catch (err) {
146 const info = error_1.FaastError.info(err);
147 t.true(typeof info.logUrl === "string" && info.logUrl.startsWith(" file:///"), (0, util_1.inspect)(err));
148 }
149 finally {
150 await faastModule.cleanup();
151 }
152});
153(0, ava_1.default)("local provider child process crashes should result in errors with logUrl", async (t) => {
154 const faastModule = await (0, index_1.faastLocal)(funcs, {
155 childProcess: true,
156 concurrency: 1,
157 maxRetries: 1,
158 gc: "off"
159 });
160 t.plan(1);
161 try {
162 await faastModule.functions.processExit(-1);
163 }
164 catch (err) {
165 const info = error_1.FaastError.info(err);
166 t.true(typeof info.logUrl === "string" && info.logUrl.startsWith(" file:///"), (0, util_1.inspect)(err));
167 }
168 finally {
169 await faastModule.cleanup();
170 }
171});
172(0, ava_1.default)("local provider concurrent executions with child processes", async (t) => {
173 await testConcurrency(t, {
174 options: {
175 childProcess: true
176 },
177 maxConcurrency: 5,
178 expectedConcurrency: 5
179 });
180});
181(0, ava_1.default)("local provider no concurrency for cpu bound work without child processes", async (t) => {
182 await testConcurrency(t, {
183 options: {
184 childProcess: false
185 },
186 maxConcurrency: 5,
187 expectedConcurrency: 1
188 });
189});
190(0, ava_1.default)("local provider cleanup waits for all child processes to exit", async (t) => {
191 const faastModule = await (0, index_1.faastLocal)(funcs, {
192 childProcess: true,
193 gc: "off"
194 });
195 faastModule.functions.spin(5000).catch(_ => { });
196 while (true) {
197 await (0, util_2.sleep)(100);
198 if (faastModule.state.executors.length > 0) {
199 break;
200 }
201 }
202 t.is(faastModule.state.executors.length, 1, "executor is not running");
203 await faastModule.cleanup({ gcTimeout: 60 });
204 t.is(faastModule.state.executors.length, 0, "executors are running after cleanup");
205});
206(0, ava_1.default)("local unresolved module", async (t) => {
207 t.plan(1);
208 try {
209 await (0, index_1.faastLocal)({});
210 }
211 catch (err) {
212 t.regex(err.message, /Could not find file/);
213 }
214});
215(0, ava_1.default)("local issue #37", async (t) => {
216 // Previously this code caused an exception about module wrapper not being
217 // re-entrant. The problem was a race condition between wrapper selection
218 // and execution in local provider. Solved by making wrapper selector a
219 // regular function instead of an async function.
220 const m = await (0, index_1.faastLocal)(funcs);
221 try {
222 const { identityString: identity } = m.functions;
223 await identity("a");
224 const b = identity("b");
225 const c = identity("c");
226 await b;
227 await c;
228 // Test succeeds if no exceptions are thrown.
229 t.true(true);
230 }
231 finally {
232 await m.cleanup();
233 }
234});
235//# 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"]}
\No newline at end of file