1 |
|
2 | ;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | require("source-map-support").install();
|
5 | const commander_1 = require("commander");
|
6 | const fs_extra_1 = require("fs-extra");
|
7 | const googleapis_1 = require("googleapis");
|
8 | const ora = require("ora");
|
9 | const os_1 = require("os");
|
10 | const path = require("path");
|
11 | const awsFaast = require("./aws/aws-faast");
|
12 | const cache_1 = require("./cache");
|
13 | const googleFaast = require("./google/google-faast");
|
14 | const shared_1 = require("./shared");
|
15 | const throttle_1 = require("./throttle");
|
16 | const warn = console.warn;
|
17 | const log = console.log;
|
18 | async function deleteResources(name, matchingResources, doRemove, { concurrency = 10, rate = 5, burst = 5 } = {}) {
|
19 | if (matchingResources.length > 0) {
|
20 | const timeEstimate = (nResources) => nResources <= 5 ? "" : `(est: ${(nResources / 5).toFixed(0)}s)`;
|
21 | const updateSpinnerText = (nResources = 0) => `Deleting ${matchingResources.length} ${name} ${timeEstimate(nResources)}`;
|
22 | const spinner = ora(updateSpinnerText(matchingResources.length)).start();
|
23 | let done = 0;
|
24 | const scheduleRemove = (0, throttle_1.throttle)({
|
25 | concurrency,
|
26 | rate,
|
27 | burst,
|
28 | retry: 5
|
29 | }, async (arg) => {
|
30 | await doRemove(arg);
|
31 | done++;
|
32 | });
|
33 | const timer = setInterval(() => (spinner.text = updateSpinnerText(matchingResources.length - done)), 1000);
|
34 | try {
|
35 | await Promise.all(matchingResources.map(resource => scheduleRemove(resource).catch(err => console.warn(`Could not remove resource ${resource}: ${err}`))));
|
36 | }
|
37 | finally {
|
38 | clearInterval(timer);
|
39 | spinner.text = updateSpinnerText();
|
40 | }
|
41 | spinner.stopAndPersist({ symbol: "✔" });
|
42 | }
|
43 | }
|
44 | async function cleanupAWS({ region, execute }) {
|
45 | let nResources = 0;
|
46 | const output = (msg) => !execute && log(msg);
|
47 | const { cloudwatch, iam, lambda, sns, sqs, s3 } = await awsFaast.createAwsApis(region);
|
48 | function listAWSResource(pattern, getList, extractList, extractElement) {
|
49 | const allResources = [];
|
50 | return new Promise((resolve, reject) => {
|
51 | getList().eachPage((err, page) => {
|
52 | if (err) {
|
53 | reject(err);
|
54 | return false;
|
55 | }
|
56 | const elems = (page && extractList(page)) || [];
|
57 | allResources.push(...elems.map(elem => extractElement(elem) || ""));
|
58 | if (page === null) {
|
59 | // console.log(`allResources: ${allResources.join("\n")}`);
|
60 | // console.log(`pattern: ${pattern}`);
|
61 | const matchingResources = allResources.filter(t => t.match(pattern));
|
62 | matchingResources.forEach(resource => output(` ${resource}`));
|
63 | resolve(matchingResources);
|
64 | }
|
65 | return true;
|
66 | });
|
67 | });
|
68 | }
|
69 | async function deleteAWSResource(name, pattern, getList, extractList, extractElement, doRemove) {
|
70 | const allResources = await listAWSResource(pattern, getList, extractList, extractElement);
|
71 | nResources += allResources.length;
|
72 | if (execute) {
|
73 | await deleteResources(name, allResources, doRemove, {
|
74 | concurrency: 10,
|
75 | rate: 5,
|
76 | burst: 5
|
77 | });
|
78 | }
|
79 | }
|
80 | output(`SNS subscriptions`);
|
81 | await deleteAWSResource("SNS subscription(s)", new RegExp(`:faast-${shared_1.uuidv4Pattern}`), () => sns.listSubscriptions(), page => page.Subscriptions, subscription => subscription.SubscriptionArn, SubscriptionArn => sns.unsubscribe({ SubscriptionArn }).promise());
|
82 | output(`SNS topics`);
|
83 | await deleteAWSResource("SNS topic(s)", new RegExp(`:faast-${shared_1.uuidv4Pattern}`), () => sns.listTopics(), page => page.Topics, topic => topic.TopicArn, TopicArn => sns.deleteTopic({ TopicArn }).promise());
|
84 | output(`SQS queues`);
|
85 | await deleteAWSResource("SQS queue(s)", new RegExp(`/faast-${shared_1.uuidv4Pattern}`), () => sqs.listQueues(), page => page.QueueUrls, queueUrl => queueUrl, QueueUrl => sqs.deleteQueue({ QueueUrl }).promise());
|
86 | output(`S3 buckets`);
|
87 | await deleteAWSResource("S3 bucket(s)", new RegExp(`^faast-${shared_1.uuidv4Pattern}`), () => s3.listBuckets(), page => page.Buckets, Bucket => Bucket.Name, async (Bucket) => {
|
88 | const objects = await s3
|
89 | .listObjectsV2({ Bucket, Prefix: "faast-" })
|
90 | .promise();
|
91 | const keys = (objects.Contents || []).map(entry => ({ Key: entry.Key }));
|
92 | if (keys.length > 0) {
|
93 | await s3.deleteObjects({ Bucket, Delete: { Objects: keys } }).promise();
|
94 | }
|
95 | await s3.deleteBucket({ Bucket }).promise();
|
96 | });
|
97 | output(`Lambda functions`);
|
98 | await deleteAWSResource("Lambda function(s)", new RegExp(`^faast-${shared_1.uuidv4Pattern}`), () => lambda.listFunctions(), page => page.Functions, func => func.FunctionName, FunctionName => lambda.deleteFunction({ FunctionName }).promise());
|
99 | output(`IAM roles`);
|
100 | await deleteAWSResource("IAM role(s)", /^faast-cached-lambda-role$/, () => iam.listRoles(), page => page.Roles, role => role.RoleName, RoleName => awsFaast.deleteRole(RoleName, iam));
|
101 | output(`IAM test roles`);
|
102 | await deleteAWSResource("IAM test role(s)", new RegExp(`^faast-test-.*${shared_1.uuidv4Pattern}$`), () => iam.listRoles(), page => page.Roles, role => role.RoleName, RoleName => awsFaast.deleteRole(RoleName, iam));
|
103 | output(`Lambda layers`);
|
104 | await deleteAWSResource("Lambda layer(s)", new RegExp(`^faast-(${shared_1.uuidv4Pattern})|([a-f0-9]{64})`), () => lambda.listLayers({ CompatibleRuntime: "nodejs" }), page => page.Layers, layer => layer.LayerName, async (LayerName) => {
|
105 | const versions = await lambda.listLayerVersions({ LayerName }).promise();
|
106 | for (const layerVersion of versions.LayerVersions || []) {
|
107 | await lambda
|
108 | .deleteLayerVersion({
|
109 | LayerName,
|
110 | VersionNumber: layerVersion.Version
|
111 | })
|
112 | .promise();
|
113 | }
|
114 | });
|
115 | async function cleanupCacheDir(cache) {
|
116 | output(`Persistent cache: ${cache.dir}`);
|
117 | const entries = await cache.entries();
|
118 | if (!execute) {
|
119 | output(` cache entries: ${entries.length}`);
|
120 | }
|
121 | nResources += entries.length;
|
122 | if (execute) {
|
123 | cache.clear({ leaveEmptyDir: false });
|
124 | }
|
125 | }
|
126 | for (const cache of (0, shared_1.keysOf)(cache_1.caches)) {
|
127 | await cleanupCacheDir(await cache_1.caches[cache]);
|
128 | }
|
129 | output(`Cloudwatch log groups`);
|
130 | await deleteAWSResource("Cloudwatch log group(s)", new RegExp(`/faast-${shared_1.uuidv4Pattern}$`), () => cloudwatch.describeLogGroups(), page => page.logGroups, logGroup => logGroup.logGroupName, logGroupName => cloudwatch.deleteLogGroup({ logGroupName }).promise());
|
131 | return nResources;
|
132 | }
|
133 | async function iterate(getPage, each) {
|
134 | let token;
|
135 | do {
|
136 | const result = await getPage(token);
|
137 | each(result.data);
|
138 | token = result.data.nextPageToken;
|
139 | } while (token);
|
140 | }
|
141 | async function cleanupGoogle({ execute }) {
|
142 | let nResources = 0;
|
143 | const output = (msg) => !execute && log(msg);
|
144 | async function listGoogleResource(pattern, getList, extractList, extractElement) {
|
145 | const allResources = [];
|
146 | await iterate(pageToken => getList(pageToken), result => {
|
147 | const resources = extractList(result) || [];
|
148 | allResources.push(...resources.map(elem => extractElement(elem) || ""));
|
149 | });
|
150 | const matchingResources = allResources.filter(t => t.match(pattern));
|
151 | matchingResources.forEach(resource => output(` ${resource}`));
|
152 | return matchingResources;
|
153 | }
|
154 | async function deleteGoogleResource(name, pattern, getList, extractList, extractElement, doRemove) {
|
155 | const allResources = await listGoogleResource(pattern, getList, extractList, extractElement);
|
156 | nResources += allResources.length;
|
157 | if (execute) {
|
158 | await deleteResources(name, allResources, doRemove, {
|
159 | concurrency: 20,
|
160 | rate: 20,
|
161 | burst: 20
|
162 | });
|
163 | }
|
164 | }
|
165 | const { cloudFunctions, pubsub } = await googleFaast.initializeGoogleServices();
|
166 | const project = await googleapis_1.google.auth.getProjectId();
|
167 | log(`Default project: ${project}`);
|
168 | output(`Cloud functions`);
|
169 | await deleteGoogleResource("Cloud Function(s)", new RegExp(`faast-${shared_1.uuidv4Pattern}`), (pageToken) => cloudFunctions.projects.locations.functions.list({
|
170 | pageToken,
|
171 | parent: `projects/${project}/locations/-`
|
172 | }), page => page.functions, func => func.name ?? undefined, name => cloudFunctions.projects.locations.functions.delete({ name }));
|
173 | output(`Pub/Sub subscriptions`);
|
174 | await deleteGoogleResource("Pub/Sub Subscription(s)", new RegExp(`faast-${shared_1.uuidv4Pattern}`), pageToken => pubsub.projects.subscriptions.list({
|
175 | pageToken,
|
176 | project: `projects/${project}`
|
177 | }), page => page.subscriptions, subscription => subscription.name ?? undefined, subscriptionName => pubsub.projects.subscriptions.delete({ subscription: subscriptionName }));
|
178 | output(`Pub/Sub topics`);
|
179 | await deleteGoogleResource("Pub/Sub topic(s)", new RegExp(`topics/faast-${shared_1.uuidv4Pattern}`), pageToken => pubsub.projects.topics.list({ pageToken, project: `projects/${project}` }), page => page.topics, topic => topic.name ?? undefined, topicName => pubsub.projects.topics.delete({ topic: topicName }));
|
180 | return nResources;
|
181 | }
|
182 | async function cleanupLocal({ execute }) {
|
183 | const output = (msg) => !execute && log(msg);
|
184 | const tmpDir = (0, os_1.tmpdir)();
|
185 | const dir = await (0, fs_extra_1.readdir)(tmpDir);
|
186 | let nResources = 0;
|
187 | output(`Temporary directories:`);
|
188 | const entryRegexp = new RegExp(`^faast-${shared_1.uuidv4Pattern}$`);
|
189 | for (const entry of dir) {
|
190 | if (entry.match(entryRegexp)) {
|
191 | nResources++;
|
192 | const faastDir = path.join(tmpDir, entry);
|
193 | output(`${faastDir}`);
|
194 | if (execute) {
|
195 | await (0, fs_extra_1.remove)(faastDir);
|
196 | }
|
197 | }
|
198 | }
|
199 | return nResources;
|
200 | }
|
201 | const readline = require("readline");
|
202 | async function prompt() {
|
203 | const rl = readline.createInterface({
|
204 | input: process.stdin,
|
205 | output: process.stdout
|
206 | });
|
207 | await new Promise(resolve => {
|
208 | rl.question("WARNING: this operation will delete resources. Confirm? [y/N] ", answer => {
|
209 | if (answer !== "y") {
|
210 | log(`Execution aborted.`);
|
211 | process.exit(0);
|
212 | }
|
213 | rl.close();
|
214 | resolve();
|
215 | });
|
216 | });
|
217 | }
|
218 | async function runCleanup(cloud, options) {
|
219 | let nResources = 0;
|
220 | if (cloud === "aws") {
|
221 | nResources = await cleanupAWS(options);
|
222 | }
|
223 | else if (cloud === "google") {
|
224 | nResources = await cleanupGoogle(options);
|
225 | }
|
226 | else if (cloud === "local") {
|
227 | nResources = await cleanupLocal(options);
|
228 | }
|
229 | else {
|
230 | warn(`Unknown cloud name "${cloud}". Must specify "aws" or "google", or "local".`);
|
231 | process.exit(-1);
|
232 | }
|
233 | if (options.execute) {
|
234 | log(`Done.`);
|
235 | }
|
236 | else {
|
237 | if (nResources === 0) {
|
238 | log(`No resources to clean up.`);
|
239 | }
|
240 | }
|
241 | return nResources;
|
242 | }
|
243 | async function main() {
|
244 | let cloud;
|
245 | let command;
|
246 | commander_1.program
|
247 | .version("0.1.0")
|
248 | .option("-v, --verbose", "Verbose mode")
|
249 | .option("-r, --region <region>", "Cloud region to operate on. Defaults to us-west-2 for AWS, and us-central1 for Google.")
|
250 | .option("-x, --execute", "Execute the cleanup process. If this option is not specified, the output will be a dry run.")
|
251 | .option("-f, --force", "When used with -x, skips the prompt")
|
252 | .command("cleanup <cloud>")
|
253 | .description(`Cleanup faast.js resources that may have leaked. The <cloud> argument must be "aws", "google", or "local".
|
254 | By default the output is a dry run and will only print the actions that would be performed if '-x' is specified.`)
|
255 | .action((arg) => {
|
256 | command = "cleanup";
|
257 | cloud = arg;
|
258 | });
|
259 | const opts = commander_1.program.parse(process.argv).opts();
|
260 | if (opts.verbose) {
|
261 | process.env.DEBUG = "faast:*";
|
262 | }
|
263 | const execute = opts.execute || false;
|
264 | let region = opts.region;
|
265 | if (!region) {
|
266 | switch (cloud) {
|
267 | case "aws":
|
268 | region = awsFaast.defaults.region;
|
269 | break;
|
270 | case "google":
|
271 | region = googleFaast.defaults.region;
|
272 | break;
|
273 | }
|
274 | }
|
275 | const force = opts.force || false;
|
276 | region && log(`Region: ${region}`);
|
277 | const options = { region, execute };
|
278 | let nResources = 0;
|
279 | if (command === "cleanup") {
|
280 | if (execute && !force) {
|
281 | nResources = await runCleanup(cloud, { ...options, execute: false });
|
282 | if (nResources > 0) {
|
283 | await prompt();
|
284 | }
|
285 | else {
|
286 | process.exit(0);
|
287 | }
|
288 | }
|
289 | nResources = await runCleanup(cloud, options);
|
290 | if (!execute && nResources > 0) {
|
291 | log(`(dryrun mode, no resources will be deleted, specify -x to execute cleanup)`);
|
292 | }
|
293 | }
|
294 | else {
|
295 | log(`No command specified.`);
|
296 | commander_1.program.help();
|
297 | }
|
298 | }
|
299 | main();
|
300 | //# 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();\n\nimport { Request as AWSRequest, AWSError } from \"aws-sdk\";\nimport { program } from \"commander\";\nimport { readdir, remove } from \"fs-extra\";\nimport { GaxiosPromise, GaxiosResponse } from \"gaxios\";\nimport { google } from \"googleapis\";\nimport * as ora from \"ora\";\nimport { tmpdir } from \"os\";\nimport * as path from \"path\";\nimport * as awsFaast from \"./aws/aws-faast\";\nimport { caches, PersistentCache } from \"./cache\";\nimport * as googleFaast from \"./google/google-faast\";\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 and Google 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    function listAWSResource<T, U>(\n        pattern: RegExp,\n        getList: () => AWSRequest<T, AWSError>,\n        extractList: (arg: T) => U[] | undefined,\n        extractElement: (arg: U) => string | undefined\n    ) {\n        const allResources: string[] = [];\n        return new Promise<string[]>((resolve, reject) => {\n            getList().eachPage((err, page) => {\n                if (err) {\n                    reject(err);\n                    return false;\n                }\n                const elems = (page && extractList(page)) || [];\n                allResources.push(...elems.map(elem => extractElement(elem) || \"\"));\n                if (page === null) {\n                    // console.log(`allResources: ${allResources.join(\"\\n\")}`);\n                    // console.log(`pattern: ${pattern}`);\n                    const matchingResources = allResources.filter(t => t.match(pattern));\n                    matchingResources.forEach(resource => output(`  ${resource}`));\n                    resolve(matchingResources);\n                }\n                return true;\n            });\n        });\n    }\n\n    async function deleteAWSResource<T, U>(\n        name: string,\n        pattern: RegExp,\n        getList: () => AWSRequest<T, AWSError>,\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        () => sns.listSubscriptions(),\n        page => page.Subscriptions,\n        subscription => subscription.SubscriptionArn,\n        SubscriptionArn => sns.unsubscribe({ SubscriptionArn }).promise()\n    );\n\n    output(`SNS topics`);\n    await deleteAWSResource(\n        \"SNS topic(s)\",\n        new RegExp(`:faast-${uuidv4Pattern}`),\n        () => sns.listTopics(),\n        page => page.Topics,\n        topic => topic.TopicArn,\n        TopicArn => sns.deleteTopic({ TopicArn }).promise()\n    );\n\n    output(`SQS queues`);\n    await deleteAWSResource(\n        \"SQS queue(s)\",\n        new RegExp(`/faast-${uuidv4Pattern}`),\n        () => sqs.listQueues(),\n        page => page.QueueUrls,\n        queueUrl => queueUrl,\n        QueueUrl => sqs.deleteQueue({ QueueUrl }).promise()\n    );\n\n    output(`S3 buckets`);\n    await deleteAWSResource(\n        \"S3 bucket(s)\",\n        new RegExp(`^faast-${uuidv4Pattern}`),\n        () => s3.listBuckets(),\n        page => page.Buckets,\n        Bucket => Bucket.Name,\n        async Bucket => {\n            const objects = await s3\n                .listObjectsV2({ Bucket, Prefix: \"faast-\" })\n                .promise();\n            const keys = (objects.Contents || []).map(entry => ({ Key: entry.Key! }));\n            if (keys.length > 0) {\n                await s3.deleteObjects({ Bucket, Delete: { Objects: keys } }).promise();\n            }\n            await s3.deleteBucket({ Bucket }).promise();\n        }\n    );\n\n    output(`Lambda functions`);\n    await deleteAWSResource(\n        \"Lambda function(s)\",\n        new RegExp(`^faast-${uuidv4Pattern}`),\n        () => lambda.listFunctions(),\n        page => page.Functions,\n        func => func.FunctionName,\n        FunctionName => lambda.deleteFunction({ FunctionName }).promise()\n    );\n\n    output(`IAM roles`);\n    await deleteAWSResource(\n        \"IAM role(s)\",\n        /^faast-cached-lambda-role$/,\n        () => iam.listRoles(),\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        () => iam.listRoles(),\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        () => lambda.listLayers({ CompatibleRuntime: \"nodejs\" }),\n        page => page.Layers,\n        layer => layer.LayerName,\n        async LayerName => {\n            const versions = await lambda.listLayerVersions({ LayerName }).promise();\n            for (const layerVersion of versions.LayerVersions || []) {\n                await lambda\n                    .deleteLayerVersion({\n                        LayerName,\n                        VersionNumber: layerVersion.Version!\n                    })\n                    .promise();\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        () => cloudwatch.describeLogGroups(),\n        page => page.logGroups,\n        logGroup => logGroup.logGroupName,\n        logGroupName => cloudwatch.deleteLogGroup({ logGroupName }).promise()\n    );\n\n    return nResources;\n}\n\ninterface HasNextPageToken {\n    nextPageToken?: string;\n}\n\nasync function iterate<T extends HasNextPageToken>(\n    getPage: (token?: string) => GaxiosPromise<T>,\n    each: (val: T) => void\n) {\n    let token;\n    do {\n        const result: GaxiosResponse<T> = await getPage(token);\n        each(result.data);\n        token = result.data.nextPageToken;\n    } while (token);\n}\n\nasync function cleanupGoogle({ execute }: CleanupOptions) {\n    let nResources = 0;\n    const output = (msg: string) => !execute && log(msg);\n\n    async function listGoogleResource<T, U>(\n        pattern: RegExp,\n        getList: (pageToken?: string) => GaxiosPromise<T>,\n        extractList: (arg: T) => U[] | undefined,\n        extractElement: (arg: U) => string | undefined\n    ) {\n        const allResources: string[] = [];\n        await iterate(\n            pageToken => getList(pageToken),\n            result => {\n                const resources = extractList(result) || [];\n                allResources.push(...resources.map(elem => extractElement(elem) || \"\"));\n            }\n        );\n\n        const matchingResources = allResources.filter(t => t.match(pattern));\n        matchingResources.forEach(resource => output(`  ${resource}`));\n        return matchingResources;\n    }\n\n    async function deleteGoogleResource<T, U>(\n        name: string,\n        pattern: RegExp,\n        getList: (pageToken?: string) => GaxiosPromise<T>,\n        extractList: (arg: T) => U[] | undefined,\n        extractElement: (arg: U) => string | undefined,\n        doRemove: (arg: string) => Promise<any>\n    ) {\n        const allResources = await listGoogleResource(\n            pattern,\n            getList,\n            extractList,\n            extractElement\n        );\n        nResources += allResources.length;\n        if (execute) {\n            await deleteResources(name, allResources, doRemove, {\n                concurrency: 20,\n                rate: 20,\n                burst: 20\n            });\n        }\n    }\n    const { cloudFunctions, pubsub } = await googleFaast.initializeGoogleServices();\n    const project = await google.auth.getProjectId();\n    log(`Default project: ${project}`);\n\n    output(`Cloud functions`);\n    await deleteGoogleResource(\n        \"Cloud Function(s)\",\n        new RegExp(`faast-${uuidv4Pattern}`),\n        (pageToken?: string) =>\n            cloudFunctions.projects.locations.functions.list({\n                pageToken,\n                parent: `projects/${project}/locations/-`\n            }),\n        page => page.functions,\n        func => func.name ?? undefined,\n        name => cloudFunctions.projects.locations.functions.delete({ name })\n    );\n\n    output(`Pub/Sub subscriptions`);\n    await deleteGoogleResource(\n        \"Pub/Sub Subscription(s)\",\n        new RegExp(`faast-${uuidv4Pattern}`),\n        pageToken =>\n            pubsub.projects.subscriptions.list({\n                pageToken,\n                project: `projects/${project}`\n            }),\n        page => page.subscriptions,\n        subscription => subscription.name ?? undefined,\n        subscriptionName =>\n            pubsub.projects.subscriptions.delete({ subscription: subscriptionName })\n    );\n\n    output(`Pub/Sub topics`);\n    await deleteGoogleResource(\n        \"Pub/Sub topic(s)\",\n        new RegExp(`topics/faast-${uuidv4Pattern}`),\n        pageToken =>\n            pubsub.projects.topics.list({ pageToken, project: `projects/${project}` }),\n        page => page.topics,\n        topic => topic.name ?? undefined,\n        topicName => pubsub.projects.topics.delete({ topic: topicName })\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\nimport * as readline from \"readline\";\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 === \"google\") {\n        nResources = await cleanupGoogle(options);\n    } else if (cloud === \"local\") {\n        nResources = await cleanupLocal(options);\n    } else {\n        warn(\n            `Unknown cloud name \"${cloud}\". Must specify \"aws\" or \"google\", or \"local\".`\n        );\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, and us-central1 for Google.\"\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\", \"google\", 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            case \"google\":\n                region = googleFaast.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"]} |
\ | No newline at end of file |