UNPKG

@aws-cdk/aws-bedrock-agentcore-alpha

Version:

The CDK Construct Library for Amazon Bedrock

209 lines 29.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.ManagedMemoryStrategy = void 0; const jsiiDeprecationWarnings = require("../../../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const memory_strategy_1 = require("../memory-strategy"); const validation_helpers_1 = require("../validation-helpers"); /****************************************************************************** * CONSTANTS *****************************************************************************/ /** * Minimum length for prompt * @internal */ const PROMPT_MIN_LENGTH = 1; /** * Maximum length for prompt * @internal */ const PROMPT_MAX_LENGTH = 30000; /** * Managed memory strategy that handles both built-in and override configurations. * This strategy can be used for quick setup with built-in defaults or customized * with specific models and prompt templates. */ class ManagedMemoryStrategy { static [JSII_RTTI_SYMBOL_1] = { fqn: "@aws-cdk/aws-bedrock-agentcore-alpha.ManagedMemoryStrategy", version: "2.227.0-alpha.0" }; name; description; /** * The namespaces for the strategy */ namespaces; /** * The configuration for the custom consolidation. */ consolidationOverride; /** * The configuration for the custom extraction. */ extractionOverride; strategyType; /** * Constructor to create a new managed memory strategy * @param strategyType the strategy type * @param props the properties for the strategy */ constructor(strategyType, props) { try { jsiiDeprecationWarnings._aws_cdk_aws_bedrock_agentcore_alpha_MemoryStrategyType(strategyType); jsiiDeprecationWarnings._aws_cdk_aws_bedrock_agentcore_alpha_ManagedStrategyProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, ManagedMemoryStrategy); } throw error; } // ------------------------------------------------------ // Set properties and defaults // ------------------------------------------------------ this.name = props.name; this.description = props.description; this.namespaces = props.namespaces; this.strategyType = strategyType; this.consolidationOverride = props.customConsolidation; this.extractionOverride = props.customExtraction; // ------------------------------------------------------ // Validations // ------------------------------------------------------ (0, validation_helpers_1.throwIfInvalid)(this._validatePrompt, this.consolidationOverride?.appendToPrompt); (0, validation_helpers_1.throwIfInvalid)(this._validatePrompt, this.extractionOverride?.appendToPrompt); (0, validation_helpers_1.throwIfInvalid)(this._validateMemoryStrategyName, this.name); if (this.namespaces) { (0, validation_helpers_1.throwIfInvalid)(this._validateMemoryStrategyNamespaces, this.namespaces); } } /** * Renders the network configuration as a CloudFormation property. * @returns The CloudFormation property for the memory strategy. */ render() { // If no overrides, use built-in strategy format if (!this.consolidationOverride && !this.extractionOverride) { const cfnStrategyMap = { [memory_strategy_1.MemoryStrategyType.USER_PREFERENCE]: 'userPreferenceMemoryStrategy', [memory_strategy_1.MemoryStrategyType.SEMANTIC]: 'semanticMemoryStrategy', [memory_strategy_1.MemoryStrategyType.SUMMARIZATION]: 'summaryMemoryStrategy', [memory_strategy_1.MemoryStrategyType.CUSTOM]: 'customMemoryStrategy', }; const strategyKey = cfnStrategyMap[this.strategyType]; return { [strategyKey]: { name: this.name, description: this.description, namespaces: this.namespaces, type: this.strategyType, }, }; } // If overrides are provided, use custom strategy format const cfnStrategyMap = { [memory_strategy_1.MemoryStrategyType.USER_PREFERENCE]: 'userPreferenceOverride', [memory_strategy_1.MemoryStrategyType.SEMANTIC]: 'semanticOverride', [memory_strategy_1.MemoryStrategyType.SUMMARIZATION]: 'summaryOverride', [memory_strategy_1.MemoryStrategyType.CUSTOM]: '', }; const strategyKey = cfnStrategyMap[this.strategyType]; return { customMemoryStrategy: { name: this.name, description: this.description, namespaces: this.namespaces, type: this.strategyType, configuration: { [strategyKey]: { ...(this.consolidationOverride && { consolidation: { modelId: aws_cdk_lib_1.Arn.split(this.consolidationOverride.model.invokableArn, aws_cdk_lib_1.ArnFormat.SLASH_RESOURCE_NAME) .resourceName, appendToPrompt: this.consolidationOverride.appendToPrompt, }, }), ...(this.extractionOverride && { extraction: { modelId: aws_cdk_lib_1.Arn.split(this.extractionOverride.model.invokableArn, aws_cdk_lib_1.ArnFormat.SLASH_RESOURCE_NAME) .resourceName, appendToPrompt: this.extractionOverride.appendToPrompt, }, }), }, }, }, }; } /** * Grants the necessary permissions to the role * @param grantee - The grantee to grant permissions to * @returns The Grant object for chaining */ grant(grantee) { const grant1 = this.consolidationOverride?.model.grantInvoke(grantee); const grant2 = this.extractionOverride?.model.grantInvoke(grantee); return grant1 && grant2 ? grant1.combine(grant2) : grant1 || grant2; } // ------------------------------------------------------ // VALIDATORS // ------------------------------------------------------ /** * Validates the memory strategy name * @param name - The name to validate * @returns Array of validation error messages, empty if valid */ _validateMemoryStrategyName = (name, scope) => { let errors = []; errors.push(...(0, validation_helpers_1.validateStringFieldLength)({ value: name, fieldName: 'Memory name', minLength: memory_strategy_1.MEMORY_NAME_MIN_LENGTH, maxLength: memory_strategy_1.MEMORY_NAME_MAX_LENGTH, }, scope)); // Check if name matches the AWS API pattern: [a-zA-Z][a-zA-Z0-9_]{0,47} // Must start with a letter, followed by up to 47 letters, numbers, or underscores const validNamePattern = /^[a-zA-Z][a-zA-Z0-9_]{0,47}$/; errors.push(...(0, validation_helpers_1.validateFieldPattern)(name, 'Memory name', validNamePattern, undefined, scope)); return errors; }; /** * Validates the prompt * @param prompt - The prompt to validate * @returns Array of validation error messages, empty if valid */ _validatePrompt = (prompt, scope) => { let errors = []; if (!prompt) { return errors; } errors.push(...(0, validation_helpers_1.validateStringFieldLength)({ value: prompt, fieldName: 'Prompt', minLength: PROMPT_MIN_LENGTH, maxLength: PROMPT_MAX_LENGTH, }, scope)); return errors; }; /** * Validates the memory strategy namespaces * @param namespaces - The namespaces to validate * @returns Array of validation error messages, empty if valid */ _validateMemoryStrategyNamespaces = (namespaces) => { let errors = []; if (namespaces.length === 0) { return errors; } for (const namespace of namespaces) { // Only check for template variables in namespace definition if (namespace.includes('{') && !(namespace.includes('{actorId}') || namespace.includes('{sessionId}') || namespace.includes('{memoryStrategyId}'))) { errors.push(`Namespace with templates should contain valid variables: ${namespace}`); } } return errors; }; } exports.ManagedMemoryStrategy = ManagedMemoryStrategy; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"managed-strategy.js","sourceRoot":"","sources":["managed-strategy.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n *  Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n *\n *  Licensed under the Apache License, Version 2.0 (the \"License\"). You may not use this file except in compliance\n *  with the License. A copy of the License is located at\n *\n *      http://www.apache.org/licenses/LICENSE-2.0\n *\n *  or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES\n *  OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions\n *  and limitations under the License.\n */\n\nimport { IBedrockInvokable } from '@aws-cdk/aws-bedrock-alpha';\nimport { Arn, ArnFormat } from 'aws-cdk-lib';\nimport { IConstruct } from 'constructs';\nimport * as bedrockagentcore from 'aws-cdk-lib/aws-bedrockagentcore';\nimport { Grant, IGrantable } from 'aws-cdk-lib/aws-iam';\nimport { MemoryStrategyCommonProps, IMemoryStrategy, MemoryStrategyType, MEMORY_NAME_MIN_LENGTH, MEMORY_NAME_MAX_LENGTH } from '../memory-strategy';\nimport { validateStringFieldLength, throwIfInvalid, validateFieldPattern } from '../validation-helpers';\n\n/******************************************************************************\n *                              CONSTANTS\n *****************************************************************************/\n/**\n * Minimum length for prompt\n * @internal\n */\nconst PROMPT_MIN_LENGTH = 1;\n/**\n * Maximum length for prompt\n * @internal\n */\nconst PROMPT_MAX_LENGTH = 30000;\n\n/**\n * Configuration for overriding model and prompt template\n */\nexport interface OverrideConfig {\n  /**\n   * The model to use for consolidation/extraction\n   */\n  readonly model: IBedrockInvokable;\n  /**\n   * The prompt that will be appended to the system prompt to define\n   * the model's memory consolidation/extraction strategy.\n   * This configuration provides customization to how the model identifies and extracts\n   * relevant information for memory storage. You can use the default user prompt or create a customized one.\n   *\n   * @see https://docs.aws.amazon.com/bedrock-agentcore/latest/devguide/system-prompts.html\n   */\n  readonly appendToPrompt: string;\n}\n\n/**\n * Configuration parameters for a memory strategy that can override\n * existing built-in default prompts/models\n */\nexport interface ManagedStrategyProps extends MemoryStrategyCommonProps {\n  /**\n   * The configuration for the custom extraction.\n   * This configuration provides customization to how the model identifies\n   * and extracts relevant information for memory storage.\n   * @default - No custom extraction\n   */\n  readonly customExtraction?: OverrideConfig;\n  /**\n   * The configuration for the custom consolidation.\n   * This configuration provides customization to how the model identifies\n   * and extracts relevant information for memory storage.\n   * @default - No custom extraction\n   */\n  readonly customConsolidation?: OverrideConfig;\n  /**\n   * The namespaces for the strategy\n   * Represents a namespace for organizing memory data\n   * Use a hierarchical format separated by forward slashes (/)\n   *\n   * Use a hierarchical format separated by forward slashes (/) to organize namespaces logically.\n   * You can include these defined variables:\n   *\n   * - {sessionId} - the user identifier to be created in the CreateEvent API\n   * - {memoryStrategyId} - an identifier for an extraction strategy\n   * - {sessionId} - an identifier for each session\n   *\n   * Example namespace path:\n   * /strategies/{memoryStrategyId}/actions/{actionId}/sessions/{sessionId}\n   *\n   * After memory creation, this namespace might look like:\n   * /actor/actor-3afc5aa8fef9/strategy/summarization-fy5c5fwc7/session/session-qj7tpd1kvr8\n   */\n  readonly namespaces: string[];\n}\n\n/**\n * Managed memory strategy that handles both built-in and override configurations.\n * This strategy can be used for quick setup with built-in defaults or customized\n * with specific models and prompt templates.\n */\nexport class ManagedMemoryStrategy implements IMemoryStrategy {\n  public readonly name: string;\n  public readonly description?: string;\n  /**\n   * The namespaces for the strategy\n   */\n  public readonly namespaces: string[];\n  /**\n   * The configuration for the custom consolidation.\n   */\n  public readonly consolidationOverride?: OverrideConfig;\n  /**\n   * The configuration for the custom extraction.\n   */\n  public readonly extractionOverride?: OverrideConfig;\n  public readonly strategyType: MemoryStrategyType;\n\n  /**\n   * Constructor to create a new managed memory strategy\n   * @param strategyType the strategy type\n   * @param props the properties for the strategy\n   */\n  constructor(strategyType: MemoryStrategyType, props: ManagedStrategyProps) {\n    // ------------------------------------------------------\n    // Set properties and defaults\n    // ------------------------------------------------------\n    this.name = props.name;\n    this.description = props.description;\n    this.namespaces = props.namespaces;\n    this.strategyType = strategyType;\n    this.consolidationOverride = props.customConsolidation;\n    this.extractionOverride = props.customExtraction;\n\n    // ------------------------------------------------------\n    // Validations\n    // ------------------------------------------------------\n    throwIfInvalid(this._validatePrompt, this.consolidationOverride?.appendToPrompt);\n    throwIfInvalid(this._validatePrompt, this.extractionOverride?.appendToPrompt);\n    throwIfInvalid(this._validateMemoryStrategyName, this.name);\n    if (this.namespaces) {\n      throwIfInvalid(this._validateMemoryStrategyNamespaces, this.namespaces);\n    }\n  }\n\n  /**\n   * Renders the network configuration as a CloudFormation property.\n   * @returns The CloudFormation property for the memory strategy.\n   */\n  public render(): bedrockagentcore.CfnMemory.MemoryStrategyProperty {\n    // If no overrides, use built-in strategy format\n    if (!this.consolidationOverride && !this.extractionOverride) {\n      const cfnStrategyMap: Record<MemoryStrategyType, string> = {\n        [MemoryStrategyType.USER_PREFERENCE]: 'userPreferenceMemoryStrategy',\n        [MemoryStrategyType.SEMANTIC]: 'semanticMemoryStrategy',\n        [MemoryStrategyType.SUMMARIZATION]: 'summaryMemoryStrategy',\n        [MemoryStrategyType.CUSTOM]: 'customMemoryStrategy',\n      };\n      const strategyKey = cfnStrategyMap[this.strategyType];\n      return {\n        [strategyKey]: {\n          name: this.name,\n          description: this.description,\n          namespaces: this.namespaces,\n          type: this.strategyType,\n        },\n      };\n    }\n\n    // If overrides are provided, use custom strategy format\n    const cfnStrategyMap: Record<MemoryStrategyType, string> = {\n      [MemoryStrategyType.USER_PREFERENCE]: 'userPreferenceOverride',\n      [MemoryStrategyType.SEMANTIC]: 'semanticOverride',\n      [MemoryStrategyType.SUMMARIZATION]: 'summaryOverride',\n      [MemoryStrategyType.CUSTOM]: '',\n    };\n\n    const strategyKey = cfnStrategyMap[this.strategyType];\n    return {\n      customMemoryStrategy: {\n        name: this.name,\n        description: this.description,\n        namespaces: this.namespaces,\n        type: this.strategyType,\n        configuration: {\n          [strategyKey]: {\n            ...(this.consolidationOverride && {\n              consolidation: {\n                modelId: Arn.split(this.consolidationOverride.model.invokableArn, ArnFormat.SLASH_RESOURCE_NAME)\n                  .resourceName,\n                appendToPrompt: this.consolidationOverride.appendToPrompt,\n              },\n            }),\n            ...(this.extractionOverride && {\n              extraction: {\n                modelId: Arn.split(this.extractionOverride.model.invokableArn, ArnFormat.SLASH_RESOURCE_NAME)\n                  .resourceName,\n                appendToPrompt: this.extractionOverride.appendToPrompt,\n              },\n            }),\n          },\n        },\n      },\n    };\n  }\n\n  /**\n   * Grants the necessary permissions to the role\n   * @param grantee - The grantee to grant permissions to\n   * @returns The Grant object for chaining\n   */\n  public grant(grantee: IGrantable): Grant | undefined {\n    const grant1 = this.consolidationOverride?.model.grantInvoke(grantee);\n    const grant2 = this.extractionOverride?.model.grantInvoke(grantee);\n    return grant1 && grant2 ? grant1.combine(grant2) : grant1 || grant2;\n  }\n\n  // ------------------------------------------------------\n  // VALIDATORS\n  // ------------------------------------------------------\n  /**\n   * Validates the memory strategy name\n   * @param name - The name to validate\n   * @returns Array of validation error messages, empty if valid\n   */\n  private _validateMemoryStrategyName = (name: string, scope?: IConstruct): string[] => {\n    let errors: string[] = [];\n\n    errors.push(...validateStringFieldLength({\n      value: name,\n      fieldName: 'Memory name',\n      minLength: MEMORY_NAME_MIN_LENGTH,\n      maxLength: MEMORY_NAME_MAX_LENGTH,\n    }, scope));\n\n    // Check if name matches the AWS API pattern: [a-zA-Z][a-zA-Z0-9_]{0,47}\n    // Must start with a letter, followed by up to 47 letters, numbers, or underscores\n    const validNamePattern = /^[a-zA-Z][a-zA-Z0-9_]{0,47}$/;\n    errors.push(...validateFieldPattern(name, 'Memory name', validNamePattern, undefined, scope));\n\n    return errors;\n  };\n\n  /**\n   * Validates the prompt\n   * @param prompt - The prompt to validate\n   * @returns Array of validation error messages, empty if valid\n   */\n  private _validatePrompt = (prompt?: string, scope?: IConstruct): string[] => {\n    let errors: string[] = [];\n    if (!prompt) {\n      return errors;\n    }\n    errors.push(...validateStringFieldLength({\n      value: prompt,\n      fieldName: 'Prompt',\n      minLength: PROMPT_MIN_LENGTH,\n      maxLength: PROMPT_MAX_LENGTH,\n    }, scope));\n    return errors;\n  };\n\n  /**\n   * Validates the memory strategy namespaces\n   * @param namespaces - The namespaces to validate\n   * @returns Array of validation error messages, empty if valid\n   */\n  private _validateMemoryStrategyNamespaces = (namespaces: string[]): string[] => {\n    let errors: string[] = [];\n\n    if (namespaces.length === 0) {\n      return errors;\n    }\n\n    for (const namespace of namespaces) {\n      // Only check for template variables in namespace definition\n      if (namespace.includes('{') && !(\n        namespace.includes('{actorId}') ||\n            namespace.includes('{sessionId}') ||\n            namespace.includes('{memoryStrategyId}')\n      )) {\n        errors.push(`Namespace with templates should contain valid variables: ${namespace}`);\n      }\n    }\n\n    return errors;\n  };\n}\n"]}