@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
148 lines • 27.7 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.InfrastructureTsProject = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.
SPDX-License-Identifier: Apache-2.0 */
const fs = require("fs");
const path = require("path");
const monorepo_1 = require("../../../monorepo");
const Mustache = require("mustache");
const projen_1 = require("projen");
const awscdk_1 = require("projen/lib/awscdk");
const javascript_1 = require("projen/lib/javascript");
const infrastructure_commands_1 = require("../../components/infrastructure-commands");
const consts_1 = require("../../consts");
/**
* Synthesizes a Infrastructure Typescript Project.
*/
class InfrastructureTsProject extends awscdk_1.AwsCdkTypeScriptApp {
constructor(options) {
super({
...options,
defaultReleaseBranch: options.defaultReleaseBranch ?? "main",
prettier: options.prettier || true,
packageManager: options.parent && options.parent instanceof javascript_1.NodeProject
? options.parent.package.packageManager
: options.packageManager,
cdkVersion: options.cdkVersion ?? "2.1.0",
name: options.name,
sampleCode: false,
readme: {
contents: fs
.readFileSync(path.resolve(__dirname, "../../../samples/infrastructure/typescript/README.md"))
.toString(),
},
});
infrastructure_commands_1.InfrastructureCommands.ensure(this);
this.addDeps("@aws/pdk", "cdk-nag");
const srcDir = path.resolve(__dirname, "../../../samples/infrastructure/typescript/src");
const testDir = path.resolve(__dirname, "../../../samples/infrastructure/typescript/test");
const typeSafeApis = [
...(options.typeSafeApis || []),
...(options.typeSafeApi ? [options.typeSafeApi] : []),
];
const typeSafeWebSocketApis = options.typeSafeWebSocketApis ?? [];
const cloudscapeReactTsWebsites = [
...(options.cloudscapeReactTsWebsites || []),
...(options.cloudscapeReactTsWebsite
? [options.cloudscapeReactTsWebsite]
: []),
];
[...typeSafeApis, ...typeSafeWebSocketApis].forEach((tsApi) => {
if (!tsApi.infrastructure.typescript) {
throw new Error("Cannot pass in a Type Safe Api without Typescript Infrastructure configured!");
}
this.addDeps(`${tsApi.infrastructure.typescript?.package.packageName}@${tsApi.infrastructure.typescript?.package.manifest.version}`);
// Ensure handlers are built before infra
tsApi.all.handlers?.forEach((handler) => {
monorepo_1.NxProject.ensure(this).addImplicitDependency(handler);
});
});
cloudscapeReactTsWebsites.forEach((csWebsite) => {
// Ensure website is built before infra
this.addDevDeps(`${csWebsite.package.packageName}@${csWebsite.package.manifest.version}`);
});
const mustacheConfig = {
stackName: options.stackName || consts_1.DEFAULT_STACK_NAME,
allowSignup: options.allowSignup ?? false,
typeSafeApis: this.generateTypeSafeMustacheConfig(typeSafeApis),
typeSafeWebSocketApis: this.generateTypeSafeMustacheConfig(typeSafeWebSocketApis),
stages: options.stages || [],
cloudscapeReactTsWebsites: cloudscapeReactTsWebsites.map((csWebsite) => {
const websiteName = this.capitalize(csWebsite.package.packageName
.replace(/[^a-z0-9_]+/gi, "")
.replace(/^[0-9]+/gi, ""));
return {
websiteName,
websiteNameLowercase: websiteName.toLowerCase(),
websiteDistRelativePath: path.relative(this.outdir, `${csWebsite.outdir}/build`),
typeSafeApis: this.generateTypeSafeMustacheConfig(csWebsite.typeSafeApis),
typeSafeWebSocketApis: this.generateTypeSafeMustacheConfig(csWebsite.typeSafeWebSocketApis),
};
}),
};
options.sampleCode !== false &&
this.emitSampleFiles(srcDir, ["src"], mustacheConfig);
options.sampleCode !== false &&
this.emitSampleFiles(testDir, ["test"], mustacheConfig);
const eslintTask = this.tasks.tryFind("eslint");
this.testTask.reset();
this.testTask.exec("jest --passWithNoTests ${CI:-'--updateSnapshot'}");
eslintTask && this.testTask.spawn(eslintTask);
}
generateTypeSafeMustacheConfig(typeSafeApis) {
return (typeSafeApis ?? []).map((tsApi, idx) => {
const apiName = this.capitalize(tsApi.model
.apiName.replace(/[^a-z0-9_]+/gi, "")
.replace(/^[0-9]+/gi, ""));
return {
apiName,
apiNameLowercase: apiName?.toLowerCase(),
infraPackage: tsApi.infrastructure.typescript?.package.packageName,
isLast: idx === typeSafeApis.length - 1,
};
});
}
capitalize(word) {
return word.charAt(0).toUpperCase() + word.slice(1);
}
emitSampleFiles(dir, pathPrefixes = [], mustacheConfig) {
fs.readdirSync(dir, { withFileTypes: true }).forEach((f) => {
if (f.isDirectory()) {
this.emitSampleFiles(`${dir}/${f.name}`, [...pathPrefixes, f.name], mustacheConfig);
}
else if (f.name.endsWith("websocketapi.ts.mustache")) {
mustacheConfig.typeSafeWebSocketApis.forEach((tsApi) => {
new projen_1.SampleFile(this, `${path.join(...pathPrefixes, `${tsApi.apiNameLowercase}.ts`)}`, {
contents: Mustache.render(fs.readFileSync(`${dir}/${f.name}`).toString(), tsApi),
});
});
}
else if (f.name.endsWith("api.ts.mustache")) {
mustacheConfig.typeSafeApis.forEach((tsApi) => {
new projen_1.SampleFile(this, `${path.join(...pathPrefixes, `${tsApi.apiNameLowercase}.ts`)}`, {
contents: Mustache.render(fs.readFileSync(`${dir}/${f.name}`).toString(), tsApi),
});
});
}
else if (f.name.endsWith("website.ts.mustache")) {
mustacheConfig.cloudscapeReactTsWebsites.forEach((csWebsite) => {
new projen_1.SampleFile(this, `${path.join(...pathPrefixes, `${csWebsite.websiteNameLowercase}.ts`)}`, {
contents: Mustache.render(fs.readFileSync(`${dir}/${f.name}`).toString(), csWebsite),
});
});
}
else {
new projen_1.SampleFile(this, `${path.join(...pathPrefixes, f.name.replace(".mustache", ""))}`, {
contents: Mustache.render(fs.readFileSync(`${dir}/${f.name}`).toString(), mustacheConfig),
});
}
});
}
}
exports.InfrastructureTsProject = InfrastructureTsProject;
_a = JSII_RTTI_SYMBOL_1;
InfrastructureTsProject[_a] = { fqn: "@aws/pdk.infrastructure.InfrastructureTsProject", version: "0.26.14" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"infrastructure-ts-project.js","sourceRoot":"","sources":["infrastructure-ts-project.ts"],"names":[],"mappings":"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","sourcesContent":["/*! Copyright [Amazon.com](http://amazon.com/), Inc. or its affiliates. All Rights Reserved.\nSPDX-License-Identifier: Apache-2.0 */\nimport * as fs from \"fs\";\nimport * as path from \"path\";\nimport { CloudscapeReactTsWebsiteProject } from \"@aws/cloudscape-react-ts-website\";\nimport { NxProject } from \"@aws/monorepo\";\nimport {\n  TypeSafeApiProject,\n  TypeSafeWebSocketApiProject,\n} from \"@aws/type-safe-api\";\nimport * as Mustache from \"mustache\";\nimport { SampleFile } from \"projen\";\nimport { AwsCdkTypeScriptApp } from \"projen/lib/awscdk\";\nimport { NodeProject } from \"projen/lib/javascript\";\nimport { AwsCdkTypeScriptAppOptions } from \"./aws-cdk-ts-app-options\";\nimport { DeploymentStage } from \"../../components/deployment-stage\";\nimport { InfrastructureCommands } from \"../../components/infrastructure-commands\";\nimport { DEFAULT_STACK_NAME } from \"../../consts\";\n\n/**\n * Configuration options for the InfrastructureTsProject.\n */\nexport interface InfrastructureTsProjectOptions\n  extends AwsCdkTypeScriptAppOptions {\n  /**\n   * Stack name.\n   *\n   * @default infra-dev\n   */\n  readonly stackName?: string;\n\n  /**\n   * List of deployment stages.\n   */\n  readonly stages?: DeploymentStage[];\n\n  /**\n   * Allow self sign up for the UserIdentity construct.\n   *\n   * @default - false\n   */\n  readonly allowSignup?: boolean;\n\n  /**\n   * TypeSafeApi instance to use when setting up the initial project sample code.\n   * @deprecated use typeSafeApis\n   */\n  readonly typeSafeApi?: TypeSafeApiProject;\n\n  /**\n   * CloudscapeReactTsWebsiteProject instance to use when setting up the initial project sample code.\n   * @deprecated use cloudscapeReactTsWebsites\n   */\n  readonly cloudscapeReactTsWebsite?: CloudscapeReactTsWebsiteProject;\n\n  /**\n   * TypeSafeApi instances to use when setting up the initial project sample code.\n   */\n  readonly typeSafeApis?: TypeSafeApiProject[];\n\n  /**\n   * TypeSafeWebSocketApi instances to use when setting up the initial project sample code.\n   */\n  readonly typeSafeWebSocketApis?: TypeSafeWebSocketApiProject[];\n\n  /**\n   * CloudscapeReactTsWebsiteProject instances to use when setting up the initial project sample code.\n   */\n  readonly cloudscapeReactTsWebsites?: CloudscapeReactTsWebsiteProject[];\n}\n\n/**\n * Synthesizes a Infrastructure Typescript Project.\n */\nexport class InfrastructureTsProject extends AwsCdkTypeScriptApp {\n  constructor(options: InfrastructureTsProjectOptions) {\n    super({\n      ...options,\n      defaultReleaseBranch: options.defaultReleaseBranch ?? \"main\",\n      prettier: options.prettier || true,\n      packageManager:\n        options.parent && options.parent instanceof NodeProject\n          ? options.parent.package.packageManager\n          : options.packageManager,\n      cdkVersion: options.cdkVersion ?? \"2.1.0\",\n      name: options.name,\n      sampleCode: false,\n      readme: {\n        contents: fs\n          .readFileSync(\n            path.resolve(\n              __dirname,\n              \"../../../samples/infrastructure/typescript/README.md\"\n            )\n          )\n          .toString(),\n      },\n    });\n\n    InfrastructureCommands.ensure(this);\n\n    this.addDeps(\"@aws/pdk\", \"cdk-nag\");\n\n    const srcDir = path.resolve(\n      __dirname,\n      \"../../../samples/infrastructure/typescript/src\"\n    );\n    const testDir = path.resolve(\n      __dirname,\n      \"../../../samples/infrastructure/typescript/test\"\n    );\n\n    const typeSafeApis = [\n      ...(options.typeSafeApis || []),\n      ...(options.typeSafeApi ? [options.typeSafeApi] : []),\n    ];\n    const typeSafeWebSocketApis = options.typeSafeWebSocketApis ?? [];\n    const cloudscapeReactTsWebsites = [\n      ...(options.cloudscapeReactTsWebsites || []),\n      ...(options.cloudscapeReactTsWebsite\n        ? [options.cloudscapeReactTsWebsite]\n        : []),\n    ];\n\n    [...typeSafeApis, ...typeSafeWebSocketApis].forEach((tsApi) => {\n      if (!tsApi.infrastructure.typescript) {\n        throw new Error(\n          \"Cannot pass in a Type Safe Api without Typescript Infrastructure configured!\"\n        );\n      }\n      this.addDeps(\n        `${tsApi.infrastructure.typescript?.package.packageName!}@${\n          tsApi.infrastructure.typescript?.package.manifest.version\n        }`\n      );\n      // Ensure handlers are built before infra\n      tsApi.all.handlers?.forEach((handler) => {\n        NxProject.ensure(this).addImplicitDependency(handler);\n      });\n    });\n\n    cloudscapeReactTsWebsites.forEach((csWebsite) => {\n      // Ensure website is built before infra\n      this.addDevDeps(\n        `${csWebsite.package.packageName}@${csWebsite.package.manifest.version}`\n      );\n    });\n\n    const mustacheConfig = {\n      stackName: options.stackName || DEFAULT_STACK_NAME,\n      allowSignup: options.allowSignup ?? false,\n      typeSafeApis: this.generateTypeSafeMustacheConfig(typeSafeApis),\n      typeSafeWebSocketApis: this.generateTypeSafeMustacheConfig(\n        typeSafeWebSocketApis\n      ),\n      stages: options.stages || [],\n      cloudscapeReactTsWebsites: cloudscapeReactTsWebsites.map((csWebsite) => {\n        const websiteName = this.capitalize(\n          csWebsite.package.packageName\n            .replace(/[^a-z0-9_]+/gi, \"\")\n            .replace(/^[0-9]+/gi, \"\")\n        );\n        return {\n          websiteName,\n          websiteNameLowercase: websiteName.toLowerCase(),\n          websiteDistRelativePath: path.relative(\n            this.outdir,\n            `${csWebsite.outdir}/build`\n          ),\n          typeSafeApis: this.generateTypeSafeMustacheConfig(\n            csWebsite.typeSafeApis\n          ),\n          typeSafeWebSocketApis: this.generateTypeSafeMustacheConfig(\n            csWebsite.typeSafeWebSocketApis\n          ),\n        };\n      }),\n    };\n\n    options.sampleCode !== false &&\n      this.emitSampleFiles(srcDir, [\"src\"], mustacheConfig);\n    options.sampleCode !== false &&\n      this.emitSampleFiles(testDir, [\"test\"], mustacheConfig);\n\n    const eslintTask = this.tasks.tryFind(\"eslint\");\n    this.testTask.reset();\n    this.testTask.exec(\"jest --passWithNoTests ${CI:-'--updateSnapshot'}\");\n    eslintTask && this.testTask.spawn(eslintTask);\n  }\n\n  private generateTypeSafeMustacheConfig(\n    typeSafeApis?: (TypeSafeApiProject | TypeSafeWebSocketApiProject)[]\n  ) {\n    return (typeSafeApis ?? []).map((tsApi, idx) => {\n      const apiName = this.capitalize(\n        tsApi.model\n          .apiName!.replace(/[^a-z0-9_]+/gi, \"\")\n          .replace(/^[0-9]+/gi, \"\")\n      );\n      return {\n        apiName,\n        apiNameLowercase: apiName?.toLowerCase(),\n        infraPackage: tsApi.infrastructure.typescript?.package.packageName,\n        isLast: idx === typeSafeApis!.length - 1,\n      };\n    });\n  }\n\n  private capitalize(word: string) {\n    return word.charAt(0).toUpperCase() + word.slice(1);\n  }\n\n  private emitSampleFiles(\n    dir: string,\n    pathPrefixes: string[] = [],\n    mustacheConfig: any\n  ) {\n    fs.readdirSync(dir, { withFileTypes: true }).forEach((f) => {\n      if (f.isDirectory()) {\n        this.emitSampleFiles(\n          `${dir}/${f.name}`,\n          [...pathPrefixes, f.name],\n          mustacheConfig\n        );\n      } else if (f.name.endsWith(\"websocketapi.ts.mustache\")) {\n        mustacheConfig.typeSafeWebSocketApis.forEach((tsApi: any) => {\n          new SampleFile(\n            this,\n            `${path.join(...pathPrefixes, `${tsApi.apiNameLowercase}.ts`)}`,\n            {\n              contents: Mustache.render(\n                fs.readFileSync(`${dir}/${f.name}`).toString(),\n                tsApi\n              ),\n            }\n          );\n        });\n      } else if (f.name.endsWith(\"api.ts.mustache\")) {\n        mustacheConfig.typeSafeApis.forEach((tsApi: any) => {\n          new SampleFile(\n            this,\n            `${path.join(...pathPrefixes, `${tsApi.apiNameLowercase}.ts`)}`,\n            {\n              contents: Mustache.render(\n                fs.readFileSync(`${dir}/${f.name}`).toString(),\n                tsApi\n              ),\n            }\n          );\n        });\n      } else if (f.name.endsWith(\"website.ts.mustache\")) {\n        mustacheConfig.cloudscapeReactTsWebsites.forEach((csWebsite: any) => {\n          new SampleFile(\n            this,\n            `${path.join(\n              ...pathPrefixes,\n              `${csWebsite.websiteNameLowercase}.ts`\n            )}`,\n            {\n              contents: Mustache.render(\n                fs.readFileSync(`${dir}/${f.name}`).toString(),\n                csWebsite\n              ),\n            }\n          );\n        });\n      } else {\n        new SampleFile(\n          this,\n          `${path.join(...pathPrefixes, f.name.replace(\".mustache\", \"\"))}`,\n          {\n            contents: Mustache.render(\n              fs.readFileSync(`${dir}/${f.name}`).toString(),\n              mustacheConfig\n            ),\n          }\n        );\n      }\n    });\n  }\n}\n"]}