UNPKG

cdk-efs-assets

Version:

Amazon EFS assets from Github repositories or S3 buckets

309 lines 43.1 kB
"use strict"; var _a, _b, _c, _d; Object.defineProperty(exports, "__esModule", { value: true }); exports.SyncedAccessPoint = exports.SyncEngine = exports.S3ArchiveSyncSource = exports.GithubSyncSource = exports.SyncSource = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const path = require("path"); const url_1 = require("url"); const cdk = require("aws-cdk-lib"); const ec2 = require("aws-cdk-lib/aws-ec2"); const ecs = require("aws-cdk-lib/aws-ecs"); const efs = require("aws-cdk-lib/aws-efs"); const aws_events_targets_1 = require("aws-cdk-lib/aws-events-targets"); const aws_iam_1 = require("aws-cdk-lib/aws-iam"); const lambda = require("aws-cdk-lib/aws-lambda"); const secretsmanager = require("aws-cdk-lib/aws-secretsmanager"); const cr = require("aws-cdk-lib/custom-resources"); const cdk_fargate_run_task_1 = require("cdk-fargate-run-task"); const efs_fargate_task_1 = require("./efs-fargate-task"); class SyncSource { static github(props) { return new GithubSyncSource(props); } static s3Archive(props) { return new S3ArchiveSyncSource(props); } } exports.SyncSource = SyncSource; _a = JSII_RTTI_SYMBOL_1; SyncSource[_a] = { fqn: "cdk-efs-assets.SyncSource", version: "0.3.95" }; class GithubSyncSource extends SyncSource { constructor(props) { super(); this.props = props; } /** * @internal * @param accessPoint The EFS Access Point */ _createHandler(accessPoint) { const stack = cdk.Stack.of(accessPoint); const region = stack.region; const vpcSubnets = this.props.vpcSubnets ?? { subnetType: ec2.SubnetType.PRIVATE_WITH_NAT }; const timeout = this.props.timeout ?? cdk.Duration.minutes(3); let syncDirectoryPath; if (this.props.syncDirectoryPath === undefined) { // if property is unspecified, use repository name as output directory const parsed = new url_1.URL(this.props.repository); syncDirectoryPath = '/' + path.basename(parsed.pathname, '.git'); } else { syncDirectoryPath = this.props.syncDirectoryPath; } const lambdaEnv = { REPOSITORY_URI: this.props.repository, MOUNT_TARGET: '/mnt/efsmount', SYNC_PATH: syncDirectoryPath, }; if (this.props.secret) { lambdaEnv.GITHUB_SECRET_ID = this.props.secret.id; lambdaEnv.GITHUB_SECRET_KEY = this.props.secret.key; } const handler = new lambda.Function(accessPoint, 'GithubHandler', { runtime: lambda.Runtime.PYTHON_3_8, code: lambda.Code.fromAsset(path.join(__dirname, '../lambda-handler', 'github-sync')), handler: 'index.on_event', layers: [ lambda.LayerVersion.fromLayerVersionArn(accessPoint, 'GitLayer', `arn:aws:lambda:${region}:553035198032:layer:git-lambda2:7`), ], filesystem: lambda.FileSystem.fromEfsAccessPoint(accessPoint, '/mnt/efsmount'), vpcSubnets: vpcSubnets, vpc: this.props.vpc, memorySize: 512, timeout: timeout, environment: lambdaEnv, currentVersionOptions: { provisionedConcurrentExecutions: 1, }, }); if (this.props.secret?.id) { // format the arn e.g. 'arn:aws:secretsmanager:eu-west-1:111111111111:secret:MySecret'; const secretPartialArn = stack.formatArn({ service: 'secretsmanager', resource: 'secret', resourceName: this.props.secret?.id, arnFormat: cdk.ArnFormat.COLON_RESOURCE_NAME, }); const secret = secretsmanager.Secret.fromSecretAttributes(stack, 'GithubSecret', { secretPartialArn, }); // allow lambda to read the secret secret.grantRead(handler); } return handler; } /** * @internal * @param id The task ID. * @param accessPoint The EFS access point. */ _createFargateTask(id, accessPoint) { const stack = cdk.Stack.of(accessPoint); const mountTarget = '/mnt/efsmount'; let syncDirectoryPath; if (this.props.syncDirectoryPath === undefined) { // if property is unspecified, use repository name as output directory const parsed = new url_1.URL(this.props.repository); syncDirectoryPath = '/' + path.basename(parsed.pathname, '.git'); } else { syncDirectoryPath = this.props.syncDirectoryPath; } const environment = { REPOSITORY_URI: this.props.repository, MOUNT_TARGET: mountTarget, SYNC_PATH: syncDirectoryPath, }; if (this.props.secret) { environment.GITHUB_SECRET_ID = this.props.secret.id; environment.GITHUB_SECRET_KEY = this.props.secret.key; } let secret; if (this.props.secret?.id) { // format the arn e.g. 'arn:aws:secretsmanager:eu-west-1:111111111111:secret:MySecret'; const secretPartialArn = stack.formatArn({ service: 'secretsmanager', resource: 'secret', resourceName: this.props.secret?.id, arnFormat: cdk.ArnFormat.COLON_RESOURCE_NAME, }); secret = secretsmanager.Secret.fromSecretAttributes(stack, 'GithubSecret', { secretPartialArn, }); } const fargateSyncTask = new efs_fargate_task_1.EfsFargateTask(stack, id, { vpc: this.props.vpc, accessPoint, efsMountTarget: mountTarget, syncContainer: { image: ecs.ContainerImage.fromAsset(path.join(__dirname, '../docker.d')), command: ['/root/githubsync.sh'], environment, secrets: secret ? { OAUTH_TOKEN: ecs.Secret.fromSecretsManager(secret, 'oauth_token'), } : undefined, }, }); // allow task to read the secret secret?.grantRead(fargateSyncTask.task.taskRole); return { task: fargateSyncTask.task, securityGroup: fargateSyncTask.securityGroup, }; } ; } exports.GithubSyncSource = GithubSyncSource; _b = JSII_RTTI_SYMBOL_1; GithubSyncSource[_b] = { fqn: "cdk-efs-assets.GithubSyncSource", version: "0.3.95" }; class S3ArchiveSyncSource extends SyncSource { constructor(props) { super(); this.props = props; } /** * @internal * @param accessPoint The EFS access point. */ _createHandler(accessPoint) { const vpcSubnets = this.props.vpcSubnets ?? { subnetType: ec2.SubnetType.PRIVATE_WITH_NAT }; const syncOnUpdate = this.props.syncOnUpdate ?? true; const timeout = this.props.timeout ?? cdk.Duration.minutes(3); const filename = path.basename(this.props.zipFilePath, '.zip'); let syncDirectoryPath; if (this.props.syncDirectoryPath === undefined) { // if property is unspecified, use zip file name as output directory syncDirectoryPath = '/' + filename; } else { syncDirectoryPath = this.props.syncDirectoryPath; } const handler = new lambda.Function(accessPoint, 'SyncHandler', { runtime: lambda.Runtime.PYTHON_3_8, code: lambda.Code.fromAsset(path.join(__dirname, '../lambda-handler', 's3-archive-sync')), handler: 'index.on_event', filesystem: lambda.FileSystem.fromEfsAccessPoint(accessPoint, '/mnt/efsmount'), vpcSubnets: vpcSubnets, vpc: this.props.vpc, memorySize: 512, timeout: timeout, environment: { MOUNT_TARGET: '/mnt/efsmount', BUCKET_NAME: this.props.bucket.bucketName, ZIPPED_KEY: this.props.zipFilePath, SYNC_PATH: syncDirectoryPath, }, currentVersionOptions: { provisionedConcurrentExecutions: 1, }, initialPolicy: [ new aws_iam_1.PolicyStatement({ actions: ['s3:GetObject*'], resources: ['arn:aws:s3:::' + this.props.bucket.bucketName + '/' + this.props.zipFilePath], }), ], }); if (syncOnUpdate) { // In order to support bucket notifications for imported IBucket objects, onCloudTrailWriteObject is used. // TODO: When https://github.com/aws/aws-cdk/issues/2004 is closed, can use handler.addEventSource instead. /* handler.addEventSource( new S3EventSource(props.bucket, { events: [s3.EventType.OBJECT_CREATED], filters: [{ prefix: props.zipFilePath }] }) ); */ this.props.bucket.onCloudTrailWriteObject('S3FileListener-' + filename, { paths: [this.props.zipFilePath], target: new aws_events_targets_1.LambdaFunction(handler), }); } return handler; } /** * @internal * @param id The Fargate task ID. * @param accessPoint The EFS access point. */ _createFargateTask(id, accessPoint) { const stack = cdk.Stack.of(accessPoint); const mountTarget = '/mnt/efsmount'; const filename = path.basename(this.props.zipFilePath, '.zip'); const syncDirectoryPath = this.props.syncDirectoryPath === undefined ? '/' + filename : this.props.syncDirectoryPath; const environment = { MOUNT_TARGET: '/mnt/efsmount', BUCKET_NAME: this.props.bucket.bucketName, ZIPPED_KEY: this.props.zipFilePath, SYNC_PATH: syncDirectoryPath, }; const fargateSyncTask = new efs_fargate_task_1.EfsFargateTask(stack, id, { vpc: this.props.vpc, accessPoint, efsMountTarget: mountTarget, syncContainer: { image: ecs.ContainerImage.fromAsset(path.join(__dirname, '../docker.d')), command: ['/root/s3sync.sh'], environment, }, }); // allow ecs task to get the s3 object fargateSyncTask.task.addToTaskRolePolicy(new aws_iam_1.PolicyStatement({ actions: ['s3:GetObject*'], resources: [ stack.formatArn({ service: 's3', resource: this.props.bucket.bucketName, region: '', account: '', arnFormat: cdk.ArnFormat.SLASH_RESOURCE_NAME, resourceName: this.props.zipFilePath, }), ], })); return { task: fargateSyncTask.task, securityGroup: fargateSyncTask.securityGroup, }; } ; } exports.S3ArchiveSyncSource = S3ArchiveSyncSource; _c = JSII_RTTI_SYMBOL_1; S3ArchiveSyncSource[_c] = { fqn: "cdk-efs-assets.S3ArchiveSyncSource", version: "0.3.95" }; var SyncEngine; (function (SyncEngine) { SyncEngine[SyncEngine["FARGATE"] = 0] = "FARGATE"; SyncEngine[SyncEngine["LAMBDA"] = 1] = "LAMBDA"; })(SyncEngine = exports.SyncEngine || (exports.SyncEngine = {})); class SyncedAccessPoint extends efs.AccessPoint { constructor(scope, id, props) { super(scope, id, props); if (props.engine === SyncEngine.LAMBDA) { const handler = props.syncSource._createHandler(this); // create a custom resource to trigger the sync const myProvider = new cr.Provider(this, 'Provider', { onEventHandler: handler, }); new cdk.CustomResource(this, 'SyncTrigger', { serviceToken: myProvider.serviceToken }); // ensure the mount targets are available as dependency for the sync function handler.node.addDependency(props.fileSystem.mountTargetsAvailable); } else { const taskConfig = props.syncSource._createFargateTask(`${id}FargateTask`, this); const cluster = new ecs.Cluster(this, 'Cluster', { vpc: props.vpc }); const runTask = new cdk_fargate_run_task_1.RunTask(this, 'SyncTrigger', { task: taskConfig.task, securityGroup: taskConfig.securityGroup, cluster, fargatePlatformVersion: cdk_fargate_run_task_1.PlatformVersion.V1_4_0, }); runTask.node.addDependency(props.fileSystem.mountTargetsAvailable); } } } exports.SyncedAccessPoint = SyncedAccessPoint; _d = JSII_RTTI_SYMBOL_1; SyncedAccessPoint[_d] = { fqn: "cdk-efs-assets.SyncedAccessPoint", version: "0.3.95" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"synced-access-point.js","sourceRoot":"","sources":["../src/synced-access-point.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { URL } from 'url';\nimport * as cdk from 'aws-cdk-lib';\nimport * as ec2 from 'aws-cdk-lib/aws-ec2';\nimport * as ecs from 'aws-cdk-lib/aws-ecs';\nimport * as efs from 'aws-cdk-lib/aws-efs';\nimport { LambdaFunction } from 'aws-cdk-lib/aws-events-targets';\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport * as secretsmanager from 'aws-cdk-lib/aws-secretsmanager';\nimport * as cr from 'aws-cdk-lib/custom-resources';\nimport { PlatformVersion, RunTask } from 'cdk-fargate-run-task';\nimport { Construct } from 'constructs';\nimport { EfsFargateTask } from './efs-fargate-task';\n\n\nexport interface SyncSourceProps {\n  /**\n   * The VPC of the Amazon EFS Filesystem.\n   */\n  readonly vpc: ec2.IVpc;\n  /**\n   * Where to place the network interfaces within the VPC.\n   */\n  readonly vpcSubnets?: ec2.SubnetSelection;\n  /**\n   * Timeout duration for sync Lambda function. (optional, default: Duration.minutes(3))\n   */\n  readonly timeout?: cdk.Duration;\n  /**\n   * The (absolute) directory path inside the EFS AccessPoint to sync files to. Specify '/' to restore synced files to the root\n   * directory. (optional, default: a source-specific directory path. For example, for the GitHub source, the default\n   * behavior is to restore to a directory matching the name of the repository)\n   */\n  readonly syncDirectoryPath?: string;\n}\n\nexport interface GithubSecret {\n  /**\n   * The secret ID from AWS Secrets Manager\n   */\n  readonly id: string;\n  /**\n   * The key of the secret\n   */\n  readonly key: string;\n}\n\nexport interface GithubSourceProps extends SyncSourceProps {\n  /**\n   * The github repository HTTP URI.\n   */\n  readonly repository: string;\n  /**\n   * The github secret for the private repository\n   */\n  readonly secret?: GithubSecret;\n}\n\nexport interface S3ArchiveSourceProps extends SyncSourceProps {\n  /**\n   * The S3 bucket containing the archive file.\n   */\n  readonly bucket: s3.IBucket;\n\n  /**\n   * The path of the zip file to extract in the S3 bucket.\n   */\n  readonly zipFilePath: string;\n\n  /**\n   * If this is set to true, then whenever a new object is uploaded to the specified path,\n   * an EFS sync will be triggered. Currently, this functionality depends on at least one CloudTrail Trail\n   * existing in your account that captures the S3 event.\n   *\n   * The option is only available with the `LAMBDA` sync engine.\n   *\n   * @default true\n   */\n  readonly syncOnUpdate?: boolean;\n}\n\nexport interface FargateTaskConfig {\n  readonly task: ecs.TaskDefinition;\n  /**\n   * The security group of the fargate task\n   */\n  readonly securityGroup: ec2.ISecurityGroup;\n}\n\nexport abstract class SyncSource {\n  public static github(props: GithubSourceProps): SyncSource {\n    return new GithubSyncSource(props);\n  }\n\n  public static s3Archive(props: S3ArchiveSourceProps): SyncSource {\n    return new S3ArchiveSyncSource(props);\n  }\n\n  /** @internal */\n  abstract _createHandler(accessPoint: efs.AccessPoint): lambda.Function;\n\n  /** @internal */\n  abstract _createFargateTask(id: string, accessPoint: efs.AccessPoint): FargateTaskConfig;\n}\n\nexport class GithubSyncSource extends SyncSource {\n  private readonly props: GithubSourceProps;\n\n  constructor(props: GithubSourceProps) {\n    super();\n    this.props = props;\n  }\n\n  /**\n   * @internal\n   * @param accessPoint The EFS Access Point\n   */\n  _createHandler(accessPoint: efs.AccessPoint): lambda.Function {\n    const stack = cdk.Stack.of(accessPoint);\n    const region = stack.region;\n\n    const vpcSubnets = this.props.vpcSubnets ?? { subnetType: ec2.SubnetType.PRIVATE_WITH_NAT };\n    const timeout = this.props.timeout ?? cdk.Duration.minutes(3);\n\n    let syncDirectoryPath;\n    if (this.props.syncDirectoryPath === undefined) {\n      // if property is unspecified, use repository name as output directory\n\n      const parsed = new URL(this.props.repository);\n      syncDirectoryPath = '/' + path.basename(parsed.pathname, '.git');\n    } else {\n      syncDirectoryPath = this.props.syncDirectoryPath;\n    }\n\n    const lambdaEnv: { [key: string]: string } = {\n      REPOSITORY_URI: this.props.repository,\n      MOUNT_TARGET: '/mnt/efsmount',\n      SYNC_PATH: syncDirectoryPath,\n    };\n\n    if (this.props.secret) {\n      lambdaEnv.GITHUB_SECRET_ID = this.props.secret.id;\n      lambdaEnv.GITHUB_SECRET_KEY = this.props.secret.key;\n    }\n\n    const handler = new lambda.Function(accessPoint, 'GithubHandler', {\n      runtime: lambda.Runtime.PYTHON_3_8,\n      code: lambda.Code.fromAsset(path.join(__dirname, '../lambda-handler', 'github-sync')),\n      handler: 'index.on_event',\n      layers: [\n        lambda.LayerVersion.fromLayerVersionArn(\n          accessPoint,\n          'GitLayer',\n          `arn:aws:lambda:${region}:553035198032:layer:git-lambda2:7`,\n        ),\n      ],\n      filesystem: lambda.FileSystem.fromEfsAccessPoint(accessPoint, '/mnt/efsmount'),\n      vpcSubnets: vpcSubnets,\n      vpc: this.props.vpc,\n      memorySize: 512,\n      timeout: timeout,\n      environment: lambdaEnv,\n      currentVersionOptions: {\n        provisionedConcurrentExecutions: 1,\n      },\n    });\n\n    if (this.props.secret?.id) {\n      // format the arn e.g. 'arn:aws:secretsmanager:eu-west-1:111111111111:secret:MySecret';\n      const secretPartialArn = stack.formatArn({\n        service: 'secretsmanager',\n        resource: 'secret',\n        resourceName: this.props.secret?.id,\n        arnFormat: cdk.ArnFormat.COLON_RESOURCE_NAME,\n      });\n      const secret = secretsmanager.Secret.fromSecretAttributes(stack, 'GithubSecret', {\n        secretPartialArn,\n      });\n      // allow lambda to read the secret\n      secret.grantRead(handler);\n    }\n\n    return handler;\n  }\n\n  /**\n   * @internal\n   * @param id The task ID.\n   * @param accessPoint The EFS access point.\n   */\n  _createFargateTask(id: string, accessPoint: efs.AccessPoint): FargateTaskConfig {\n    const stack = cdk.Stack.of(accessPoint);\n\n    const mountTarget = '/mnt/efsmount';\n\n    let syncDirectoryPath;\n    if (this.props.syncDirectoryPath === undefined) {\n      // if property is unspecified, use repository name as output directory\n      const parsed = new URL(this.props.repository);\n      syncDirectoryPath = '/' + path.basename(parsed.pathname, '.git');\n    } else {\n      syncDirectoryPath = this.props.syncDirectoryPath;\n    }\n\n    const environment: { [key: string]: string } = {\n      REPOSITORY_URI: this.props.repository,\n      MOUNT_TARGET: mountTarget,\n      SYNC_PATH: syncDirectoryPath,\n    };\n\n    if (this.props.secret) {\n      environment.GITHUB_SECRET_ID = this.props.secret.id;\n      environment.GITHUB_SECRET_KEY = this.props.secret.key;\n    }\n\n    let secret: secretsmanager.ISecret | undefined;\n\n    if (this.props.secret?.id) {\n      // format the arn e.g. 'arn:aws:secretsmanager:eu-west-1:111111111111:secret:MySecret';\n      const secretPartialArn = stack.formatArn({\n        service: 'secretsmanager',\n        resource: 'secret',\n        resourceName: this.props.secret?.id,\n        arnFormat: cdk.ArnFormat.COLON_RESOURCE_NAME,\n      });\n      secret = secretsmanager.Secret.fromSecretAttributes(stack, 'GithubSecret', {\n        secretPartialArn,\n      });\n    }\n\n    const fargateSyncTask = new EfsFargateTask(stack, id, {\n      vpc: this.props.vpc,\n      accessPoint,\n      efsMountTarget: mountTarget,\n      syncContainer: {\n        image: ecs.ContainerImage.fromAsset(path.join(__dirname, '../docker.d')),\n        command: ['/root/githubsync.sh'],\n        environment,\n        secrets: secret ? {\n          OAUTH_TOKEN: ecs.Secret.fromSecretsManager(secret, 'oauth_token'),\n        } : undefined,\n      },\n    });\n\n    // allow task to read the secret\n    secret?.grantRead(fargateSyncTask.task.taskRole);\n\n    return {\n      task: fargateSyncTask.task,\n      securityGroup: fargateSyncTask.securityGroup,\n    };\n  };\n}\n\nexport class S3ArchiveSyncSource extends SyncSource {\n  private readonly props: S3ArchiveSourceProps;\n\n  constructor(props: S3ArchiveSourceProps) {\n    super();\n    this.props = props;\n  }\n\n  /**\n   * @internal\n   * @param accessPoint The EFS access point.\n   */\n  _createHandler(accessPoint: efs.AccessPoint): lambda.Function {\n    const vpcSubnets = this.props.vpcSubnets ?? { subnetType: ec2.SubnetType.PRIVATE_WITH_NAT };\n    const syncOnUpdate = this.props.syncOnUpdate ?? true;\n    const timeout = this.props.timeout ?? cdk.Duration.minutes(3);\n\n    const filename = path.basename(this.props.zipFilePath, '.zip');\n\n    let syncDirectoryPath;\n    if (this.props.syncDirectoryPath === undefined) {\n      // if property is unspecified, use zip file name as output directory\n\n      syncDirectoryPath = '/' + filename;\n    } else {\n      syncDirectoryPath = this.props.syncDirectoryPath;\n    }\n\n    const handler = new lambda.Function(accessPoint, 'SyncHandler', {\n      runtime: lambda.Runtime.PYTHON_3_8,\n      code: lambda.Code.fromAsset(path.join(__dirname, '../lambda-handler', 's3-archive-sync')),\n      handler: 'index.on_event',\n      filesystem: lambda.FileSystem.fromEfsAccessPoint(accessPoint, '/mnt/efsmount'),\n      vpcSubnets: vpcSubnets,\n      vpc: this.props.vpc,\n      memorySize: 512,\n      timeout: timeout,\n      environment: {\n        MOUNT_TARGET: '/mnt/efsmount',\n        BUCKET_NAME: this.props.bucket.bucketName,\n        ZIPPED_KEY: this.props.zipFilePath,\n        SYNC_PATH: syncDirectoryPath,\n      },\n      currentVersionOptions: {\n        provisionedConcurrentExecutions: 1,\n      },\n      initialPolicy: [\n        new PolicyStatement({\n          actions: ['s3:GetObject*'],\n          resources: ['arn:aws:s3:::' + this.props.bucket.bucketName + '/' + this.props.zipFilePath],\n        }),\n      ],\n    });\n\n    if (syncOnUpdate) {\n      // In order to support bucket notifications for imported IBucket objects, onCloudTrailWriteObject is used.\n      // TODO: When https://github.com/aws/aws-cdk/issues/2004 is closed, can use handler.addEventSource instead.\n      /*\n      handler.addEventSource(\n        new S3EventSource(props.bucket, {\n          events: [s3.EventType.OBJECT_CREATED],\n          filters: [{ prefix: props.zipFilePath }]\n        })\n      );\n       */\n\n      this.props.bucket.onCloudTrailWriteObject('S3FileListener-' + filename, {\n        paths: [this.props.zipFilePath],\n        target: new LambdaFunction(handler),\n      });\n    }\n\n    return handler;\n  }\n\n  /**\n   * @internal\n   * @param id The Fargate task ID.\n   * @param accessPoint The EFS access point.\n   */\n  _createFargateTask(id: string, accessPoint: efs.AccessPoint): FargateTaskConfig {\n    const stack = cdk.Stack.of(accessPoint);\n\n    const mountTarget = '/mnt/efsmount';\n\n    const filename = path.basename(this.props.zipFilePath, '.zip');\n\n    const syncDirectoryPath = this.props.syncDirectoryPath === undefined ?\n      '/' + filename : this.props.syncDirectoryPath;\n\n    const environment: { [key: string]: string } = {\n      MOUNT_TARGET: '/mnt/efsmount',\n      BUCKET_NAME: this.props.bucket.bucketName,\n      ZIPPED_KEY: this.props.zipFilePath,\n      SYNC_PATH: syncDirectoryPath,\n    };\n\n    const fargateSyncTask = new EfsFargateTask(stack, id, {\n      vpc: this.props.vpc,\n      accessPoint,\n      efsMountTarget: mountTarget,\n      syncContainer: {\n        image: ecs.ContainerImage.fromAsset(path.join(__dirname, '../docker.d')),\n        command: ['/root/s3sync.sh'],\n        environment,\n      },\n    });\n\n    // allow ecs task to get the s3 object\n    fargateSyncTask.task.addToTaskRolePolicy(new PolicyStatement({\n      actions: ['s3:GetObject*'],\n      resources: [\n        stack.formatArn({\n          service: 's3',\n          resource: this.props.bucket.bucketName,\n          region: '',\n          account: '',\n          arnFormat: cdk.ArnFormat.SLASH_RESOURCE_NAME,\n          resourceName: this.props.zipFilePath,\n        }),\n      ],\n    }));\n\n    return {\n      task: fargateSyncTask.task,\n      securityGroup: fargateSyncTask.securityGroup,\n    };\n  };\n}\n\nexport enum SyncEngine {\n  FARGATE,\n  LAMBDA,\n}\n\nexport interface SyncedAccessPointProps extends efs.AccessPointProps {\n  readonly syncSource: SyncSource;\n  /**\n   * The VPC to run the sync job\n   */\n  readonly vpc: ec2.IVpc;\n  /**\n   * Trigger the sync with AWS Lambda or AWS Fargate.\n   */\n  readonly engine?: SyncEngine;\n}\n\nexport class SyncedAccessPoint extends efs.AccessPoint implements efs.IAccessPoint {\n  constructor(scope: Construct, id: string, props: SyncedAccessPointProps) {\n    super(scope, id, props);\n\n    if (props.engine === SyncEngine.LAMBDA) {\n      const handler = props.syncSource._createHandler(this);\n\n      // create a custom resource to trigger the sync\n      const myProvider = new cr.Provider(this, 'Provider', {\n        onEventHandler: handler,\n      });\n\n      new cdk.CustomResource(this, 'SyncTrigger', { serviceToken: myProvider.serviceToken });\n\n      // ensure the mount targets are available as dependency for the sync function\n      handler.node.addDependency(props.fileSystem.mountTargetsAvailable);\n    } else {\n      const taskConfig = props.syncSource._createFargateTask(`${id}FargateTask`, this);\n\n      const cluster = new ecs.Cluster(this, 'Cluster', { vpc: props.vpc });\n\n      const runTask = new RunTask(this, 'SyncTrigger', {\n        task: taskConfig.task,\n        securityGroup: taskConfig.securityGroup,\n        cluster,\n        fargatePlatformVersion: PlatformVersion.V1_4_0,\n      });\n\n      runTask.node.addDependency(props.fileSystem.mountTargetsAvailable);\n    }\n  }\n}\n"]}