@lambda-kata/cdk 0.1.3-rc.90 → 0.1.3-rc.93
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
"use strict";var h=Object.defineProperty;var $=Object.getOwnPropertyDescriptor;var v=Object.getOwnPropertyNames;var F=Object.prototype.hasOwnProperty;var y=(t,e)=>{for(var p in e)h(t,p,{get:e[p],enumerable:!0})},C=(t,e,p,a)=>{if(e&&typeof e=="object"||typeof e=="function")for(let c of v(e))!F.call(t,c)&&c!==p&&h(t,c,{get:()=>e[c],enumerable:!(a=$(e,c))||a.enumerable});return t};var I=t=>C(h({},"__esModule",{value:!0}),t);var U={};y(U,{_testable:()=>m,activateSnapStart:()=>E,handler:()=>T});module.exports=I(U);var n=require("@aws-sdk/client-lambda"),d=5,N=15e3,P=3e3,D={snapshotTimeoutSeconds:180,pollingIntervalSeconds:2,aliasName:"kata",_prePublishDelayMs:P,_retryDelayMs:N},m={sleep(t){return new Promise(e=>setTimeout(e,t))}};function A(t){if(t instanceof n.ResourceNotFoundException)return!0;if(t instanceof Error){if(t.name==="ResourceNotFoundException")return!0;let e=t.cause;if(e instanceof n.ResourceNotFoundException||e instanceof Error&&e.name==="ResourceNotFoundException")return!0}return!1}var R=["lambda:GetFunction","lambda:GetFunctionConfiguration","lambda:UpdateFunctionConfiguration","lambda:PublishVersion","lambda:GetAlias","lambda:CreateAlias","lambda:UpdateAlias"];function w(t){if(t instanceof Error){if(t.name==="AccessDeniedException")return!0;let e=t.cause;if(e instanceof Error&&e.name==="AccessDeniedException")return!0}return!1}async function E(t,e,p){let a={...D,...p},c=Math.ceil(a.snapshotTimeoutSeconds/a.pollingIntervalSeconds);console.log("=".repeat(60)),console.log("SNAPSTART ACTIVATION CYCLE"),console.log("=".repeat(60)),console.log(`Function: ${e}`),console.log(`Timeout: ${a.snapshotTimeoutSeconds}s, Polling: ${a.pollingIntervalSeconds}s`),console.log(`Max publish retries: ${d}`);try{console.log(`
|
|
2
|
+
[0/5] Ensuring function is Active...`);try{await(0,n.waitUntilFunctionActiveV2)({client:t,maxWaitTime:60},{FunctionName:e})}catch(o){throw A(o)?new Error(`Lambda function '${e}' does not exist. Verify the function name or ARN is correct.`):w(o)?new Error(`Insufficient permissions to activate SnapStart on '${e}'. Ensure the execution role has the following permissions: ${R.join(", ")}`):o}console.log(" Function is Active"),console.log(`
|
|
3
|
+
[1/5] Enabling SnapStart (ApplyOn: PublishedVersions)...`),await t.send(new n.UpdateFunctionConfigurationCommand({FunctionName:e,SnapStart:{ApplyOn:"PublishedVersions"}})),console.log(" SnapStart configuration sent"),console.log("[2/5] Waiting for configuration update...");try{await(0,n.waitUntilFunctionUpdatedV2)({client:t,maxWaitTime:120},{FunctionName:e})}catch(o){throw A(o)?new Error(`Lambda function '${e}' does not exist. Verify the function name or ARN is correct.`):w(o)?new Error(`Insufficient permissions to activate SnapStart on '${e}'. Ensure the execution role has the following permissions: ${R.join(", ")}`):o}console.log(" Configuration updated successfully"),console.log(` Waiting ${a._prePublishDelayMs/1e3}s for configuration propagation...`),await m.sleep(a._prePublishDelayMs);let i="",u="Unknown",s="Unknown",S="";for(let o=1;o<=d;o++){console.log(`[3/5] Publishing new version (attempt ${o}/${d})...`),o>1&&(console.log(` Waiting ${a._retryDelayMs/1e3}s before retry...`),await m.sleep(a._retryDelayMs),console.log(" Ensuring function is Active before re-publish..."),await(0,n.waitUntilFunctionActiveV2)({client:t,maxWaitTime:60},{FunctionName:e}),console.log(" Re-applying SnapStart config to force new version..."),await t.send(new n.UpdateFunctionConfigurationCommand({FunctionName:e,SnapStart:{ApplyOn:"PublishedVersions"}})),await(0,n.waitUntilFunctionUpdatedV2)({client:t,maxWaitTime:120},{FunctionName:e}),console.log(` Waiting ${a._prePublishDelayMs/1e3}s for configuration propagation...`),await m.sleep(a._prePublishDelayMs)),i=(await t.send(new n.PublishVersionCommand({FunctionName:e,Description:`SnapStart enabled - ${new Date().toISOString()} (attempt ${o})`}))).Version,console.log(` Published version: ${i}`),console.log("[4/5] Waiting for SnapStart snapshot creation..."),s="Unknown";for(let g=0;g<c;g++){let f=await t.send(new n.GetFunctionConfigurationCommand({FunctionName:e,Qualifier:i}));if(u=(f.SnapStart??{}).OptimizationStatus??"Unknown",s=f.State??"Unknown",s==="Active"){console.log(" SnapStart snapshot ready!"),console.log(` OptimizationStatus: ${u}`),console.log(` State: ${s}`);break}else if(s==="Failed"){S=f.StateReason??"Unknown",console.log(` Snapshot creation failed: ${S}`);break}else if(g%10===0||g<5){let b=g*a.pollingIntervalSeconds;console.log(` Creating snapshot... Status: ${u}, State: ${s} (${b}s elapsed)`)}await m.sleep(a.pollingIntervalSeconds*1e3)}if(s==="Active"||s!=="Failed")break;o<d&&(console.log(` Snapshot failed on attempt ${o}, will retry with new version...`),console.log(` Reason: ${S}`))}s==="Failed"&&(console.log(` All ${d} snapshot attempts failed.`),console.log(` Reason: ${S}`),console.log(""),console.log(" [Lambda Kata] SnapStart optimization failed. Alias will be created pointing"),console.log(" to the latest version. Function remains operational without SnapStart."),console.log(" Review CloudWatch logs for initialization errors.")),s!=="Active"&&s!=="Failed"&&(console.log(` Warning: Snapshot creation timeout after ${a.snapshotTimeoutSeconds}s`),console.log(` Final status: OptimizationStatus=${u}, State=${s}`)),console.log(`[5/5] Creating/updating alias '${a.aliasName}' -> version ${i}...`);let r;try{await t.send(new n.GetAliasCommand({FunctionName:e,Name:a.aliasName})),r=(await t.send(new n.UpdateAliasCommand({FunctionName:e,Name:a.aliasName,FunctionVersion:i,Description:"Lambda Kata SnapStart-enabled version"}))).AliasArn,console.log(` Updated existing alias: ${r}`)}catch(o){if(o instanceof n.ResourceNotFoundException||o instanceof Error&&o.name==="ResourceNotFoundException")r=(await t.send(new n.CreateAliasCommand({FunctionName:e,Name:a.aliasName,FunctionVersion:i,Description:"Lambda Kata SnapStart-enabled version"}))).AliasArn,console.log(` Created new alias: ${r}`);else throw o}return console.log(`
|
|
4
|
+
`+"=".repeat(60)),console.log("SNAPSTART ACTIVATION COMPLETE"),console.log("=".repeat(60)),console.log(`Version: ${i}`),console.log(`Alias: ${a.aliasName} -> ${r}`),console.log(`OptimizationStatus: ${u}`),{version:i,aliasName:a.aliasName,aliasArn:r,optimizationStatus:u}}catch(i){throw w(i)?new Error(`Insufficient permissions to activate SnapStart on '${e}'. Ensure the execution role has the following permissions: ${R.join(", ")}`):i}}async function T(t){console.log("Custom Resource Event:",JSON.stringify(t,null,2));let{RequestType:e,StackId:p,RequestId:a,LogicalResourceId:c,ResourceProperties:i}=t,u=i.FunctionName,s=i.AliasName??"kata",r={PhysicalResourceId:t.PhysicalResourceId??`${u}:snapstart:${s}`,StackId:p,RequestId:a,LogicalResourceId:c};try{if(e==="Delete")return console.log("Delete request - no action needed (Lambda will be deleted by CloudFormation)"),{...r,Status:"SUCCESS"};let o=new n.LambdaClient({}),l=await E(o,u,{aliasName:s});return{...r,Status:"SUCCESS",Data:{Version:l.version,AliasName:l.aliasName,AliasArn:l.aliasArn,OptimizationStatus:l.optimizationStatus}}}catch(o){let l=o instanceof Error?o.message:String(o);return console.error("SnapStart activation failed:",l),e==="Update"?(console.log("Returning SUCCESS for Update request to prevent rollback deadlock."),console.log("The function will continue with its previous SnapStart configuration."),{...r,Status:"SUCCESS",Reason:`SnapStart activation failed (non-blocking): ${l}`}):{...r,Status:"FAILED",Reason:`SnapStart activation failed: ${l}`}}}0&&(module.exports={_testable,activateSnapStart,handler});
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/snapstart-activator.ts"],
  "sourcesContent": ["/*\n * Apache-2.0\n * Copyright (C) 2025\u2013present Raman Marozau, Work Target Insight Function. All rights reserved.\n * Contact: raman@worktif.com\n *\n * This file is part of the Licensed Work: lambda_kata_npm_cdk, <worktif_lambda_kata_npm_cdk>.\n * Use of this software is governed by the Apache-2.0; see the LICENSE file\n * or https://www.apache.org/licenses/LICENSE-2.0 for details.\n *\n * SPDX-License-Identifier: Apache-2.0\n */\n\n/**\n * SnapStart Activator - Custom Resource Handler\n *\n * This module provides the Lambda handler for a CloudFormation Custom Resource\n * that enables SnapStart on Lambda functions after deployment. SnapStart requires\n * asynchronous waiting for snapshot creation, which cannot be done during CDK synthesis.\n *\n * The activation process:\n * 1. Wait for function to be Active\n * 2. Enable SnapStart configuration\n * 3. Wait for configuration update\n * 4. Publish new version (triggers snapshot creation)\n * 5. Wait for snapshot to be ready (up to 3 minutes)\n * 6. Create/update 'kata' alias pointing to the new version\n *\n * @module snapstart-activator\n */\n\nimport {\n    LambdaClient,\n    UpdateFunctionConfigurationCommand,\n    PublishVersionCommand,\n    GetFunctionConfigurationCommand,\n    CreateAliasCommand,\n    UpdateAliasCommand,\n    GetAliasCommand,\n    ResourceNotFoundException,\n    waitUntilFunctionUpdatedV2,\n    waitUntilFunctionActiveV2,\n} from '@aws-sdk/client-lambda';\n\n/**\n * Custom Resource event from CloudFormation.\n */\nexport interface CustomResourceEvent {\n    RequestType: 'Create' | 'Update' | 'Delete';\n    ServiceToken: string;\n    ResponseURL: string;\n    StackId: string;\n    RequestId: string;\n    ResourceType: string;\n    LogicalResourceId: string;\n    PhysicalResourceId?: string;\n    ResourceProperties: {\n        ServiceToken: string;\n        FunctionName: string;\n        AliasName?: string;\n    };\n    OldResourceProperties?: {\n        FunctionName: string;\n        AliasName?: string;\n    };\n}\n\n/**\n * Custom Resource response to CloudFormation.\n */\nexport interface CustomResourceResponse {\n    Status: 'SUCCESS' | 'FAILED';\n    Reason?: string;\n    PhysicalResourceId: string;\n    StackId: string;\n    RequestId: string;\n    LogicalResourceId: string;\n    Data?: {\n        Version?: string;\n        AliasName?: string;\n        AliasArn?: string;\n        OptimizationStatus?: string;\n    };\n}\n\n/**\n * Result of a successful SnapStart activation cycle.\n *\n * Returned by {@link activateSnapStart} after enabling SnapStart,\n * publishing a version, waiting for snapshot readiness, and creating/updating an alias.\n *\n * @see {@link SnapStartActivatorConfig} for configuration options\n */\nexport interface SnapStartActivationResult {\n    /** The published Lambda version number (e.g. \"42\"). */\n    version: string;\n    /** The alias name that was created or updated (e.g. \"kata\"). */\n    aliasName: string;\n    /** The full ARN of the alias (e.g. \"arn:aws:lambda:us-east-1:123456789012:function:my-fn:kata\"). */\n    aliasArn: string;\n    /** The SnapStart optimization status of the published version (\"On\", \"Off\", or \"Unknown\"). */\n    optimizationStatus: string;\n}\n\n/**\n * Configuration options for the SnapStart activation cycle.\n *\n * All properties are optional and fall back to sensible defaults.\n *\n * @see {@link activateSnapStart} for the function that consumes this config\n */\nexport interface SnapStartActivatorConfig {\n    /**\n     * Maximum time in seconds to wait for snapshot creation before proceeding.\n     * If exceeded, a warning is logged and alias creation continues.\n     * @default 180\n     */\n    snapshotTimeoutSeconds?: number;\n    /**\n     * Interval in seconds between polling attempts for snapshot readiness.\n     * @default 2\n     */\n    pollingIntervalSeconds?: number;\n    /**\n     * The Lambda alias name to create or update after publishing a version.\n     * @default 'kata'\n     */\n    aliasName?: string;\n    /**\n     * @internal Override pre-publish delay in milliseconds (for testing only).\n     * @default 3000\n     */\n    _prePublishDelayMs?: number;\n    /**\n     * @internal Override retry delay in milliseconds (for testing only).\n     * @default 5000\n     */\n    _retryDelayMs?: number;\n}\n\n/**\n * Maximum number of PublishVersion + snapshot polling attempts.\n * Each retry publishes a NEW version (new snapshot attempt).\n * Bounded to prevent infinite loops; total worst-case time:\n * 5 * (15s delay + publish + 180s poll) \u2248 16min, within Lambda 15min limit\n * (but snapshot timeout is typically much shorter than 180s per attempt).\n */\nconst MAX_PUBLISH_RETRIES = 5;\n\n/**\n * Delay in milliseconds before retrying a failed snapshot creation.\n * Set to 15s to give transient Lambda init failures time to clear.\n */\nconst RETRY_DELAY_MS = 15000;\n\n/**\n * Delay in milliseconds after waitUntilFunctionUpdatedV2 completes,\n * before the first PublishVersion call. Mitigates eventual consistency\n * between Lambda config update propagation and PublishVersion init phase.\n */\nconst PRE_PUBLISH_DELAY_MS = 3000;\n\nconst DEFAULT_CONFIG: Required<SnapStartActivatorConfig> = {\n    snapshotTimeoutSeconds: 180,\n    pollingIntervalSeconds: 2,\n    aliasName: 'kata',\n    _prePublishDelayMs: PRE_PUBLISH_DELAY_MS,\n    _retryDelayMs: RETRY_DELAY_MS,\n};\n\n/**\n * Sleep for specified milliseconds.\n * @internal Exported for testability \u2014 tests can jest.spyOn to avoid real delays.\n */\nexport const _testable = {\n    sleep(ms: number): Promise<void> {\n        return new Promise(resolve => setTimeout(resolve, ms));\n    },\n};\n\n/**\n * Checks whether an error (or its cause chain) is a ResourceNotFoundException.\n *\n * AWS SDK v3 waiters may throw the underlying service exception directly,\n * or wrap it in a waiter-state error. This helper inspects both the error\n * itself and its `cause` property to detect either case.\n */\nfunction isResourceNotFoundError(error: unknown): boolean {\n    if (error instanceof ResourceNotFoundException) {\n        return true;\n    }\n    if (error instanceof Error) {\n        if (error.name === 'ResourceNotFoundException') {\n            return true;\n        }\n        // Waiters may wrap the service exception in a cause chain\n        const cause = (error as Error & { cause?: unknown }).cause;\n        if (cause instanceof ResourceNotFoundException) {\n            return true;\n        }\n        if (cause instanceof Error && cause.name === 'ResourceNotFoundException') {\n            return true;\n        }\n    }\n    return false;\n}\n\n/**\n * Required IAM permissions for SnapStart activation.\n * Used in error messages to guide users when AccessDeniedException occurs.\n */\nconst REQUIRED_PERMISSIONS: readonly string[] = [\n    'lambda:GetFunction',\n    'lambda:GetFunctionConfiguration',\n    'lambda:UpdateFunctionConfiguration',\n    'lambda:PublishVersion',\n    'lambda:GetAlias',\n    'lambda:CreateAlias',\n    'lambda:UpdateAlias',\n];\n\n/**\n * Checks whether an error is an AccessDeniedException.\n *\n * AWS SDK v3 may throw AccessDeniedException directly or wrap it.\n * This helper inspects both the error itself and its `cause` property.\n */\nfunction isAccessDeniedError(error: unknown): boolean {\n    if (error instanceof Error) {\n        if (error.name === 'AccessDeniedException') {\n            return true;\n        }\n        const cause = (error as Error & { cause?: unknown }).cause;\n        if (cause instanceof Error && cause.name === 'AccessDeniedException') {\n            return true;\n        }\n    }\n    return false;\n}\n\n\n/**\n * Activates SnapStart on a Lambda function.\n *\n * This function performs the full SnapStart activation cycle:\n * 1. Ensures function is Active\n * 2. Enables SnapStart configuration\n * 3. Waits for configuration update\n * 4. Publishes new version (with retry on snapshot failure)\n * 5. Waits for snapshot creation\n * 6. Creates/updates alias\n *\n * Steps 3+4 are wrapped in a retry loop: on State: Failed, a new version\n * is published (up to MAX_PUBLISH_RETRIES attempts) to handle transient\n * initialization failures during snapshot creation.\n *\n * @param lambdaClient - AWS Lambda client\n * @param functionName - Name or ARN of the Lambda function\n * @param config - Optional configuration\n * @returns Activation result with version and alias information\n */\nexport async function activateSnapStart(\n    lambdaClient: LambdaClient,\n    functionName: string,\n    config?: SnapStartActivatorConfig,\n): Promise<SnapStartActivationResult> {\n    const cfg = { ...DEFAULT_CONFIG, ...config };\n    const maxAttempts = Math.ceil(cfg.snapshotTimeoutSeconds / cfg.pollingIntervalSeconds);\n\n    console.log('='.repeat(60));\n    console.log('SNAPSTART ACTIVATION CYCLE');\n    console.log('='.repeat(60));\n    console.log(`Function: ${functionName}`);\n    console.log(`Timeout: ${cfg.snapshotTimeoutSeconds}s, Polling: ${cfg.pollingIntervalSeconds}s`);\n    console.log(`Max publish retries: ${MAX_PUBLISH_RETRIES}`);\n\n    try {\n        // Step 0: Ensure function is Active before starting\n        console.log('\\n[0/5] Ensuring function is Active...');\n        try {\n            await waitUntilFunctionActiveV2(\n                { client: lambdaClient, maxWaitTime: 60 },\n                { FunctionName: functionName }\n            );\n        } catch (error) {\n            if (isResourceNotFoundError(error)) {\n                throw new Error(`Lambda function '${functionName}' does not exist. Verify the function name or ARN is correct.`);\n            }\n            if (isAccessDeniedError(error)) {\n                throw new Error(\n                    `Insufficient permissions to activate SnapStart on '${functionName}'. ` +\n                    `Ensure the execution role has the following permissions: ${REQUIRED_PERMISSIONS.join(', ')}`\n                );\n            }\n            throw error;\n        }\n        console.log('      Function is Active');\n\n        // Step 1: Enable SnapStart on the function\n        console.log('\\n[1/5] Enabling SnapStart (ApplyOn: PublishedVersions)...');\n        await lambdaClient.send(new UpdateFunctionConfigurationCommand({\n            FunctionName: functionName,\n            SnapStart: { ApplyOn: 'PublishedVersions' },\n        }));\n        console.log('      SnapStart configuration sent');\n\n        // Step 2: Wait for configuration update to complete\n        console.log('[2/5] Waiting for configuration update...');\n        try {\n            await waitUntilFunctionUpdatedV2(\n                { client: lambdaClient, maxWaitTime: 120 },\n                { FunctionName: functionName }\n            );\n        } catch (error) {\n            if (isResourceNotFoundError(error)) {\n                throw new Error(`Lambda function '${functionName}' does not exist. Verify the function name or ARN is correct.`);\n            }\n            if (isAccessDeniedError(error)) {\n                throw new Error(\n                    `Insufficient permissions to activate SnapStart on '${functionName}'. ` +\n                    `Ensure the execution role has the following permissions: ${REQUIRED_PERMISSIONS.join(', ')}`\n                );\n            }\n            throw error;\n        }\n        console.log('      Configuration updated successfully');\n\n        // Pre-publish delay: mitigate eventual consistency between\n        // Lambda config update propagation and PublishVersion init phase\n        console.log(`      Waiting ${cfg._prePublishDelayMs / 1000}s for configuration propagation...`);\n        await _testable.sleep(cfg._prePublishDelayMs);\n\n        // Steps 3+4: Publish version and wait for snapshot (WITH RETRY)\n        let version: string = '';\n        let optimizationStatus = 'Unknown';\n        let state = 'Unknown';\n        let lastFailReason = '';\n\n        for (let publishAttempt = 1; publishAttempt <= MAX_PUBLISH_RETRIES; publishAttempt++) {\n            // Step 3: Publish new version to create snapshot\n            console.log(`[3/5] Publishing new version (attempt ${publishAttempt}/${MAX_PUBLISH_RETRIES})...`);\n\n            // Before re-publish (attempt > 1), wait and ensure function is ready\n            if (publishAttempt > 1) {\n                console.log(`      Waiting ${cfg._retryDelayMs / 1000}s before retry...`);\n                await _testable.sleep(cfg._retryDelayMs);\n                console.log('      Ensuring function is Active before re-publish...');\n                await waitUntilFunctionActiveV2(\n                    { client: lambdaClient, maxWaitTime: 60 },\n                    { FunctionName: functionName }\n                );\n                // Force a config update so PublishVersion creates a genuinely new version.\n                // Without this, PublishVersion returns the same (Failed) version number\n                // because Lambda deduplicates identical configurations.\n                console.log('      Re-applying SnapStart config to force new version...');\n                await lambdaClient.send(new UpdateFunctionConfigurationCommand({\n                    FunctionName: functionName,\n                    SnapStart: { ApplyOn: 'PublishedVersions' },\n                }));\n                await waitUntilFunctionUpdatedV2(\n                    { client: lambdaClient, maxWaitTime: 120 },\n                    { FunctionName: functionName }\n                );\n                console.log(`      Waiting ${cfg._prePublishDelayMs / 1000}s for configuration propagation...`);\n                await _testable.sleep(cfg._prePublishDelayMs);\n            }\n\n            const publishResponse = await lambdaClient.send(new PublishVersionCommand({\n                FunctionName: functionName,\n                Description: `SnapStart enabled - ${new Date().toISOString()} (attempt ${publishAttempt})`,\n            }));\n            version = publishResponse.Version!;\n            console.log(`      Published version: ${version}`);\n\n            // Step 4: Wait for snapshot optimization and verify status\n            console.log('[4/5] Waiting for SnapStart snapshot creation...');\n            state = 'Unknown';\n\n            for (let attempt = 0; attempt < maxAttempts; attempt++) {\n                const versionConfig = await lambdaClient.send(new GetFunctionConfigurationCommand({\n                    FunctionName: functionName,\n                    Qualifier: version,\n                }));\n\n                const snapStartStatus = versionConfig.SnapStart ?? {};\n                optimizationStatus = snapStartStatus.OptimizationStatus ?? 'Unknown';\n                state = versionConfig.State ?? 'Unknown';\n\n                if (state === 'Active') {\n                    console.log('      SnapStart snapshot ready!');\n                    console.log(`      OptimizationStatus: ${optimizationStatus}`);\n                    console.log(`      State: ${state}`);\n                    break;\n                } else if (state === 'Failed') {\n                    lastFailReason = versionConfig.StateReason ?? 'Unknown';\n                    console.log(`      Snapshot creation failed: ${lastFailReason}`);\n                    break;\n                } else {\n                    // Show progress every 10 attempts or first 5\n                    if (attempt % 10 === 0 || attempt < 5) {\n                        const elapsed = attempt * cfg.pollingIntervalSeconds;\n                        console.log(`      Creating snapshot... Status: ${optimizationStatus}, State: ${state} (${elapsed}s elapsed)`);\n                    }\n                }\n\n                await _testable.sleep(cfg.pollingIntervalSeconds * 1000);\n            }\n\n            // If snapshot succeeded, break out of retry loop\n            if (state === 'Active') {\n                break;\n            }\n\n            // Only retry on Failed state \u2014 timeout (Pending) should not retry\n            if (state !== 'Failed') {\n                break;\n            }\n\n            // If failed and we have retries left, log and continue\n            if (publishAttempt < MAX_PUBLISH_RETRIES) {\n                console.log(`      Snapshot failed on attempt ${publishAttempt}, will retry with new version...`);\n                console.log(`      Reason: ${lastFailReason}`);\n            }\n        }\n\n        // After all retries, check final state\n        if (state === 'Failed') {\n            console.log(`      All ${MAX_PUBLISH_RETRIES} snapshot attempts failed.`);\n            console.log(`      Reason: ${lastFailReason}`);\n            console.log('');\n            console.log('      [Lambda Kata] SnapStart optimization failed. Alias will be created pointing');\n            console.log('      to the latest version. Function remains operational without SnapStart.');\n            console.log('      Review CloudWatch logs for initialization errors.');\n        }\n\n        // Check if we timed out\n        if (state !== 'Active' && state !== 'Failed') {\n            console.log(`      Warning: Snapshot creation timeout after ${cfg.snapshotTimeoutSeconds}s`);\n            console.log(`      Final status: OptimizationStatus=${optimizationStatus}, State=${state}`);\n        }\n\n        // Step 5: Create or update alias\n        console.log(`[5/5] Creating/updating alias '${cfg.aliasName}' -> version ${version}...`);\n        let aliasArn: string;\n\n        try {\n            // Try to get existing alias\n            await lambdaClient.send(new GetAliasCommand({\n                FunctionName: functionName,\n                Name: cfg.aliasName,\n            }));\n\n            // Alias exists, update it\n            const updateResponse = await lambdaClient.send(new UpdateAliasCommand({\n                FunctionName: functionName,\n                Name: cfg.aliasName,\n                FunctionVersion: version,\n                Description: 'Lambda Kata SnapStart-enabled version',\n            }));\n            aliasArn = updateResponse.AliasArn!;\n            console.log(`      Updated existing alias: ${aliasArn}`);\n        } catch (error) {\n            // Check by error name for better testability\n            const isResourceNotFound = error instanceof ResourceNotFoundException ||\n                (error instanceof Error && error.name === 'ResourceNotFoundException');\n\n            if (isResourceNotFound) {\n                // Alias doesn't exist, create it\n                const createResponse = await lambdaClient.send(new CreateAliasCommand({\n                    FunctionName: functionName,\n                    Name: cfg.aliasName,\n                    FunctionVersion: version,\n                    Description: 'Lambda Kata SnapStart-enabled version',\n                }));\n                aliasArn = createResponse.AliasArn!;\n                console.log(`      Created new alias: ${aliasArn}`);\n            } else {\n                throw error;\n            }\n        }\n\n        console.log('\\n' + '='.repeat(60));\n        console.log('SNAPSTART ACTIVATION COMPLETE');\n        console.log('='.repeat(60));\n        console.log(`Version: ${version}`);\n        console.log(`Alias: ${cfg.aliasName} -> ${aliasArn}`);\n        console.log(`OptimizationStatus: ${optimizationStatus}`);\n\n        return {\n            version,\n            aliasName: cfg.aliasName,\n            aliasArn,\n            optimizationStatus,\n        };\n    } catch (error) {\n        // Catch AccessDeniedException from any Lambda API call (send calls)\n        // that isn't already handled by the waiter try-catch blocks\n        if (isAccessDeniedError(error)) {\n            throw new Error(\n                `Insufficient permissions to activate SnapStart on '${functionName}'. ` +\n                `Ensure the execution role has the following permissions: ${REQUIRED_PERMISSIONS.join(', ')}`\n            );\n        }\n        throw error;\n    }\n}\n\n/**\n * Lambda handler for CloudFormation Custom Resource.\n *\n * This handler is invoked by CloudFormation when the custom resource\n * is created, updated, or deleted.\n */\nexport async function handler(event: CustomResourceEvent): Promise<CustomResourceResponse> {\n    console.log('Custom Resource Event:', JSON.stringify(event, null, 2));\n\n    const { RequestType, StackId, RequestId, LogicalResourceId, ResourceProperties } = event;\n    const functionName = ResourceProperties.FunctionName;\n    const aliasName = ResourceProperties.AliasName ?? 'kata';\n\n    // Physical resource ID format: {functionName}:snapstart:{aliasName}\n    const physicalResourceId = event.PhysicalResourceId ?? `${functionName}:snapstart:${aliasName}`;\n\n    const baseResponse: Omit<CustomResourceResponse, 'Status' | 'Reason' | 'Data'> = {\n        PhysicalResourceId: physicalResourceId,\n        StackId,\n        RequestId,\n        LogicalResourceId,\n    };\n\n    try {\n        if (RequestType === 'Delete') {\n            // On delete, we don't remove SnapStart or the alias\n            // The Lambda function itself will be deleted by CloudFormation\n            console.log('Delete request - no action needed (Lambda will be deleted by CloudFormation)');\n            return {\n                ...baseResponse,\n                Status: 'SUCCESS',\n            };\n        }\n\n        // Create or Update - activate SnapStart\n        const lambdaClient = new LambdaClient({});\n\n        const result = await activateSnapStart(lambdaClient, functionName, { aliasName });\n\n        return {\n            ...baseResponse,\n            Status: 'SUCCESS',\n            Data: {\n                Version: result.version,\n                AliasName: result.aliasName,\n                AliasArn: result.aliasArn,\n                OptimizationStatus: result.optimizationStatus,\n            },\n        };\n    } catch (error) {\n        const errorMessage = error instanceof Error ? error.message : String(error);\n        console.error('SnapStart activation failed:', errorMessage);\n\n        // On Update requests, return SUCCESS even on failure to prevent\n        // CloudFormation rollback from getting stuck in UPDATE_ROLLBACK_FAILED.\n        // Returning FAILED from an Update during rollback blocks the entire stack.\n        // The function remains operational with its previous configuration.\n        if (RequestType === 'Update') {\n            console.log('Returning SUCCESS for Update request to prevent rollback deadlock.');\n            console.log('The function will continue with its previous SnapStart configuration.');\n            return {\n                ...baseResponse,\n                Status: 'SUCCESS',\n                Reason: `SnapStart activation failed (non-blocking): ${errorMessage}`,\n            };\n        }\n\n        return {\n            ...baseResponse,\n            Status: 'FAILED',\n            Reason: `SnapStart activation failed: ${errorMessage}`,\n        };\n    }\n}\n"],
  "mappings": "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",
  "names": ["snapstart_activator_exports", "__export", "_testable", "activateSnapStart", "handler", "__toCommonJS", "import_client_lambda", "MAX_PUBLISH_RETRIES", "RETRY_DELAY_MS", "PRE_PUBLISH_DELAY_MS", "DEFAULT_CONFIG", "ms", "resolve", "isResourceNotFoundError", "error", "cause", "REQUIRED_PERMISSIONS", "isAccessDeniedError", "lambdaClient", "functionName", "config", "cfg", "maxAttempts", "version", "optimizationStatus", "state", "lastFailReason", "publishAttempt", "attempt", "versionConfig", "elapsed", "aliasArn", "event", "RequestType", "StackId", "RequestId", "LogicalResourceId", "ResourceProperties", "aliasName", "baseResponse", "result", "errorMessage"]
}

|
|
@@ -1,12 +1,3 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* SnapStart Construct - CDK Custom Resource for SnapStart Activation
|
|
3
|
-
*
|
|
4
|
-
* This module provides a CDK construct that creates a Custom Resource
|
|
5
|
-
* to enable SnapStart on Lambda functions after deployment. The construct
|
|
6
|
-
* handles the asynchronous nature of SnapStart snapshot creation.
|
|
7
|
-
*
|
|
8
|
-
* @module snapstart-construct
|
|
9
|
-
*/
|
|
10
1
|
import { Construct } from 'constructs';
|
|
11
2
|
import { CustomResource } from 'aws-cdk-lib';
|
|
12
3
|
import { IFunction } from 'aws-cdk-lib/aws-lambda';
|
|
@@ -102,11 +93,4 @@ export declare class SnapStartActivator extends Construct {
|
|
|
102
93
|
* IAM propagation race conditions.
|
|
103
94
|
*/
|
|
104
95
|
private createProviderFunction;
|
|
105
|
-
/**
|
|
106
|
-
* Generates the inline handler code for the Custom Resource.
|
|
107
|
-
*
|
|
108
|
-
* This is a self-contained version of the snapstart-activator logic
|
|
109
|
-
* that can be deployed as inline Lambda code.
|
|
110
|
-
*/
|
|
111
|
-
private generateHandlerCode;
|
|
112
96
|
}
|
package/package.json
CHANGED
|
@@ -1,11 +1,12 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@lambda-kata/cdk",
|
|
3
|
-
"version": "0.1.3-rc.
|
|
3
|
+
"version": "0.1.3-rc.93",
|
|
4
4
|
"description": "AWS CDK integration for Lambda Kata - Node.js Lambdas running via Lambda Kata runtime",
|
|
5
5
|
"main": "out/dist/index.js",
|
|
6
6
|
"types": "out/tsc/src/index.d.ts",
|
|
7
7
|
"scripts": {
|
|
8
|
-
"build": "rm -rf ./dist && rm -rf ./out && rm -rf ./lib && yarn run build:cdk && yarn run types",
|
|
8
|
+
"build": "rm -rf ./dist && rm -rf ./out && rm -rf ./lib && yarn run build:cdk && yarn run build:snapstart && yarn run types",
|
|
9
|
+
"build:snapstart": "yarn run build:config && node dist/utils/build.js --target=snapstartHandler",
|
|
9
10
|
"types": "tsc --emitDeclarationOnly && tsc-alias",
|
|
10
11
|
"build:cdk": "yarn run build:config && node dist/utils/build.js --target=cdk",
|
|
11
12
|
"build:minify": "yarn run build:config && node dist/utils/build.js",
|