UNPKG

turbo-remote-cache-construct

Version:

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

258 lines 37.6 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.APIGateway = void 0; const apigateway = require("aws-cdk-lib/aws-apigateway"); const logs = require("aws-cdk-lib/aws-logs"); const cdk = require("aws-cdk-lib"); const constructs_1 = require("constructs"); const get_artifact_1 = require("./get-artifact"); const head_artifact_1 = require("./head-artifact"); const put_artifact_1 = require("./put-artifact"); class APIGateway extends constructs_1.Construct { constructor(scope, id, props) { super(scope, id); const logGroup = logs.LogGroup.fromLogGroupName(this, 'LogGroup', '/aws/apigateway/turbo-remote-cache-api'); let tokenAuthorizer; if (props.lambdaFunctions.authorizerFunction) { tokenAuthorizer = new apigateway.TokenAuthorizer(this, 'TokenAuthorizer', { handler: props.lambdaFunctions.authorizerFunction, resultsCacheTtl: cdk.Duration.minutes(60), identitySource: 'method.request.header.Authorization', }); } const api = new apigateway.RestApi(this, 'TurboRemoteCacheApi', { restApiName: 'Turborepo Remote Cache API', description: 'Turborepo is an intelligent build system optimized for JavaScript and TypeScript codebases.', cloudWatchRole: true, deployOptions: { documentationVersion: '8.0.0', loggingLevel: apigateway.MethodLoggingLevel.INFO, accessLogDestination: new apigateway.LogGroupLogDestination(logGroup), accessLogFormat: apigateway.AccessLogFormat.jsonWithStandardFields(), dataTraceEnabled: true, tracingEnabled: true, }, binaryMediaTypes: ['application/octet-stream'], ...(tokenAuthorizer ? { defaultMethodOptions: { authorizationType: apigateway.AuthorizationType.CUSTOM, authorizer: tokenAuthorizer } } : {}), ...props.apiProps, }); new apigateway.CfnDocumentationVersion(this, 'DocumentationVersion', { documentationVersion: '8.0.0', restApiId: api.restApiId, }); const v8Resource = api.root.addResource('v8'); const artifactsResource = v8Resource.addResource('artifacts'); const eventsResource = artifactsResource.addResource('events'); eventsResource.addMethod('POST', new apigateway.LambdaIntegration(props.lambdaFunctions.recordEventsFunction), { operationName: 'recordEvents', methodResponses: [{ statusCode: '200' }], requestParameters: { 'method.request.header.x-artifact-client-ci': false, 'method.request.header.x-artifact-client-interactive': false, }, requestModels: { 'application/json': new apigateway.Model(this, 'RecordEventsModel', { restApi: api, contentType: 'application/json', modelName: 'RecordEventsModel', schema: { type: apigateway.JsonSchemaType.ARRAY, items: { type: apigateway.JsonSchemaType.OBJECT, properties: { sessionId: { type: apigateway.JsonSchemaType.STRING }, source: { type: apigateway.JsonSchemaType.STRING, enum: ['LOCAL', 'REMOTE'] }, event: { type: apigateway.JsonSchemaType.STRING, enum: ['HIT', 'MISS'] }, hash: { type: apigateway.JsonSchemaType.STRING }, duration: { type: apigateway.JsonSchemaType.NUMBER }, }, required: ['sessionId', 'source', 'hash', 'event'], }, }, }), }, }); const recordEventDocumentationPart = { description: 'Records an artifacts cache usage event. The body of this request is an array of cache usage events. The supported event types are `HIT` and `MISS`. The source is either `LOCAL` the cache event was on the users filesystem cache or `REMOTE` if the cache event is for a remote cache. When the event is a `HIT` the request also accepts a number `duration` which is the time taken to generate the artifact in the cache.', summary: 'Record an artifacts cache usage event', tags: ['artifacts'], }; new apigateway.CfnDocumentationPart(this, 'ArtifactsEventsDocumentationPart', { location: { type: 'METHOD', method: 'POST', path: '/v8/artifacts/events', }, properties: JSON.stringify(recordEventDocumentationPart), restApiId: api.restApiId, }); const statusResource = artifactsResource.addResource('status'); statusResource.addMethod('GET', new apigateway.LambdaIntegration(props.lambdaFunctions.statusFunction), { operationName: 'status', methodResponses: [ { statusCode: '200', responseModels: { 'application/json': new apigateway.Model(this, 'StatusResponseModel', { restApi: api, contentType: 'application/json', modelName: 'StatusResponseModel', schema: { type: apigateway.JsonSchemaType.OBJECT, properties: { status: { type: apigateway.JsonSchemaType.STRING, enum: ['disabled', 'enabled', 'over_limit', 'paused'] }, }, required: ['status'], }, }), }, }, ], }); const statusDocumentationPart = { description: 'Check the status of Remote Caching for this principal. Returns a JSON-encoded status indicating if Remote Caching is enabled, disabled, or disabled due to usage limits.', summary: 'Get status of Remote Caching for this principal', tags: ['artifacts'], }; new apigateway.CfnDocumentationPart(this, 'ArtifactsStatusDocumentationPart', { location: { type: 'METHOD', method: 'GET', path: '/v8/artifacts/status', }, properties: JSON.stringify(statusDocumentationPart), restApiId: api.restApiId, }); const hashResource = artifactsResource.addResource('{hash}'); hashResource.addMethod('OPTIONS', new apigateway.LambdaIntegration(props.lambdaFunctions.preflightArtifactFunction), { operationName: 'preflightArtifact', methodResponses: [{ statusCode: '200' }], }); // GET /v8/artifacts/{hash} (0, get_artifact_1.getArtifactIntegration)(this, { artifactsBucket: props.artifactsBucket, s3Credentials: props.s3Credentials, api, hashResource, }); // HEAD /v8/artifacts/{hash} // Check that a cache artifact with the given `hash` exists. This request returns response headers only // and is equivalent to a `GET` request to this endpoint where the response contains no body. (0, head_artifact_1.headArtifactIntegration)(this, { artifactsBucket: props.artifactsBucket, s3Credentials: props.s3Credentials, api, hashResource, }); // PUT /v8/artifacts/{hash} // Uploads a cache artifact identified by the `hash` specified on the path. // The cache artifact can then be downloaded with the provided `hash`. (0, put_artifact_1.putArtifactIntegration)(this, { artifactsBucket: props.artifactsBucket, s3Credentials: props.s3Credentials, api, hashResource, }); // POST /v8/artifacts // Query information about an array of artifacts. artifactsResource.addMethod('POST', new apigateway.LambdaIntegration(props.lambdaFunctions.artifactQueryFunction), { operationName: 'artifactQuery', methodResponses: [{ statusCode: '200' }], requestModels: { 'application/json': new apigateway.Model(this, 'ArtifactQueryModel', { restApi: api, contentType: 'application/json', modelName: 'ArtifactQueryModel', schema: { type: apigateway.JsonSchemaType.OBJECT, properties: { hashes: { type: apigateway.JsonSchemaType.ARRAY, items: { type: apigateway.JsonSchemaType.STRING } }, }, required: ['hashes'], }, }), }, }); const artifactQueryDocumentationPart = { description: 'Query information about an array of artifacts.', summary: 'Query information about an artifact', tags: ['artifacts'], }; new apigateway.CfnDocumentationPart(this, 'ArtifactsQueryDocumentationPart', { location: { type: 'METHOD', method: 'POST', path: '/v8/artifacts', }, properties: JSON.stringify(artifactQueryDocumentationPart), restApiId: api.restApiId, }); // turbo login // TODO: implement turbo login for third party JWTs // const turborepoResource = api.root.addResource('turborepo'); // const tokenResource = turborepoResource.addResource('token'); // // GET /v8/turborepo/token // tokenResource.addMethod('GET', new apigateway.LambdaIntegration(props.lambdaFunctions.initiateLoginFunction), { // operationName: 'initiateLogin', // }); // const initiateLoginDocumentationPart = { // description: 'Initiates a login process for Turborepo.', // summary: 'Initiate login', // tags: ['login'], // } // new apigateway.CfnDocumentationPart(this, 'TurborepoInitiateLoginDocumentationPart', { // location: { // type: 'METHOD', // method: 'GET', // path: '/v8/turborepo/token', // }, // properties: JSON.stringify(initiateLoginDocumentationPart), // restApiId: api.restApiId, // }); // // GET /v8/turborepo/success // const successResource = turborepoResource.addResource('success'); // successResource.addMethod('GET', new apigateway.LambdaIntegration(props.lambdaFunctions.loginSuccessFunction), { // operationName: 'loginSuccess', // }); // const loginSuccessDocumentationPart = { // description: 'Handles the success of a login process for Turborepo.', // summary: 'Login success', // tags: ['login'], // } // new apigateway.CfnDocumentationPart(this, 'TurborepoLoginSuccessDocumentationPart', { // location: { // type: 'METHOD', // method: 'GET', // path: '/v8/turborepo/success', // }, // properties: JSON.stringify(loginSuccessDocumentationPart), // restApiId: api.restApiId, // }); const v2Resource = api.root.addResource('v2'); const userResource = v2Resource.addResource('user'); // GET /v2/user userResource.addMethod('GET', new apigateway.LambdaIntegration(props.lambdaFunctions.getUserInfoFunction), { operationName: 'getUserInfo', }); const getUserInfoDocumentationPart = { description: 'Retrieves information about the authenticated user.', summary: 'Get user info', tags: ['login'], }; new apigateway.CfnDocumentationPart(this, 'TurborepoUserInfoDocumentationPart', { location: { type: 'METHOD', method: 'GET', path: '/v2/user', }, properties: JSON.stringify(getUserInfoDocumentationPart), restApiId: api.restApiId, }); // cloudfront domain name for CNAME new cdk.CfnOutput(this, 'CloudfrontAliasDomainName', { value: api.domainName?.domainNameAliasDomainName ?? 'N/A', }); } } exports.APIGateway = APIGateway; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api.js","sourceRoot":"","sources":["../src/api.ts"],"names":[],"mappings":"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","sourcesContent":["import * as apigateway from 'aws-cdk-lib/aws-apigateway';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport * as cdk from 'aws-cdk-lib';\nimport { Construct } from \"constructs\";\nimport { LambdaFunctions } from './lambda';\nimport { getArtifactIntegration } from './get-artifact';\nimport { headArtifactIntegration } from './head-artifact';\nimport { putArtifactIntegration } from './put-artifact';\n\ninterface APIGatewayProps {\n  apiProps?: apigateway.RestApiProps;\n  lambdaFunctions: LambdaFunctions;\n  artifactsBucket: s3.Bucket;\n  s3Credentials: iam.Role;\n}\n\nexport class APIGateway extends Construct {\n  constructor(scope: Construct, id: string, props: APIGatewayProps) {\n    super(scope, id);\n    const logGroup = logs.LogGroup.fromLogGroupName(this, 'LogGroup', '/aws/apigateway/turbo-remote-cache-api');\n\n    let tokenAuthorizer: apigateway.TokenAuthorizer | undefined;\n    if (props.lambdaFunctions.authorizerFunction) {\n      tokenAuthorizer = new apigateway.TokenAuthorizer(this, 'TokenAuthorizer', {\n        handler: props.lambdaFunctions.authorizerFunction,\n        resultsCacheTtl: cdk.Duration.minutes(60),\n        identitySource: 'method.request.header.Authorization',\n      });\n    }\n\n    const api = new apigateway.RestApi(this, 'TurboRemoteCacheApi', {\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      ...(tokenAuthorizer ? { defaultMethodOptions: { authorizationType: apigateway.AuthorizationType.CUSTOM, authorizer: tokenAuthorizer } } : {}),\n      ...props.apiProps,\n    });\n\n    new apigateway.CfnDocumentationVersion(this, 'DocumentationVersion', {\n      documentationVersion: '8.0.0',\n      restApiId: api.restApiId,\n    });\n\n    const v8Resource = api.root.addResource('v8');\n    const artifactsResource = v8Resource.addResource('artifacts');\n\n    const eventsResource = artifactsResource.addResource('events');\n    eventsResource.addMethod('POST', new apigateway.LambdaIntegration(props.lambdaFunctions.recordEventsFunction), {\n      operationName: 'recordEvents',\n      methodResponses: [{ statusCode: '200' }],\n      requestParameters: {\n        'method.request.header.x-artifact-client-ci': false,\n        'method.request.header.x-artifact-client-interactive': false,\n      },\n      requestModels: {\n        'application/json': new apigateway.Model(this, 'RecordEventsModel', {\n          restApi: api,\n          contentType: 'application/json',\n          modelName: 'RecordEventsModel',\n          schema: {\n            type: apigateway.JsonSchemaType.ARRAY,\n            items: {\n              type: apigateway.JsonSchemaType.OBJECT,\n              properties: {\n                sessionId: { type: apigateway.JsonSchemaType.STRING },\n                source: { type: apigateway.JsonSchemaType.STRING, enum: ['LOCAL', 'REMOTE'] },\n                event: { type: apigateway.JsonSchemaType.STRING, enum: ['HIT', 'MISS'] },\n                hash: { type: apigateway.JsonSchemaType.STRING },\n                duration: { type: apigateway.JsonSchemaType.NUMBER },\n              },\n              required: ['sessionId', 'source', 'hash', 'event'],\n            },\n          },\n        }),\n      },\n    });\n\n    const recordEventDocumentationPart = {\n      description: 'Records an artifacts cache usage event. The body of this request is an array of cache usage events. The supported event types are `HIT` and `MISS`. The source is either `LOCAL` the cache event was on the users filesystem cache or `REMOTE` if the cache event is for a remote cache. When the event is a `HIT` the request also accepts a number `duration` which is the time taken to generate the artifact in the cache.',\n      summary: 'Record an artifacts cache usage event',\n      tags: ['artifacts'],\n    }\n\n    new apigateway.CfnDocumentationPart(this, 'ArtifactsEventsDocumentationPart', {\n      location: {\n        type: 'METHOD',\n        method: 'POST',\n        path: '/v8/artifacts/events',\n      },\n      properties: JSON.stringify(recordEventDocumentationPart),\n      restApiId: api.restApiId,\n    });\n\n    const statusResource = artifactsResource.addResource('status');\n    statusResource.addMethod('GET', new apigateway.LambdaIntegration(props.lambdaFunctions.statusFunction), {\n      operationName: 'status',\n      methodResponses: [\n        {\n          statusCode: '200',\n          responseModels: {\n            'application/json': new apigateway.Model(this, 'StatusResponseModel', {\n              restApi: api,\n              contentType: 'application/json',\n              modelName: 'StatusResponseModel',\n              schema: {\n                type: apigateway.JsonSchemaType.OBJECT,\n                properties: {\n                  status: { type: apigateway.JsonSchemaType.STRING, enum: ['disabled', 'enabled', 'over_limit', 'paused'] },\n                },\n                required: ['status'],\n              },\n            }),\n          },\n        },\n      ],\n    });\n\n    const statusDocumentationPart = {\n      description: 'Check the status of Remote Caching for this principal. Returns a JSON-encoded status indicating if Remote Caching is enabled, disabled, or disabled due to usage limits.',\n      summary: 'Get status of Remote Caching for this principal',\n      tags: ['artifacts'],\n    }\n\n    new apigateway.CfnDocumentationPart(this, 'ArtifactsStatusDocumentationPart', {\n      location: {\n        type: 'METHOD',\n        method: 'GET',\n        path: '/v8/artifacts/status',\n      },\n      properties: JSON.stringify(statusDocumentationPart),\n      restApiId: api.restApiId,\n    });\n\n    const hashResource = artifactsResource.addResource('{hash}');\n\n    hashResource.addMethod('OPTIONS', new apigateway.LambdaIntegration(props.lambdaFunctions.preflightArtifactFunction), {\n      operationName: 'preflightArtifact',\n      methodResponses: [{ statusCode: '200' }],\n    });\n\n    // GET /v8/artifacts/{hash}\n    getArtifactIntegration(this, {\n      artifactsBucket: props.artifactsBucket,\n      s3Credentials: props.s3Credentials,\n      api,\n      hashResource,\n    });\n\n    // HEAD /v8/artifacts/{hash}\n    // Check that a cache artifact with the given `hash` exists. This request returns response headers only\n    // and is equivalent to a `GET` request to this endpoint where the response contains no body.\n    headArtifactIntegration(this, {\n      artifactsBucket: props.artifactsBucket,\n      s3Credentials: props.s3Credentials,\n      api,\n      hashResource,\n    });\n\n    // PUT /v8/artifacts/{hash}\n    // Uploads a cache artifact identified by the `hash` specified on the path.\n    // The cache artifact can then be downloaded with the provided `hash`.\n    putArtifactIntegration(this, {\n      artifactsBucket: props.artifactsBucket,\n      s3Credentials: props.s3Credentials,\n      api,\n      hashResource,\n    });\n\n    // POST /v8/artifacts\n    // Query information about an array of artifacts.\n    artifactsResource.addMethod('POST', new apigateway.LambdaIntegration(props.lambdaFunctions.artifactQueryFunction), {\n      operationName: 'artifactQuery',\n      methodResponses: [{ statusCode: '200' }],\n      requestModels: {\n        'application/json': new apigateway.Model(this, 'ArtifactQueryModel', {\n          restApi: api,\n          contentType: 'application/json',\n          modelName: 'ArtifactQueryModel',\n          schema: {\n            type: apigateway.JsonSchemaType.OBJECT,\n            properties: {\n              hashes: { type: apigateway.JsonSchemaType.ARRAY, items: { type: apigateway.JsonSchemaType.STRING } },\n            },\n            required: ['hashes'],\n          },\n        }),\n      },\n    });\n\n    const artifactQueryDocumentationPart = {\n      description: 'Query information about an array of artifacts.',\n      summary: 'Query information about an artifact',\n      tags: ['artifacts'],\n    }\n\n    new apigateway.CfnDocumentationPart(this, 'ArtifactsQueryDocumentationPart', {\n      location: {\n        type: 'METHOD',\n        method: 'POST',\n        path: '/v8/artifacts',\n      },\n      properties: JSON.stringify(artifactQueryDocumentationPart),\n      restApiId: api.restApiId,\n    });\n\n    // turbo login\n    // TODO: implement turbo login for third party JWTs\n    // const turborepoResource = api.root.addResource('turborepo');\n    // const tokenResource = turborepoResource.addResource('token');\n\n    // // GET /v8/turborepo/token\n    // tokenResource.addMethod('GET', new apigateway.LambdaIntegration(props.lambdaFunctions.initiateLoginFunction), {\n    //   operationName: 'initiateLogin',\n    // });\n\n    // const initiateLoginDocumentationPart = {\n    //   description: 'Initiates a login process for Turborepo.',\n    //   summary: 'Initiate login',\n    //   tags: ['login'],\n    // }\n\n    // new apigateway.CfnDocumentationPart(this, 'TurborepoInitiateLoginDocumentationPart', {\n    //   location: {\n    //     type: 'METHOD',\n    //     method: 'GET',\n    //     path: '/v8/turborepo/token',\n    //   },\n    //   properties: JSON.stringify(initiateLoginDocumentationPart),\n    //   restApiId: api.restApiId,\n    // });\n\n    // // GET /v8/turborepo/success\n    // const successResource = turborepoResource.addResource('success');\n    // successResource.addMethod('GET', new apigateway.LambdaIntegration(props.lambdaFunctions.loginSuccessFunction), {\n    //   operationName: 'loginSuccess',\n    // });\n\n    // const loginSuccessDocumentationPart = {\n    //   description: 'Handles the success of a login process for Turborepo.',\n    //   summary: 'Login success',\n    //   tags: ['login'],\n    // }\n\n    // new apigateway.CfnDocumentationPart(this, 'TurborepoLoginSuccessDocumentationPart', {\n    //   location: {\n    //     type: 'METHOD',\n    //     method: 'GET',\n    //     path: '/v8/turborepo/success',\n    //   },\n    //   properties: JSON.stringify(loginSuccessDocumentationPart),\n    //   restApiId: api.restApiId,\n    // });\n\n    const v2Resource = api.root.addResource('v2');\n\n    const userResource = v2Resource.addResource('user');\n\n    // GET /v2/user\n    userResource.addMethod('GET', new apigateway.LambdaIntegration(props.lambdaFunctions.getUserInfoFunction), {\n      operationName: 'getUserInfo',\n    });\n\n    const getUserInfoDocumentationPart = {\n      description: 'Retrieves information about the authenticated user.',\n      summary: 'Get user info',\n      tags: ['login'],\n    }\n\n    new apigateway.CfnDocumentationPart(this, 'TurborepoUserInfoDocumentationPart', {\n      location: {\n        type: 'METHOD',\n        method: 'GET',\n        path: '/v2/user',\n      },\n      properties: JSON.stringify(getUserInfoDocumentationPart),\n      restApiId: api.restApiId,\n    });\n\n    // cloudfront domain name for CNAME\n    new cdk.CfnOutput(this, 'CloudfrontAliasDomainName', {\n      value: api.domainName?.domainNameAliasDomainName ?? 'N/A',\n    });\n  }\n}"]}