UNPKG

@aws-solutions-constructs/core

Version:
217 lines 29.8 kB
"use strict"; /** * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"). You may not use this file except in compliance * with the License. A copy of the License is located at * * http://www.apache.org/licenses/LICENSE-2.0 * * or in the 'license' file accompanying this file. This file is distributed on an 'AS IS' BASIS, WITHOUT WARRANTIES * OR CONDITIONS OF ANY KIND, express or implied. See the License for the specific language governing permissions * and limitations under the License. */ Object.defineProperty(exports, "__esModule", { value: true }); exports.buildKendraIndex = buildKendraIndex; exports.AddMultipleKendraDataSources = AddMultipleKendraDataSources; exports.AddKendraDataSource = AddKendraDataSource; exports.normalizeKendraPermissions = normalizeKendraPermissions; /* * The functions found here in the core library are for internal use and can be changed * or removed outside of a major release. We recommend against calling them directly from client code. */ const kendra = require("aws-cdk-lib/aws-kendra"); const iam = require("aws-cdk-lib/aws-iam"); const utils_1 = require("./utils"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const kendra_defaults_1 = require("./kendra-defaults"); /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function buildKendraIndex(scope, id, props) { // Conditional lambda function creation if (props.existingIndexObj) { // The client provided an Index, so we'll do nothing and return it to them return props.existingIndexObj; } else { let indexRoleArn = ""; // If the client provided a role, then don't bother creating a new one that we don't need if (!props.kendraIndexProps?.roleArn) { indexRoleArn = CreateKendraIndexLoggingRole(scope, id); } const defaultIndexProperties = (0, kendra_defaults_1.DefaultKendraIndexProps)(id, indexRoleArn); const consolidatedIndexProperties = (0, utils_1.consolidateProps)(defaultIndexProperties, props.kendraIndexProps); const newIndex = new kendra.CfnIndex(scope, `kendra-index-${id}`, consolidatedIndexProperties); (0, utils_1.addCfnSuppressRules)(newIndex, [{ id: "W80", reason: "We consulted the Kendra TFC and they confirmed the default encryption is sufficient for general use cases" }]); return newIndex; } } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function AddMultipleKendraDataSources(scope, id, kendraIndex, clientDataSourceProps) { const returnDataSources = []; clientDataSourceProps.forEach((props, index) => { returnDataSources.push(AddKendraDataSource(scope, `${id}${index}`, kendraIndex, props)); }); return returnDataSources; } /** * @internal This is an internal core function and should not be called directly by Solutions Constructs clients. */ function AddKendraDataSource(scope, id, index, clientDataSourceProps) { if (clientDataSourceProps.type === 'S3') { return CreateS3DataSource(scope, index, id, clientDataSourceProps); } else { if (clientDataSourceProps.indexId) { throw new Error('Invalid DataSource prop specified - Construct must set the indexId prop'); } return new kendra.CfnDataSource(scope, `kendra-data-source-${id}`, { ...clientDataSourceProps, indexId: index.attrId }); } } function CreateS3DataSource(scope, targetIndex, id, clientProps) { // We go through some hoops here to extract the various inputs, because we need to narrow // the type to remove the union with IResolvable const dataSourceConfig = clientProps.dataSourceConfiguration; if (!dataSourceConfig) { throw new Error('Error - an S3 Kendra DataSource requires an DataSourceConfiguration prop'); } const s3DataSourceConfig = dataSourceConfig.s3Configuration; if (!s3DataSourceConfig) { throw new Error('Error - an S3 Kendra DataSource requires an DataSourceConfiguration.S3Configuration prop'); } // No Bucket name is an error if (!s3DataSourceConfig.bucketName) { throw new Error('Error - an S3 Kendra DataSource requires the DataSourceConfiguration.S3Configuration.bucketName prop'); } // If there's no role, make a role and put it into defaultProps // Put bucket name in default props let defaultProps = { indexId: targetIndex.ref, name: (0, utils_1.generatePhysicalKendraIndexName)('', ['s3-datasource', id]), type: 'S3' }; // Return consolidated default and user props if (!clientProps.roleArn) { const s3CrawlPolicy = new iam.PolicyDocument({ statements: [ new iam.PolicyStatement({ actions: [ "s3:GetObject" ], resources: [ `arn:aws:s3:::${s3DataSourceConfig.bucketName}/*` ], effect: iam.Effect.ALLOW }), new iam.PolicyStatement({ actions: [ "s3:ListBucket" ], resources: [ `arn:aws:s3:::${s3DataSourceConfig.bucketName}` ], effect: iam.Effect.ALLOW }), new iam.PolicyStatement({ effect: iam.Effect.ALLOW, actions: [ "kendra:BatchPutDocument", "kendra:BatchDeleteDocument" ], resources: [ targetIndex.attrArn ] }), ] }); const dataSourceRole = new iam.Role(scope, `data-source-role-${id}`, { assumedBy: new iam.ServicePrincipal('kendra.amazonaws.com'), description: 'Policy for Kendra S3 Data Source', inlinePolicies: { s3CrawlPolicy, }, }); defaultProps = (0, utils_1.overrideProps)(defaultProps, { roleArn: dataSourceRole.roleArn }); (0, utils_1.addCfnGuardSuppressRules)(dataSourceRole, ["IAM_NO_INLINE_POLICY_CHECK"]); } const consolidatedProps = (0, utils_1.consolidateProps)(defaultProps, clientProps); return new kendra.CfnDataSource(scope, `data-source-${id}`, consolidatedProps); } function CreateKendraIndexLoggingRole(scope, id) { const allowKendraToLogPolicy = new iam.PolicyDocument({ statements: [ new iam.PolicyStatement({ resources: ['*'], actions: [ "cloudwatch:PutMetricData" ], effect: iam.Effect.ALLOW, conditions: { StringEquals: { "cloudwatch:namespace": "AWS/Kendra" } } }), new iam.PolicyStatement({ resources: [`arn:aws:logs:${aws_cdk_lib_1.Aws.REGION}:${aws_cdk_lib_1.Aws.ACCOUNT_ID}:log-group:/aws/kendra/*`], actions: [ "logs:CreateLogGroup" ], effect: iam.Effect.ALLOW, }), new iam.PolicyStatement({ resources: [`arn:${aws_cdk_lib_1.Aws.PARTITION}:logs:${aws_cdk_lib_1.Aws.REGION}:${aws_cdk_lib_1.Aws.ACCOUNT_ID}:log-group:/aws/kendra/*`], actions: [ "logs:DescribeLogGroups" ], effect: iam.Effect.ALLOW, }), new iam.PolicyStatement({ resources: [`arn:${aws_cdk_lib_1.Aws.PARTITION}:logs:${aws_cdk_lib_1.Aws.REGION}:${aws_cdk_lib_1.Aws.ACCOUNT_ID}:log-group:/aws/kendra/*:log-stream:*`], actions: [ 'logs:CreateLogStream', 'logs:PutLogEvents', 'logs:DescribeLogStream', ], effect: iam.Effect.ALLOW, }), ], }); const indexRole = new iam.Role(scope, `kendra-index-role-${id}`, { assumedBy: new iam.ServicePrincipal('kendra.amazonaws.com'), description: 'Allow Kendra index to write CloudWatch Logs', inlinePolicies: { AllowLogging: allowKendraToLogPolicy, }, }); (0, utils_1.addCfnSuppressRules)(indexRole, [{ id: "W11", reason: "PutMetricData does not allow resource specification, " + "scope is narrowed by the namespace condition. " + "https://docs.aws.amazon.com/service-authorization/latest/reference/list_amazoncloudwatch.html" }]); (0, utils_1.addCfnGuardSuppressRules)(indexRole, ["IAM_NO_INLINE_POLICY_CHECK"]); return indexRole.roleArn; } // @summary Confirm each entry is a correct value, uppercase each entry function normalizeKendraPermissions(rawPermissions) { const validPermissions = ["READ", "SUBMITFEEDBACK", "WRITE"]; const result = rawPermissions.map((s) => { const upperCaseValue = s.toUpperCase(); if (!validPermissions.includes(upperCaseValue)) { throw new Error(`Invalid indexPermission value - valid values are "READ", "SUBMITFEEDBACK" and "WRITE"`); } return upperCaseValue; }); return result; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kendra-helper.js","sourceRoot":"","sources":["kendra-helper.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\n/*\n *  The functions found here in the core library are for internal use and can be changed\n *  or removed outside of a major release. We recommend against calling them directly from client code.\n */\n\nimport * as kendra from 'aws-cdk-lib/aws-kendra';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport { addCfnGuardSuppressRules, addCfnSuppressRules, consolidateProps, generatePhysicalKendraIndexName, overrideProps } from \"./utils\";\nimport { Aws } from 'aws-cdk-lib';\n\n// Note: To ensure CDKv2 compatibility, keep the import statement for Construct separate\nimport { Construct } from 'constructs';\nimport { DefaultKendraIndexProps } from './kendra-defaults';\n\nexport interface BuildKendraIndexProps {\n  readonly kendraIndexProps?: kendra.CfnIndexProps | any;\n  /**\n   * Existing instance of Kendra Index object, Providing both this and kendraIndexProps will cause an error.\n   *\n   * @default - None\n   */\n  readonly existingIndexObj?: kendra.CfnIndex;\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function buildKendraIndex(scope: Construct, id: string, props: BuildKendraIndexProps): kendra.CfnIndex {\n  // Conditional lambda function creation\n  if (props.existingIndexObj) {\n    // The client provided an Index, so we'll do nothing and return it to them\n    return props.existingIndexObj;\n  } else {\n\n    let indexRoleArn: string = \"\";\n\n    // If the client provided a role, then don't bother creating a new one that we don't need\n    if (!props.kendraIndexProps?.roleArn) {\n      indexRoleArn = CreateKendraIndexLoggingRole(scope, id);\n    }\n    const defaultIndexProperties = DefaultKendraIndexProps(id, indexRoleArn);\n\n    const consolidatedIndexProperties = consolidateProps(defaultIndexProperties, props.kendraIndexProps);\n    const newIndex = new kendra.CfnIndex(scope, `kendra-index-${id}`, consolidatedIndexProperties);\n    addCfnSuppressRules(newIndex, [{\n      id: \"W80\",\n      reason: \"We consulted the Kendra TFC and they confirmed the default encryption is sufficient for general use cases\"\n    }]);\n\n    return newIndex;\n  }\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function AddMultipleKendraDataSources(scope: Construct,\n  id: string,\n  kendraIndex: kendra.CfnIndex,\n  clientDataSourceProps: Partial<kendra.CfnDataSourceProps>[]): kendra.CfnDataSource[] {\n\n  const returnDataSources: kendra.CfnDataSource[] = [];\n  clientDataSourceProps.forEach((props, index) => {\n    returnDataSources.push(AddKendraDataSource(scope, `${id}${index}`, kendraIndex, props));\n  });\n  return returnDataSources;\n}\n\n/**\n * @internal This is an internal core function and should not be called directly by Solutions Constructs clients.\n */\nexport function AddKendraDataSource(scope: Construct,\n  id: string, index: kendra.CfnIndex,\n  clientDataSourceProps: kendra.CfnDataSourceProps | any): kendra.CfnDataSource {\n\n  if  (clientDataSourceProps.type === 'S3') {\n    return CreateS3DataSource(scope, index, id, clientDataSourceProps);\n  } else {\n    if (clientDataSourceProps.indexId) {\n      throw new Error('Invalid DataSource prop specified - Construct must set the indexId prop');\n    }\n    return new kendra.CfnDataSource(scope, `kendra-data-source-${id}`, {\n      ...clientDataSourceProps,\n      indexId: index.attrId\n    });\n  }\n}\n\nfunction CreateS3DataSource(scope: Construct,\n  targetIndex: kendra.CfnIndex,\n  id: string,\n  clientProps: Partial<kendra.CfnDataSourceProps>): kendra.CfnDataSource {\n\n  // We go through some hoops here to extract the various inputs, because we need to narrow\n  // the type to remove the union with IResolvable\n  const dataSourceConfig = clientProps.dataSourceConfiguration as kendra.CfnDataSource.DataSourceConfigurationProperty;\n  if (!dataSourceConfig) {\n    throw new Error('Error - an S3 Kendra DataSource requires an DataSourceConfiguration prop');\n  }\n\n  const s3DataSourceConfig = dataSourceConfig.s3Configuration as kendra.CfnDataSource.S3DataSourceConfigurationProperty;\n\n  if (!s3DataSourceConfig) {\n    throw new Error('Error - an S3 Kendra DataSource requires an DataSourceConfiguration.S3Configuration prop');\n  }\n\n  // No Bucket name is an error\n  if (!s3DataSourceConfig.bucketName) {\n    throw new Error('Error - an S3 Kendra DataSource requires the DataSourceConfiguration.S3Configuration.bucketName prop');\n  }\n\n  // If there's no role, make a role and put it into defaultProps\n  // Put bucket name in default props\n  let defaultProps: kendra.CfnDataSourceProps = {\n    indexId: targetIndex.ref,\n    name: generatePhysicalKendraIndexName('', ['s3-datasource', id]),\n    type: 'S3'\n  };\n\n  // Return consolidated default and user props\n  if (!clientProps.roleArn) {\n    const s3CrawlPolicy = new iam.PolicyDocument({\n      statements: [\n        new iam.PolicyStatement({\n          actions: [\n            \"s3:GetObject\"\n          ],\n          resources: [\n            `arn:aws:s3:::${s3DataSourceConfig.bucketName}/*`\n          ],\n          effect: iam.Effect.ALLOW\n        }),\n        new iam.PolicyStatement({\n          actions: [\n            \"s3:ListBucket\"\n          ],\n          resources: [\n            `arn:aws:s3:::${s3DataSourceConfig.bucketName}`\n          ],\n          effect: iam.Effect.ALLOW\n        }),\n        new iam.PolicyStatement({\n          effect: iam.Effect.ALLOW,\n          actions: [\n            \"kendra:BatchPutDocument\",\n            \"kendra:BatchDeleteDocument\"\n          ],\n          resources: [\n            targetIndex.attrArn\n          ]\n        }),\n      ]\n    });\n\n    const dataSourceRole: iam.Role = new iam.Role(scope, `data-source-role-${id}`, {\n      assumedBy: new iam.ServicePrincipal('kendra.amazonaws.com'),\n      description: 'Policy for Kendra S3 Data Source',\n      inlinePolicies: {\n        s3CrawlPolicy,\n      },\n    });\n    defaultProps = overrideProps(defaultProps, { roleArn: dataSourceRole.roleArn });\n    addCfnGuardSuppressRules(dataSourceRole, [\"IAM_NO_INLINE_POLICY_CHECK\"]);\n  }\n\n  const consolidatedProps: kendra.CfnDataSourceProps = consolidateProps(defaultProps, clientProps);\n\n  return new kendra.CfnDataSource(scope, `data-source-${id}`, consolidatedProps);\n\n}\n\nfunction CreateKendraIndexLoggingRole(scope: Construct, id: string): string {\n  const allowKendraToLogPolicy = new iam.PolicyDocument({\n    statements: [\n      new iam.PolicyStatement({\n        resources: ['*'],\n        actions: [\n          \"cloudwatch:PutMetricData\"\n        ],\n        effect: iam.Effect.ALLOW,\n        conditions: {\n          StringEquals: {\n            \"cloudwatch:namespace\": \"AWS/Kendra\"\n          }\n        }\n      }),\n      new iam.PolicyStatement({\n        resources: [`arn:aws:logs:${Aws.REGION}:${Aws.ACCOUNT_ID}:log-group:/aws/kendra/*`],\n        actions: [\n          \"logs:CreateLogGroup\"\n        ],\n        effect: iam.Effect.ALLOW,\n      }),\n      new iam.PolicyStatement({\n        resources: [`arn:${Aws.PARTITION}:logs:${Aws.REGION}:${Aws.ACCOUNT_ID}:log-group:/aws/kendra/*`],\n        actions: [\n          \"logs:DescribeLogGroups\"\n        ],\n        effect: iam.Effect.ALLOW,\n      }),\n      new iam.PolicyStatement({\n        resources: [`arn:${Aws.PARTITION}:logs:${Aws.REGION}:${Aws.ACCOUNT_ID}:log-group:/aws/kendra/*:log-stream:*`],\n        actions: [\n          'logs:CreateLogStream',\n          'logs:PutLogEvents',\n          'logs:DescribeLogStream',\n        ],\n        effect: iam.Effect.ALLOW,\n      }),\n    ],\n  });\n\n  const indexRole: iam.Role = new iam.Role(scope, `kendra-index-role-${id}`, {\n    assumedBy: new iam.ServicePrincipal('kendra.amazonaws.com'),\n    description: 'Allow Kendra index to write CloudWatch Logs',\n    inlinePolicies: {\n      AllowLogging: allowKendraToLogPolicy,\n    },\n  });\n  addCfnSuppressRules(indexRole, [{\n    id: \"W11\",\n    reason: \"PutMetricData does not allow resource specification, \" +\n      \"scope is narrowed by the namespace condition. \" +\n      \"https://docs.aws.amazon.com/service-authorization/latest/reference/list_amazoncloudwatch.html\"\n  }]);\n  addCfnGuardSuppressRules(indexRole, [\"IAM_NO_INLINE_POLICY_CHECK\"]);\n\n  return indexRole.roleArn;\n}\n\n// @summary Confirm each entry is a correct value, uppercase each entry\nexport function normalizeKendraPermissions(rawPermissions: string[]): string[] {\n  const validPermissions = [\"READ\", \"SUBMITFEEDBACK\", \"WRITE\"];\n\n  const result = rawPermissions.map<string>((s) => {\n    const upperCaseValue = s.toUpperCase();\n    if (!validPermissions.includes(upperCaseValue)) {\n      throw new Error(`Invalid indexPermission value - valid values are \"READ\", \"SUBMITFEEDBACK\" and \"WRITE\"`);\n    }\n    return upperCaseValue;\n  });\n  return result;\n}\n"]}