UNPKG

turbo-remote-cache-construct

Version:

A Turborepo Remote Cache implementation using AWS API Gateway, Lambda, S3, and DynamoDB.

74 lines 12.9 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.TurboRemoteCache = void 0; const constructs_1 = require("constructs"); const s3 = require("aws-cdk-lib/aws-s3"); const dynamodb = require("aws-cdk-lib/aws-dynamodb"); const iam = require("aws-cdk-lib/aws-iam"); const cdk = require("aws-cdk-lib"); const lambda_1 = require("./lambda"); const api_1 = require("./api"); class TurboRemoteCache extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); const artifactsBucket = new s3.Bucket(this, 'ArtifactsBucket', { versioned: false, bucketName: 'turbo-remote-cache-artifacts', encryption: s3.BucketEncryption.S3_MANAGED, lifecycleRules: [ { expiration: cdk.Duration.days(30), }, ], cors: [ { allowedHeaders: ['*'], allowedMethods: [s3.HttpMethods.GET, s3.HttpMethods.PUT, s3.HttpMethods.HEAD], allowedOrigins: ['*'], exposedHeaders: [ 'User-Agent', 'Content-Type', 'Content-Length', 'x-artifact-duration', 'x-artifact-tag', ], } ], removalPolicy: cdk.RemovalPolicy.DESTROY, ...props.artifactsBucketProps, }); const s3Credentials = new iam.Role(this, 'S3CredentialsRole', { assumedBy: new iam.ServicePrincipal('apigateway.amazonaws.com'), managedPolicies: [ iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonS3FullAccess'), ], }); artifactsBucket.grantReadWrite(s3Credentials); const eventsTable = new dynamodb.Table(this, 'EventsTable', { tableName: 'turbo-remote-cache-events', partitionKey: { name: 'hash', type: dynamodb.AttributeType.STRING }, sortKey: { name: 'sessionId', type: dynamodb.AttributeType.STRING }, billingMode: dynamodb.BillingMode.PROVISIONED, readCapacity: 5, writeCapacity: 5, timeToLiveAttribute: 'ttl', removalPolicy: cdk.RemovalPolicy.DESTROY, ...props.eventsTableProps, }); const lambdaFunctions = new lambda_1.LambdaFunctions(this, 'LambdaFunctions', { artifactsBucket, eventsTable, lambdaProps: props.lambdaProps, authorizerFunction: props.authorizerFunction, userInfoFunction: props.userInfoFunction, }); const api = new api_1.APIGateway(this, 'APIGateway', { lambdaFunctions, artifactsBucket, s3Credentials, apiProps: props.apiProps, }); } } exports.TurboRemoteCache = TurboRemoteCache; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;AAAA,2CAAuC;AACvC,yCAAyC;AACzC,qDAAqD;AAErD,2CAA2C;AAC3C,mCAAmC;AACnC,qCAA2C;AAC3C,+BAAmC;AAuFnC,MAAa,gBAAiB,SAAQ,sBAAS;IAC7C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA4B;QACpE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,eAAe,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,iBAAiB,EAAE;YAC7D,SAAS,EAAE,KAAK;YAChB,UAAU,EAAE,8BAA8B;YAC1C,UAAU,EAAE,EAAE,CAAC,gBAAgB,CAAC,UAAU;YAC1C,cAAc,EAAE;gBACd;oBACE,UAAU,EAAE,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;iBAClC;aACF;YACD,IAAI,EAAE;gBACJ;oBACE,cAAc,EAAE,CAAC,GAAG,CAAC;oBACrB,cAAc,EAAE,CAAC,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC;oBAC7E,cAAc,EAAE,CAAC,GAAG,CAAC;oBACrB,cAAc,EAAE;wBACd,YAAY;wBACZ,cAAc;wBACd,gBAAgB;wBAChB,qBAAqB;wBACrB,gBAAgB;qBACjB;iBACF;aACF;YACD,aAAa,EAAE,GAAG,CAAC,aAAa,CAAC,OAAO;YACxC,GAAG,KAAK,CAAC,oBAAoB;SAC9B,CAAC,CAAC;QAEH,MAAM,aAAa,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,mBAAmB,EAAE;YAC5D,SAAS,EAAE,IAAI,GAAG,CAAC,gBAAgB,CAAC,0BAA0B,CAAC;YAC/D,eAAe,EAAE;gBACf,GAAG,CAAC,aAAa,CAAC,wBAAwB,CAAC,oBAAoB,CAAC;aACjE;SACF,CAAC,CAAC;QAEH,eAAe,CAAC,cAAc,CAAC,aAAa,CAAC,CAAC;QAE9C,MAAM,WAAW,GAAG,IAAI,QAAQ,CAAC,KAAK,CAAC,IAAI,EAAE,aAAa,EAAE;YAC1D,SAAS,EAAE,2BAA2B;YACtC,YAAY,EAAE,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YACnE,OAAO,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,IAAI,EAAE,QAAQ,CAAC,aAAa,CAAC,MAAM,EAAE;YACnE,WAAW,EAAE,QAAQ,CAAC,WAAW,CAAC,WAAW;YAC7C,YAAY,EAAE,CAAC;YACf,aAAa,EAAE,CAAC;YAChB,mBAAmB,EAAE,KAAK;YAC1B,aAAa,EAAE,GAAG,CAAC,aAAa,CAAC,OAAO;YACxC,GAAG,KAAK,CAAC,gBAAgB;SAC1B,CAAC,CAAC;QAGH,MAAM,eAAe,GAAG,IAAI,wBAAe,CAAC,IAAI,EAAE,iBAAiB,EAAE;YACnE,eAAe;YACf,WAAW;YACX,WAAW,EAAE,KAAK,CAAC,WAAW;YAC9B,kBAAkB,EAAE,KAAK,CAAC,kBAAkB;YAC5C,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;SACzC,CAAC,CAAC;QAEH,MAAM,GAAG,GAAG,IAAI,gBAAU,CAAC,IAAI,EAAE,YAAY,EAAE;YAC7C,eAAe;YACf,eAAe;YACf,aAAa;YACb,QAAQ,EAAE,KAAK,CAAC,QAAQ;SACzB,CAAC,CAAC;IACL,CAAC;CACF;AApED,4CAoEC","sourcesContent":["import { Construct } from 'constructs';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport * as dynamodb from 'aws-cdk-lib/aws-dynamodb';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as cdk from 'aws-cdk-lib';\nimport { LambdaFunctions } from './lambda';\nimport { APIGateway } from './api';\nimport * as apigateway from 'aws-cdk-lib/aws-apigateway';\nexport interface TurboRemoteCacheProps {\n  /**\n   * Turbo Token\n   * @deprecated Use authorizerFunction instead for authentication\n   * Used only if custom authorizer is not provided\n   * @example generate a random string using `openssl rand -base64 32`\n   */\n  turboToken?: string;\n\n  /**\n   * Custom authorizer lambda function for API authentication\n   * This function should return the teamId in the authorizer context\n   * Replaces the deprecated turboToken authentication\n   * @example\n   * ```typescript\n   * new TurboRemoteCache(this, 'Cache', {\n   *   authorizerFunction: myCustomAuthorizer,\n   * });\n   * ```\n   */\n  authorizerFunction?: lambda.Function;\n\n  /**\n   * Custom user info lambda function\n   * Implements the /v2/user endpoint for retrieving authenticated user information\n   * @example\n   * ```typescript\n   * new TurboRemoteCache(this, 'Cache', {\n   *   userInfoFunction: myUserInfoFunction,\n   * });\n   * ```\n   */\n  userInfoFunction?: lambda.Function;\n\n  /**\n   * API Gateway props\n   * @default\n   *  restApiName: 'Turborepo Remote Cache API',\n   *  description: 'Turborepo is an intelligent build system optimized for JavaScript and TypeScript codebases.',\n   *  cloudWatchRole: true,\n   *  deployOptions: {\n   *    documentationVersion: '8.0.0',\n   *    loggingLevel: apigateway.MethodLoggingLevel.INFO,\n   *    accessLogDestination: new apigateway.LogGroupLogDestination(logGroup),\n   *    accessLogFormat: apigateway.AccessLogFormat.jsonWithStandardFields(),\n   *    dataTraceEnabled: true,\n   *    tracingEnabled: true,\n   *  },\n   *  binaryMediaTypes: ['application/octet-stream'],\n   */\n  apiProps?: Partial<apigateway.RestApiProps>;\n\n  /**\n   * S3 bucket props for the artifacts bucket\n   * @default\n   *  bucketName: 'turbo-remote-cache-artifacts',\n   *  lifecycleRules: [{ expiration: cdk.Duration.days(30) }],\n   *  cors: [{\n   *    allowedHeaders: ['*'],\n   *    allowedMethods: [s3.HttpMethods.GET, s3.HttpMethods.PUT, s3.HttpMethods.HEAD],\n   *    allowedOrigins: ['*'],\n   *    exposedHeaders: ['User-Agent', 'Content-Type', 'Content-Length', 'x-artifact-duration', 'x-artifact-tag'],\n   *  }],\n   *  removalPolicy: cdk.RemovalPolicy.DESTROY,\n   */\n  artifactsBucketProps?: Partial<s3.BucketProps>;\n\n  /**\n   * DynamoDB table props for the events table\n   * @default\n   *  tableName: 'turbo-remote-cache-events',\n   *  billingMode: dynamodb.BillingMode.PROVISIONED,\n   *  readCapacity: 5,\n   *  writeCapacity: 5,\n   *  timeToLiveAttribute: 'ttl',\n   *  removalPolicy: cdk.RemovalPolicy.DESTROY,\n   */\n  eventsTableProps?: Partial<dynamodb.TableProps>;\n\n  /**\n   * Common lambda function props for all lambda functions\n   */\n  lambdaProps?: Partial<lambda.FunctionProps>;\n}\n\nexport class TurboRemoteCache extends Construct {\n  constructor(scope: Construct, id: string, props: TurboRemoteCacheProps) {\n    super(scope, id);\n\n    const artifactsBucket = new s3.Bucket(this, 'ArtifactsBucket', {\n      versioned: false,\n      bucketName: 'turbo-remote-cache-artifacts',\n      encryption: s3.BucketEncryption.S3_MANAGED,\n      lifecycleRules: [\n        {\n          expiration: cdk.Duration.days(30),\n        },\n      ],\n      cors: [\n        {\n          allowedHeaders: ['*'],\n          allowedMethods: [s3.HttpMethods.GET, s3.HttpMethods.PUT, s3.HttpMethods.HEAD],\n          allowedOrigins: ['*'],\n          exposedHeaders: [\n            'User-Agent',\n            'Content-Type',\n            'Content-Length',\n            'x-artifact-duration',\n            'x-artifact-tag',\n          ],\n        }\n      ],\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n      ...props.artifactsBucketProps,\n    });\n\n    const s3Credentials = new iam.Role(this, 'S3CredentialsRole', {\n      assumedBy: new iam.ServicePrincipal('apigateway.amazonaws.com'),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName('AmazonS3FullAccess'),\n      ],\n    });\n\n    artifactsBucket.grantReadWrite(s3Credentials);\n\n    const eventsTable = new dynamodb.Table(this, 'EventsTable', {\n      tableName: 'turbo-remote-cache-events',\n      partitionKey: { name: 'hash', type: dynamodb.AttributeType.STRING },\n      sortKey: { name: 'sessionId', type: dynamodb.AttributeType.STRING },\n      billingMode: dynamodb.BillingMode.PROVISIONED,\n      readCapacity: 5,\n      writeCapacity: 5,\n      timeToLiveAttribute: 'ttl',\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n      ...props.eventsTableProps,\n    });\n\n\n    const lambdaFunctions = new LambdaFunctions(this, 'LambdaFunctions', {\n      artifactsBucket,\n      eventsTable,\n      lambdaProps: props.lambdaProps,\n      authorizerFunction: props.authorizerFunction,\n      userInfoFunction: props.userInfoFunction,\n    });\n\n    const api = new APIGateway(this, 'APIGateway', {\n      lambdaFunctions,\n      artifactsBucket,\n      s3Credentials,\n      apiProps: props.apiProps,\n    });\n  }\n}"]}