@aws-cdk/aws-ec2
Version:
The CDK Construct Library for AWS::EC2
275 lines • 37.9 kB
JavaScript
;
var _a, _b;
Object.defineProperty(exports, "__esModule", { value: true });
exports.ClientVpnEndpoint = exports.ClientVpnUserBasedAuthentication = exports.ClientVpnSessionTimeout = void 0;
const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const logs = require("@aws-cdk/aws-logs");
const core_1 = require("@aws-cdk/core");
const client_vpn_authorization_rule_1 = require("./client-vpn-authorization-rule");
const client_vpn_route_1 = require("./client-vpn-route");
const connections_1 = require("./connections");
const ec2_generated_1 = require("./ec2.generated");
const network_util_1 = require("./network-util");
const security_group_1 = require("./security-group");
/**
* Maximum VPN session duration time
*/
var ClientVpnSessionTimeout;
(function (ClientVpnSessionTimeout) {
/** 8 hours */
ClientVpnSessionTimeout[ClientVpnSessionTimeout["EIGHT_HOURS"] = 8] = "EIGHT_HOURS";
/** 10 hours */
ClientVpnSessionTimeout[ClientVpnSessionTimeout["TEN_HOURS"] = 10] = "TEN_HOURS";
/** 12 hours */
ClientVpnSessionTimeout[ClientVpnSessionTimeout["TWELVE_HOURS"] = 12] = "TWELVE_HOURS";
/** 24 hours */
ClientVpnSessionTimeout[ClientVpnSessionTimeout["TWENTY_FOUR_HOURS"] = 24] = "TWENTY_FOUR_HOURS";
})(ClientVpnSessionTimeout = exports.ClientVpnSessionTimeout || (exports.ClientVpnSessionTimeout = {}));
/**
* User-based authentication for a client VPN endpoint
*/
class ClientVpnUserBasedAuthentication {
/**
* Active Directory authentication
*/
static activeDirectory(directoryId) {
return new ActiveDirectoryAuthentication(directoryId);
}
/** Federated authentication */
static federated(samlProvider, selfServiceSamlProvider) {
return new FederatedAuthentication(samlProvider, selfServiceSamlProvider);
}
}
exports.ClientVpnUserBasedAuthentication = ClientVpnUserBasedAuthentication;
_a = JSII_RTTI_SYMBOL_1;
ClientVpnUserBasedAuthentication[_a] = { fqn: "@aws-cdk/aws-ec2.ClientVpnUserBasedAuthentication", version: "1.204.0" };
/**
* Active Directory authentication
*/
class ActiveDirectoryAuthentication extends ClientVpnUserBasedAuthentication {
constructor(directoryId) {
super();
this.directoryId = directoryId;
}
render() {
return {
type: 'directory-service-authentication',
activeDirectory: { directoryId: this.directoryId },
};
}
}
/**
* Federated authentication
*/
class FederatedAuthentication extends ClientVpnUserBasedAuthentication {
constructor(samlProvider, selfServiceSamlProvider) {
super();
this.samlProvider = samlProvider;
this.selfServiceSamlProvider = selfServiceSamlProvider;
}
render() {
return {
type: 'federated-authentication',
federatedAuthentication: {
samlProviderArn: this.samlProvider.samlProviderArn,
selfServiceSamlProviderArn: this.selfServiceSamlProvider?.samlProviderArn,
},
};
}
}
/**
* A client VPN connnection
*/
class ClientVpnEndpoint extends core_1.Resource {
constructor(scope, id, props) {
super(scope, id);
this._targetNetworksAssociated = new core_1.ConcreteDependable();
try {
jsiiDeprecationWarnings._aws_cdk_aws_ec2_ClientVpnEndpointProps(props);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, ClientVpnEndpoint);
}
throw error;
}
if (!core_1.Token.isUnresolved(props.vpc.vpcCidrBlock)) {
const clientCidr = new network_util_1.CidrBlock(props.cidr);
const vpcCidr = new network_util_1.CidrBlock(props.vpc.vpcCidrBlock);
if (vpcCidr.containsCidr(clientCidr)) {
throw new Error('The client CIDR cannot overlap with the local CIDR of the VPC');
}
}
if (props.dnsServers && props.dnsServers.length > 2) {
throw new Error('A client VPN endpoint can have up to two DNS servers');
}
if (props.logging == false && (props.logGroup || props.logStream)) {
throw new Error('Cannot specify `logGroup` or `logStream` when logging is disabled');
}
if (props.clientConnectionHandler
&& !core_1.Token.isUnresolved(props.clientConnectionHandler.functionName)
&& !props.clientConnectionHandler.functionName.startsWith('AWSClientVPN-')) {
throw new Error('The name of the Lambda function must begin with the `AWSClientVPN-` prefix');
}
if (props.clientLoginBanner
&& !core_1.Token.isUnresolved(props.clientLoginBanner)
&& props.clientLoginBanner.length > 1400) {
throw new Error(`The maximum length for the client login banner is 1400, got ${props.clientLoginBanner.length}`);
}
const logging = props.logging ?? true;
const logGroup = logging
? props.logGroup ?? new logs.LogGroup(this, 'LogGroup')
: undefined;
const securityGroups = props.securityGroups ?? [new security_group_1.SecurityGroup(this, 'SecurityGroup', {
vpc: props.vpc,
})];
this.connections = new connections_1.Connections({ securityGroups });
const endpoint = new ec2_generated_1.CfnClientVpnEndpoint(this, 'Resource', {
authenticationOptions: renderAuthenticationOptions(props.clientCertificateArn, props.userBasedAuthentication),
clientCidrBlock: props.cidr,
clientConnectOptions: props.clientConnectionHandler
? {
enabled: true,
lambdaFunctionArn: props.clientConnectionHandler.functionArn,
}
: undefined,
connectionLogOptions: {
enabled: logging,
cloudwatchLogGroup: logGroup?.logGroupName,
cloudwatchLogStream: props.logStream?.logStreamName,
},
description: props.description,
dnsServers: props.dnsServers,
securityGroupIds: securityGroups.map(s => s.securityGroupId),
selfServicePortal: booleanToEnabledDisabled(props.selfServicePortal),
serverCertificateArn: props.serverCertificateArn,
splitTunnel: props.splitTunnel,
transportProtocol: props.transportProtocol,
vpcId: props.vpc.vpcId,
vpnPort: props.port,
sessionTimeoutHours: props.sessionTimeout,
clientLoginBannerOptions: props.clientLoginBanner
? {
enabled: true,
bannerText: props.clientLoginBanner,
}
: undefined,
});
this.endpointId = endpoint.ref;
if (props.userBasedAuthentication && (props.selfServicePortal ?? true)) {
// Output self-service portal URL
new core_1.CfnOutput(this, 'SelfServicePortalUrl', {
value: `https://self-service.clientvpn.amazonaws.com/endpoints/${this.endpointId}`,
});
}
// Associate subnets
const subnetIds = props.vpc.selectSubnets(props.vpcSubnets).subnetIds;
if (core_1.Token.isUnresolved(subnetIds)) {
throw new Error('Cannot associate subnets when VPC are imported from parameters or exports containing lists of subnet IDs.');
}
for (const [idx, subnetId] of Object.entries(subnetIds)) {
this._targetNetworksAssociated.add(new ec2_generated_1.CfnClientVpnTargetNetworkAssociation(this, `Association${idx}`, {
clientVpnEndpointId: this.endpointId,
subnetId,
}));
}
this.targetNetworksAssociated = this._targetNetworksAssociated;
if (props.authorizeAllUsersToVpcCidr ?? true) {
this.addAuthorizationRule('AuthorizeAll', {
cidr: props.vpc.vpcCidrBlock,
});
}
}
/**
* Import an existing client VPN endpoint
*/
static fromEndpointAttributes(scope, id, attrs) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_ec2_ClientVpnEndpointAttributes(attrs);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.fromEndpointAttributes);
}
throw error;
}
class Import extends core_1.Resource {
constructor() {
super(...arguments);
this.endpointId = attrs.endpointId;
this.connections = new connections_1.Connections({ securityGroups: attrs.securityGroups });
this.targetNetworksAssociated = new core_1.ConcreteDependable();
}
}
return new Import(scope, id);
}
/**
* Adds an authorization rule to this endpoint
*/
addAuthorizationRule(id, props) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_ec2_ClientVpnAuthorizationRuleOptions(props);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.addAuthorizationRule);
}
throw error;
}
return new client_vpn_authorization_rule_1.ClientVpnAuthorizationRule(this, id, {
...props,
clientVpnEndpoint: this,
});
}
/**
* Adds a route to this endpoint
*/
addRoute(id, props) {
try {
jsiiDeprecationWarnings._aws_cdk_aws_ec2_ClientVpnRouteOptions(props);
}
catch (error) {
if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
Error.captureStackTrace(error, this.addRoute);
}
throw error;
}
return new client_vpn_route_1.ClientVpnRoute(this, id, {
...props,
clientVpnEndpoint: this,
});
}
}
exports.ClientVpnEndpoint = ClientVpnEndpoint;
_b = JSII_RTTI_SYMBOL_1;
ClientVpnEndpoint[_b] = { fqn: "@aws-cdk/aws-ec2.ClientVpnEndpoint", version: "1.204.0" };
function renderAuthenticationOptions(clientCertificateArn, userBasedAuthentication) {
const authenticationOptions = [];
if (clientCertificateArn) {
authenticationOptions.push({
type: 'certificate-authentication',
mutualAuthentication: {
clientRootCertificateChainArn: clientCertificateArn,
},
});
}
if (userBasedAuthentication) {
authenticationOptions.push(userBasedAuthentication.render());
}
if (authenticationOptions.length === 0) {
throw new Error('A client VPN endpoint must use at least one authentication option');
}
return authenticationOptions;
}
function booleanToEnabledDisabled(val) {
switch (val) {
case undefined:
return undefined;
case true:
return 'enabled';
case false:
return 'disabled';
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client-vpn-endpoint.js","sourceRoot":"","sources":["client-vpn-endpoint.ts"],"names":[],"mappings":"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","sourcesContent":["import { ISamlProvider } from '@aws-cdk/aws-iam';\nimport * as logs from '@aws-cdk/aws-logs';\nimport { CfnOutput, ConcreteDependable, IDependable, Resource, Token } from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { ClientVpnAuthorizationRule, ClientVpnAuthorizationRuleOptions } from './client-vpn-authorization-rule';\nimport { IClientVpnConnectionHandler, IClientVpnEndpoint, TransportProtocol, VpnPort } from './client-vpn-endpoint-types';\nimport { ClientVpnRoute, ClientVpnRouteOptions } from './client-vpn-route';\nimport { Connections } from './connections';\nimport { CfnClientVpnEndpoint, CfnClientVpnTargetNetworkAssociation } from './ec2.generated';\nimport { CidrBlock } from './network-util';\nimport { ISecurityGroup, SecurityGroup } from './security-group';\nimport { IVpc, SubnetSelection } from './vpc';\n\n/**\n * Options for a client VPN endpoint\n */\nexport interface ClientVpnEndpointOptions {\n  /**\n   * The IPv4 address range, in CIDR notation, from which to assign client IP\n   * addresses. The address range cannot overlap with the local CIDR of the VPC\n   * in which the associated subnet is located, or the routes that you add manually.\n   *\n   * Changing the address range will replace the Client VPN endpoint.\n   *\n   * The CIDR block should be /22 or greater.\n   */\n  readonly cidr: string;\n\n  /**\n   * The ARN of the client certificate for mutual authentication.\n   *\n   * The certificate must be signed by a certificate authority (CA) and it must\n   * be provisioned in AWS Certificate Manager (ACM).\n   *\n   * @default - use user-based authentication\n   */\n  readonly clientCertificateArn?: string;\n\n  /**\n   * The type of user-based authentication to use.\n   *\n   * @see https://docs.aws.amazon.com/vpn/latest/clientvpn-admin/client-authentication.html\n   *\n   * @default - use mutual authentication\n   */\n  readonly userBasedAuthentication?: ClientVpnUserBasedAuthentication;\n\n  /**\n   * Whether to enable connections logging\n   *\n   * @default true\n   */\n  readonly logging?: boolean;\n\n  /**\n   * A CloudWatch Logs log group for connection logging\n   *\n   * @default - a new group is created\n   */\n  readonly logGroup?: logs.ILogGroup;\n\n  /**\n   * A CloudWatch Logs log stream for connection logging\n   *\n   * @default - a new stream is created\n   */\n  readonly logStream?: logs.ILogStream;\n\n  /**\n   * The AWS Lambda function used for connection authorization\n   *\n   * The name of the Lambda function must begin with the `AWSClientVPN-` prefix\n   *\n   * @default - no connection handler\n   */\n  readonly clientConnectionHandler?: IClientVpnConnectionHandler;\n\n  /**\n   * A brief description of the Client VPN endpoint.\n   *\n   * @default - no description\n   */\n  readonly description?: string;\n\n  /**\n   * The security groups to apply to the target network.\n   *\n   * @default - a new security group is created\n   */\n  readonly securityGroups?: ISecurityGroup[];\n\n  /**\n   * Specify whether to enable the self-service portal for the Client VPN endpoint.\n   *\n   * @default true\n   */\n  readonly selfServicePortal?: boolean;\n\n  /**\n   * The ARN of the server certificate\n   */\n  readonly serverCertificateArn: string;\n\n  /**\n   * Indicates whether split-tunnel is enabled on the AWS Client VPN endpoint.\n   *\n   * @see https://docs.aws.amazon.com/vpn/latest/clientvpn-admin/split-tunnel-vpn.html\n   *\n   * @default false\n   */\n  readonly splitTunnel?: boolean;\n\n  /**\n   * The transport protocol to be used by the VPN session.\n   *\n   * @default TransportProtocol.UDP\n   */\n  readonly transportProtocol?: TransportProtocol;\n\n  /**\n   * The port number to assign to the Client VPN endpoint for TCP and UDP\n   * traffic.\n   *\n   * @default VpnPort.HTTPS\n   */\n  readonly port?: VpnPort;\n\n  /**\n   * Information about the DNS servers to be used for DNS resolution.\n   *\n   * A Client VPN endpoint can have up to two DNS servers.\n   *\n   * @default - use the DNS address configured on the device\n   */\n  readonly dnsServers?: string[];\n\n  /**\n   * Subnets to associate to the client VPN endpoint.\n   *\n   * @default - the VPC default strategy\n   */\n  readonly vpcSubnets?: SubnetSelection;\n\n  /**\n   * Whether to authorize all users to the VPC CIDR\n   *\n   * This automatically creates an authorization rule. Set this to `false` and\n   * use `addAuthorizationRule()` to create your own rules instead.\n   *\n   * @default true\n   */\n  readonly authorizeAllUsersToVpcCidr?: boolean;\n\n  /**\n   * The maximum VPN session duration time.\n   *\n   * @default ClientVpnSessionTimeout.TWENTY_FOUR_HOURS\n   */\n  readonly sessionTimeout?: ClientVpnSessionTimeout;\n\n  /**\n   * Customizable text that will be displayed in a banner on AWS provided clients\n   * when a VPN session is established.\n   *\n   * UTF-8 encoded characters only. Maximum of 1400 characters.\n   *\n   * @default - no banner is presented to the client\n   */\n  readonly clientLoginBanner?: string;\n}\n\n/**\n * Maximum VPN session duration time\n */\nexport enum ClientVpnSessionTimeout {\n  /** 8 hours */\n  EIGHT_HOURS = 8,\n  /** 10 hours */\n  TEN_HOURS = 10,\n  /** 12 hours */\n  TWELVE_HOURS = 12,\n  /** 24 hours */\n  TWENTY_FOUR_HOURS = 24,\n}\n\n/**\n * User-based authentication for a client VPN endpoint\n */\nexport abstract class ClientVpnUserBasedAuthentication {\n  /**\n   * Active Directory authentication\n   */\n  public static activeDirectory(directoryId: string): ClientVpnUserBasedAuthentication {\n    return new ActiveDirectoryAuthentication(directoryId);\n  }\n\n  /** Federated authentication */\n  public static federated(samlProvider: ISamlProvider, selfServiceSamlProvider?: ISamlProvider): ClientVpnUserBasedAuthentication {\n    return new FederatedAuthentication(samlProvider, selfServiceSamlProvider);\n  }\n\n  /** Renders the user based authentication */\n  public abstract render(): any;\n}\n\n/**\n * Active Directory authentication\n */\nclass ActiveDirectoryAuthentication extends ClientVpnUserBasedAuthentication {\n  constructor(private readonly directoryId: string) {\n    super();\n  }\n\n  render(): any {\n    return {\n      type: 'directory-service-authentication',\n      activeDirectory: { directoryId: this.directoryId },\n    };\n  }\n}\n\n/**\n * Federated authentication\n */\nclass FederatedAuthentication extends ClientVpnUserBasedAuthentication {\n  constructor(private readonly samlProvider: ISamlProvider, private readonly selfServiceSamlProvider?: ISamlProvider) {\n    super();\n  }\n\n  render(): any {\n    return {\n      type: 'federated-authentication',\n      federatedAuthentication: {\n        samlProviderArn: this.samlProvider.samlProviderArn,\n        selfServiceSamlProviderArn: this.selfServiceSamlProvider?.samlProviderArn,\n      },\n    };\n  }\n}\n\n/**\n * Properties for a client VPN endpoint\n */\nexport interface ClientVpnEndpointProps extends ClientVpnEndpointOptions {\n  /**\n   * The VPC to connect to.\n   */\n  readonly vpc: IVpc;\n}\n\n/**\n * Attributes when importing an existing client VPN endpoint\n */\nexport interface ClientVpnEndpointAttributes {\n  /**\n   * The endpoint ID\n   */\n  readonly endpointId: string;\n\n  /**\n   * The security groups associated with the endpoint\n   */\n  readonly securityGroups: ISecurityGroup[];\n}\n\n/**\n * A client VPN connnection\n */\nexport class ClientVpnEndpoint extends Resource implements IClientVpnEndpoint {\n  /**\n   * Import an existing client VPN endpoint\n   */\n  public static fromEndpointAttributes(scope: Construct, id: string, attrs: ClientVpnEndpointAttributes): IClientVpnEndpoint {\n    class Import extends Resource implements IClientVpnEndpoint {\n      public readonly endpointId = attrs.endpointId;\n      public readonly connections = new Connections({ securityGroups: attrs.securityGroups });\n      public readonly targetNetworksAssociated: IDependable = new ConcreteDependable();\n    }\n    return new Import(scope, id);\n  }\n\n  public readonly endpointId: string;\n\n  /**\n   * Allows specify security group connections for the endpoint.\n   */\n  public readonly connections: Connections;\n\n  public readonly targetNetworksAssociated: IDependable;\n\n  private readonly _targetNetworksAssociated = new ConcreteDependable();\n\n  constructor(scope: Construct, id: string, props: ClientVpnEndpointProps) {\n    super(scope, id);\n\n    if (!Token.isUnresolved(props.vpc.vpcCidrBlock)) {\n      const clientCidr = new CidrBlock(props.cidr);\n      const vpcCidr = new CidrBlock(props.vpc.vpcCidrBlock);\n      if (vpcCidr.containsCidr(clientCidr)) {\n        throw new Error('The client CIDR cannot overlap with the local CIDR of the VPC');\n      }\n    }\n\n    if (props.dnsServers && props.dnsServers.length > 2) {\n      throw new Error('A client VPN endpoint can have up to two DNS servers');\n    }\n\n    if (props.logging == false && (props.logGroup || props.logStream)) {\n      throw new Error('Cannot specify `logGroup` or `logStream` when logging is disabled');\n    }\n\n    if (props.clientConnectionHandler\n      && !Token.isUnresolved(props.clientConnectionHandler.functionName)\n      && !props.clientConnectionHandler.functionName.startsWith('AWSClientVPN-')) {\n      throw new Error('The name of the Lambda function must begin with the `AWSClientVPN-` prefix');\n    }\n\n    if (props.clientLoginBanner\n      && !Token.isUnresolved(props.clientLoginBanner)\n      && props.clientLoginBanner.length > 1400) {\n      throw new Error(`The maximum length for the client login banner is 1400, got ${props.clientLoginBanner.length}`);\n    }\n\n    const logging = props.logging ?? true;\n    const logGroup = logging\n      ? props.logGroup ?? new logs.LogGroup(this, 'LogGroup')\n      : undefined;\n\n    const securityGroups = props.securityGroups ?? [new SecurityGroup(this, 'SecurityGroup', {\n      vpc: props.vpc,\n    })];\n    this.connections = new Connections({ securityGroups });\n\n    const endpoint = new CfnClientVpnEndpoint(this, 'Resource', {\n      authenticationOptions: renderAuthenticationOptions(props.clientCertificateArn, props.userBasedAuthentication),\n      clientCidrBlock: props.cidr,\n      clientConnectOptions: props.clientConnectionHandler\n        ? {\n          enabled: true,\n          lambdaFunctionArn: props.clientConnectionHandler.functionArn,\n        }\n        : undefined,\n      connectionLogOptions: {\n        enabled: logging,\n        cloudwatchLogGroup: logGroup?.logGroupName,\n        cloudwatchLogStream: props.logStream?.logStreamName,\n      },\n      description: props.description,\n      dnsServers: props.dnsServers,\n      securityGroupIds: securityGroups.map(s => s.securityGroupId),\n      selfServicePortal: booleanToEnabledDisabled(props.selfServicePortal),\n      serverCertificateArn: props.serverCertificateArn,\n      splitTunnel: props.splitTunnel,\n      transportProtocol: props.transportProtocol,\n      vpcId: props.vpc.vpcId,\n      vpnPort: props.port,\n      sessionTimeoutHours: props.sessionTimeout,\n      clientLoginBannerOptions: props.clientLoginBanner\n        ? {\n          enabled: true,\n          bannerText: props.clientLoginBanner,\n        }\n        : undefined,\n    });\n\n    this.endpointId = endpoint.ref;\n\n    if (props.userBasedAuthentication && (props.selfServicePortal ?? true)) {\n      // Output self-service portal URL\n      new CfnOutput(this, 'SelfServicePortalUrl', {\n        value: `https://self-service.clientvpn.amazonaws.com/endpoints/${this.endpointId}`,\n      });\n    }\n\n    // Associate subnets\n    const subnetIds = props.vpc.selectSubnets(props.vpcSubnets).subnetIds;\n\n    if (Token.isUnresolved(subnetIds)) {\n      throw new Error('Cannot associate subnets when VPC are imported from parameters or exports containing lists of subnet IDs.');\n    }\n\n    for (const [idx, subnetId] of Object.entries(subnetIds)) {\n      this._targetNetworksAssociated.add(new CfnClientVpnTargetNetworkAssociation(this, `Association${idx}`, {\n        clientVpnEndpointId: this.endpointId,\n        subnetId,\n      }));\n    }\n    this.targetNetworksAssociated = this._targetNetworksAssociated;\n\n    if (props.authorizeAllUsersToVpcCidr ?? true) {\n      this.addAuthorizationRule('AuthorizeAll', {\n        cidr: props.vpc.vpcCidrBlock,\n      });\n    }\n  }\n\n  /**\n   * Adds an authorization rule to this endpoint\n   */\n  public addAuthorizationRule(id: string, props: ClientVpnAuthorizationRuleOptions): ClientVpnAuthorizationRule {\n    return new ClientVpnAuthorizationRule(this, id, {\n      ...props,\n      clientVpnEndpoint: this,\n    });\n  }\n\n  /**\n   * Adds a route to this endpoint\n   */\n  public addRoute(id: string, props: ClientVpnRouteOptions): ClientVpnRoute {\n    return new ClientVpnRoute(this, id, {\n      ...props,\n      clientVpnEndpoint: this,\n    });\n  }\n}\n\nfunction renderAuthenticationOptions(\n  clientCertificateArn?: string,\n  userBasedAuthentication?: ClientVpnUserBasedAuthentication): CfnClientVpnEndpoint.ClientAuthenticationRequestProperty[] {\n  const authenticationOptions: CfnClientVpnEndpoint.ClientAuthenticationRequestProperty[] = [];\n\n  if (clientCertificateArn) {\n    authenticationOptions.push({\n      type: 'certificate-authentication',\n      mutualAuthentication: {\n        clientRootCertificateChainArn: clientCertificateArn,\n      },\n    });\n  }\n\n  if (userBasedAuthentication) {\n    authenticationOptions.push(userBasedAuthentication.render());\n  }\n\n  if (authenticationOptions.length === 0) {\n    throw new Error('A client VPN endpoint must use at least one authentication option');\n  }\n  return authenticationOptions;\n}\n\nfunction booleanToEnabledDisabled(val?: boolean): 'enabled' | 'disabled' | undefined {\n  switch (val) {\n    case undefined:\n      return undefined;\n    case true:\n      return 'enabled';\n    case false:\n      return 'disabled';\n  }\n}\n"]}