UNPKG

@aws-cdk/aws-ec2

Version:

The CDK Construct Library for AWS::EC2

240 lines 45.3 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.Instance = void 0; const jsiiDeprecationWarnings = require("../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const crypto = require("crypto"); const iam = require("@aws-cdk/aws-iam"); const core_1 = require("@aws-cdk/core"); const aspects_1 = require("./aspects"); const connections_1 = require("./connections"); const ec2_generated_1 = require("./ec2.generated"); const ebs_util_1 = require("./private/ebs-util"); const security_group_1 = require("./security-group"); const vpc_1 = require("./vpc"); /** * Name tag constant */ const NAME_TAG = 'Name'; /** * This represents a single EC2 instance */ class Instance extends core_1.Resource { constructor(scope, id, props) { super(scope, id); this.securityGroups = []; try { jsiiDeprecationWarnings._aws_cdk_aws_ec2_InstanceProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, Instance); } throw error; } if (props.initOptions && !props.init) { throw new Error('Setting \'initOptions\' requires that \'init\' is also set'); } if (props.securityGroup) { this.securityGroup = props.securityGroup; } else { this.securityGroup = new security_group_1.SecurityGroup(this, 'InstanceSecurityGroup', { vpc: props.vpc, allowAllOutbound: props.allowAllOutbound !== false, }); } this.connections = new connections_1.Connections({ securityGroups: [this.securityGroup] }); this.securityGroups.push(this.securityGroup); core_1.Tags.of(this).add(NAME_TAG, props.instanceName || this.node.path); this.role = props.role || new iam.Role(this, 'InstanceRole', { assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'), }); this.grantPrincipal = this.role; const iamProfile = new iam.CfnInstanceProfile(this, 'InstanceProfile', { roles: [this.role.roleName], }); // use delayed evaluation const imageConfig = props.machineImage.getImage(this); this.userData = props.userData ?? imageConfig.userData; const userDataToken = core_1.Lazy.string({ produce: () => core_1.Fn.base64(this.userData.render()) }); const securityGroupsToken = core_1.Lazy.list({ produce: () => this.securityGroups.map(sg => sg.securityGroupId) }); const { subnets } = props.vpc.selectSubnets(props.vpcSubnets); let subnet; if (props.availabilityZone) { const selected = subnets.filter(sn => sn.availabilityZone === props.availabilityZone); if (selected.length === 1) { subnet = selected[0]; } else { core_1.Annotations.of(this).addError(`Need exactly 1 subnet to match AZ '${props.availabilityZone}', found ${selected.length}. Use a different availabilityZone.`); } } else { if (subnets.length > 0) { subnet = subnets[0]; } else { core_1.Annotations.of(this).addError(`Did not find any subnets matching '${JSON.stringify(props.vpcSubnets)}', please use a different selection.`); } } if (!subnet) { // We got here and we don't have a subnet because of validation errors. // Invent one on the spot so the code below doesn't fail. subnet = vpc_1.Subnet.fromSubnetAttributes(this, 'DummySubnet', { subnetId: 's-notfound', availabilityZone: 'az-notfound', }); } this.instance = new ec2_generated_1.CfnInstance(this, 'Resource', { imageId: imageConfig.imageId, keyName: props.keyName, instanceType: props.instanceType.toString(), securityGroupIds: securityGroupsToken, iamInstanceProfile: iamProfile.ref, userData: userDataToken, subnetId: subnet.subnetId, availabilityZone: subnet.availabilityZone, sourceDestCheck: props.sourceDestCheck, blockDeviceMappings: props.blockDevices !== undefined ? ebs_util_1.instanceBlockDeviceMappings(this, props.blockDevices) : undefined, privateIpAddress: props.privateIpAddress, propagateTagsToVolumeOnCreation: props.propagateTagsToVolumeOnCreation, monitoring: props.detailedMonitoring, }); this.instance.node.addDependency(this.role); this.osType = imageConfig.osType; this.node.defaultChild = this.instance; this.instanceId = this.instance.ref; this.instanceAvailabilityZone = this.instance.attrAvailabilityZone; this.instancePrivateDnsName = this.instance.attrPrivateDnsName; this.instancePrivateIp = this.instance.attrPrivateIp; this.instancePublicDnsName = this.instance.attrPublicDnsName; this.instancePublicIp = this.instance.attrPublicIp; if (props.init) { this.applyCloudFormationInit(props.init, props.initOptions); } this.applyUpdatePolicies(props); // Trigger replacement (via new logical ID) on user data change, if specified or cfn-init is being used. // // This is slightly tricky -- we need to resolve the UserData string (in order to get at actual Asset hashes, // instead of the Token stringifications of them ('${Token[1234]}'). However, in the case of CFN Init usage, // a UserData is going to contain the logicalID of the resource itself, which means infinite recursion if we // try to naively resolve. We need a recursion breaker in this. const originalLogicalId = core_1.Stack.of(this).getLogicalId(this.instance); let recursing = false; this.instance.overrideLogicalId(core_1.Lazy.uncachedString({ produce: (context) => { if (recursing) { return originalLogicalId; } if (!(props.userDataCausesReplacement ?? props.initOptions)) { return originalLogicalId; } const md5 = crypto.createHash('md5'); recursing = true; try { md5.update(JSON.stringify(context.resolve(this.userData.render()))); } finally { recursing = false; } const digest = md5.digest('hex').slice(0, 16); return `${originalLogicalId}${digest}`; }, })); if (props.requireImdsv2) { core_1.Aspects.of(this).add(new aspects_1.InstanceRequireImdsv2Aspect()); } } /** * Add the security group to the instance. * * @param securityGroup: The security group to add */ addSecurityGroup(securityGroup) { try { jsiiDeprecationWarnings._aws_cdk_aws_ec2_ISecurityGroup(securityGroup); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.addSecurityGroup); } throw error; } this.securityGroups.push(securityGroup); } /** * Add command to the startup script of the instance. * The command must be in the scripting language supported by the instance's OS (i.e. Linux/Windows). */ addUserData(...commands) { this.userData.addCommands(...commands); } /** * Adds a statement to the IAM role assumed by the instance. */ addToRolePolicy(statement) { this.role.addToPrincipalPolicy(statement); } /** * Use a CloudFormation Init configuration at instance startup * * This does the following: * * - Attaches the CloudFormation Init metadata to the Instance resource. * - Add commands to the instance UserData to run `cfn-init` and `cfn-signal`. * - Update the instance's CreationPolicy to wait for the `cfn-signal` commands. */ applyCloudFormationInit(init, options = {}) { init.attach(this.instance, { platform: this.osType, instanceRole: this.role, userData: this.userData, configSets: options.configSets, embedFingerprint: options.embedFingerprint, printLog: options.printLog, ignoreFailures: options.ignoreFailures, includeRole: options.includeRole, includeUrl: options.includeUrl, }); this.waitForResourceSignal(options.timeout ?? core_1.Duration.minutes(5)); } /** * Wait for a single additional resource signal * * Add 1 to the current ResourceSignal Count and add the given timeout to the current timeout. * * Use this to pause the CloudFormation deployment to wait for the instances * in the AutoScalingGroup to report successful startup during * creation and updates. The UserData script needs to invoke `cfn-signal` * with a success or failure code after it is done setting up the instance. */ waitForResourceSignal(timeout) { const oldResourceSignal = this.instance.cfnOptions.creationPolicy?.resourceSignal; this.instance.cfnOptions.creationPolicy = { ...this.instance.cfnOptions.creationPolicy, resourceSignal: { count: (oldResourceSignal?.count ?? 0) + 1, timeout: (oldResourceSignal?.timeout ? core_1.Duration.parse(oldResourceSignal?.timeout).plus(timeout) : timeout).toIsoString(), }, }; } /** * Apply CloudFormation update policies for the instance */ applyUpdatePolicies(props) { if (props.resourceSignalTimeout !== undefined) { this.instance.cfnOptions.creationPolicy = { ...this.instance.cfnOptions.creationPolicy, resourceSignal: { timeout: props.resourceSignalTimeout && props.resourceSignalTimeout.toIsoString(), }, }; } } } exports.Instance = Instance; _a = JSII_RTTI_SYMBOL_1; Instance[_a] = { fqn: "@aws-cdk/aws-ec2.Instance", version: "1.204.0" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"instance.js","sourceRoot":"","sources":["instance.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport * as iam from '@aws-cdk/aws-iam';\n\nimport { Annotations, Aspects, Duration, Fn, IResource, Lazy, Resource, Stack, Tags } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { InstanceRequireImdsv2Aspect } from './aspects';\nimport { CloudFormationInit } from './cfn-init';\nimport { Connections, IConnectable } from './connections';\nimport { CfnInstance } from './ec2.generated';\nimport { InstanceType } from './instance-types';\nimport { IMachineImage, OperatingSystemType } from './machine-image';\nimport { instanceBlockDeviceMappings } from './private/ebs-util';\nimport { ISecurityGroup, SecurityGroup } from './security-group';\nimport { UserData } from './user-data';\nimport { BlockDevice } from './volume';\nimport { IVpc, Subnet, SubnetSelection } from './vpc';\n\n/**\n * Name tag constant\n */\nconst NAME_TAG: string = 'Name';\n\nexport interface IInstance extends IResource, IConnectable, iam.IGrantable {\n  /**\n   * The instance's ID\n   *\n   * @attribute\n   */\n  readonly instanceId: string;\n\n  /**\n   * The availability zone the instance was launched in\n   *\n   * @attribute\n   */\n  readonly instanceAvailabilityZone: string;\n\n  /**\n   * Private DNS name for this instance\n   * @attribute\n   */\n  readonly instancePrivateDnsName: string;\n\n  /**\n   * Private IP for this instance\n   *\n   * @attribute\n   */\n  readonly instancePrivateIp: string;\n\n  /**\n   * Publicly-routable DNS name for this instance.\n   *\n   * (May be an empty string if the instance does not have a public name).\n   *\n   * @attribute\n   */\n  readonly instancePublicDnsName: string;\n\n  /**\n   * Publicly-routable IP  address for this instance.\n   *\n   * (May be an empty string if the instance does not have a public IP).\n   *\n   * @attribute\n   */\n  readonly instancePublicIp: string;\n}\n\n/**\n * Properties of an EC2 Instance\n */\nexport interface InstanceProps {\n\n  /**\n   * Name of SSH keypair to grant access to instance\n   *\n   * @default - No SSH access will be possible.\n   */\n  readonly keyName?: string;\n\n  /**\n   * Where to place the instance within the VPC\n   *\n   * @default - Private subnets.\n   */\n  readonly vpcSubnets?: SubnetSelection;\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   * Whether the instance could initiate connections to anywhere by default.\n   * This property is only used when you do not provide a security group.\n   *\n   * @default true\n   */\n  readonly allowAllOutbound?: boolean;\n\n  /**\n   * The length of time to wait for the resourceSignalCount\n   *\n   * The maximum value is 43200 (12 hours).\n   *\n   * @default Duration.minutes(5)\n   */\n  readonly resourceSignalTimeout?: Duration;\n\n  /**\n   * VPC to launch the instance in.\n   */\n  readonly vpc: IVpc;\n\n  /**\n   * Security Group to assign to this instance\n   *\n   * @default - create new security group\n   */\n  readonly securityGroup?: ISecurityGroup;\n\n  /**\n   * Type of instance to launch\n   */\n  readonly instanceType: InstanceType;\n\n  /**\n   * AMI to launch\n   */\n  readonly machineImage: IMachineImage;\n\n  /**\n   * Specific UserData to use\n   *\n   * The UserData may still be mutated after creation.\n   *\n   * @default - A UserData object appropriate for the MachineImage's\n   * Operating System is created.\n   */\n  readonly userData?: UserData;\n\n  /**\n   * Changes to the UserData force replacement\n   *\n   * Depending the EC2 instance type, changing UserData either\n   * restarts the instance or replaces the instance.\n   *\n   * - Instance store-backed instances are replaced.\n   * - EBS-backed instances are restarted.\n   *\n   * By default, restarting does not execute the new UserData so you\n   * will need a different mechanism to ensure the instance is restarted.\n   *\n   * Setting this to `true` will make the instance's Logical ID depend on the\n   * UserData, which will cause CloudFormation to replace it if the UserData\n   * changes.\n   *\n   * @default - true iff `initOptions` is specified, false otherwise.\n   */\n  readonly userDataCausesReplacement?: boolean;\n\n  /**\n   * An IAM role to associate with the instance profile assigned to this Auto Scaling Group.\n   *\n   * The role must be assumable by the service principal `ec2.amazonaws.com`:\n   *\n   * @example\n   * const role = new iam.Role(this, 'MyRole', {\n   *   assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com')\n   * });\n   *\n   * @default - A role will automatically be created, it can be accessed via the `role` property\n   */\n  readonly role?: iam.IRole;\n\n  /**\n   * The name of the instance\n   *\n   * @default - CDK generated name\n   */\n  readonly instanceName?: string;\n\n  /**\n   * Specifies whether to enable an instance launched in a VPC to perform NAT.\n   * This controls whether source/destination checking is enabled on the instance.\n   * A value of true means that checking is enabled, and false means that checking is disabled.\n   * The value must be false for the instance to perform NAT.\n   *\n   * @default true\n   */\n  readonly sourceDestCheck?: boolean;\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   * Defines a private IP address to associate with an instance.\n   *\n   * Private IP should be available within the VPC that the instance is build within.\n   *\n   * @default - no association\n   */\n  readonly privateIpAddress?: string;\n\n  /**\n   * Propagate the EC2 instance tags to the EBS volumes.\n   *\n   * @default - false\n   */\n  readonly propagateTagsToVolumeOnCreation?: boolean;\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   * Whether \"Detailed Monitoring\" is enabled for this instance\n   * Keep in mind that Detailed Monitoring results in extra charges\n   *\n   * @see http://aws.amazon.com/cloudwatch/pricing/\n   * @default - false\n   */\n  readonly detailedMonitoring?: boolean;\n}\n\n/**\n * This represents a single EC2 instance\n */\nexport class Instance extends Resource implements IInstance {\n\n  /**\n   * The type of OS the instance is running.\n   */\n  public readonly osType: OperatingSystemType;\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: iam.IRole;\n\n  /**\n   * The principal to grant permissions to\n   */\n  public readonly grantPrincipal: iam.IPrincipal;\n\n  /**\n   * UserData for the instance\n   */\n  public readonly userData: UserData;\n\n  /**\n   * the underlying instance resource\n   */\n  public readonly instance: CfnInstance;\n  /**\n   * @attribute\n   */\n  public readonly instanceId: string;\n  /**\n   * @attribute\n   */\n  public readonly instanceAvailabilityZone: string;\n  /**\n   * @attribute\n   */\n  public readonly instancePrivateDnsName: string;\n  /**\n   * @attribute\n   */\n  public readonly instancePrivateIp: string;\n  /**\n   * @attribute\n   */\n  public readonly instancePublicDnsName: string;\n  /**\n   * @attribute\n   */\n  public readonly instancePublicIp: string;\n\n  private readonly securityGroup: ISecurityGroup;\n  private readonly securityGroups: ISecurityGroup[] = [];\n\n  constructor(scope: Construct, id: string, props: InstanceProps) {\n    super(scope, id);\n\n    if (props.initOptions && !props.init) {\n      throw new Error('Setting \\'initOptions\\' requires that \\'init\\' is also set');\n    }\n\n    if (props.securityGroup) {\n      this.securityGroup = props.securityGroup;\n    } else {\n      this.securityGroup = new SecurityGroup(this, 'InstanceSecurityGroup', {\n        vpc: props.vpc,\n        allowAllOutbound: props.allowAllOutbound !== false,\n      });\n    }\n    this.connections = new Connections({ securityGroups: [this.securityGroup] });\n    this.securityGroups.push(this.securityGroup);\n    Tags.of(this).add(NAME_TAG, props.instanceName || this.node.path);\n\n    this.role = props.role || new iam.Role(this, 'InstanceRole', {\n      assumedBy: new iam.ServicePrincipal('ec2.amazonaws.com'),\n    });\n    this.grantPrincipal = this.role;\n\n    const iamProfile = new iam.CfnInstanceProfile(this, 'InstanceProfile', {\n      roles: [this.role.roleName],\n    });\n\n    // use delayed evaluation\n    const imageConfig = props.machineImage.getImage(this);\n    this.userData = props.userData ?? imageConfig.userData;\n    const userDataToken = Lazy.string({ produce: () => Fn.base64(this.userData.render()) });\n    const securityGroupsToken = Lazy.list({ produce: () => this.securityGroups.map(sg => sg.securityGroupId) });\n\n    const { subnets } = props.vpc.selectSubnets(props.vpcSubnets);\n    let subnet;\n    if (props.availabilityZone) {\n      const selected = subnets.filter(sn => sn.availabilityZone === props.availabilityZone);\n      if (selected.length === 1) {\n        subnet = selected[0];\n      } else {\n        Annotations.of(this).addError(`Need exactly 1 subnet to match AZ '${props.availabilityZone}', found ${selected.length}. Use a different availabilityZone.`);\n      }\n    } else {\n      if (subnets.length > 0) {\n        subnet = subnets[0];\n      } else {\n        Annotations.of(this).addError(`Did not find any subnets matching '${JSON.stringify(props.vpcSubnets)}', please use a different selection.`);\n      }\n    }\n    if (!subnet) {\n      // We got here and we don't have a subnet because of validation errors.\n      // Invent one on the spot so the code below doesn't fail.\n      subnet = Subnet.fromSubnetAttributes(this, 'DummySubnet', {\n        subnetId: 's-notfound',\n        availabilityZone: 'az-notfound',\n      });\n    }\n\n    this.instance = new CfnInstance(this, 'Resource', {\n      imageId: imageConfig.imageId,\n      keyName: props.keyName,\n      instanceType: props.instanceType.toString(),\n      securityGroupIds: securityGroupsToken,\n      iamInstanceProfile: iamProfile.ref,\n      userData: userDataToken,\n      subnetId: subnet.subnetId,\n      availabilityZone: subnet.availabilityZone,\n      sourceDestCheck: props.sourceDestCheck,\n      blockDeviceMappings: props.blockDevices !== undefined ? instanceBlockDeviceMappings(this, props.blockDevices) : undefined,\n      privateIpAddress: props.privateIpAddress,\n      propagateTagsToVolumeOnCreation: props.propagateTagsToVolumeOnCreation,\n      monitoring: props.detailedMonitoring,\n    });\n    this.instance.node.addDependency(this.role);\n\n    this.osType = imageConfig.osType;\n    this.node.defaultChild = this.instance;\n\n    this.instanceId = this.instance.ref;\n    this.instanceAvailabilityZone = this.instance.attrAvailabilityZone;\n    this.instancePrivateDnsName = this.instance.attrPrivateDnsName;\n    this.instancePrivateIp = this.instance.attrPrivateIp;\n    this.instancePublicDnsName = this.instance.attrPublicDnsName;\n    this.instancePublicIp = this.instance.attrPublicIp;\n\n    if (props.init) {\n      this.applyCloudFormationInit(props.init, props.initOptions);\n    }\n\n    this.applyUpdatePolicies(props);\n\n    // Trigger replacement (via new logical ID) on user data change, if specified or cfn-init is being used.\n    //\n    // This is slightly tricky -- we need to resolve the UserData string (in order to get at actual Asset hashes,\n    // instead of the Token stringifications of them ('${Token[1234]}'). However, in the case of CFN Init usage,\n    // a UserData is going to contain the logicalID of the resource itself, which means infinite recursion if we\n    // try to naively resolve. We need a recursion breaker in this.\n    const originalLogicalId = Stack.of(this).getLogicalId(this.instance);\n    let recursing = false;\n    this.instance.overrideLogicalId(Lazy.uncachedString({\n      produce: (context) => {\n        if (recursing) { return originalLogicalId; }\n        if (!(props.userDataCausesReplacement ?? props.initOptions)) { return originalLogicalId; }\n\n        const md5 = crypto.createHash('md5');\n        recursing = true;\n        try {\n          md5.update(JSON.stringify(context.resolve(this.userData.render())));\n        } finally {\n          recursing = false;\n        }\n        const digest = md5.digest('hex').slice(0, 16);\n        return `${originalLogicalId}${digest}`;\n      },\n    }));\n\n    if (props.requireImdsv2) {\n      Aspects.of(this).add(new InstanceRequireImdsv2Aspect());\n    }\n  }\n\n  /**\n   * Add the security group to the instance.\n   *\n   * @param securityGroup: The security group to add\n   */\n  public addSecurityGroup(securityGroup: ISecurityGroup): void {\n    this.securityGroups.push(securityGroup);\n  }\n\n  /**\n   * Add command to the startup script of the instance.\n   * The command must be in the scripting language supported by the instance's OS (i.e. Linux/Windows).\n   */\n  public addUserData(...commands: string[]) {\n    this.userData.addCommands(...commands);\n  }\n\n  /**\n   * Adds a statement to the IAM role assumed by the instance.\n   */\n  public addToRolePolicy(statement: iam.PolicyStatement) {\n    this.role.addToPrincipalPolicy(statement);\n  }\n\n  /**\n   * Use a CloudFormation Init configuration at instance startup\n   *\n   * This does the following:\n   *\n   * - Attaches the CloudFormation Init metadata to the Instance resource.\n   * - Add commands to the instance UserData to run `cfn-init` and `cfn-signal`.\n   * - Update the instance's CreationPolicy to wait for the `cfn-signal` commands.\n   */\n  private applyCloudFormationInit(init: CloudFormationInit, options: ApplyCloudFormationInitOptions = {}) {\n    init.attach(this.instance, {\n      platform: this.osType,\n      instanceRole: this.role,\n      userData: this.userData,\n      configSets: options.configSets,\n      embedFingerprint: options.embedFingerprint,\n      printLog: options.printLog,\n      ignoreFailures: options.ignoreFailures,\n      includeRole: options.includeRole,\n      includeUrl: options.includeUrl,\n    });\n    this.waitForResourceSignal(options.timeout ?? Duration.minutes(5));\n  }\n\n  /**\n   * Wait for a single additional resource signal\n   *\n   * Add 1 to the current ResourceSignal Count and add the given timeout to the current timeout.\n   *\n   * Use this to pause the CloudFormation deployment to wait for the instances\n   * in the AutoScalingGroup to report successful startup during\n   * creation and updates. The UserData script needs to invoke `cfn-signal`\n   * with a success or failure code after it is done setting up the instance.\n   */\n  private waitForResourceSignal(timeout: Duration) {\n    const oldResourceSignal = this.instance.cfnOptions.creationPolicy?.resourceSignal;\n    this.instance.cfnOptions.creationPolicy = {\n      ...this.instance.cfnOptions.creationPolicy,\n      resourceSignal: {\n        count: (oldResourceSignal?.count ?? 0) + 1,\n        timeout: (oldResourceSignal?.timeout ? Duration.parse(oldResourceSignal?.timeout).plus(timeout) : timeout).toIsoString(),\n      },\n    };\n  }\n\n  /**\n   * Apply CloudFormation update policies for the instance\n   */\n  private applyUpdatePolicies(props: InstanceProps) {\n    if (props.resourceSignalTimeout !== undefined) {\n      this.instance.cfnOptions.creationPolicy = {\n        ...this.instance.cfnOptions.creationPolicy,\n        resourceSignal: {\n          timeout: props.resourceSignalTimeout && props.resourceSignalTimeout.toIsoString(),\n        },\n      };\n    }\n  }\n}\n\n/**\n * Options for applying CloudFormation init to an instance or instance group\n */\nexport interface ApplyCloudFormationInitOptions {\n  /**\n   * ConfigSet to activate\n   *\n   * @default ['default']\n   */\n  readonly configSets?: string[];\n\n  /**\n   * Timeout waiting for the configuration to be applied\n   *\n   * @default Duration.minutes(5)\n   */\n  readonly timeout?: Duration;\n\n  /**\n   * Force instance replacement by embedding a config fingerprint\n   *\n   * If `true` (the default), a hash of the config will be embedded into the\n   * UserData, so that if the config changes, the UserData changes.\n   *\n   * - If the EC2 instance is instance-store backed or\n   *   `userDataCausesReplacement` is set, this will cause the instance to be\n   *   replaced and the new configuration to be applied.\n   * - If the instance is EBS-backed and `userDataCausesReplacement` is not\n   *   set, the change of UserData will make the instance restart but not be\n   *   replaced, and the configuration will not be applied automatically.\n   *\n   * If `false`, no hash will be embedded, and if the CloudFormation Init\n   * config changes nothing will happen to the running instance. If a\n   * config update introduces errors, you will not notice until after the\n   * CloudFormation deployment successfully finishes and the next instance\n   * fails to launch.\n   *\n   * @default true\n   */\n  readonly embedFingerprint?: boolean;\n\n  /**\n   * Print the results of running cfn-init to the Instance System Log\n   *\n   * By default, the output of running cfn-init is written to a log file\n   * on the instance. Set this to `true` to print it to the System Log\n   * (visible from the EC2 Console), `false` to not print it.\n   *\n   * (Be aware that the system log is refreshed at certain points in\n   * time of the instance life cycle, and successful execution may\n   * not always show up).\n   *\n   * @default true\n   */\n  readonly printLog?: boolean;\n\n  /**\n   * Don't fail the instance creation when cfn-init fails\n   *\n   * You can use this to prevent CloudFormation from rolling back when\n   * instances fail to start up, to help in debugging.\n   *\n   * @default false\n   */\n  readonly ignoreFailures?: boolean;\n\n  /**\n   * Include --url argument when running cfn-init and cfn-signal commands\n   *\n   * This will be the cloudformation endpoint in the deployed region\n   * e.g. https://cloudformation.us-east-1.amazonaws.com\n   *\n   * @default false\n   */\n  readonly includeUrl?: boolean;\n\n  /**\n   * Include --role argument when running cfn-init and cfn-signal commands\n   *\n   * This will be the IAM instance profile attached to the EC2 instance\n   *\n   * @default false\n   */\n  readonly includeRole?: boolean;\n}\n"]}