@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
154 lines • 26.4 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.SonarCodeScanner = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0 */
const pdk_nag_1 = require("../../pdk-nag");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const aws_codebuild_1 = require("aws-cdk-lib/aws-codebuild");
const aws_events_1 = require("aws-cdk-lib/aws-events");
const aws_events_targets_1 = require("aws-cdk-lib/aws-events-targets");
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
const aws_secretsmanager_1 = require("aws-cdk-lib/aws-secretsmanager");
const cdk_nag_1 = require("cdk-nag");
const constructs_1 = require("constructs");
const sonarqube_commands_1 = require("./sonarqube-commands");
const unpackSourceAndArtifacts = (includeGlobsForScan) => [
'export BUILT_ARTIFACT_URI=`aws codebuild batch-get-builds --ids $SYNTH_BUILD_ID | jq -r \'.builds[0].secondaryArtifacts[] | select(.artifactIdentifier == "Synth__") | .location\' | awk \'{sub("arn:aws:s3:::","s3://")}1\' $1`',
"export SYNTH_SOURCE_URI=`aws codebuild batch-get-builds --ids $SYNTH_BUILD_ID | jq -r '.builds[0].sourceVersion' | awk '{sub(\"arn:aws:s3:::\",\"s3://\")}1' $1`",
"aws s3 cp $SYNTH_SOURCE_URI source.zip",
"aws s3 cp $BUILT_ARTIFACT_URI built.zip",
"unzip source.zip -d src",
"unzip built.zip -d built",
"rm source.zip built.zip",
`rsync -a built/* src --include="*/" ${includeGlobsForScan
? includeGlobsForScan.map((g) => `--include ${g}`).join(" ")
: ""} --include="**/coverage/**" --include="**/cdk.out/**" --exclude="**/node_modules/**/*" --exclude="**/.env/**" --exclude="*" --prune-empty-dirs`,
];
const owaspScan = () => `npx owasp-dependency-check --format HTML --out src/reports --exclude '**/.git/**/*' --scan src --enableExperimental --bin /tmp/dep-check --disableRetireJS`;
const cfnNagScan = (cdkOutDir, cfnNagIgnorePath) => cdkOutDir
? `cfn_nag ${cfnNagIgnorePath ? `--deny-list-path=${cfnNagIgnorePath}` : ""} built/${cdkOutDir}/**/*.template.json --output-format=json > src/reports/cfn-nag-report.json`
: 'echo "skipping cfn_nag as no cdkOutDir was specified.';
class SonarCodeScanner extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
const sonarQubeToken = new aws_secretsmanager_1.Secret(this, "SonarQubeToken");
const synthBuildProject = aws_codebuild_1.Project.fromProjectArn(this, "SynthBuildProject", props.synthBuildArn);
const validationProject = new aws_codebuild_1.Project(this, "ValidationProject", {
environment: {
buildImage: aws_codebuild_1.LinuxBuildImage.STANDARD_5_0,
},
environmentVariables: {
SONARQUBE_TOKEN: {
type: aws_codebuild_1.BuildEnvironmentVariableType.SECRETS_MANAGER,
value: sonarQubeToken.secretArn,
},
SONARQUBE_ENDPOINT: {
type: aws_codebuild_1.BuildEnvironmentVariableType.PLAINTEXT,
value: props.sonarqubeEndpoint,
},
PROJECT_NAME: {
type: aws_codebuild_1.BuildEnvironmentVariableType.PLAINTEXT,
value: props.sonarqubeProjectName,
},
},
buildSpec: aws_codebuild_1.BuildSpec.fromObject({
version: "0.2",
env: {
shell: "bash",
},
phases: {
install: {
commands: ["npm install -g aws-cdk", "gem install cfn-nag"],
},
build: {
commands: [
"export RESOLVED_SOURCE_VERSION=`aws codebuild batch-get-builds --ids $SYNTH_BUILD_ID | jq -r '.builds[0].resolvedSourceVersion'`",
...unpackSourceAndArtifacts(props.includeGlobsForScan),
...(0, sonarqube_commands_1.createSonarqubeProject)(props),
"mkdir -p src/reports",
owaspScan(),
cfnNagScan(props.cdkOutDir, props.cfnNagIgnorePath),
"cd src",
(0, sonarqube_commands_1.sonarqubeScanner)(props.excludeGlobsForScan),
...(0, sonarqube_commands_1.generateSonarqubeReports)(),
...(props.preArchiveCommands || []),
],
},
},
}),
});
validationProject.addToRolePolicy(new aws_iam_1.PolicyStatement({
actions: ["codebuild:BatchGetBuilds"],
effect: aws_iam_1.Effect.ALLOW,
resources: [synthBuildProject.projectArn],
}));
validationProject.addToRolePolicy(new aws_iam_1.PolicyStatement({
actions: ["s3:GetObject*"],
effect: aws_iam_1.Effect.ALLOW,
resources: [props.artifactBucketArn, `${props.artifactBucketArn}/**`],
}));
props.artifactBucketKeyArn &&
validationProject.addToRolePolicy(new aws_iam_1.PolicyStatement({
actions: ["kms:Decrypt", "kms:DescribeKey"],
effect: aws_iam_1.Effect.ALLOW,
resources: [props.artifactBucketKeyArn],
}));
synthBuildProject.onBuildSucceeded("OnSynthSuccess", {
target: new aws_events_targets_1.CodeBuildProject(validationProject, {
event: aws_events_1.RuleTargetInput.fromObject({
environmentVariablesOverride: [
{
name: "SYNTH_BUILD_ID",
type: "PLAINTEXT",
value: aws_events_1.EventField.fromPath("$.detail.build-id"),
},
],
}),
}),
});
new aws_cdk_lib_1.CfnOutput(this, "SonarqubeSecretArn", {
value: sonarQubeToken.secretArn,
});
[
"AwsSolutions-SMG4",
"AwsPrototyping-SecretsManagerRotationEnabled",
].forEach((RuleId) => {
cdk_nag_1.NagSuppressions.addResourceSuppressions(sonarQubeToken, [
{
id: RuleId,
reason: "Key rotation is not possible as a user token needs to be generated from Sonarqube",
},
]);
});
const stack = aws_cdk_lib_1.Stack.of(this);
["AwsSolutions-IAM5", "AwsPrototyping-IAMNoWildcardPermissions"].forEach((RuleId) => {
cdk_nag_1.NagSuppressions.addResourceSuppressions(validationProject.role, [
{
id: RuleId,
reason: "Validation CodeBuild project requires access to the ArtifactsBucket and ability to create logs.",
appliesTo: [
{
regex: `/^Resource::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:logs:${pdk_nag_1.PDKNag.getStackRegionRegex(stack)}:${pdk_nag_1.PDKNag.getStackAccountRegex(stack)}:log-group:/aws/codebuild/<.*SonarCodeScannerValidationProject.*>:\\*$/g`,
},
{
regex: `/^Resource::arn:${pdk_nag_1.PDKNag.getStackPartitionRegex(stack)}:codebuild:${pdk_nag_1.PDKNag.getStackRegionRegex(stack)}:${pdk_nag_1.PDKNag.getStackAccountRegex(stack)}:report-group/<.*SonarCodeScannerValidationProject.*>-\\*$/g`,
},
{
regex: `/^Action::s3:GetObject\\*$/g`,
},
{
regex: "/^Resource::<ArtifactsBucket.*.Arn>/\\*\\*$/g",
},
],
},
], true);
});
}
}
exports.SonarCodeScanner = SonarCodeScanner;
_a = JSII_RTTI_SYMBOL_1;
SonarCodeScanner[_a] = { fqn: "@aws/pdk.pipeline.SonarCodeScanner", version: "0.26.14" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sonar-code-scanner.js","sourceRoot":"","sources":["sonar-code-scanner.ts"],"names":[],"mappings":"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","sourcesContent":["/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.\nSPDX-License-Identifier: Apache-2.0 */\nimport { PDKNag } from \"@aws/pdk-nag\";\nimport { CfnOutput, Stack } from \"aws-cdk-lib\";\nimport {\n  BuildEnvironmentVariableType,\n  BuildSpec,\n  LinuxBuildImage,\n  Project,\n} from \"aws-cdk-lib/aws-codebuild\";\nimport { EventField, RuleTargetInput } from \"aws-cdk-lib/aws-events\";\nimport { CodeBuildProject } from \"aws-cdk-lib/aws-events-targets\";\nimport { Effect, PolicyStatement } from \"aws-cdk-lib/aws-iam\";\nimport { Secret } from \"aws-cdk-lib/aws-secretsmanager\";\nimport { NagSuppressions } from \"cdk-nag\";\nimport { Construct } from \"constructs\";\nimport {\n  createSonarqubeProject,\n  generateSonarqubeReports,\n  sonarqubeScanner,\n} from \"./sonarqube-commands\";\n\nexport interface SonarCodeScannerConfig {\n  /**\n   * path to a file containing the cfn nag suppression rules.\n   */\n  readonly cfnNagIgnorePath?: string;\n\n  /**\n   * directory containing the synthesized cdk resources.\n   */\n  readonly cdkOutDir?: string;\n\n  /**\n   * glob patterns to exclude from sonar scan.\n   */\n  readonly excludeGlobsForScan?: string[];\n\n  /**\n   * glob patterns to include from sonar scan.\n   */\n  readonly includeGlobsForScan?: string[];\n\n  /**\n   * endpoint of the sonarqube instance i.e: https://<your-sonarqube-endpoint>.\n   *\n   * Note: Ensure a trailing '/' is not included.\n   */\n  readonly sonarqubeEndpoint: string;\n\n  /**\n   * Default profile/gate name i.e: your org profile.\n   *\n   * Note: These need to be set up in Sonarqube manually.\n   */\n  readonly sonarqubeDefaultProfileOrGateName: string;\n\n  /**\n   * Specific profile/gate name i.e: language specific.\n   *\n   * Note: These need to be set up in Sonarqube manually.\n   */\n  readonly sonarqubeSpecificProfileOrGateName?: string;\n\n  /**\n   * Group name in Sonarqube with access to administer this project.\n   */\n  readonly sonarqubeAuthorizedGroup: string;\n\n  /**\n   * Name of the project to create in Sonarqube.\n   */\n  readonly sonarqubeProjectName: string;\n\n  /**\n   * Tags to associate with this project.\n   */\n  readonly sonarqubeTags?: string[];\n\n  /**\n   * Hook which allows custom commands to be executed before the process commences the archival process.\n   */\n  readonly preArchiveCommands?: string[];\n}\n\n/**\n * SonarCodeScanners properties.\n */\nexport interface SonarCodeScannerProps extends SonarCodeScannerConfig {\n  /**\n   * ARN for the CodeBuild task responsible for executing the synth command.\n   */\n  readonly synthBuildArn: string;\n\n  /**\n   * S3 bucket ARN containing the built artifacts from the synth build.\n   */\n  readonly artifactBucketArn: string;\n\n  /**\n   * Artifact bucket key ARN used to encrypt the artifacts.\n   */\n  readonly artifactBucketKeyArn?: string;\n}\n\nconst unpackSourceAndArtifacts = (includeGlobsForScan?: string[]) => [\n  'export BUILT_ARTIFACT_URI=`aws codebuild batch-get-builds --ids $SYNTH_BUILD_ID | jq -r \\'.builds[0].secondaryArtifacts[] | select(.artifactIdentifier == \"Synth__\") | .location\\' | awk \\'{sub(\"arn:aws:s3:::\",\"s3://\")}1\\' $1`',\n  \"export SYNTH_SOURCE_URI=`aws codebuild batch-get-builds --ids $SYNTH_BUILD_ID | jq -r '.builds[0].sourceVersion' | awk '{sub(\\\"arn:aws:s3:::\\\",\\\"s3://\\\")}1' $1`\",\n  \"aws s3 cp $SYNTH_SOURCE_URI source.zip\",\n  \"aws s3 cp $BUILT_ARTIFACT_URI built.zip\",\n  \"unzip source.zip -d src\",\n  \"unzip built.zip -d built\",\n  \"rm source.zip built.zip\",\n  `rsync -a built/* src --include=\"*/\" ${\n    includeGlobsForScan\n      ? includeGlobsForScan.map((g) => `--include ${g}`).join(\" \")\n      : \"\"\n  } --include=\"**/coverage/**\" --include=\"**/cdk.out/**\" --exclude=\"**/node_modules/**/*\" --exclude=\"**/.env/**\" --exclude=\"*\" --prune-empty-dirs`,\n];\n\nconst owaspScan = () =>\n  `npx owasp-dependency-check --format HTML --out src/reports --exclude '**/.git/**/*' --scan src --enableExperimental --bin /tmp/dep-check --disableRetireJS`;\n\nconst cfnNagScan = (cdkOutDir?: string, cfnNagIgnorePath?: string) =>\n  cdkOutDir\n    ? `cfn_nag ${\n        cfnNagIgnorePath ? `--deny-list-path=${cfnNagIgnorePath}` : \"\"\n      } built/${cdkOutDir}/**/*.template.json --output-format=json > src/reports/cfn-nag-report.json`\n    : 'echo \"skipping cfn_nag as no cdkOutDir was specified.';\n\nexport class SonarCodeScanner extends Construct {\n  constructor(scope: Construct, id: string, props: SonarCodeScannerProps) {\n    super(scope, id);\n\n    const sonarQubeToken = new Secret(this, \"SonarQubeToken\");\n\n    const synthBuildProject = Project.fromProjectArn(\n      this,\n      \"SynthBuildProject\",\n      props.synthBuildArn\n    );\n\n    const validationProject = new Project(this, \"ValidationProject\", {\n      environment: {\n        buildImage: LinuxBuildImage.STANDARD_5_0,\n      },\n      environmentVariables: {\n        SONARQUBE_TOKEN: {\n          type: BuildEnvironmentVariableType.SECRETS_MANAGER,\n          value: sonarQubeToken.secretArn,\n        },\n        SONARQUBE_ENDPOINT: {\n          type: BuildEnvironmentVariableType.PLAINTEXT,\n          value: props.sonarqubeEndpoint,\n        },\n        PROJECT_NAME: {\n          type: BuildEnvironmentVariableType.PLAINTEXT,\n          value: props.sonarqubeProjectName,\n        },\n      },\n      buildSpec: BuildSpec.fromObject({\n        version: \"0.2\",\n        env: {\n          shell: \"bash\",\n        },\n        phases: {\n          install: {\n            commands: [\"npm install -g aws-cdk\", \"gem install cfn-nag\"],\n          },\n          build: {\n            commands: [\n              \"export RESOLVED_SOURCE_VERSION=`aws codebuild batch-get-builds --ids $SYNTH_BUILD_ID | jq -r '.builds[0].resolvedSourceVersion'`\",\n              ...unpackSourceAndArtifacts(props.includeGlobsForScan),\n              ...createSonarqubeProject(props),\n              \"mkdir -p src/reports\",\n              owaspScan(),\n              cfnNagScan(props.cdkOutDir, props.cfnNagIgnorePath),\n              \"cd src\",\n              sonarqubeScanner(props.excludeGlobsForScan),\n              ...generateSonarqubeReports(),\n              ...(props.preArchiveCommands || []),\n            ],\n          },\n        },\n      }),\n    });\n\n    validationProject.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\"codebuild:BatchGetBuilds\"],\n        effect: Effect.ALLOW,\n        resources: [synthBuildProject.projectArn],\n      })\n    );\n\n    validationProject.addToRolePolicy(\n      new PolicyStatement({\n        actions: [\"s3:GetObject*\"],\n        effect: Effect.ALLOW,\n        resources: [props.artifactBucketArn, `${props.artifactBucketArn}/**`],\n      })\n    );\n\n    props.artifactBucketKeyArn &&\n      validationProject.addToRolePolicy(\n        new PolicyStatement({\n          actions: [\"kms:Decrypt\", \"kms:DescribeKey\"],\n          effect: Effect.ALLOW,\n          resources: [props.artifactBucketKeyArn],\n        })\n      );\n\n    synthBuildProject.onBuildSucceeded(\"OnSynthSuccess\", {\n      target: new CodeBuildProject(validationProject, {\n        event: RuleTargetInput.fromObject({\n          environmentVariablesOverride: [\n            {\n              name: \"SYNTH_BUILD_ID\",\n              type: \"PLAINTEXT\",\n              value: EventField.fromPath(\"$.detail.build-id\"),\n            },\n          ],\n        }),\n      }),\n    });\n\n    new CfnOutput(this, \"SonarqubeSecretArn\", {\n      value: sonarQubeToken.secretArn,\n    });\n\n    [\n      \"AwsSolutions-SMG4\",\n      \"AwsPrototyping-SecretsManagerRotationEnabled\",\n    ].forEach((RuleId) => {\n      NagSuppressions.addResourceSuppressions(sonarQubeToken, [\n        {\n          id: RuleId,\n          reason:\n            \"Key rotation is not possible as a user token needs to be generated from Sonarqube\",\n        },\n      ]);\n    });\n\n    const stack = Stack.of(this);\n\n    [\"AwsSolutions-IAM5\", \"AwsPrototyping-IAMNoWildcardPermissions\"].forEach(\n      (RuleId) => {\n        NagSuppressions.addResourceSuppressions(\n          validationProject.role!,\n          [\n            {\n              id: RuleId,\n              reason:\n                \"Validation CodeBuild project requires access to the ArtifactsBucket and ability to create logs.\",\n              appliesTo: [\n                {\n                  regex: `/^Resource::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:logs:${PDKNag.getStackRegionRegex(\n                    stack\n                  )}:${PDKNag.getStackAccountRegex(\n                    stack\n                  )}:log-group:/aws/codebuild/<.*SonarCodeScannerValidationProject.*>:\\\\*$/g`,\n                },\n                {\n                  regex: `/^Resource::arn:${PDKNag.getStackPartitionRegex(\n                    stack\n                  )}:codebuild:${PDKNag.getStackRegionRegex(\n                    stack\n                  )}:${PDKNag.getStackAccountRegex(\n                    stack\n                  )}:report-group/<.*SonarCodeScannerValidationProject.*>-\\\\*$/g`,\n                },\n                {\n                  regex: `/^Action::s3:GetObject\\\\*$/g`,\n                },\n                {\n                  regex: \"/^Resource::<ArtifactsBucket.*.Arn>/\\\\*\\\\*$/g\",\n                },\n              ],\n            },\n          ],\n          true\n        );\n      }\n    );\n  }\n}\n"]}