@lambda-kata/cdk 0.1.4 → 0.1.6
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.
- package/out/dist/index.js +38 -46
- package/out/dist/snapstart-handler.js +3 -5
- package/out/tsc/src/index.d.ts +1 -3
- package/out/tsc/src/kata-wrapper.d.ts +3 -15
- package/out/tsc/src/types.d.ts +0 -6
- package/package.json +2 -2
- package/out/tsc/src/licensing.d.ts +0 -151
- package/out/tsc/src/mock-licensing.d.ts +0 -218
|
@@ -1,5 +1,3 @@
|
|
|
1
|
-
"use strict";var
|
|
2
|
-
[
|
|
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
|
+
"use strict";var R=Object.defineProperty;var F=Object.getOwnPropertyDescriptor;var $=Object.getOwnPropertyNames;var y=Object.prototype.hasOwnProperty;var v=(t,e)=>{for(var u in e)R(t,u,{get:e[u],enumerable:!0})},C=(t,e,u,a)=>{if(e&&typeof e=="object"||typeof e=="function")for(let c of $(e))!y.call(t,c)&&c!==u&&R(t,c,{get:()=>e[c],enumerable:!(a=F(e,c))||a.enumerable});return t};var I=t=>C(R({},"__esModule",{value:!0}),t);var x={};v(x,{_testable:()=>g,activateSnapStart:()=>A,handler:()=>P});module.exports=I(x);var o=require("@aws-sdk/client-lambda"),S=5,N=15e3,L=3e3,D={snapshotTimeoutSeconds:180,pollingIntervalSeconds:2,aliasName:"kata",_prePublishDelayMs:L,_retryDelayMs:N},g={sleep(t){return new Promise(e=>setTimeout(e,t))}};function b(t){if(t instanceof o.ResourceNotFoundException)return!0;if(t instanceof Error){if(t.name==="ResourceNotFoundException")return!0;let e=t.cause;if(e instanceof o.ResourceNotFoundException||e instanceof Error&&e.name==="ResourceNotFoundException")return!0}return!1}var h=["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 A(t,e,u){let a={...D,...u},c=Math.ceil(a.snapshotTimeoutSeconds/a.pollingIntervalSeconds);console.log("[Lambda Kata] Deterministic initialization"),console.log(`Function: ${e}`),console.log(`Timeout: ${a.snapshotTimeoutSeconds}s, Polling: ${a.pollingIntervalSeconds}s`),console.log(`Max publish retries: ${S}`);try{try{await(0,o.waitUntilFunctionActiveV2)({client:t,maxWaitTime:60},{FunctionName:e})}catch(n){throw b(n)?new Error(`Lambda function '${e}' does not exist. Verify the function name or ARN is correct.`):w(n)?new Error(`Insufficient permissions to activate SnapStart on '${e}'. Ensure the execution role has the following permissions: ${h.join(", ")}`):n}console.log(`
|
|
2
|
+
[Lambda Kata] Preparing low-latency cold-start runtime...`),await t.send(new o.UpdateFunctionConfigurationCommand({FunctionName:e,SnapStart:{ApplyOn:"PublishedVersions"}}));try{await(0,o.waitUntilFunctionUpdatedV2)({client:t,maxWaitTime:120},{FunctionName:e})}catch(n){throw b(n)?new Error(`Lambda function '${e}' does not exist. Verify the function name or ARN is correct.`):w(n)?new Error(`Insufficient permissions to activate SnapStart on '${e}'. Ensure the execution role has the following permissions: ${h.join(", ")}`):n}await g.sleep(a._prePublishDelayMs);let i="",p="Unknown",s="Unknown",m="";for(let n=1;n<=S;n++){console.log(`[Lambda Kata] Publishing optimized runtime version (attempt ${n}/${S})...`),n>1&&(console.log(` Waiting ${a._retryDelayMs/1e3}s before retry...`),await g.sleep(a._retryDelayMs),console.log(" Ensuring function is Active before re-publish..."),await(0,o.waitUntilFunctionActiveV2)({client:t,maxWaitTime:60},{FunctionName:e}),console.log(" Re-applying SnapStart config to force new version..."),await t.send(new o.UpdateFunctionConfigurationCommand({FunctionName:e,SnapStart:{ApplyOn:"PublishedVersions"}})),await(0,o.waitUntilFunctionUpdatedV2)({client:t,maxWaitTime:120},{FunctionName:e}),await g.sleep(a._prePublishDelayMs)),i=(await t.send(new o.PublishVersionCommand({FunctionName:e,Description:`SnapStart enabled - ${new Date().toISOString()} (attempt ${n})`}))).Version,console.log(` Published version: ${i}`),console.log("[Lambda Kata] Preparing pre-initialized runtime image..."),s="Unknown";for(let d=0;d<c;d++){let f=await t.send(new o.GetFunctionConfigurationCommand({FunctionName:e,Qualifier:i}));if(p=(f.SnapStart??{}).OptimizationStatus??"Unknown",s=f.State??"Unknown",s==="Active"){console.log(" SnapStart snapshot ready!"),console.log(` OptimizationStatus: ${p}`),console.log(` State: ${s}`);break}else if(s==="Failed"){m=f.StateReason??"Unknown",console.log(` Snapshot creation failed: ${m}`),console.log(` Snapshot creation failed: ${f.StateReasonCode}`);break}else if(d%10===0||d<5){let E=d*a.pollingIntervalSeconds;console.log(` Creating snapshot... Status: ${p}, State: ${s} (${E}s elapsed)`)}await g.sleep(a.pollingIntervalSeconds*1e3)}if(s==="Active"||s!=="Failed")break;n<S&&(console.log(` Snapshot failed on attempt ${n}, will retry with new version...`),console.log(` Reason: ${m}`))}if(s==="Failed")throw console.log(` All ${S} snapshot attempts failed.`),console.log(` Reason: ${m}`),console.log(""),console.log(" [Lambda Kata] SnapStart optimization failed. Cannot create alias on Failed version."),console.log(" Review CloudWatch logs for initialization errors."),console.log(" Common causes: missing Lambda layers, import errors, or runtime issues."),new Error(`SnapStart activation failed: All ${S} snapshot attempts failed. Reason: ${m}. Cannot create alias on a Failed version. Check CloudWatch logs for the Lambda function to diagnose initialization errors.`);s!=="Active"&&s!=="Failed"&&(console.log(` Warning: Snapshot creation timeout after ${a.snapshotTimeoutSeconds}s`),console.log(` Final status: OptimizationStatus=${p}, State=${s}`));let r;try{await t.send(new o.GetAliasCommand({FunctionName:e,Name:a.aliasName})),r=(await t.send(new o.UpdateAliasCommand({FunctionName:e,Name:a.aliasName,FunctionVersion:i,Description:"Lambda Kata SnapStart-enabled version"}))).AliasArn,console.log(` Updated existing alias: ${r}`)}catch(n){if(n instanceof o.ResourceNotFoundException||n instanceof Error&&n.name==="ResourceNotFoundException")r=(await t.send(new o.CreateAliasCommand({FunctionName:e,Name:a.aliasName,FunctionVersion:i,Description:"Lambda Kata SnapStart-enabled version"}))).AliasArn,console.log(` Created new alias: ${r}`);else throw n}return console.log(`[Lambda Kata] Version: ${i}`),console.log(`[Lambda Kata] Alias: ${a.aliasName} | ${r}`),{version:i,aliasName:a.aliasName,aliasArn:r,optimizationStatus:p}}catch(i){throw w(i)?new Error(`Insufficient permissions to activate SnapStart on '${e}'. Ensure the execution role has the following permissions: ${h.join(", ")}`):i}}async function P(t){console.log("[Lambda Kata] Custom Resource Event:",JSON.stringify(t,null,2));let{RequestType:e,StackId:u,RequestId:a,LogicalResourceId:c,ResourceProperties:i}=t,p=i.FunctionName,s=i.AliasName??"kata",r={PhysicalResourceId:t.PhysicalResourceId??`${p}:snapstart:${s}`,StackId:u,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 n=new o.LambdaClient({}),l=await A(n,p,{aliasName:s});return{...r,Status:"SUCCESS",Data:{Version:l.version,AliasName:l.aliasName,AliasArn:l.aliasArn,OptimizationStatus:l.optimizationStatus}}}catch(n){let l=n instanceof Error?n.message:String(n);return console.error("[Lambda Kata] Optimization error:",l),e==="Update"?{...r,Status:"SUCCESS",Reason:`SnapStart activation failed (non-blocking): ${l}`}:{...r,Status:"FAILED",Reason:`SnapStart activation failed: ${l}`}}}0&&(module.exports={_testable,activateSnapStart,handler});
|
|
3
|
+
//# 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('[Lambda Kata] Deterministic initialization');\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[Lambda Kata] Preparing low-latency cold-start runtime...');\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(`[Lambda Kata] Publishing optimized runtime 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('[Lambda Kata]   Preparing pre-initialized runtime image...');\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                    console.log(`      Snapshot creation failed: ${versionConfig.StateReasonCode}`);\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. Cannot create alias on Failed version.');\n            console.log('      Review CloudWatch logs for initialization errors.');\n            console.log('      Common causes: missing Lambda layers, import errors, or runtime issues.');\n\n            // Cannot create alias on Failed version - AWS Lambda rejects this operation.\n            // Throw error to signal the failure clearly.\n            throw new Error(\n                `SnapStart activation failed: All ${MAX_PUBLISH_RETRIES} snapshot attempts failed. ` +\n                `Reason: ${lastFailReason}. ` +\n                `Cannot create alias on a Failed version. ` +\n                `Check CloudWatch logs for the Lambda function to diagnose initialization errors.`\n            );\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 (only if version is Active or Pending)\n        // Note: AWS Lambda does not allow creating aliases on Failed versions\n        // console.log(`[Lambda Kata]  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(`[Lambda Kata] Version: ${version}`);\n        console.log(`[Lambda Kata] Alias: ${cfg.aliasName} | ${aliasArn}`);\n        // console.log(`[Lambda Kata] 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('[Lambda Kata] 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('[Lambda Kata] Optimization error:', 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"]
}

|
package/out/tsc/src/index.d.ts
CHANGED
|
@@ -18,11 +18,9 @@
|
|
|
18
18
|
* ```
|
|
19
19
|
*/
|
|
20
20
|
export { KataProps, LicensingResponse, TransformationConfig } from './types';
|
|
21
|
-
export { LicensingService, LicenseCheckParams, HttpLicensingService, createLicensingService, isValidAccountId, } from './licensing';
|
|
22
|
-
export { MockLicensingService, createMockLicensingService, } from './mock-licensing';
|
|
23
21
|
export { resolveAccountId, resolveAccountIdWithSource, isValidAccountIdFormat, AccountResolutionError, AccountResolutionResult, AccountResolverOptions, } from './account-resolver';
|
|
24
22
|
export { resolveAccountIdSync, resolveAccountIdSyncWithSource, resolveRegionSync, SyncAccountResolutionError, SyncAccountResolutionResult, SyncAccountResolverOptions, } from './sync-account-resolver';
|
|
25
|
-
export { kata,
|
|
23
|
+
export { kata, applyTransformation, handleUnlicensed, isKataTransformed, getKataPromise, extractBundlePathFromHandler, extractNodeVersion, KataWrapperOptions, KataResult, } from './kata-wrapper';
|
|
26
24
|
export { SnapStartActivator, SnapStartActivatorProps, } from './snapstart-construct';
|
|
27
25
|
export { activateSnapStart, SnapStartActivationResult, SnapStartActivatorConfig, } from './snapstart-activator';
|
|
28
26
|
export { createKataConfigLayer, generateConfigContent, KataConfigLayerProps, CONFIG_DIR_NAME, CONFIG_FILE_NAME, HANDLER_CONFIG_KEY, } from './config-layer';
|
|
@@ -1,24 +1,10 @@
|
|
|
1
1
|
import { Function as LambdaFunction } from 'aws-cdk-lib/aws-lambda';
|
|
2
2
|
import { NodejsFunction } from 'aws-cdk-lib/aws-lambda-nodejs';
|
|
3
3
|
import { KataProps, LicensingResponse, TransformationConfig } from './types';
|
|
4
|
-
import { LicensingService } from './licensing';
|
|
5
|
-
import { LicensingService as NativeLicensingServiceInterface } from '@lambda-kata/licensing';
|
|
6
4
|
/**
|
|
7
5
|
* Options for the kata wrapper, extending KataProps with internal options.
|
|
8
6
|
*/
|
|
9
7
|
export interface KataWrapperOptions extends KataProps {
|
|
10
|
-
/**
|
|
11
|
-
* Optional: Custom licensing service for testing.
|
|
12
|
-
* If not provided, the default HTTP licensing service will be used.
|
|
13
|
-
* @internal
|
|
14
|
-
*/
|
|
15
|
-
licensingService?: LicensingService;
|
|
16
|
-
/**
|
|
17
|
-
* Optional: Custom synchronous licensing service for testing.
|
|
18
|
-
* Must implement checkEntitlementSync(accountId: string): LicensingResponse
|
|
19
|
-
* @internal
|
|
20
|
-
*/
|
|
21
|
-
syncLicensingService?: NativeLicensingServiceInterface;
|
|
22
8
|
/**
|
|
23
9
|
* Custom bundle path.
|
|
24
10
|
* If not specified, uses the default /opt/js_runtime/bundle.js
|
|
@@ -133,6 +119,8 @@ export declare function kata<T extends NodejsFunction | LambdaFunction>(lambda:
|
|
|
133
119
|
* @param props - Optional configuration for the transformation
|
|
134
120
|
* @returns Promise resolving to the transformation result
|
|
135
121
|
*
|
|
122
|
+
* @deprecated
|
|
123
|
+
*
|
|
136
124
|
* @internal
|
|
137
125
|
*/
|
|
138
126
|
export declare function kataWithAccountId<T extends NodejsFunction | LambdaFunction>(lambda: T, accountId: string, region: string, props?: KataWrapperOptions): Promise<KataResult>;
|
|
@@ -224,7 +212,7 @@ export declare function applyTransformation(lambda: NodejsFunction | LambdaFunct
|
|
|
224
212
|
*
|
|
225
213
|
* @internal
|
|
226
214
|
*/
|
|
227
|
-
export declare function handleUnlicensed(lambda: NodejsFunction | LambdaFunction, props:
|
|
215
|
+
export declare function handleUnlicensed(lambda: NodejsFunction | LambdaFunction, props: KataProps | undefined, licensingResponse: LicensingResponse): void;
|
|
228
216
|
/**
|
|
229
217
|
* Checks if a Lambda function has been transformed by kata().
|
|
230
218
|
*
|
package/out/tsc/src/types.d.ts
CHANGED
|
@@ -12,16 +12,10 @@ import { Runtime } from 'aws-cdk-lib/aws-lambda';
|
|
|
12
12
|
* ```typescript
|
|
13
13
|
* kata(myFunction, {
|
|
14
14
|
* unlicensedBehavior: 'fail',
|
|
15
|
-
* licensingEndpoint: 'https://custom-endpoint.example.com'
|
|
16
15
|
* });
|
|
17
16
|
* ```
|
|
18
17
|
*/
|
|
19
18
|
export interface KataProps {
|
|
20
|
-
/**
|
|
21
|
-
* Optional: Override the licensing service endpoint
|
|
22
|
-
* Default: Lambda Kata production licensing endpoint
|
|
23
|
-
*/
|
|
24
|
-
licensingEndpoint?: string;
|
|
25
19
|
/**
|
|
26
20
|
* Optional: Behavior when account is not licensed
|
|
27
21
|
* 'warn' - Keep original Lambda, emit warning (default)
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@lambda-kata/cdk",
|
|
3
|
-
"version": "0.1.
|
|
3
|
+
"version": "0.1.6",
|
|
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",
|
|
@@ -84,4 +84,4 @@
|
|
|
84
84
|
"optional": true
|
|
85
85
|
}
|
|
86
86
|
}
|
|
87
|
-
}
|
|
87
|
+
}
|
|
@@ -1,151 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Licensing Service for Lambda Kata CDK Integration
|
|
3
|
-
*
|
|
4
|
-
* This module provides the interface and implementation for validating
|
|
5
|
-
* AWS Marketplace entitlements at CDK synthesis/deploy time.
|
|
6
|
-
*
|
|
7
|
-
* @module licensing
|
|
8
|
-
*/
|
|
9
|
-
import { LicensingResponse } from './types';
|
|
10
|
-
/**
|
|
11
|
-
* Supported Node.js versions for licensing requests.
|
|
12
|
-
* Valid values: "18", "20", "22", "24"
|
|
13
|
-
*/
|
|
14
|
-
export type NodeVersion = '18' | '20' | '22' | '24';
|
|
15
|
-
/**
|
|
16
|
-
* Supported architectures for licensing requests.
|
|
17
|
-
* Valid values: "x86_64", "arm64"
|
|
18
|
-
*/
|
|
19
|
-
export type LicenseArchitecture = 'x86_64' | 'arm64';
|
|
20
|
-
/**
|
|
21
|
-
* Parameters for license entitlement check.
|
|
22
|
-
*/
|
|
23
|
-
export interface LicenseCheckParams {
|
|
24
|
-
/**
|
|
25
|
-
* AWS account ID (12-digit string)
|
|
26
|
-
*/
|
|
27
|
-
accountId: string;
|
|
28
|
-
/**
|
|
29
|
-
* Node.js version number.
|
|
30
|
-
* Valid values: "18", "20", "22", "24"
|
|
31
|
-
* @default "20"
|
|
32
|
-
*/
|
|
33
|
-
nodeVersion?: NodeVersion;
|
|
34
|
-
/**
|
|
35
|
-
* Lambda architecture.
|
|
36
|
-
* Valid values: "x86_64", "arm64"
|
|
37
|
-
* @default "x86_64"
|
|
38
|
-
*/
|
|
39
|
-
architecture?: LicenseArchitecture;
|
|
40
|
-
}
|
|
41
|
-
/**
|
|
42
|
-
* Interface for the Lambda Kata licensing service.
|
|
43
|
-
*
|
|
44
|
-
* Implementations of this interface are responsible for validating
|
|
45
|
-
* AWS account entitlements and returning customer-specific Layer ARNs.
|
|
46
|
-
*
|
|
47
|
-
* @example
|
|
48
|
-
* ```typescript
|
|
49
|
-
* const licensingService = new HttpLicensingService();
|
|
50
|
-
* const response = await licensingService.checkEntitlement({
|
|
51
|
-
* accountId: '123456789012',
|
|
52
|
-
* nodeVersion: '20',
|
|
53
|
-
* architecture: 'x86_64'
|
|
54
|
-
* });
|
|
55
|
-
* if (response.entitled) {
|
|
56
|
-
* console.log(`Layer ARN: ${response.layerVersionArn}`);
|
|
57
|
-
* }
|
|
58
|
-
* ```
|
|
59
|
-
*/
|
|
60
|
-
export interface LicensingService {
|
|
61
|
-
/**
|
|
62
|
-
* Check if an AWS account is entitled to use Lambda Kata.
|
|
63
|
-
*
|
|
64
|
-
* @param params - License check parameters (accountId, nodeVersion, architecture)
|
|
65
|
-
* For backward compatibility, can also be just accountId string
|
|
66
|
-
* @returns Promise resolving to entitlement status and Layer ARN if entitled
|
|
67
|
-
*
|
|
68
|
-
* @remarks
|
|
69
|
-
* - If the account is entitled, the response will include the customer-specific Layer ARN
|
|
70
|
-
* - If the account is not entitled, the response will have `entitled: false`
|
|
71
|
-
* - Network errors are handled gracefully and treated as unlicensed (Requirement 6.5)
|
|
72
|
-
*/
|
|
73
|
-
checkEntitlement(params: LicenseCheckParams | string): Promise<LicensingResponse>;
|
|
74
|
-
}
|
|
75
|
-
/**
|
|
76
|
-
* HTTP-based implementation of the LicensingService interface.
|
|
77
|
-
*
|
|
78
|
-
* This implementation calls the Lambda Kata licensing backend to validate
|
|
79
|
-
* AWS Marketplace entitlements. Network errors are handled gracefully
|
|
80
|
-
* by treating unreachable services as unlicensed (per Requirement 6.5).
|
|
81
|
-
*
|
|
82
|
-
* @example
|
|
83
|
-
* ```typescript
|
|
84
|
-
* // Use default endpoint
|
|
85
|
-
* const service = new HttpLicensingService();
|
|
86
|
-
*
|
|
87
|
-
* // Use custom endpoint
|
|
88
|
-
* const customService = new HttpLicensingService('https://custom.endpoint.com');
|
|
89
|
-
* ```
|
|
90
|
-
*/
|
|
91
|
-
export declare class HttpLicensingService implements LicensingService {
|
|
92
|
-
private readonly endpoint;
|
|
93
|
-
private readonly timeoutMs;
|
|
94
|
-
/**
|
|
95
|
-
* Creates a new HttpLicensingService instance.
|
|
96
|
-
*
|
|
97
|
-
* @param endpoint - Optional custom licensing endpoint URL
|
|
98
|
-
* @param timeoutMs - Optional request timeout in milliseconds (default: 5000)
|
|
99
|
-
*/
|
|
100
|
-
constructor(endpoint?: string, timeoutMs?: number);
|
|
101
|
-
/**
|
|
102
|
-
* Check if an AWS account is entitled to use Lambda Kata.
|
|
103
|
-
*
|
|
104
|
-
* Makes an HTTP request to the licensing service to validate the account's
|
|
105
|
-
* AWS Marketplace entitlement. If the service is unreachable or returns
|
|
106
|
-
* an error, the account is treated as unlicensed (Requirement 6.5).
|
|
107
|
-
*
|
|
108
|
-
* @param params - License check parameters or accountId string for backward compatibility
|
|
109
|
-
* @returns Promise resolving to entitlement status and Layer ARN if entitled
|
|
110
|
-
*
|
|
111
|
-
* @remarks
|
|
112
|
-
* Validates: Requirements 3.2, 3.3, 6.5
|
|
113
|
-
* - 3.2: Calls the Licensing_Service to validate the account's entitlement
|
|
114
|
-
* - 3.3: Returns the customer-specific Layer_ARN if entitled
|
|
115
|
-
* - 6.5: Treats unreachable service as unlicensed with appropriate warning
|
|
116
|
-
*/
|
|
117
|
-
checkEntitlement(params: LicenseCheckParams | string): Promise<LicensingResponse>;
|
|
118
|
-
/**
|
|
119
|
-
* Makes the HTTP request to the licensing service.
|
|
120
|
-
*
|
|
121
|
-
* @param accountId - The AWS account ID to check
|
|
122
|
-
* @param nodeVersion - Optional Node.js version (defaults to "20")
|
|
123
|
-
* @param architecture - Optional architecture (defaults to "x86_64")
|
|
124
|
-
* @returns Promise resolving to the licensing response
|
|
125
|
-
* @throws Error if the request fails or times out
|
|
126
|
-
*/
|
|
127
|
-
private makeRequest;
|
|
128
|
-
}
|
|
129
|
-
/**
|
|
130
|
-
* Validates that an AWS account ID is in the correct format.
|
|
131
|
-
*
|
|
132
|
-
* @param accountId - The account ID to validate
|
|
133
|
-
* @returns true if the account ID is a valid 12-digit string
|
|
134
|
-
*/
|
|
135
|
-
export declare function isValidAccountId(accountId: string): boolean;
|
|
136
|
-
/**
|
|
137
|
-
* Creates a default LicensingService instance.
|
|
138
|
-
*
|
|
139
|
-
* This factory function provides a convenient way to create a licensing
|
|
140
|
-
* service with default configuration.
|
|
141
|
-
*
|
|
142
|
-
* @param endpoint - Optional custom licensing endpoint URL
|
|
143
|
-
* @returns A new LicensingService instance
|
|
144
|
-
*
|
|
145
|
-
* @example
|
|
146
|
-
* ```typescript
|
|
147
|
-
* const service = createLicensingService();
|
|
148
|
-
* const response = await service.checkEntitlement('123456789012');
|
|
149
|
-
* ```
|
|
150
|
-
*/
|
|
151
|
-
export declare function createLicensingService(endpoint?: string): LicensingService;
|