UNPKG

@aws-cdk/aws-ec2

Version:

The CDK Construct Library for AWS::EC2

216 lines 29.8 kB
"use strict"; var _a, _b, _c; Object.defineProperty(exports, "__esModule", { value: true }); exports.RESERVED_TUNNEL_INSIDE_CIDR = exports.VpnConnection = exports.VpnConnectionBase = exports.VpnGateway = exports.VpnConnectionType = void 0; const jsiiDeprecationWarnings = require("../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const net = require("net"); const cloudwatch = require("@aws-cdk/aws-cloudwatch"); const core_1 = require("@aws-cdk/core"); const ec2_generated_1 = require("./ec2.generated"); /** * The VPN connection type. */ var VpnConnectionType; (function (VpnConnectionType) { /** * The IPsec 1 VPN connection type. */ VpnConnectionType["IPSEC_1"] = "ipsec.1"; /** * Dummy member * TODO: remove once https://github.com/aws/jsii/issues/231 is fixed */ VpnConnectionType["DUMMY"] = "dummy"; })(VpnConnectionType = exports.VpnConnectionType || (exports.VpnConnectionType = {})); /** * The VPN Gateway that shall be added to the VPC * * @resource AWS::EC2::VPNGateway */ class VpnGateway extends core_1.Resource { constructor(scope, id, props) { super(scope, id); try { jsiiDeprecationWarnings._aws_cdk_aws_ec2_VpnGatewayProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, VpnGateway); } throw error; } // This is 'Default' instead of 'Resource', because using 'Default' will generate // a logical ID for a VpnGateway which is exactly the same as the logical ID that used // to be created for the CfnVPNGateway (and 'Resource' would not do that). const vpnGW = new ec2_generated_1.CfnVPNGateway(this, 'Default', props); this.gatewayId = vpnGW.ref; } } exports.VpnGateway = VpnGateway; _a = JSII_RTTI_SYMBOL_1; VpnGateway[_a] = { fqn: "@aws-cdk/aws-ec2.VpnGateway", version: "1.204.0" }; /** * Base class for Vpn connections. */ class VpnConnectionBase extends core_1.Resource { } exports.VpnConnectionBase = VpnConnectionBase; _b = JSII_RTTI_SYMBOL_1; VpnConnectionBase[_b] = { fqn: "@aws-cdk/aws-ec2.VpnConnectionBase", version: "1.204.0" }; /** * Define a VPN Connection * * @resource AWS::EC2::VPNConnection */ class VpnConnection extends VpnConnectionBase { constructor(scope, id, props) { super(scope, id); try { jsiiDeprecationWarnings._aws_cdk_aws_ec2_VpnConnectionProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, VpnConnection); } throw error; } if (!props.vpc.vpnGatewayId) { props.vpc.enableVpnGateway({ type: 'ipsec.1', amazonSideAsn: props.asn, }); } if (!core_1.Token.isUnresolved(props.ip) && !net.isIPv4(props.ip)) { throw new Error(`The \`ip\` ${props.ip} is not a valid IPv4 address.`); } const type = VpnConnectionType.IPSEC_1; const bgpAsn = props.asn || 65000; const customerGateway = new ec2_generated_1.CfnCustomerGateway(this, 'CustomerGateway', { bgpAsn, ipAddress: props.ip, type, }); this.customerGatewayId = customerGateway.ref; this.customerGatewayAsn = bgpAsn; this.customerGatewayIp = props.ip; // Validate tunnel options if (props.tunnelOptions) { if (props.tunnelOptions.length > 2) { throw new Error('Cannot specify more than two `tunnelOptions`'); } if (props.tunnelOptions.length === 2 && props.tunnelOptions[0].tunnelInsideCidr === props.tunnelOptions[1].tunnelInsideCidr) { throw new Error(`Same ${props.tunnelOptions[0].tunnelInsideCidr} \`tunnelInsideCidr\` cannot be used for both tunnels.`); } props.tunnelOptions.forEach((options, index) => { if (options.preSharedKey && options.preSharedKeySecret) { throw new Error("Specify at most one of 'preSharedKey' and 'preSharedKeySecret'."); } if (options.preSharedKey && !core_1.Token.isUnresolved(options.preSharedKey) && !/^[a-zA-Z1-9._][a-zA-Z\d._]{7,63}$/.test(options.preSharedKey)) { /* eslint-disable max-len */ throw new Error(`The \`preSharedKey\` ${options.preSharedKey} for tunnel ${index + 1} is invalid. Allowed characters are alphanumeric characters and ._. Must be between 8 and 64 characters in length and cannot start with zero (0).`); /* eslint-enable max-len */ } if (options.tunnelInsideCidr) { if (exports.RESERVED_TUNNEL_INSIDE_CIDR.includes(options.tunnelInsideCidr)) { throw new Error(`The \`tunnelInsideCidr\` ${options.tunnelInsideCidr} for tunnel ${index + 1} is a reserved inside CIDR.`); } if (!/^169\.254\.\d{1,3}\.\d{1,3}\/30$/.test(options.tunnelInsideCidr)) { /* eslint-disable-next-line max-len */ throw new Error(`The \`tunnelInsideCidr\` ${options.tunnelInsideCidr} for tunnel ${index + 1} is not a size /30 CIDR block from the 169.254.0.0/16 range.`); } } }); } const vpnConnection = new ec2_generated_1.CfnVPNConnection(this, 'Resource', { type, customerGatewayId: customerGateway.ref, staticRoutesOnly: props.staticRoutes ? true : false, vpnGatewayId: props.vpc.vpnGatewayId, vpnTunnelOptionsSpecifications: props.tunnelOptions?.map(t => ({ preSharedKey: t.preSharedKeySecret?.unsafeUnwrap() ?? t.preSharedKey, tunnelInsideCidr: t.tunnelInsideCidr, })), }); this.vpnId = vpnConnection.ref; if (props.staticRoutes) { props.staticRoutes.forEach(route => { new ec2_generated_1.CfnVPNConnectionRoute(this, `Route${route.replace(/[^\d]/g, '')}`, { destinationCidrBlock: route, vpnConnectionId: this.vpnId, }); }); } } /** * Import a VPN connection by supplying all attributes directly */ static fromVpnConnectionAttributes(scope, id, attrs) { try { jsiiDeprecationWarnings._aws_cdk_aws_ec2_VpnConnectionAttributes(attrs); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, this.fromVpnConnectionAttributes); } throw error; } class Import extends VpnConnectionBase { constructor() { super(...arguments); this.vpnId = attrs.vpnId; this.customerGatewayId = attrs.customerGatewayId; this.customerGatewayIp = attrs.customerGatewayIp; this.customerGatewayAsn = attrs.customerGatewayAsn; } } return new Import(scope, id); } /** * Return the given named metric for all VPN connections in the account/region. */ static metricAll(metricName, props) { return new cloudwatch.Metric({ namespace: 'AWS/VPN', metricName, ...props, }); } /** * Metric for the tunnel state of all VPN connections in the account/region. * * @default average over 5 minutes */ static metricAllTunnelState(props) { return this.metricAll('TunnelState', { statistic: 'avg', ...props }); } /** * Metric for the tunnel data in of all VPN connections in the account/region. * * @default sum over 5 minutes */ static metricAllTunnelDataIn(props) { return this.metricAll('TunnelDataIn', { statistic: 'sum', ...props }); } /** * Metric for the tunnel data out of all VPN connections. * * @default sum over 5 minutes */ static metricAllTunnelDataOut(props) { return this.metricAll('TunnelDataOut', { statistic: 'sum', ...props }); } } exports.VpnConnection = VpnConnection; _c = JSII_RTTI_SYMBOL_1; VpnConnection[_c] = { fqn: "@aws-cdk/aws-ec2.VpnConnection", version: "1.204.0" }; exports.RESERVED_TUNNEL_INSIDE_CIDR = [ '169.254.0.0/30', '169.254.1.0/30', '169.254.2.0/30', '169.254.3.0/30', '169.254.4.0/30', '169.254.5.0/30', '169.254.169.252/30', ]; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vpn.js","sourceRoot":"","sources":["vpn.ts"],"names":[],"mappings":"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","sourcesContent":["import * as net from 'net';\nimport * as cloudwatch from '@aws-cdk/aws-cloudwatch';\nimport { IResource, Resource, SecretValue, Token } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport {\n  CfnCustomerGateway,\n  CfnVPNConnection,\n  CfnVPNConnectionRoute,\n  CfnVPNGateway,\n} from './ec2.generated';\nimport { IVpc, SubnetSelection } from './vpc';\n\nexport interface IVpnConnection extends IResource {\n  /**\n   * The id of the VPN connection.\n   * @attribute VpnConnectionId\n   */\n  readonly vpnId: string;\n\n  /**\n   * The id of the customer gateway.\n   */\n  readonly customerGatewayId: string;\n\n  /**\n   * The ip address of the customer gateway.\n   */\n  readonly customerGatewayIp: string;\n\n  /**\n   * The ASN of the customer gateway.\n   */\n  readonly customerGatewayAsn: number;\n}\n\n/**\n * The virtual private gateway interface\n */\nexport interface IVpnGateway extends IResource {\n\n  /**\n   * The virtual private gateway Id\n   */\n  readonly gatewayId: string\n}\n\nexport interface VpnTunnelOption {\n  /**\n   * The pre-shared key (PSK) to establish initial authentication between the\n   * virtual private gateway and customer gateway. Allowed characters are\n   * alphanumeric characters period `.` and underscores `_`. Must be between 8\n   * and 64 characters in length and cannot start with zero (0).\n   *\n   * @default an Amazon generated pre-shared key\n   * @deprecated Use `preSharedKeySecret` instead\n   */\n  readonly preSharedKey?: string;\n\n  /**\n   * The pre-shared key (PSK) to establish initial authentication between the\n   * virtual private gateway and customer gateway. Allowed characters are\n   * alphanumeric characters period `.` and underscores `_`. Must be between 8\n   * and 64 characters in length and cannot start with zero (0).\n   *\n   * @default an Amazon generated pre-shared key\n   */\n  readonly preSharedKeySecret?: SecretValue;\n\n  /**\n   * The range of inside IP addresses for the tunnel. Any specified CIDR blocks must be\n   * unique across all VPN connections that use the same virtual private gateway.\n   * A size /30 CIDR block from the 169.254.0.0/16 range.\n   *\n   * @default an Amazon generated inside IP CIDR\n   */\n  readonly tunnelInsideCidr?: string;\n}\n\nexport interface VpnConnectionOptions {\n  /**\n   * The ip address of the customer gateway.\n   */\n  readonly ip: string;\n\n  /**\n   * The ASN of the customer gateway.\n   *\n   * @default 65000\n   */\n  readonly asn?: number;\n\n  /**\n   * The static routes to be routed from the VPN gateway to the customer gateway.\n   *\n   * @default Dynamic routing (BGP)\n   */\n  readonly staticRoutes?: string[];\n\n  /**\n   * The tunnel options for the VPN connection. At most two elements (one per tunnel).\n   * Duplicates not allowed.\n   *\n   * @default Amazon generated tunnel options\n   */\n  readonly tunnelOptions?: VpnTunnelOption[];\n}\n\n/**\n * The VpnGateway Properties\n */\nexport interface VpnGatewayProps {\n\n  /**\n   * Default type ipsec.1\n   */\n  readonly type: string;\n\n  /**\n   * Explicitly specify an Asn or let aws pick an Asn for you.\n   * @default 65000\n   */\n  readonly amazonSideAsn?: number;\n}\n\n/**\n * Options for the Vpc.enableVpnGateway() method\n */\nexport interface EnableVpnGatewayOptions extends VpnGatewayProps {\n  /**\n   * Provide an array of subnets where the route propagation should be added.\n   * @default noPropagation\n   */\n  readonly vpnRoutePropagation?: SubnetSelection[]\n}\n\nexport interface VpnConnectionProps extends VpnConnectionOptions {\n  /**\n   * The VPC to connect to.\n   */\n  readonly vpc: IVpc;\n}\n\n/**\n * The VPN connection type.\n */\nexport enum VpnConnectionType {\n  /**\n   * The IPsec 1 VPN connection type.\n   */\n  IPSEC_1 = 'ipsec.1',\n\n  /**\n   * Dummy member\n   * TODO: remove once https://github.com/aws/jsii/issues/231 is fixed\n   */\n  DUMMY = 'dummy'\n}\n\n/**\n * The VPN Gateway that shall be added to the VPC\n *\n * @resource AWS::EC2::VPNGateway\n */\nexport class VpnGateway extends Resource implements IVpnGateway {\n\n  /**\n   * The virtual private gateway Id\n   */\n  public readonly gatewayId: string;\n\n  constructor(scope: Construct, id: string, props: VpnGatewayProps) {\n    super(scope, id);\n\n    // This is 'Default' instead of 'Resource', because using 'Default' will generate\n    // a logical ID for a VpnGateway which is exactly the same as the logical ID that used\n    // to be created for the CfnVPNGateway (and 'Resource' would not do that).\n    const vpnGW = new CfnVPNGateway(this, 'Default', props);\n    this.gatewayId = vpnGW.ref;\n  }\n}\n\n/**\n * Attributes of an imported VpnConnection.\n */\nexport interface VpnConnectionAttributes {\n\n  /**\n   * The id of the VPN connection.\n   */\n  readonly vpnId: string;\n\n  /**\n   * The id of the customer gateway.\n   */\n  readonly customerGatewayId: string;\n\n  /**\n   * The ip address of the customer gateway.\n   */\n  readonly customerGatewayIp: string;\n\n  /**\n   * The ASN of the customer gateway.\n   */\n  readonly customerGatewayAsn: number;\n\n}\n\n/**\n * Base class for Vpn connections.\n */\nexport abstract class VpnConnectionBase extends Resource implements IVpnConnection {\n\n  public abstract readonly vpnId: string;\n  public abstract readonly customerGatewayId: string;\n  public abstract readonly customerGatewayIp: string;\n  public abstract readonly customerGatewayAsn: number;\n\n}\n\n/**\n * Define a VPN Connection\n *\n * @resource AWS::EC2::VPNConnection\n */\nexport class VpnConnection extends VpnConnectionBase {\n\n  /**\n   * Import a VPN connection by supplying all attributes directly\n   */\n  public static fromVpnConnectionAttributes(scope: Construct, id: string, attrs: VpnConnectionAttributes): IVpnConnection {\n\n    class Import extends VpnConnectionBase {\n\n      public readonly vpnId: string = attrs.vpnId;\n      public readonly customerGatewayId: string = attrs.customerGatewayId;\n      public readonly customerGatewayIp: string = attrs.customerGatewayIp;\n      public readonly customerGatewayAsn: number = attrs.customerGatewayAsn;\n\n    }\n\n    return new Import(scope, id);\n\n  }\n\n  /**\n   * Return the given named metric for all VPN connections in the account/region.\n   */\n  public static metricAll(metricName: string, props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return new cloudwatch.Metric({\n      namespace: 'AWS/VPN',\n      metricName,\n      ...props,\n    });\n  }\n\n  /**\n   * Metric for the tunnel state of all VPN connections in the account/region.\n   *\n   * @default average over 5 minutes\n   */\n  public static metricAllTunnelState(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.metricAll('TunnelState', { statistic: 'avg', ...props });\n  }\n\n  /**\n   * Metric for the tunnel data in of all VPN connections in the account/region.\n   *\n   * @default sum over 5 minutes\n   */\n  public static metricAllTunnelDataIn(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.metricAll('TunnelDataIn', { statistic: 'sum', ...props });\n  }\n\n  /**\n   * Metric for the tunnel data out of all VPN connections.\n   *\n   * @default sum over 5 minutes\n   */\n  public static metricAllTunnelDataOut(props?: cloudwatch.MetricOptions): cloudwatch.Metric {\n    return this.metricAll('TunnelDataOut', { statistic: 'sum', ...props });\n  }\n\n  public readonly vpnId: string;\n  public readonly customerGatewayId: string;\n  public readonly customerGatewayIp: string;\n  public readonly customerGatewayAsn: number;\n\n  constructor(scope: Construct, id: string, props: VpnConnectionProps) {\n    super(scope, id);\n\n    if (!props.vpc.vpnGatewayId) {\n      props.vpc.enableVpnGateway({\n        type: 'ipsec.1',\n        amazonSideAsn: props.asn,\n      });\n    }\n\n    if (!Token.isUnresolved(props.ip) && !net.isIPv4(props.ip)) {\n      throw new Error(`The \\`ip\\` ${props.ip} is not a valid IPv4 address.`);\n    }\n\n    const type = VpnConnectionType.IPSEC_1;\n    const bgpAsn = props.asn || 65000;\n\n    const customerGateway = new CfnCustomerGateway(this, 'CustomerGateway', {\n      bgpAsn,\n      ipAddress: props.ip,\n      type,\n    });\n\n    this.customerGatewayId = customerGateway.ref;\n    this.customerGatewayAsn = bgpAsn;\n    this.customerGatewayIp = props.ip;\n\n    // Validate tunnel options\n    if (props.tunnelOptions) {\n      if (props.tunnelOptions.length > 2) {\n        throw new Error('Cannot specify more than two `tunnelOptions`');\n      }\n\n      if (props.tunnelOptions.length === 2 && props.tunnelOptions[0].tunnelInsideCidr === props.tunnelOptions[1].tunnelInsideCidr) {\n        throw new Error(`Same ${props.tunnelOptions[0].tunnelInsideCidr} \\`tunnelInsideCidr\\` cannot be used for both tunnels.`);\n      }\n\n      props.tunnelOptions.forEach((options, index) => {\n        if (options.preSharedKey && options.preSharedKeySecret) {\n          throw new Error(\"Specify at most one of 'preSharedKey' and 'preSharedKeySecret'.\");\n        }\n\n        if (options.preSharedKey && !Token.isUnresolved(options.preSharedKey) && !/^[a-zA-Z1-9._][a-zA-Z\\d._]{7,63}$/.test(options.preSharedKey)) {\n          /* eslint-disable max-len */\n          throw new Error(`The \\`preSharedKey\\` ${options.preSharedKey} for tunnel ${index + 1} is invalid. Allowed characters are alphanumeric characters and ._. Must be between 8 and 64 characters in length and cannot start with zero (0).`);\n          /* eslint-enable max-len */\n        }\n\n        if (options.tunnelInsideCidr) {\n          if (RESERVED_TUNNEL_INSIDE_CIDR.includes(options.tunnelInsideCidr)) {\n            throw new Error(`The \\`tunnelInsideCidr\\` ${options.tunnelInsideCidr} for tunnel ${index + 1} is a reserved inside CIDR.`);\n          }\n\n          if (!/^169\\.254\\.\\d{1,3}\\.\\d{1,3}\\/30$/.test(options.tunnelInsideCidr)) {\n            /* eslint-disable-next-line max-len */\n            throw new Error(`The \\`tunnelInsideCidr\\` ${options.tunnelInsideCidr} for tunnel ${index + 1} is not a size /30 CIDR block from the 169.254.0.0/16 range.`);\n          }\n        }\n      });\n    }\n\n    const vpnConnection = new CfnVPNConnection(this, 'Resource', {\n      type,\n      customerGatewayId: customerGateway.ref,\n      staticRoutesOnly: props.staticRoutes ? true : false,\n      vpnGatewayId: props.vpc.vpnGatewayId,\n      vpnTunnelOptionsSpecifications: props.tunnelOptions?.map(t => ({\n        preSharedKey: t.preSharedKeySecret?.unsafeUnwrap() ?? t.preSharedKey,\n        tunnelInsideCidr: t.tunnelInsideCidr,\n      })),\n    });\n\n    this.vpnId = vpnConnection.ref;\n\n    if (props.staticRoutes) {\n      props.staticRoutes.forEach(route => {\n        new CfnVPNConnectionRoute(this, `Route${route.replace(/[^\\d]/g, '')}`, {\n          destinationCidrBlock: route,\n          vpnConnectionId: this.vpnId,\n        });\n      });\n    }\n  }\n}\n\nexport const RESERVED_TUNNEL_INSIDE_CIDR = [\n  '169.254.0.0/30',\n  '169.254.1.0/30',\n  '169.254.2.0/30',\n  '169.254.3.0/30',\n  '169.254.4.0/30',\n  '169.254.5.0/30',\n  '169.254.169.252/30',\n];\n"]}