UNPKG

faastjs

Version:

Serverless batch computing made simple.

239 lines 41.9 kB
#!/usr/bin/env node "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); const tslib_1 = require("tslib"); require("source-map-support").install(); const client_cloudwatch_logs_1 = require("@aws-sdk/client-cloudwatch-logs"); const client_iam_1 = require("@aws-sdk/client-iam"); const client_lambda_1 = require("@aws-sdk/client-lambda"); const client_sns_1 = require("@aws-sdk/client-sns"); const client_sqs_1 = require("@aws-sdk/client-sqs"); const commander_1 = require("commander"); const fs_extra_1 = require("fs-extra"); const ora_1 = tslib_1.__importDefault(require("ora")); const os_1 = require("os"); const path_1 = tslib_1.__importDefault(require("path")); const readline = tslib_1.__importStar(require("readline")); const awsFaast = tslib_1.__importStar(require("./aws/aws-faast")); const cache_1 = require("./cache"); const shared_1 = require("./shared"); const throttle_1 = require("./throttle"); const warn = console.warn; const log = console.log; async function deleteResources(name, matchingResources, doRemove, { concurrency = 10, rate = 5, burst = 5 } = {}) { if (matchingResources.length > 0) { const timeEstimate = (nResources) => nResources <= 5 ? "" : `(est: ${(nResources / 5).toFixed(0)}s)`; const updateSpinnerText = (nResources = 0) => `Deleting ${matchingResources.length} ${name} ${timeEstimate(nResources)}`; const spinner = (0, ora_1.default)(updateSpinnerText(matchingResources.length)).start(); let done = 0; const scheduleRemove = (0, throttle_1.throttle)({ concurrency, rate, burst, retry: 5 }, async (arg) => { await doRemove(arg); done++; }); const timer = setInterval(() => (spinner.text = updateSpinnerText(matchingResources.length - done)), 1000); try { await Promise.all(matchingResources.map(resource => scheduleRemove(resource).catch(err => console.warn(`Could not remove resource ${resource}: ${err}`)))); } finally { clearInterval(timer); spinner.text = updateSpinnerText(); } spinner.stopAndPersist({ symbol: "✔" }); } } async function cleanupAWS({ region, execute }) { let nResources = 0; const output = (msg) => !execute && log(msg); const { cloudwatch, iam, lambda, sns, sqs, s3 } = await awsFaast.createAwsApis(region); async function listAWSResource(pattern, getList, extractList, extractElement) { const allResources = []; for await (const page of getList()) { const elems = (page && extractList(page)) || []; allResources.push(...elems.map(elem => extractElement(elem) || "")); } const matchingResources = allResources.filter(t => t.match(pattern)); matchingResources.forEach(resource => output(` ${resource}`)); return matchingResources; } async function deleteAWSResource(name, pattern, getList, extractList, extractElement, doRemove) { const allResources = await listAWSResource(pattern, getList, extractList, extractElement); nResources += allResources.length; if (execute) { await deleteResources(name, allResources, doRemove, { concurrency: 10, rate: 5, burst: 5 }); } } output(`SNS subscriptions`); await deleteAWSResource("SNS subscription(s)", new RegExp(`:faast-${shared_1.uuidv4Pattern}`), () => (0, client_sns_1.paginateListSubscriptions)({ client: sns }, {}), page => page.Subscriptions, subscription => subscription.SubscriptionArn, SubscriptionArn => sns.unsubscribe({ SubscriptionArn })); output(`SNS topics`); await deleteAWSResource("SNS topic(s)", new RegExp(`:faast-${shared_1.uuidv4Pattern}`), () => (0, client_sns_1.paginateListTopics)({ client: sns }, {}), page => page.Topics, topic => topic.TopicArn, TopicArn => sns.deleteTopic({ TopicArn })); output(`SQS queues`); await deleteAWSResource("SQS queue(s)", new RegExp(`/faast-${shared_1.uuidv4Pattern}`), () => (0, client_sqs_1.paginateListQueues)({ client: sqs }, {}), page => page.QueueUrls, queueUrl => queueUrl, QueueUrl => sqs.deleteQueue({ QueueUrl })); async function* listBuckets() { const result = s3.listBuckets({}); yield result; return result; } output(`S3 buckets`); await deleteAWSResource("S3 bucket(s)", new RegExp(`^faast-${shared_1.uuidv4Pattern}`), () => listBuckets(), page => page.Buckets, Bucket => Bucket.Name, async (Bucket) => { const objects = await s3.listObjectsV2({ Bucket, Prefix: "faast-" }); const keys = (objects.Contents || []).map(entry => ({ Key: entry.Key })); if (keys.length > 0) { await s3.deleteObjects({ Bucket, Delete: { Objects: keys } }); } await s3.deleteBucket({ Bucket }); }); output(`Lambda functions`); await deleteAWSResource("Lambda function(s)", new RegExp(`^faast-${shared_1.uuidv4Pattern}`), () => (0, client_lambda_1.paginateListFunctions)({ client: lambda }, {}), page => page.Functions, func => func.FunctionName, FunctionName => lambda.deleteFunction({ FunctionName })); output(`IAM roles`); await deleteAWSResource("IAM role(s)", /^faast-cached-lambda-role$/, () => (0, client_iam_1.paginateListRoles)({ client: iam }, {}), page => page.Roles, role => role.RoleName, RoleName => awsFaast.deleteRole(RoleName, iam)); output(`IAM test roles`); await deleteAWSResource("IAM test role(s)", new RegExp(`^faast-test-.*${shared_1.uuidv4Pattern}$`), () => (0, client_iam_1.paginateListRoles)({ client: iam }, {}), page => page.Roles, role => role.RoleName, RoleName => awsFaast.deleteRole(RoleName, iam)); output(`Lambda layers`); await deleteAWSResource("Lambda layer(s)", new RegExp(`^faast-(${shared_1.uuidv4Pattern})|([a-f0-9]{64})`), () => (0, client_lambda_1.paginateListLayers)({ client: lambda }, { CompatibleRuntime: "nodejs" }), page => page.Layers, layer => layer.LayerName, async (LayerName) => { const versions = await lambda.listLayerVersions({ LayerName }); for (const layerVersion of versions.LayerVersions || []) { await lambda.deleteLayerVersion({ LayerName, VersionNumber: layerVersion.Version }); } }); async function cleanupCacheDir(cache) { output(`Persistent cache: ${cache.dir}`); const entries = await cache.entries(); if (!execute) { output(` cache entries: ${entries.length}`); } nResources += entries.length; if (execute) { cache.clear({ leaveEmptyDir: false }); } } for (const cache of (0, shared_1.keysOf)(cache_1.caches)) { await cleanupCacheDir(await cache_1.caches[cache]); } output(`Cloudwatch log groups`); await deleteAWSResource("Cloudwatch log group(s)", new RegExp(`/faast-${shared_1.uuidv4Pattern}$`), () => (0, client_cloudwatch_logs_1.paginateDescribeLogGroups)({ client: cloudwatch }, {}), page => page.logGroups, logGroup => logGroup.logGroupName, logGroupName => cloudwatch.deleteLogGroup({ logGroupName })); return nResources; } async function cleanupLocal({ execute }) { const output = (msg) => !execute && log(msg); const tmpDir = (0, os_1.tmpdir)(); const dir = await (0, fs_extra_1.readdir)(tmpDir); let nResources = 0; output(`Temporary directories:`); const entryRegexp = new RegExp(`^faast-${shared_1.uuidv4Pattern}$`); for (const entry of dir) { if (entry.match(entryRegexp)) { nResources++; const faastDir = path_1.default.join(tmpDir, entry); output(`${faastDir}`); if (execute) { await (0, fs_extra_1.remove)(faastDir); } } } return nResources; } async function prompt() { const rl = readline.createInterface({ input: process.stdin, output: process.stdout }); await new Promise(resolve => { rl.question("WARNING: this operation will delete resources. Confirm? [y/N] ", answer => { if (answer !== "y") { log(`Execution aborted.`); process.exit(0); } rl.close(); resolve(); }); }); } async function runCleanup(cloud, options) { let nResources = 0; if (cloud === "aws") { nResources = await cleanupAWS(options); } else if (cloud === "local") { nResources = await cleanupLocal(options); } else { warn(`Unknown cloud name "${cloud}". Must specify "aws" or "local".`); process.exit(-1); } if (options.execute) { log(`Done.`); } else { if (nResources === 0) { log(`No resources to clean up.`); } } return nResources; } async function main() { let cloud; let command; commander_1.program .version("0.1.0") .option("-v, --verbose", "Verbose mode") .option("-r, --region <region>", "Cloud region to operate on. Defaults to us-west-2 for AWS.") .option("-x, --execute", "Execute the cleanup process. If this option is not specified, the output will be a dry run.") .option("-f, --force", "When used with -x, skips the prompt") .command("cleanup <cloud>") .description(`Cleanup faast.js resources that may have leaked. The <cloud> argument must be "aws" or "local". By default the output is a dry run and will only print the actions that would be performed if '-x' is specified.`) .action((arg) => { command = "cleanup"; cloud = arg; }); const opts = commander_1.program.parse(process.argv).opts(); if (opts.verbose) { process.env.DEBUG = "faast:*"; } const execute = opts.execute || false; let region = opts.region; if (!region) { switch (cloud) { case "aws": region = awsFaast.defaults.region; break; } } const force = opts.force || false; region && log(`Region: ${region}`); const options = { region, execute }; let nResources = 0; if (command === "cleanup") { if (execute && !force) { nResources = await runCleanup(cloud, { ...options, execute: false }); if (nResources > 0) { await prompt(); } else { process.exit(0); } } nResources = await runCleanup(cloud, options); if (!execute && nResources > 0) { log(`(dryrun mode, no resources will be deleted, specify -x to execute cleanup)`); } } else { log(`No command specified.`); commander_1.program.help(); } } main(); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli.js","sourceRoot":"","sources":["../../src/cli.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\n\nrequire(\"source-map-support\").install();\nimport { paginateDescribeLogGroups } from \"@aws-sdk/client-cloudwatch-logs\";\nimport { paginateListRoles } from \"@aws-sdk/client-iam\";\nimport { paginateListFunctions, paginateListLayers } from \"@aws-sdk/client-lambda\";\nimport { paginateListSubscriptions, paginateListTopics } from \"@aws-sdk/client-sns\";\nimport { paginateListQueues } from \"@aws-sdk/client-sqs\";\nimport { Paginator } from \"@aws-sdk/types\";\nimport { program } from \"commander\";\nimport { readdir, remove } from \"fs-extra\";\nimport ora from \"ora\";\nimport { tmpdir } from \"os\";\nimport path from \"path\";\nimport * as readline from \"readline\";\nimport * as awsFaast from \"./aws/aws-faast\";\nimport { PersistentCache, caches } from \"./cache\";\nimport { keysOf, uuidv4Pattern } from \"./shared\";\nimport { throttle } from \"./throttle\";\n\nconst warn = console.warn;\nconst log = console.log;\n\ninterface CleanupOptions {\n    region?: string; // AWS only.\n    execute: boolean;\n}\n\nasync function deleteResources(\n    name: string,\n    matchingResources: string[],\n    doRemove: (arg: string) => Promise<any>,\n    { concurrency = 10, rate = 5, burst = 5 } = {}\n) {\n    if (matchingResources.length > 0) {\n        const timeEstimate = (nResources: number) =>\n            nResources <= 5 ? \"\" : `(est: ${(nResources / 5).toFixed(0)}s)`;\n        const updateSpinnerText = (nResources: number = 0) =>\n            `Deleting ${matchingResources.length} ${name} ${timeEstimate(nResources)}`;\n        const spinner = ora(updateSpinnerText(matchingResources.length)).start();\n        let done = 0;\n        const scheduleRemove = throttle(\n            {\n                concurrency,\n                rate,\n                burst,\n                retry: 5\n            },\n            async arg => {\n                await doRemove(arg);\n                done++;\n            }\n        );\n        const timer = setInterval(\n            () => (spinner.text = updateSpinnerText(matchingResources.length - done)),\n            1000\n        );\n        try {\n            await Promise.all(\n                matchingResources.map(resource =>\n                    scheduleRemove(resource).catch(err =>\n                        console.warn(`Could not remove resource ${resource}: ${err}`)\n                    )\n                )\n            );\n        } finally {\n            clearInterval(timer);\n            spinner.text = updateSpinnerText();\n        }\n        spinner.stopAndPersist({ symbol: \"✔\" });\n    }\n}\n\nasync function cleanupAWS({ region, execute }: CleanupOptions) {\n    let nResources = 0;\n    const output = (msg: string) => !execute && log(msg);\n    const { cloudwatch, iam, lambda, sns, sqs, s3 } = await awsFaast.createAwsApis(\n        region! as awsFaast.AwsRegion\n    );\n\n    async function listAWSResource<T, U>(\n        pattern: RegExp,\n        getList: () => Paginator<T>,\n        extractList: (arg: T) => U[] | undefined,\n        extractElement: (arg: U) => string | undefined\n    ) {\n        const allResources: string[] = [];\n        for await (const page of getList()) {\n            const elems = (page && extractList(page)) || [];\n            allResources.push(...elems.map(elem => extractElement(elem) || \"\"));\n        }\n        const matchingResources = allResources.filter(t => t.match(pattern));\n        matchingResources.forEach(resource => output(`  ${resource}`));\n        return matchingResources;\n    }\n\n    async function deleteAWSResource<T, U>(\n        name: string,\n        pattern: RegExp,\n        getList: () => Paginator<T>,\n        extractList: (arg: T) => U[] | undefined,\n        extractElement: (arg: U) => string | undefined,\n        doRemove: (arg: string) => Promise<any>\n    ) {\n        const allResources = await listAWSResource(\n            pattern,\n            getList,\n            extractList,\n            extractElement\n        );\n        nResources += allResources.length;\n        if (execute) {\n            await deleteResources(name, allResources, doRemove, {\n                concurrency: 10,\n                rate: 5,\n                burst: 5\n            });\n        }\n    }\n\n    output(`SNS subscriptions`);\n    await deleteAWSResource(\n        \"SNS subscription(s)\",\n        new RegExp(`:faast-${uuidv4Pattern}`),\n        () => paginateListSubscriptions({ client: sns }, {}),\n        page => page.Subscriptions,\n        subscription => subscription.SubscriptionArn,\n        SubscriptionArn => sns.unsubscribe({ SubscriptionArn })\n    );\n\n    output(`SNS topics`);\n    await deleteAWSResource(\n        \"SNS topic(s)\",\n        new RegExp(`:faast-${uuidv4Pattern}`),\n        () => paginateListTopics({ client: sns }, {}),\n        page => page.Topics,\n        topic => topic.TopicArn,\n        TopicArn => sns.deleteTopic({ TopicArn })\n    );\n\n    output(`SQS queues`);\n    await deleteAWSResource(\n        \"SQS queue(s)\",\n        new RegExp(`/faast-${uuidv4Pattern}`),\n        () => paginateListQueues({ client: sqs }, {}),\n        page => page.QueueUrls,\n        queueUrl => queueUrl,\n        QueueUrl => sqs.deleteQueue({ QueueUrl })\n    );\n\n    async function* listBuckets() {\n        const result = s3.listBuckets({});\n        yield result;\n        return result;\n    }\n\n    output(`S3 buckets`);\n    await deleteAWSResource(\n        \"S3 bucket(s)\",\n        new RegExp(`^faast-${uuidv4Pattern}`),\n        () => listBuckets(),\n        page => page.Buckets,\n        Bucket => Bucket.Name,\n        async Bucket => {\n            const objects = await s3.listObjectsV2({ Bucket, Prefix: \"faast-\" });\n            const keys = (objects.Contents || []).map(entry => ({ Key: entry.Key! }));\n            if (keys.length > 0) {\n                await s3.deleteObjects({ Bucket, Delete: { Objects: keys } });\n            }\n            await s3.deleteBucket({ Bucket });\n        }\n    );\n\n    output(`Lambda functions`);\n    await deleteAWSResource(\n        \"Lambda function(s)\",\n        new RegExp(`^faast-${uuidv4Pattern}`),\n        () => paginateListFunctions({ client: lambda }, {}),\n        page => page.Functions,\n        func => func.FunctionName,\n        FunctionName => lambda.deleteFunction({ FunctionName })\n    );\n\n    output(`IAM roles`);\n    await deleteAWSResource(\n        \"IAM role(s)\",\n        /^faast-cached-lambda-role$/,\n        () => paginateListRoles({ client: iam }, {}),\n        page => page.Roles,\n        role => role.RoleName,\n        RoleName => awsFaast.deleteRole(RoleName, iam)\n    );\n\n    output(`IAM test roles`);\n    await deleteAWSResource(\n        \"IAM test role(s)\",\n        new RegExp(`^faast-test-.*${uuidv4Pattern}$`),\n        () => paginateListRoles({ client: iam }, {}),\n        page => page.Roles,\n        role => role.RoleName,\n        RoleName => awsFaast.deleteRole(RoleName, iam)\n    );\n\n    output(`Lambda layers`);\n\n    await deleteAWSResource(\n        \"Lambda layer(s)\",\n        new RegExp(`^faast-(${uuidv4Pattern})|([a-f0-9]{64})`),\n        () => paginateListLayers({ client: lambda }, { CompatibleRuntime: \"nodejs\" }),\n        page => page.Layers,\n        layer => layer.LayerName,\n        async LayerName => {\n            const versions = await lambda.listLayerVersions({ LayerName });\n            for (const layerVersion of versions.LayerVersions || []) {\n                await lambda.deleteLayerVersion({\n                    LayerName,\n                    VersionNumber: layerVersion.Version!\n                });\n            }\n        }\n    );\n\n    async function cleanupCacheDir(cache: PersistentCache) {\n        output(`Persistent cache: ${cache.dir}`);\n        const entries = await cache.entries();\n        if (!execute) {\n            output(`  cache entries: ${entries.length}`);\n        }\n        nResources += entries.length;\n        if (execute) {\n            cache.clear({ leaveEmptyDir: false });\n        }\n    }\n\n    for (const cache of keysOf(caches)) {\n        await cleanupCacheDir(await caches[cache]);\n    }\n\n    output(`Cloudwatch log groups`);\n    await deleteAWSResource(\n        \"Cloudwatch log group(s)\",\n        new RegExp(`/faast-${uuidv4Pattern}$`),\n        () => paginateDescribeLogGroups({ client: cloudwatch }, {}),\n        page => page.logGroups,\n        logGroup => logGroup.logGroupName,\n        logGroupName => cloudwatch.deleteLogGroup({ logGroupName })\n    );\n\n    return nResources;\n}\n\nasync function cleanupLocal({ execute }: CleanupOptions) {\n    const output = (msg: string) => !execute && log(msg);\n    const tmpDir = tmpdir();\n    const dir = await readdir(tmpDir);\n    let nResources = 0;\n    output(`Temporary directories:`);\n    const entryRegexp = new RegExp(`^faast-${uuidv4Pattern}$`);\n    for (const entry of dir) {\n        if (entry.match(entryRegexp)) {\n            nResources++;\n            const faastDir = path.join(tmpDir, entry);\n            output(`${faastDir}`);\n            if (execute) {\n                await remove(faastDir);\n            }\n        }\n    }\n    return nResources;\n}\n\nasync function prompt() {\n    const rl = readline.createInterface({\n        input: process.stdin,\n        output: process.stdout\n    });\n\n    await new Promise<void>(resolve => {\n        rl.question(\n            \"WARNING: this operation will delete resources. Confirm? [y/N] \",\n            answer => {\n                if (answer !== \"y\") {\n                    log(`Execution aborted.`);\n                    process.exit(0);\n                }\n                rl.close();\n                resolve();\n            }\n        );\n    });\n}\n\nasync function runCleanup(cloud: string, options: CleanupOptions) {\n    let nResources = 0;\n    if (cloud === \"aws\") {\n        nResources = await cleanupAWS(options);\n    } else if (cloud === \"local\") {\n        nResources = await cleanupLocal(options);\n    } else {\n        warn(`Unknown cloud name \"${cloud}\". Must specify \"aws\" or \"local\".`);\n        process.exit(-1);\n    }\n    if (options.execute) {\n        log(`Done.`);\n    } else {\n        if (nResources === 0) {\n            log(`No resources to clean up.`);\n        }\n    }\n    return nResources;\n}\n\nasync function main() {\n    let cloud!: string;\n    let command: string | undefined;\n    program\n        .version(\"0.1.0\")\n        .option(\"-v, --verbose\", \"Verbose mode\")\n        .option(\n            \"-r, --region <region>\",\n            \"Cloud region to operate on. Defaults to us-west-2 for AWS.\"\n        )\n        .option(\n            \"-x, --execute\",\n            \"Execute the cleanup process. If this option is not specified, the output will be a dry run.\"\n        )\n        .option(\"-f, --force\", \"When used with -x, skips the prompt\")\n        .command(\"cleanup <cloud>\")\n        .description(\n            `Cleanup faast.js resources that may have leaked. The <cloud> argument must be \"aws\" or \"local\".\n        By default the output is a dry run and will only print the actions that would be performed if '-x' is specified.`\n        )\n        .action((arg: string) => {\n            command = \"cleanup\";\n            cloud = arg;\n        });\n\n    const opts = program.parse(process.argv).opts();\n    if (opts.verbose) {\n        process.env.DEBUG = \"faast:*\";\n    }\n    const execute = opts.execute || false;\n    let region = opts.region;\n\n    if (!region) {\n        switch (cloud) {\n            case \"aws\":\n                region = awsFaast.defaults.region;\n                break;\n        }\n    }\n    const force = opts.force || false;\n\n    region && log(`Region: ${region}`);\n    const options = { region, execute };\n    let nResources = 0;\n    if (command === \"cleanup\") {\n        if (execute && !force) {\n            nResources = await runCleanup(cloud, { ...options, execute: false });\n            if (nResources > 0) {\n                await prompt();\n            } else {\n                process.exit(0);\n            }\n        }\n        nResources = await runCleanup(cloud, options);\n        if (!execute && nResources > 0) {\n            log(\n                `(dryrun mode, no resources will be deleted, specify -x to execute cleanup)`\n            );\n        }\n    } else {\n        log(`No command specified.`);\n        program.help();\n    }\n}\n\nmain();\n"]}