UNPKG

@aws-cdk/aws-ec2

Version:

The CDK Construct Library for AWS::EC2

113 lines 18.1 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.BastionHostLinux = void 0; const jsiiDeprecationWarnings = require("../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const aws_iam_1 = require("@aws-cdk/aws-iam"); const core_1 = require("@aws-cdk/core"); const _1 = require("."); const instance_1 = require("./instance"); const machine_image_1 = require("./machine-image"); const port_1 = require("./port"); /** * This creates a linux bastion host you can use to connect to other instances or services in your VPC. * The recommended way to connect to the bastion host is by using AWS Systems Manager Session Manager. * * The operating system is Amazon Linux 2 with the latest SSM agent installed * * You can also configure this bastion host to allow connections via SSH * * * @resource AWS::EC2::Instance */ class BastionHostLinux extends core_1.Resource { constructor(scope, id, props) { super(scope, id); try { jsiiDeprecationWarnings._aws_cdk_aws_ec2_BastionHostLinuxProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, BastionHostLinux); } throw error; } this.stack = core_1.Stack.of(scope); const instanceType = props.instanceType ?? _1.InstanceType.of(_1.InstanceClass.T3, _1.InstanceSize.NANO); this.instance = new instance_1.Instance(this, 'Resource', { vpc: props.vpc, availabilityZone: props.availabilityZone, securityGroup: props.securityGroup, instanceName: props.instanceName ?? 'BastionHost', instanceType, machineImage: props.machineImage ?? machine_image_1.MachineImage.latestAmazonLinux({ generation: _1.AmazonLinuxGeneration.AMAZON_LINUX_2, cpuType: this.toAmazonLinuxCpuType(instanceType.architecture), }), vpcSubnets: props.subnetSelection ?? {}, blockDevices: props.blockDevices ?? undefined, init: props.init, initOptions: props.initOptions, requireImdsv2: props.requireImdsv2 ?? false, }); this.instance.addToRolePolicy(new aws_iam_1.PolicyStatement({ actions: [ 'ssmmessages:*', 'ssm:UpdateInstanceInformation', 'ec2messages:*', ], resources: ['*'], })); this.connections = this.instance.connections; this.role = this.instance.role; this.grantPrincipal = this.instance.role; this.instanceId = this.instance.instanceId; this.instancePrivateIp = this.instance.instancePrivateIp; this.instanceAvailabilityZone = this.instance.instanceAvailabilityZone; this.instancePrivateDnsName = this.instance.instancePrivateDnsName; this.instancePublicIp = this.instance.instancePublicIp; this.instancePublicDnsName = this.instance.instancePublicDnsName; new core_1.CfnOutput(this, 'BastionHostId', { description: 'Instance ID of the bastion host. Use this to connect via SSM Session Manager', value: this.instanceId, }); } /** * Returns the AmazonLinuxCpuType corresponding to the given instance architecture * @param architecture the instance architecture value to convert */ toAmazonLinuxCpuType(architecture) { if (architecture === _1.InstanceArchitecture.ARM_64) { return machine_image_1.AmazonLinuxCpuType.ARM_64; } else if (architecture === _1.InstanceArchitecture.X86_64) { return machine_image_1.AmazonLinuxCpuType.X86_64; } throw new Error(`Unsupported instance architecture '${architecture}'`); } /** * Allow SSH access from the given peer or peers * * Necessary if you want to connect to the instance using ssh. If not * called, you should use SSM Session Manager to connect to the instance. */ allowSshAccessFrom(...peer) { try { jsiiDeprecationWarnings._aws_cdk_aws_ec2_IPeer(peer); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.allowSshAccessFrom); } throw error; } peer.forEach(p => { this.connections.allowFrom(p, port_1.Port.tcp(22), 'SSH access'); }); } } exports.BastionHostLinux = BastionHostLinux; _a = JSII_RTTI_SYMBOL_1; BastionHostLinux[_a] = { fqn: "@aws-cdk/aws-ec2.BastionHostLinux", version: "1.204.0" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bastion-host.js","sourceRoot":"","sources":["bastion-host.ts"],"names":[],"mappings":"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","sourcesContent":["import { IPrincipal, IRole, PolicyStatement } from '@aws-cdk/aws-iam';\nimport { CfnOutput, Resource, Stack } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { AmazonLinuxGeneration, InstanceArchitecture, InstanceClass, InstanceSize, InstanceType } from '.';\nimport { CloudFormationInit } from './cfn-init';\nimport { Connections } from './connections';\nimport { ApplyCloudFormationInitOptions, IInstance, Instance } from './instance';\nimport { AmazonLinuxCpuType, IMachineImage, MachineImage } from './machine-image';\nimport { IPeer } from './peer';\nimport { Port } from './port';\nimport { ISecurityGroup } from './security-group';\nimport { BlockDevice } from './volume';\nimport { IVpc, SubnetSelection } from './vpc';\n\n/**\n * Properties of the bastion host\n *\n *\n */\nexport interface BastionHostLinuxProps {\n\n  /**\n   * In which AZ to place the instance within the VPC\n   *\n   * @default - Random zone.\n   */\n  readonly availabilityZone?: string;\n\n  /**\n   * VPC to launch the instance in.\n   */\n  readonly vpc: IVpc;\n\n  /**\n   * The name of the instance\n   *\n   * @default 'BastionHost'\n   */\n  readonly instanceName?: string;\n\n  /**\n   * Select the subnets to run the bastion host in.\n   * Set this to PUBLIC if you need to connect to this instance via the internet and cannot use SSM.\n   * You have to allow port 22 manually by using the connections field\n   *\n   * @default - private subnets of the supplied VPC\n   */\n  readonly subnetSelection?: SubnetSelection;\n\n  /**\n   * Security Group to assign to this instance\n   *\n   * @default - create new security group with no inbound and all outbound traffic allowed\n   */\n  readonly securityGroup?: ISecurityGroup;\n\n  /**\n   * Type of instance to launch\n   * @default 't3.nano'\n   */\n  readonly instanceType?: InstanceType;\n\n  /**\n   * The machine image to use, assumed to have SSM Agent preinstalled.\n   *\n   * @default - An Amazon Linux 2 image which is kept up-to-date automatically (the instance\n   * may be replaced on every deployment) and already has SSM Agent installed.\n   */\n  readonly machineImage?: IMachineImage;\n\n  /**\n   * Specifies how block devices are exposed to the instance. You can specify virtual devices and EBS volumes.\n   *\n   * Each instance that is launched has an associated root device volume,\n   * either an Amazon EBS volume or an instance store volume.\n   * You can use block device mappings to specify additional EBS volumes or\n   * instance store volumes to attach to an instance when it is launched.\n   *\n   * @see https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/block-device-mapping-concepts.html\n   *\n   * @default - Uses the block device mapping of the AMI\n   */\n  readonly blockDevices?: BlockDevice[];\n\n  /**\n   * Apply the given CloudFormation Init configuration to the instance at startup\n   *\n   * @default - no CloudFormation init\n   */\n  readonly init?: CloudFormationInit;\n\n  /**\n   * Use the given options for applying CloudFormation Init\n   *\n   * Describes the configsets to use and the timeout to wait\n   *\n   * @default - default options\n   */\n  readonly initOptions?: ApplyCloudFormationInitOptions;\n\n  /**\n   * Whether IMDSv2 should be required on this instance\n   *\n   * @default - false\n   */\n  readonly requireImdsv2?: boolean;\n}\n\n/**\n * This creates a linux bastion host you can use to connect to other instances or services in your VPC.\n * The recommended way to connect to the bastion host is by using AWS Systems Manager Session Manager.\n *\n * The operating system is Amazon Linux 2 with the latest SSM agent installed\n *\n * You can also configure this bastion host to allow connections via SSH\n *\n *\n * @resource AWS::EC2::Instance\n */\nexport class BastionHostLinux extends Resource implements IInstance {\n  public readonly stack: Stack;\n\n  /**\n   * Allows specify security group connections for the instance.\n   */\n  public readonly connections: Connections;\n\n  /**\n   * The IAM role assumed by the instance.\n   */\n  public readonly role: IRole;\n\n  /**\n   * The principal to grant permissions to\n   */\n  public readonly grantPrincipal: IPrincipal;\n\n  /**\n   * The underlying instance resource\n   */\n  public readonly instance: Instance;\n\n  /**\n   * @attribute\n   */\n  public readonly instanceId: string;\n\n  /**\n   * @attribute\n   */\n  public readonly instanceAvailabilityZone: string;\n\n  /**\n   * @attribute\n   */\n  public readonly instancePrivateDnsName: string;\n\n  /**\n   * @attribute\n   */\n  public readonly instancePrivateIp: string;\n\n  /**\n   * @attribute\n   */\n  public readonly instancePublicDnsName: string;\n\n  /**\n   * @attribute\n   */\n  public readonly instancePublicIp: string;\n\n  constructor(scope: Construct, id: string, props: BastionHostLinuxProps) {\n    super(scope, id);\n    this.stack = Stack.of(scope);\n    const instanceType = props.instanceType ?? InstanceType.of(InstanceClass.T3, InstanceSize.NANO);\n    this.instance = new Instance(this, 'Resource', {\n      vpc: props.vpc,\n      availabilityZone: props.availabilityZone,\n      securityGroup: props.securityGroup,\n      instanceName: props.instanceName ?? 'BastionHost',\n      instanceType,\n      machineImage: props.machineImage ?? MachineImage.latestAmazonLinux({\n        generation: AmazonLinuxGeneration.AMAZON_LINUX_2,\n        cpuType: this.toAmazonLinuxCpuType(instanceType.architecture),\n      }),\n      vpcSubnets: props.subnetSelection ?? {},\n      blockDevices: props.blockDevices ?? undefined,\n      init: props.init,\n      initOptions: props.initOptions,\n      requireImdsv2: props.requireImdsv2 ?? false,\n    });\n    this.instance.addToRolePolicy(new PolicyStatement({\n      actions: [\n        'ssmmessages:*',\n        'ssm:UpdateInstanceInformation',\n        'ec2messages:*',\n      ],\n      resources: ['*'],\n    }));\n    this.connections = this.instance.connections;\n    this.role = this.instance.role;\n    this.grantPrincipal = this.instance.role;\n    this.instanceId = this.instance.instanceId;\n    this.instancePrivateIp = this.instance.instancePrivateIp;\n    this.instanceAvailabilityZone = this.instance.instanceAvailabilityZone;\n    this.instancePrivateDnsName = this.instance.instancePrivateDnsName;\n    this.instancePublicIp = this.instance.instancePublicIp;\n    this.instancePublicDnsName = this.instance.instancePublicDnsName;\n\n    new CfnOutput(this, 'BastionHostId', {\n      description: 'Instance ID of the bastion host. Use this to connect via SSM Session Manager',\n      value: this.instanceId,\n    });\n  }\n\n  /**\n   * Returns the AmazonLinuxCpuType corresponding to the given instance architecture\n   * @param architecture the instance architecture value to convert\n   */\n  private toAmazonLinuxCpuType(architecture: InstanceArchitecture): AmazonLinuxCpuType {\n    if (architecture === InstanceArchitecture.ARM_64) {\n      return AmazonLinuxCpuType.ARM_64;\n    } else if (architecture === InstanceArchitecture.X86_64) {\n      return AmazonLinuxCpuType.X86_64;\n    }\n\n    throw new Error(`Unsupported instance architecture '${architecture}'`);\n  }\n\n  /**\n   * Allow SSH access from the given peer or peers\n   *\n   * Necessary if you want to connect to the instance using ssh. If not\n   * called, you should use SSM Session Manager to connect to the instance.\n   */\n  public allowSshAccessFrom(...peer: IPeer[]): void {\n    peer.forEach(p => {\n      this.connections.allowFrom(p, Port.tcp(22), 'SSH access');\n    });\n  }\n}\n"]}