faastjs
Version:
Serverless batch computing made simple.
221 lines • 30 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.costSnapshot = exports.localPacker = exports.logUrl = exports.LocalImpl = exports.defaults = exports.defaultGcWorker = void 0;
const sys = require("child_process");
const fs_extra_1 = require("fs-extra");
const os_1 = require("os");
const path_1 = require("path");
const util_1 = require("util");
const cost_1 = require("../cost");
const log_1 = require("../log");
const packer_1 = require("../packer");
const provider_1 = require("../provider");
const shared_1 = require("../shared");
const throttle_1 = require("../throttle");
const wrapper_1 = require("../wrapper");
const localTrampolineFactory = require("./local-trampoline");
const exec = (0, util_1.promisify)(sys.exec);
function defaultGcWorker(dir) {
return (0, fs_extra_1.remove)(dir);
}
exports.defaultGcWorker = defaultGcWorker;
exports.defaults = {
...provider_1.commonDefaults,
concurrency: 10,
memorySize: 512,
_gcWorker: defaultGcWorker
};
exports.LocalImpl = {
name: "local",
initialize,
defaults: exports.defaults,
cleanup,
costSnapshot,
logUrl,
invoke,
poll,
responseQueueId
};
async function initialize(serverModule, nonce, options) {
const wrappers = [];
const { gc, retentionInDays, _gcWorker: gcWorker } = options;
let gcPromise;
if (gc === "auto" || gc === "force") {
gcPromise = collectGarbage(gcWorker, retentionInDays);
}
const tempDir = (0, path_1.join)((0, os_1.tmpdir)(), "faast", nonce);
log_1.log.info(`tempDir: ${tempDir} [${options.description}]`);
await (0, fs_extra_1.mkdirp)(tempDir);
const logDir = (0, path_1.join)(tempDir, "logs");
await (0, fs_extra_1.mkdir)(logDir);
const url = `file://${logDir}`;
log_1.log.info(`logURL: ${url}`);
const { childProcess, memorySize, timeout, env, validateSerialization } = options;
if (!childProcess) {
process.env = { ...process.env, ...env };
}
const { wrapperVerbose } = options.debugOptions;
const getWrapperInfo = () => {
const idleWrapper = wrappers.find(w => w.wrapper.executing === false);
if (idleWrapper) {
return idleWrapper;
}
let logStream;
let childlog = (msg) => {
if (logStream.writable) {
logStream.write(msg);
logStream.write("\n");
}
else {
log_1.log.provider(`WARNING: childlog not writable: ${msg}`);
}
};
const logFile = (0, path_1.join)(logDir, `${wrappers.length}.log`);
try {
log_1.log.info(`Creating write stream ${logFile}`);
logStream = (0, fs_extra_1.createWriteStream)(logFile);
}
catch (err) {
log_1.log.warn(`ERROR: Could not create log`);
log_1.log.warn(err);
childlog = console.log;
}
const childProcessMemoryLimitMb = options.childProcessMemoryMb;
const wrapperOptions2 = {
wrapperLog: childlog,
childProcess,
childProcessMemoryLimitMb,
childProcessTimeoutMs: timeout * 1000 - (childProcess ? 50 : 0),
childProcessEnvironment: env,
childDir: tempDir,
wrapperVerbose: wrapperVerbose || log_1.log.provider.enabled,
validateSerialization
};
const wrapper = new wrapper_1.Wrapper(require(serverModule), wrapperOptions2);
const rv = { wrapper, logUrl: `file://${logFile}`, logStream };
wrappers.push(rv);
return rv;
};
const packerResult = await localPacker(serverModule, options, { wrapperVerbose }, `faast-${nonce}`);
await (0, packer_1.unzipInDir)(tempDir, packerResult.archive);
if (options.packageJson) {
log_1.log.info(`Running 'npm install'`);
await exec("npm install --no-package-lock", { cwd: tempDir }).then(x => {
log_1.log.info(x.stdout);
if (x.stderr) {
log_1.log.warn(x.stderr);
}
});
}
return {
executors: wrappers,
getExecutor: getWrapperInfo,
tempDir,
logUrl: url,
gcPromise,
queue: new throttle_1.AsyncQueue(),
options
};
}
function logUrl(state) {
return state.logUrl;
}
exports.logUrl = logUrl;
async function localPacker(functionModule, options, wrapperOptions, FunctionName) {
return (0, packer_1.packer)(localTrampolineFactory, functionModule, options, wrapperOptions, FunctionName);
}
exports.localPacker = localPacker;
async function invoke(state, call, _) {
const {} = state;
const startTime = Date.now();
const { wrapper, logUrl: url } = state.getExecutor();
await wrapper.execute({ call, startTime, logUrl: url }, { onMessage: async (msg) => state.queue.enqueue(msg) });
}
async function poll(state, cancel) {
const message = await Promise.race([state.queue.next(), cancel]);
if (!message) {
return { Messages: [] };
}
return { Messages: [message] };
}
function responseQueueId(_state) {
return "<none>";
}
async function cleanup(state, options) {
log_1.log.info(`local cleanup starting.`);
await Promise.all(state.executors.map(e => e.wrapper.stop()));
await Promise.all(state.executors.map(e => new Promise(resolve => e.logStream?.end(resolve))));
state.executors = [];
if (state.gcPromise) {
await state.gcPromise;
}
if (options.deleteResources) {
const { tempDir } = state;
const pattern = new RegExp(`/faast/${shared_1.uuidv4Pattern}$`);
if (tempDir.match(pattern) && (await (0, fs_extra_1.pathExists)(tempDir))) {
log_1.log.info(`Deleting temp dir ${tempDir}`);
await (0, fs_extra_1.remove)(tempDir);
}
}
log_1.log.info(`local cleanup done.`);
}
let garbageCollectorRunning = false;
async function collectGarbage(gcWorker, retentionInDays) {
if (gcWorker === defaultGcWorker) {
if (garbageCollectorRunning) {
return;
}
garbageCollectorRunning = true;
}
const tmp = (0, path_1.join)((0, os_1.tmpdir)(), "faast");
log_1.log.gc(tmp);
try {
const dir = await (0, fs_extra_1.readdir)(tmp);
const pattern = new RegExp(`^${shared_1.uuidv4Pattern}$`);
for (const entry of dir) {
if (entry.match(pattern)) {
const faastDir = (0, path_1.join)(tmp, entry);
try {
const stats = await (0, fs_extra_1.stat)(faastDir);
if ((0, shared_1.hasExpired)(stats.atimeMs, retentionInDays)) {
log_1.log.gc(faastDir);
await gcWorker(faastDir);
}
}
catch (err) { }
}
}
}
catch (err) {
log_1.log.gc(err);
}
finally {
if (gcWorker === defaultGcWorker) {
garbageCollectorRunning = false;
}
}
}
async function costSnapshot(state, stats) {
const billedTimeStats = stats.estimatedBilledTime;
const seconds = (billedTimeStats.mean / 1000) * billedTimeStats.samples || 0;
const costMetrics = [];
const functionCallDuration = new cost_1.CostMetric({
name: "functionCallDuration",
pricing: 0,
unit: "second",
measured: seconds,
informationalOnly: true
});
costMetrics.push(functionCallDuration);
const functionCallRequests = new cost_1.CostMetric({
name: "functionCallRequests",
pricing: 0,
measured: stats.invocations,
unit: "request",
informationalOnly: true
});
costMetrics.push(functionCallRequests);
return new cost_1.CostSnapshot("local", state.options, stats, costMetrics);
}
exports.costSnapshot = costSnapshot;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"local-faast.js","sourceRoot":"","sources":["../../../src/local/local-faast.ts"],"names":[],"mappings":"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","sourcesContent":["import * as sys from \"child_process\";\nimport {\n    createWriteStream,\n    mkdir,\n    mkdirp,\n    pathExists,\n    readdir,\n    remove,\n    stat\n} from \"fs-extra\";\nimport { tmpdir } from \"os\";\nimport { join } from \"path\";\nimport { Writable } from \"stream\";\nimport { promisify } from \"util\";\nimport { CostMetric, CostSnapshot } from \"../cost\";\nimport { log } from \"../log\";\nimport { packer, PackerResult, unzipInDir } from \"../packer\";\nimport {\n    CleanupOptions,\n    commonDefaults,\n    CommonOptions,\n    FunctionStats,\n    Message,\n    PollResult,\n    ProviderImpl,\n    UUID\n} from \"../provider\";\nimport { hasExpired, uuidv4Pattern } from \"../shared\";\nimport { AsyncQueue } from \"../throttle\";\nimport { FunctionCall, Wrapper, WrapperOptions } from \"../wrapper\";\nimport * as localTrampolineFactory from \"./local-trampoline\";\n\nconst exec = promisify(sys.exec);\n\ninterface Executor {\n    wrapper: Wrapper;\n    logUrl: string;\n    logStream?: Writable;\n}\n\n/**\n * @public\n */\nexport interface LocalState {\n    /** @internal */\n    executors: Executor[];\n    /** @internal */\n    getExecutor: () => Executor;\n    /** The temporary directory where the local function is deployed. */\n    tempDir: string;\n    /** The file:// URL for the local function log file directory.  */\n    logUrl: string;\n    /** @internal */\n    gcPromise?: Promise<void>;\n    /** @internal */\n    queue: AsyncQueue<Message>;\n    /** Options used to initialize the local function. */\n    options: Required<LocalOptions>;\n}\n\n/**\n * Local provider options for {@link faastLocal}.\n *\n * @public\n */\nexport interface LocalOptions extends CommonOptions {\n    /** @internal */\n    _gcWorker?: (tempdir: string) => Promise<void>;\n}\n\nexport function defaultGcWorker(dir: string) {\n    return remove(dir);\n}\n\nexport const defaults: Required<LocalOptions> = {\n    ...commonDefaults,\n    concurrency: 10,\n    memorySize: 512,\n    _gcWorker: defaultGcWorker\n};\n\nexport const LocalImpl: ProviderImpl<LocalOptions, LocalState> = {\n    name: \"local\",\n    initialize,\n    defaults,\n    cleanup,\n    costSnapshot,\n    logUrl,\n    invoke,\n    poll,\n    responseQueueId\n};\n\nasync function initialize(\n    serverModule: string,\n    nonce: UUID,\n    options: Required<LocalOptions>\n): Promise<LocalState> {\n    const wrappers: Executor[] = [];\n    const { gc, retentionInDays, _gcWorker: gcWorker } = options;\n\n    let gcPromise;\n    if (gc === \"auto\" || gc === \"force\") {\n        gcPromise = collectGarbage(gcWorker, retentionInDays!);\n    }\n    const tempDir = join(tmpdir(), \"faast\", nonce);\n    log.info(`tempDir: ${tempDir} [${options.description}]`);\n    await mkdirp(tempDir);\n    const logDir = join(tempDir, \"logs\");\n    await mkdir(logDir);\n    const url = `file://${logDir}`;\n\n    log.info(`logURL: ${url}`);\n\n    const { childProcess, memorySize, timeout, env, validateSerialization } = options;\n\n    if (!childProcess) {\n        process.env = { ...process.env, ...env };\n    }\n    const { wrapperVerbose } = options.debugOptions;\n    const getWrapperInfo = () => {\n        const idleWrapper = wrappers.find(w => w.wrapper.executing === false);\n        if (idleWrapper) {\n            return idleWrapper;\n        }\n        let logStream!: Writable;\n        let childlog = (msg: string) => {\n            if (logStream.writable) {\n                logStream.write(msg);\n                logStream.write(\"\\n\");\n            } else {\n                log.provider(`WARNING: childlog not writable: ${msg}`);\n            }\n        };\n        const logFile = join(logDir, `${wrappers.length}.log`);\n\n        try {\n            log.info(`Creating write stream ${logFile}`);\n            logStream = createWriteStream(logFile);\n        } catch (err: any) {\n            log.warn(`ERROR: Could not create log`);\n            log.warn(err);\n            childlog = console.log;\n        }\n        const childProcessMemoryLimitMb = options.childProcessMemoryMb;\n        const wrapperOptions2: Required<WrapperOptions> = {\n            wrapperLog: childlog,\n            childProcess,\n            childProcessMemoryLimitMb,\n            childProcessTimeoutMs: timeout * 1000 - (childProcess ? 50 : 0),\n            childProcessEnvironment: env,\n            childDir: tempDir,\n            wrapperVerbose: wrapperVerbose || log.provider.enabled,\n            validateSerialization\n        };\n        const wrapper = new Wrapper(require(serverModule), wrapperOptions2);\n        const rv = { wrapper, logUrl: `file://${logFile}`, logStream };\n        wrappers.push(rv);\n        return rv;\n    };\n\n    const packerResult = await localPacker(\n        serverModule,\n        options,\n        { wrapperVerbose },\n        `faast-${nonce}`\n    );\n\n    await unzipInDir(tempDir, packerResult.archive);\n    if (options.packageJson) {\n        log.info(`Running 'npm install'`);\n\n        await exec(\"npm install --no-package-lock\", { cwd: tempDir }).then(x => {\n            log.info(x.stdout);\n            if (x.stderr) {\n                log.warn(x.stderr);\n            }\n        });\n    }\n\n    return {\n        executors: wrappers,\n        getExecutor: getWrapperInfo,\n        tempDir,\n        logUrl: url,\n        gcPromise,\n        queue: new AsyncQueue(),\n        options\n    };\n}\n\nexport function logUrl(state: LocalState) {\n    return state.logUrl;\n}\n\nexport async function localPacker(\n    functionModule: string,\n    options: CommonOptions,\n    wrapperOptions: WrapperOptions,\n    FunctionName: string\n): Promise<PackerResult> {\n    return packer(\n        localTrampolineFactory,\n        functionModule,\n        options,\n        wrapperOptions,\n        FunctionName\n    );\n}\n\nasync function invoke(\n    state: LocalState,\n    call: FunctionCall,\n    _: Promise<void>\n): Promise<void> {\n    const {} = state;\n    const startTime = Date.now();\n    const { wrapper, logUrl: url } = state.getExecutor();\n    await wrapper.execute(\n        { call, startTime, logUrl: url },\n        { onMessage: async msg => state.queue.enqueue(msg) }\n    );\n}\n\nasync function poll(state: LocalState, cancel: Promise<void>): Promise<PollResult> {\n    const message = await Promise.race([state.queue.next(), cancel]);\n    if (!message) {\n        return { Messages: [] };\n    }\n    return { Messages: [message] };\n}\n\nfunction responseQueueId(_state: LocalState): string {\n    return \"<none>\";\n}\n\nasync function cleanup(state: LocalState, options: CleanupOptions): Promise<void> {\n    log.info(`local cleanup starting.`);\n\n    await Promise.all(state.executors.map(e => e.wrapper.stop()));\n    await Promise.all(\n        state.executors.map(e => new Promise(resolve => e.logStream?.end(resolve)))\n    );\n    state.executors = [];\n    if (state.gcPromise) {\n        await state.gcPromise;\n    }\n\n    if (options.deleteResources) {\n        const { tempDir } = state;\n        const pattern = new RegExp(`/faast/${uuidv4Pattern}$`);\n        if (tempDir.match(pattern) && (await pathExists(tempDir))) {\n            log.info(`Deleting temp dir ${tempDir}`);\n            await remove(tempDir);\n        }\n    }\n    log.info(`local cleanup done.`);\n}\n\nlet garbageCollectorRunning = false;\n\nasync function collectGarbage(\n    gcWorker: (dir: string) => Promise<void>,\n    retentionInDays: number\n) {\n    if (gcWorker === defaultGcWorker) {\n        if (garbageCollectorRunning) {\n            return;\n        }\n        garbageCollectorRunning = true;\n    }\n    const tmp = join(tmpdir(), \"faast\");\n    log.gc(tmp);\n    try {\n        const dir = await readdir(tmp);\n        const pattern = new RegExp(`^${uuidv4Pattern}$`);\n        for (const entry of dir) {\n            if (entry.match(pattern)) {\n                const faastDir = join(tmp, entry);\n                try {\n                    const stats = await stat(faastDir);\n                    if (hasExpired(stats.atimeMs, retentionInDays)) {\n                        log.gc(faastDir);\n                        await gcWorker(faastDir);\n                    }\n                } catch (err: any) {}\n            }\n        }\n    } catch (err: any) {\n        log.gc(err);\n    } finally {\n        if (gcWorker === defaultGcWorker) {\n            garbageCollectorRunning = false;\n        }\n    }\n}\n\nexport async function costSnapshot(state: LocalState, stats: FunctionStats) {\n    const billedTimeStats = stats.estimatedBilledTime;\n    const seconds = (billedTimeStats.mean / 1000) * billedTimeStats.samples || 0;\n\n    const costMetrics: CostMetric[] = [];\n    const functionCallDuration = new CostMetric({\n        name: \"functionCallDuration\",\n        pricing: 0,\n        unit: \"second\",\n        measured: seconds,\n        informationalOnly: true\n    });\n    costMetrics.push(functionCallDuration);\n\n    const functionCallRequests = new CostMetric({\n        name: \"functionCallRequests\",\n        pricing: 0,\n        measured: stats.invocations,\n        unit: \"request\",\n        informationalOnly: true\n    });\n    costMetrics.push(functionCallRequests);\n    return new CostSnapshot(\"local\", state.options, stats, costMetrics);\n}\n"]}