@aws-solutions-constructs/core
Version:
Core CDK Construct for patterns library
217 lines • 29.8 kB
JavaScript
/**
* 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"]}
;