1 | ;
|
2 | Object.defineProperty(exports, "__esModule", { value: true });
|
3 | exports.npmInstall = void 0;
|
4 | const archiver = require("archiver");
|
5 | const aws_sdk_1 = require("aws-sdk");
|
6 | const child_process_1 = require("child_process");
|
7 | const fs_extra_1 = require("fs-extra");
|
8 | const os_1 = require("os");
|
9 | const path = require("path");
|
10 | const util_1 = require("util");
|
11 | const shared_1 = require("../shared");
|
12 | async function exec(cmds) {
|
13 | let rv = "";
|
14 | for (const cmd of cmds) {
|
15 | rv += (0, child_process_1.execSync)(cmd).toString();
|
16 | }
|
17 | return rv;
|
18 | }
|
19 | async function npmInstall({ LayerName, packageJsonContents, FunctionName, region, quiet, retentionInDays }) {
|
20 | const log = quiet ? (_) => { } : console.log;
|
21 | log(`*** This faast invocation is an internal lambda call used when the packageJson option is specified to createFunction(). ***`);
|
22 | log(`*** Its purpose is to create a node_modules package and cache it, then combine with user code to form an AWS Lambda code package and upload it to S3 ***`);
|
23 | const buildParentDir = path.join((0, os_1.tmpdir)(), FunctionName);
|
24 | const buildDir = path.join(buildParentDir, "nodejs");
|
25 | await (0, fs_extra_1.ensureDir)(buildDir);
|
26 | await (0, fs_extra_1.writeFile)(path.join(buildDir, "package.json"), packageJsonContents);
|
27 | const awsconfig = { correctClockSkew: true, maxRetries: 6 };
|
28 | let installLog = "";
|
29 | log("Checking cache");
|
30 | log(`Checking faast layers for ${LayerName}`);
|
31 | const lambda = new aws_sdk_1.Lambda({ apiVersion: "2015-03-31", region, ...awsconfig });
|
32 | const cached = await lambda
|
33 | .listLayerVersions({ LayerName, CompatibleRuntime: "nodejs" })
|
34 | .promise()
|
35 | .catch(_ => undefined);
|
36 | const layerVersion = cached?.LayerVersions?.[0];
|
37 | if (layerVersion && !(0, shared_1.hasExpired)(layerVersion.CreatedDate, retentionInDays)) {
|
38 | const layerInfo = {
|
39 | LayerName,
|
40 | Version: layerVersion.Version,
|
41 | LayerVersionArn: layerVersion.LayerVersionArn
|
42 | };
|
43 | log(`CACHED, ${(0, util_1.inspect)(layerInfo)}`);
|
44 | return { installLog, layerInfo };
|
45 | }
|
46 | log("NOT CACHED, running npm install");
|
47 | installLog += await exec([`echo "hello world"`]);
|
48 | installLog += await exec([
|
49 | `export HOME=/tmp; npm install --prefix=${buildDir} --no-package-lock`
|
50 | ]);
|
51 | log(`Running archiver`);
|
52 | const cacheArchive = archiver("zip", { zlib: { level: 8 } });
|
53 | cacheArchive.directory(buildParentDir, false).finalize();
|
54 | log(`Converting archive to buffer`);
|
55 | const ZipFile = await (0, shared_1.streamToBuffer)(cacheArchive);
|
56 | log(`Code ZipFile size: ${ZipFile.length}`);
|
57 | log(`Removing ${buildParentDir}`);
|
58 | const removePromise = (0, fs_extra_1.remove)(buildParentDir);
|
59 | let Content;
|
60 | const Bucket = FunctionName;
|
61 | const s3 = new aws_sdk_1.S3({ region, ...awsconfig });
|
62 | const zipSize = ZipFile.length;
|
63 | try {
|
64 | if (ZipFile.length > 50 * 2 ** 20) {
|
65 | // Try to use S3 to allow for a larger limit
|
66 | log(`Creating s3 bucket ${Bucket}`);
|
67 | await s3
|
68 | .createBucket({ Bucket })
|
69 | .promise()
|
70 | .catch(_ => { });
|
71 | log(`Uploading bucket: ${Bucket}, object: ${LayerName}`);
|
72 | await s3.upload({ Bucket, Key: LayerName, Body: ZipFile }).promise();
|
73 | Content = { S3Bucket: Bucket, S3Key: LayerName };
|
74 | }
|
75 | else {
|
76 | Content = { ZipFile };
|
77 | }
|
78 | log(`Creating lambda layer: ${LayerName}, zip file size: ${ZipFile.length}`);
|
79 | const publishResponse = await lambda
|
80 | .publishLayerVersion({
|
81 | LayerName,
|
82 | Description: `faast packageJson layer with LayerName ${LayerName}`,
|
83 | Content,
|
84 | CompatibleRuntimes: ["nodejs"]
|
85 | })
|
86 | .promise();
|
87 | const { Version } = publishResponse;
|
88 | log(`Created lambda layer: ${LayerName}:${Version}`);
|
89 | log(`DONE`);
|
90 | return {
|
91 | installLog,
|
92 | layerInfo: {
|
93 | LayerName,
|
94 | LayerVersionArn: publishResponse.LayerVersionArn,
|
95 | Version: publishResponse.Version
|
96 | },
|
97 | zipSize
|
98 | };
|
99 | }
|
100 | finally {
|
101 | if (Content?.S3Bucket) {
|
102 | try {
|
103 | await s3.deleteObject({ Bucket, Key: LayerName }).promise();
|
104 | await s3.deleteBucket({ Bucket }).promise();
|
105 | }
|
106 | catch { }
|
107 | }
|
108 | await removePromise;
|
109 | }
|
110 | }
|
111 | exports.npmInstall = npmInstall;
|
112 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws-npm.js","sourceRoot":"","sources":["../../../src/aws/aws-npm.ts"],"names":[],"mappings":"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","sourcesContent":["import * as archiver from \"archiver\";\nimport { Lambda, S3 } from \"aws-sdk\";\nimport { execSync } from \"child_process\";\nimport { ensureDir, remove, writeFile } from \"fs-extra\";\nimport { tmpdir } from \"os\";\nimport * as path from \"path\";\nimport { inspect } from \"util\";\nimport { streamToBuffer, hasExpired } from \"../shared\";\n\nasync function exec(cmds: string[]) {\n    let rv = \"\";\n    for (const cmd of cmds) {\n        rv += execSync(cmd).toString();\n    }\n    return rv;\n}\n\nexport interface NpmInstallArgs {\n    packageJsonContents: string;\n    LayerName: string;\n    FunctionName: string;\n    region: string;\n    quiet?: boolean;\n    retentionInDays: number;\n}\n\nexport interface AwsLayerInfo {\n    Version: number;\n    LayerVersionArn: string;\n    LayerName: string;\n}\n\nexport interface NpmInstallReturn {\n    installLog: string;\n    layerInfo: AwsLayerInfo;\n    zipSize?: number;\n}\n\nexport async function npmInstall({\n    LayerName,\n    packageJsonContents,\n    FunctionName,\n    region,\n    quiet,\n    retentionInDays\n}: NpmInstallArgs): Promise<NpmInstallReturn> {\n    const log = quiet ? (_: string) => {} : console.log;\n\n    log(\n        `*** This faast invocation is an internal lambda call used when the packageJson option is specified to createFunction(). ***`\n    );\n    log(\n        `*** Its purpose is to create a node_modules package and cache it, then combine with user code to form an AWS Lambda code package and upload it to S3 ***`\n    );\n\n    const buildParentDir = path.join(tmpdir(), FunctionName);\n    const buildDir = path.join(buildParentDir, \"nodejs\");\n    await ensureDir(buildDir);\n    await writeFile(path.join(buildDir, \"package.json\"), packageJsonContents);\n\n    const awsconfig = { correctClockSkew: true, maxRetries: 6 };\n\n    let installLog = \"\";\n    log(\"Checking cache\");\n    log(`Checking faast layers for ${LayerName}`);\n    const lambda = new Lambda({ apiVersion: \"2015-03-31\", region, ...awsconfig });\n\n    const cached = await lambda\n        .listLayerVersions({ LayerName, CompatibleRuntime: \"nodejs\" })\n        .promise()\n        .catch(_ => undefined);\n\n    const layerVersion = cached?.LayerVersions?.[0];\n    if (layerVersion && !hasExpired(layerVersion.CreatedDate, retentionInDays)) {\n        const layerInfo = {\n            LayerName,\n            Version: layerVersion.Version!,\n            LayerVersionArn: layerVersion.LayerVersionArn!\n        };\n        log(`CACHED, ${inspect(layerInfo)}`);\n        return { installLog, layerInfo };\n    }\n\n    log(\"NOT CACHED, running npm install\");\n    installLog += await exec([`echo \"hello world\"`]);\n    installLog += await exec([\n        `export HOME=/tmp; npm install --prefix=${buildDir} --no-package-lock`\n    ]);\n    log(`Running archiver`);\n    const cacheArchive = archiver(\"zip\", { zlib: { level: 8 } });\n    cacheArchive.directory(buildParentDir, false).finalize();\n    log(`Converting archive to buffer`);\n    const ZipFile = await streamToBuffer(cacheArchive);\n    log(`Code ZipFile size: ${ZipFile.length}`);\n    log(`Removing ${buildParentDir}`);\n    const removePromise = remove(buildParentDir);\n    let Content: Lambda.LayerVersionContentInput | undefined;\n    const Bucket = FunctionName;\n    const s3 = new S3({ region, ...awsconfig });\n    const zipSize = ZipFile.length;\n    try {\n        if (ZipFile.length > 50 * 2 ** 20) {\n            // Try to use S3 to allow for a larger limit\n            log(`Creating s3 bucket ${Bucket}`);\n            await s3\n                .createBucket({ Bucket })\n                .promise()\n                .catch(_ => {});\n            log(`Uploading bucket: ${Bucket}, object: ${LayerName}`);\n            await s3.upload({ Bucket, Key: LayerName, Body: ZipFile }).promise();\n            Content = { S3Bucket: Bucket, S3Key: LayerName };\n        } else {\n            Content = { ZipFile };\n        }\n        log(`Creating lambda layer: ${LayerName}, zip file size: ${ZipFile.length}`);\n        const publishResponse = await lambda\n            .publishLayerVersion({\n                LayerName,\n                Description: `faast packageJson layer with LayerName ${LayerName}`,\n                Content,\n                CompatibleRuntimes: [\"nodejs\"]\n            })\n            .promise();\n        const { Version } = publishResponse;\n        log(`Created lambda layer: ${LayerName}:${Version}`);\n        log(`DONE`);\n        return {\n            installLog,\n            layerInfo: {\n                LayerName,\n                LayerVersionArn: publishResponse.LayerVersionArn!,\n                Version: publishResponse.Version!\n            },\n            zipSize\n        };\n    } finally {\n        if (Content?.S3Bucket) {\n            try {\n                await s3.deleteObject({ Bucket, Key: LayerName }).promise();\n                await s3.deleteBucket({ Bucket }).promise();\n            } catch {}\n        }\n        await removePromise;\n    }\n}\n"]} |
\ | No newline at end of file |