UNPKG

faastjs

Version:

Serverless batch computing made simple.

236 lines 29.7 kB
"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":";;;AAAA,sDAA6C;AAC7C,uCAAoC;AACpC,6BAA0B;AAC1B,+BAA+B;AAC/B,oCAAoD;AACpD,oEAA8C;AAC9C,0CAA4D;AAC5D,wCAA0C;AAE1C,KAAK,UAAU,WAAW,CAAC,CAAmB,EAAE,OAAqB;IACjE,MAAM,CAAC,GAAG,MAAM,IAAA,kBAAU,EAAC,KAAK,EAAE;QAC9B,EAAE,EAAE,KAAK;QACT,GAAG,OAAO;KACb,CAAC,CAAC;IACH,IAAI,IAAI,GAAG,CAAC,CAAC;IAEb,CAAC,CAAC,SAAS;SACN,KAAK,CAAC,OAAO,CAAC;SACd,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;SACjB,KAAK,CAAC,CAAC,CAAC,EAAE,GAAE,CAAC,CAAC,CAAC;IAEpB,CAAC,CAAC,SAAS;SACN,KAAK,CAAC,IAAI,CAAC;SACX,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;SACjB,KAAK,CAAC,CAAC,CAAC,EAAE,GAAE,CAAC,CAAC,CAAC;IAEpB,MAAM,CAAC,CAAC,OAAO,EAAE,CAAC;IAClB,CAAC,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;AAClB,CAAC;AAED,KAAK,UAAU,SAAS,CAAC,CAAmB,EAAE,OAAqB;IAC/D,MAAM,WAAW,GAAG,MAAM,IAAA,kBAAU,EAAC,KAAK,EAAE;QACxC,EAAE,EAAE,KAAK;QACT,GAAG,OAAO;KACb,CAAC,CAAC;IACH,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAEV,MAAM,CAAC,GAAG,WAAW,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;IAC9C,MAAM,CAAC,GAAG,WAAW,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IACzC,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,EAAE,SAAS,CAAC,CAAC;IACzB,IAAI;QACA,MAAM,CAAC,CAAC;KACX;IAAC,OAAO,GAAQ,EAAE;QACf,CAAC,CAAC,EAAE,CAAC,GAAG,EAAE,SAAS,CAAC,CAAC;KACxB;YAAS;QACN,MAAM,WAAW,CAAC,OAAO,EAAE,CAAC;KAC/B;AACL,CAAC;AAED,KAAK,UAAU,eAAe,CAC1B,CAAmB,EACnB,EACI,OAAO,EACP,cAAc,EACd,mBAAmB,EAKtB;IAED,MAAM,WAAW,GAAG,MAAM,IAAA,kBAAU,EAAC,KAAK,EAAE;QACxC,GAAG,OAAO;QACV,EAAE,EAAE,KAAK;QACT,WAAW,EAAE,cAAc;KAC9B,CAAC,CAAC;IAEH,IAAI;QACA,MAAM,CAAC,GAAG,cAAc,GAAG,CAAC,CAAC;QAC7B,MAAM,QAAQ,GAAG,EAAE,CAAC;QACpB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE;YACxB,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;SACnD;QAED,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAC5C,CAAC,CAAC,EAAE,CAAC,IAAA,yBAAkB,EAAC,OAAO,CAAC,EAAE,mBAAmB,CAAC,CAAC;KAC1D;YAAS;QACN,MAAM,WAAW,CAAC,OAAO,EAAE,CAAC;KAC/B;AACL,CAAC;AAED,IAAA,aAAI,EAAC,yCAAyC,EAAE,WAAW,EAAE,EAAE,CAAC,CAAC;AACjE,IAAA,aAAI,EAAC,4DAA4D,EAAE,WAAW,EAAE;IAC5E,YAAY,EAAE,IAAI;CACrB,CAAC,CAAC;AAEH,MAAM,YAAY,GAAG;IACjB,EAAE,YAAY,EAAE,KAAK,EAAE,WAAW,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE;IACtD,EAAE,YAAY,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE;IACrD,EAAE,YAAY,EAAE,KAAK,EAAE,WAAW,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE;IACtD,EAAE,YAAY,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE;IACrD,EAAE,YAAY,EAAE,KAAK,EAAE,WAAW,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE;IACtD,EAAE,YAAY,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE;CACxD,CAAC;AAEF,KAAK,MAAM,MAAM,IAAI,YAAY,EAAE;IAC/B,IAAA,aAAI,EAAC,yCAAyC,IAAA,cAAO,EAAC,MAAM,CAAC,EAAE,EAAE,SAAS,EAAE,MAAM,CAAC,CAAC;CACvF;AAED,KAAK,UAAU,gBAAgB,CAAC,eAAuB;IACnD,MAAM,GAAG,GAAG,IAAI,SAAG,CAAC,eAAe,CAAC,CAAC;IACrC,MAAM,GAAG,GAAG,MAAM,IAAA,mBAAQ,EAAC,GAAG,CAAC,QAAQ,GAAG,QAAQ,CAAC,CAAC;IACpD,OAAO,GAAG;SACL,QAAQ,EAAE;SACV,KAAK,CAAC,IAAI,CAAC;SACX,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,YAAY,EAAE,QAAQ,CAAC,CAAC,CAAC;AACrD,CAAC;AAED,IAAA,aAAI,EAAC,gFAAgF,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IAC7F,MAAM,WAAW,GAAG,MAAM,IAAA,kBAAU,EAAC,KAAK,EAAE;QACxC,YAAY,EAAE,IAAI;QAClB,WAAW,EAAE,CAAC;QACd,EAAE,EAAE,KAAK;KACZ,CAAC,CAAC;IACH,IAAI;QACA,MAAM,WAAW,CAAC,SAAS,CAAC,UAAU,CAAC,2BAA2B,CAAC,CAAC;QACpE,MAAM,WAAW,CAAC,SAAS,CAAC,WAAW,CAAC,4BAA4B,CAAC,CAAC;QACtE,MAAM,WAAW,CAAC,SAAS,CAAC,YAAY,CAAC,6BAA6B,CAAC,CAAC;QACxE,MAAM,IAAA,YAAK,EAAC,IAAI,CAAC,CAAC;QAClB,MAAM,WAAW,CAAC,OAAO,CAAC,EAAE,eAAe,EAAE,KAAK,EAAE,CAAC,CAAC;QACtD,MAAM,QAAQ,GAAG,MAAM,gBAAgB,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC;QAC9D,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,mCAAmC,CAAC,CAAC,CAAC;QACxE,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,oCAAoC,CAAC,CAAC,CAAC;QACzE,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,qCAAqC,CAAC,CAAC,CAAC;KAC7E;YAAS;QACN,MAAM,WAAW,CAAC,OAAO,CAAC,EAAE,eAAe,EAAE,KAAK,EAAE,CAAC,CAAC;KACzD;AACL,CAAC,CAAC,CAAC;AAEH,IAAA,aAAI,EAAC,uFAAuF,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACpG,MAAM,WAAW,GAAG,MAAM,IAAA,kBAAU,EAAC,KAAK,EAAE;QACxC,YAAY,EAAE,IAAI;QAClB,WAAW,EAAE,CAAC;QACd,UAAU,EAAE,CAAC;QACb,EAAE,EAAE,KAAK;KACZ,CAAC,CAAC;IACH,IAAI;QACA,MAAM,WAAW,CAAC,SAAS,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;QACnD,IAAI;YACA,MAAM,WAAW,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;SAC7C;QAAC,OAAO,GAAQ,EAAE,GAAE;QACrB,MAAM,WAAW,CAAC,SAAS,CAAC,WAAW,CAAC,UAAU,CAAC,CAAC;QAEpD,iBAAiB;QACjB,MAAM,IAAA,YAAK,EAAC,GAAG,CAAC,CAAC;QACjB,MAAM,QAAQ,GAAG,MAAM,gBAAgB,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC;QAE9D,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,kBAAkB,CAAC,CAAC,CAAC;QACvD,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,kBAAkB,CAAC,CAAC,CAAC;KAC1D;YAAS;QACN,MAAM,WAAW,CAAC,OAAO,CAAC,EAAE,eAAe,EAAE,KAAK,EAAE,CAAC,CAAC;KACzD;AACL,CAAC,CAAC,CAAC;AAEH,IAAA,aAAI,EAAC,6EAA6E,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IAC1F,MAAM,WAAW,GAAG,MAAM,IAAA,kBAAU,EAAC,KAAK,EAAE;QACxC,YAAY,EAAE,IAAI;QAClB,WAAW,EAAE,CAAC;QACd,UAAU,EAAE,CAAC;QACb,EAAE,EAAE,KAAK;KACZ,CAAC,CAAC;IACH,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACV,IAAI;QACA,MAAM,WAAW,CAAC,SAAS,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAC;KACxD;IAAC,OAAO,GAAQ,EAAE;QACf,MAAM,IAAI,GAAG,kBAAU,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAClC,CAAC,CAAC,IAAI,CACF,OAAO,IAAI,CAAC,MAAM,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,WAAW,CAAC,EACtE,IAAA,cAAO,EAAC,GAAG,CAAC,CACf,CAAC;KACL;YAAS;QACN,MAAM,WAAW,CAAC,OAAO,EAAE,CAAC;KAC/B;AACL,CAAC,CAAC,CAAC;AAEH,IAAA,aAAI,EAAC,0EAA0E,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACvF,MAAM,WAAW,GAAG,MAAM,IAAA,kBAAU,EAAC,KAAK,EAAE;QACxC,YAAY,EAAE,IAAI;QAClB,WAAW,EAAE,CAAC;QACd,UAAU,EAAE,CAAC;QACb,EAAE,EAAE,KAAK;KACZ,CAAC,CAAC;IACH,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACV,IAAI;QACA,MAAM,WAAW,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC;KAC/C;IAAC,OAAO,GAAQ,EAAE;QACf,MAAM,IAAI,GAAG,kBAAU,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAClC,CAAC,CAAC,IAAI,CACF,OAAO,IAAI,CAAC,MAAM,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,WAAW,CAAC,EACtE,IAAA,cAAO,EAAC,GAAG,CAAC,CACf,CAAC;KACL;YAAS;QACN,MAAM,WAAW,CAAC,OAAO,EAAE,CAAC;KAC/B;AACL,CAAC,CAAC,CAAC;AAEH,IAAA,aAAI,EAAC,2DAA2D,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACxE,MAAM,eAAe,CAAC,CAAC,EAAE;QACrB,OAAO,EAAE;YACL,YAAY,EAAE,IAAI;SACrB;QACD,cAAc,EAAE,CAAC;QACjB,mBAAmB,EAAE,CAAC;KACzB,CAAC,CAAC;AACP,CAAC,CAAC,CAAC;AAEH,IAAA,aAAI,EAAC,0EAA0E,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACvF,MAAM,eAAe,CAAC,CAAC,EAAE;QACrB,OAAO,EAAE;YACL,YAAY,EAAE,KAAK;SACtB;QACD,cAAc,EAAE,CAAC;QACjB,mBAAmB,EAAE,CAAC;KACzB,CAAC,CAAC;AACP,CAAC,CAAC,CAAC;AAEH,IAAA,aAAI,EAAC,8DAA8D,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IAC3E,MAAM,WAAW,GAAG,MAAM,IAAA,kBAAU,EAAC,KAAK,EAAE;QACxC,YAAY,EAAE,IAAI;QAClB,EAAE,EAAE,KAAK;KACZ,CAAC,CAAC;IACH,WAAW,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,GAAE,CAAC,CAAC,CAAC;IAChD,OAAO,IAAI,EAAE;QACT,MAAM,IAAA,YAAK,EAAC,GAAG,CAAC,CAAC;QACjB,IAAI,WAAW,CAAC,KAAK,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE;YACxC,MAAM;SACT;KACJ;IACD,CAAC,CAAC,EAAE,CAAC,WAAW,CAAC,KAAK,CAAC,SAAS,CAAC,MAAM,EAAE,CAAC,EAAE,yBAAyB,CAAC,CAAC;IACvE,MAAM,WAAW,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,EAAE,EAAE,CAAC,CAAC;IAC7C,CAAC,CAAC,EAAE,CAAC,WAAW,CAAC,KAAK,CAAC,SAAS,CAAC,MAAM,EAAE,CAAC,EAAE,qCAAqC,CAAC,CAAC;AACvF,CAAC,CAAC,CAAC;AAEH,IAAA,aAAI,EAAC,yBAAyB,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IACtC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACV,IAAI;QACA,MAAM,IAAA,kBAAU,EAAC,EAAE,CAAC,CAAC;KACxB;IAAC,OAAO,GAAQ,EAAE;QACf,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,OAAO,EAAE,qBAAqB,CAAC,CAAC;KAC/C;AACL,CAAC,CAAC,CAAC;AAEH,IAAA,aAAI,EAAC,iBAAiB,EAAE,KAAK,EAAC,CAAC,EAAC,EAAE;IAC9B,0EAA0E;IAC1E,yEAAyE;IACzE,uEAAuE;IACvE,iDAAiD;IACjD,MAAM,CAAC,GAAG,MAAM,IAAA,kBAAU,EAAC,KAAK,CAAC,CAAC;IAClC,IAAI;QACA,MAAM,EAAE,cAAc,EAAE,QAAQ,EAAE,GAAG,CAAC,CAAC,SAAS,CAAC;QACjD,MAAM,QAAQ,CAAC,GAAG,CAAC,CAAC;QACpB,MAAM,CAAC,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC;QACxB,MAAM,CAAC,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC;QACxB,MAAM,CAAC,CAAC;QACR,MAAM,CAAC,CAAC;QACR,6CAA6C;QAC7C,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;KAChB;YAAS;QACN,MAAM,CAAC,CAAC,OAAO,EAAE,CAAC;KACrB;AACL,CAAC,CAAC,CAAC","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"]}