@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
144 lines • 25.5 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.InfrastructurePyProject = 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 infrastructure_commands_1 = require("../../components/infrastructure-commands");
const consts_1 = require("../../consts");
/**
* Synthesizes a Infrastructure Python Project.
*/
class InfrastructurePyProject extends awscdk_1.AwsCdkPythonApp {
constructor(options) {
const moduleName = options.moduleName ?? "infra";
super({
...options,
cdkVersion: options.cdkVersion ?? "2.1.0",
sample: false,
poetry: true,
moduleName,
appEntrypoint: "main.py",
pytest: options.pytest ?? false,
version: options.version ?? "0.0.0",
authorName: options.authorName ?? "pdkuser",
authorEmail: options.authorEmail ?? "user@pdk.com",
name: options.name,
readme: {
contents: fs
.readFileSync(path.resolve(__dirname, "../../../samples/infrastructure/python/README.md"))
.toString(),
},
});
infrastructure_commands_1.InfrastructureCommands.ensure(this);
["pytest@^7", "syrupy@^4"].forEach((devDep) => this.addDevDependency(devDep));
["aws_pdk@^0", "cdk_nag@^2", "python@^3.9"].forEach((dep) => this.addDependency(dep));
const srcDir = path.resolve(__dirname, "../../../samples/infrastructure/python/src");
const testDir = path.resolve(__dirname, "../../../samples/infrastructure/python/test");
const typeSafeApis = [
...(options.typeSafeApis || []),
...(options.typeSafeApi ? [options.typeSafeApi] : []),
];
const cloudscapeReactTsWebsites = [
...(options.cloudscapeReactTsWebsites || []),
...(options.cloudscapeReactTsWebsite
? [options.cloudscapeReactTsWebsite]
: []),
];
typeSafeApis.forEach((tsApi) => {
if (!tsApi.infrastructure.python) {
throw new Error("Cannot pass in a Type Safe Api without Python Infrastructure configured!");
}
monorepo_1.NxProject.ensure(this).addPythonPoetryDependency(tsApi.infrastructure.python);
// 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
monorepo_1.NxProject.ensure(this).addImplicitDependency(csWebsite);
});
const mustacheConfig = {
stackName: options.stackName || consts_1.DEFAULT_STACK_NAME,
allowSignup: options.allowSignup ?? false,
moduleName,
typeSafeApis: this.generateTypeSafeMustacheConfig(moduleName, typeSafeApis),
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(moduleName, csWebsite.typeSafeApis),
};
}),
};
const tstDir = "tests";
options.sample !== false &&
this.emitSampleFiles(srcDir, [this.moduleName], mustacheConfig);
options.sample !== false &&
this.emitSampleFiles(testDir, [tstDir], mustacheConfig);
this.testTask.reset(`poetry run pytest ${tstDir}/ \${CI:-'--snapshot-update'}`);
}
generateTypeSafeMustacheConfig(moduleName, 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.python?.moduleName,
moduleName,
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()) {
return this.emitSampleFiles(`${dir}/${f.name}`, [...pathPrefixes, f.name], mustacheConfig);
}
else if (f.name.endsWith("api.py.mustache")) {
mustacheConfig.typeSafeApis.forEach((tsApi) => {
new projen_1.SampleFile(this, `${path.join(...pathPrefixes, `${tsApi.apiNameLowercase}.py`)}`, {
contents: Mustache.render(fs.readFileSync(`${dir}/${f.name}`).toString(), tsApi),
});
});
}
else if (f.name.endsWith("website.py.mustache")) {
mustacheConfig.cloudscapeReactTsWebsites.forEach((csWebsite) => {
new projen_1.SampleFile(this, `${path.join(...pathPrefixes, `${csWebsite.websiteNameLowercase}.py`)}`, {
contents: Mustache.render(fs.readFileSync(`${dir}/${f.name}`).toString(), csWebsite),
});
});
}
else {
const contents = Mustache.render(fs.readFileSync(`${dir}/${f.name}`).toString(), mustacheConfig);
return new projen_1.SampleFile(this, `${path.join(...(f.name !== "main.py.mustache" ? pathPrefixes : []), // emit at the root so package imports work correctly :(
f.name.replace(".mustache", ""))}`, {
contents,
sourcePath: (!contents && `${dir}/${f.name}`) || undefined,
});
}
});
}
}
exports.InfrastructurePyProject = InfrastructurePyProject;
_a = JSII_RTTI_SYMBOL_1;
InfrastructurePyProject[_a] = { fqn: "@aws/pdk.infrastructure.InfrastructurePyProject", version: "0.26.14" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"infrastructure-py-project.js","sourceRoot":"","sources":["infrastructure-py-project.ts"],"names":[],"mappings":";;;;;AAAA;sCACsC;AACtC,yBAAyB;AACzB,6BAA6B;AAE7B,4CAA0C;AAE1C,qCAAqC;AACrC,mCAAoC;AACpC,8CAAoD;AAGpD,sFAAkF;AAClF,yCAAkD;AAgDlD;;GAEG;AACH,MAAa,uBAAwB,SAAQ,wBAAe;IAC1D,YAAY,OAAuC;QACjD,MAAM,UAAU,GAAG,OAAO,CAAC,UAAU,IAAI,OAAO,CAAC;QAEjD,KAAK,CAAC;YACJ,GAAG,OAAO;YACV,UAAU,EAAE,OAAO,CAAC,UAAU,IAAI,OAAO;YACzC,MAAM,EAAE,KAAK;YACb,MAAM,EAAE,IAAI;YACZ,UAAU;YACV,aAAa,EAAE,SAAS;YACxB,MAAM,EAAE,OAAO,CAAC,MAAM,IAAI,KAAK;YAC/B,OAAO,EAAE,OAAO,CAAC,OAAO,IAAI,OAAO;YACnC,UAAU,EAAE,OAAO,CAAC,UAAU,IAAI,SAAS;YAC3C,WAAW,EAAE,OAAO,CAAC,WAAW,IAAI,cAAc;YAClD,IAAI,EAAE,OAAO,CAAC,IAAI;YAClB,MAAM,EAAE;gBACN,QAAQ,EAAE,EAAE;qBACT,YAAY,CACX,IAAI,CAAC,OAAO,CACV,SAAS,EACT,kDAAkD,CACnD,CACF;qBACA,QAAQ,EAAE;aACd;SACF,CAAC,CAAC;QAEH,gDAAsB,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAEpC,CAAC,WAAW,EAAE,WAAW,CAAC,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE,CAC5C,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAC9B,CAAC;QACF,CAAC,YAAY,EAAE,YAAY,EAAE,aAAa,CAAC,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE,CAC1D,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,CACxB,CAAC;QAEF,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CACzB,SAAS,EACT,4CAA4C,CAC7C,CAAC;QACF,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAC1B,SAAS,EACT,6CAA6C,CAC9C,CAAC;QAEF,MAAM,YAAY,GAAG;YACnB,GAAG,CAAC,OAAO,CAAC,YAAY,IAAI,EAAE,CAAC;YAC/B,GAAG,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;SACtD,CAAC;QACF,MAAM,yBAAyB,GAAG;YAChC,GAAG,CAAC,OAAO,CAAC,yBAAyB,IAAI,EAAE,CAAC;YAC5C,GAAG,CAAC,OAAO,CAAC,wBAAwB;gBAClC,CAAC,CAAC,CAAC,OAAO,CAAC,wBAAwB,CAAC;gBACpC,CAAC,CAAC,EAAE,CAAC;SACR,CAAC;QAEF,YAAY,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;YAC7B,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,MAAM,EAAE,CAAC;gBACjC,MAAM,IAAI,KAAK,CACb,0EAA0E,CAC3E,CAAC;YACJ,CAAC;YACD,oBAAS,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,yBAAyB,CAC9C,KAAK,CAAC,cAAc,CAAC,MAAM,CAC5B,CAAC;YACF,yCAAyC;YACzC,KAAK,CAAC,GAAG,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;gBACtC,oBAAS,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,qBAAqB,CAAC,OAAO,CAAC,CAAC;YACxD,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,yBAAyB,CAAC,OAAO,CAAC,CAAC,SAAS,EAAE,EAAE;YAC9C,uCAAuC;YACvC,oBAAS,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,qBAAqB,CAAC,SAAS,CAAC,CAAC;QAC1D,CAAC,CAAC,CAAC;QAEH,MAAM,cAAc,GAAG;YACrB,SAAS,EAAE,OAAO,CAAC,SAAS,IAAI,2BAAkB;YAClD,WAAW,EAAE,OAAO,CAAC,WAAW,IAAI,KAAK;YACzC,UAAU;YACV,YAAY,EAAE,IAAI,CAAC,8BAA8B,CAC/C,UAAU,EACV,YAAY,CACb;YACD,MAAM,EAAE,OAAO,CAAC,MAAM,IAAI,EAAE;YAC5B,yBAAyB,EAAE,yBAAyB,CAAC,GAAG,CAAC,CAAC,SAAS,EAAE,EAAE;gBACrE,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU,CACjC,SAAS,CAAC,OAAO,CAAC,WAAW;qBAC1B,OAAO,CAAC,eAAe,EAAE,EAAE,CAAC;qBAC5B,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC,CAC5B,CAAC;gBACF,OAAO;oBACL,WAAW;oBACX,oBAAoB,EAAE,WAAW,CAAC,WAAW,EAAE;oBAC/C,uBAAuB,EAAE,IAAI,CAAC,QAAQ,CACpC,IAAI,CAAC,MAAM,EACX,GAAG,SAAS,CAAC,MAAM,QAAQ,CAC5B;oBACD,YAAY,EAAE,IAAI,CAAC,8BAA8B,CAC/C,UAAU,EACV,SAAS,CAAC,YAAY,CACvB;iBACF,CAAC;YACJ,CAAC,CAAC;SACH,CAAC;QAEF,MAAM,MAAM,GAAG,OAAO,CAAC;QAEvB,OAAO,CAAC,MAAM,KAAK,KAAK;YACtB,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,cAAc,CAAC,CAAC;QAClE,OAAO,CAAC,MAAM,KAAK,KAAK;YACtB,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,CAAC,MAAM,CAAC,EAAE,cAAc,CAAC,CAAC;QAE1D,IAAI,CAAC,QAAQ,CAAC,KAAK,CACjB,qBAAqB,MAAM,+BAA+B,CAC3D,CAAC;IACJ,CAAC;IAEO,8BAA8B,CACpC,UAAkB,EAClB,YAAmC;QAEnC,OAAO,YAAY,EAAE,GAAG,CAAC,CAAC,KAAK,EAAE,GAAG,EAAE,EAAE;YACtC,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,CAC7B,KAAK,CAAC,KAAK;iBACR,OAAQ,CAAC,OAAO,CAAC,eAAe,EAAE,EAAE,CAAC;iBACrC,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC,CAC5B,CAAC;YACF,OAAO;gBACL,OAAO;gBACP,gBAAgB,EAAE,OAAO,EAAE,WAAW,EAAE;gBACxC,YAAY,EAAE,KAAK,CAAC,cAAc,CAAC,MAAM,EAAE,UAAU;gBACrD,UAAU;gBACV,MAAM,EAAE,GAAG,KAAK,YAAY,CAAC,MAAM,GAAG,CAAC;aACxC,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,UAAU,CAAC,IAAY;QAC7B,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IACtD,CAAC;IAEO,eAAe,CACrB,GAAW,EACX,eAAyB,EAAE,EAC3B,cAAmB;QAEnB,EAAE,CAAC,WAAW,CAAC,GAAG,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;YACzD,IAAI,CAAC,CAAC,WAAW,EAAE,EAAE,CAAC;gBACpB,OAAO,IAAI,CAAC,eAAe,CACzB,GAAG,GAAG,IAAI,CAAC,CAAC,IAAI,EAAE,EAClB,CAAC,GAAG,YAAY,EAAE,CAAC,CAAC,IAAI,CAAC,EACzB,cAAc,CACf,CAAC;YACJ,CAAC;iBAAM,IAAI,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,iBAAiB,CAAC,EAAE,CAAC;gBAC9C,cAAc,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,KAAU,EAAE,EAAE;oBACjD,IAAI,mBAAU,CACZ,IAAI,EACJ,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,YAAY,EAAE,GAAG,KAAK,CAAC,gBAAgB,KAAK,CAAC,EAAE,EAC/D;wBACE,QAAQ,EAAE,QAAQ,CAAC,MAAM,CACvB,EAAE,CAAC,YAAY,CAAC,GAAG,GAAG,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,QAAQ,EAAE,EAC9C,KAAK,CACN;qBACF,CACF,CAAC;gBACJ,CAAC,CAAC,CAAC;YACL,CAAC;iBAAM,IAAI,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,qBAAqB,CAAC,EAAE,CAAC;gBAClD,cAAc,CAAC,yBAAyB,CAAC,OAAO,CAAC,CAAC,SAAc,EAAE,EAAE;oBAClE,IAAI,mBAAU,CACZ,IAAI,EACJ,GAAG,IAAI,CAAC,IAAI,CACV,GAAG,YAAY,EACf,GAAG,SAAS,CAAC,oBAAoB,KAAK,CACvC,EAAE,EACH;wBACE,QAAQ,EAAE,QAAQ,CAAC,MAAM,CACvB,EAAE,CAAC,YAAY,CAAC,GAAG,GAAG,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,QAAQ,EAAE,EAC9C,SAAS,CACV;qBACF,CACF,CAAC;gBACJ,CAAC,CAAC,CAAC;YACL,CAAC;iBAAM,CAAC;gBACN,MAAM,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAC9B,EAAE,CAAC,YAAY,CAAC,GAAG,GAAG,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,QAAQ,EAAE,EAC9C,cAAc,CACf,CAAC;gBACF,OAAO,IAAI,mBAAU,CACnB,IAAI,EACJ,GAAG,IAAI,CAAC,IAAI,CACV,GAAG,CAAC,CAAC,CAAC,IAAI,KAAK,kBAAkB,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,wDAAwD;gBAChH,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC,CAChC,EAAE,EACH;oBACE,QAAQ;oBACR,UAAU,EAAE,CAAC,CAAC,QAAQ,IAAI,GAAG,GAAG,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC,IAAI,SAAS;iBAC3D,CACF,CAAC;YACJ,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;;AA1MH,0DA2MC","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 { TypeSafeApiProject } from \"@aws/type-safe-api\";\nimport * as Mustache from \"mustache\";\nimport { SampleFile } from \"projen\";\nimport { AwsCdkPythonApp } from \"projen/lib/awscdk\";\nimport { AwsCdkPythonAppOptions } from \"./aws-cdk-py-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 InfrastructurePyProject.\n */\nexport interface InfrastructurePyProjectOptions extends AwsCdkPythonAppOptions {\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   * CloudscapeReactTsWebsiteProject instances to use when setting up the initial project sample code.\n   */\n  readonly cloudscapeReactTsWebsites?: CloudscapeReactTsWebsiteProject[];\n}\n\n/**\n * Synthesizes a Infrastructure Python Project.\n */\nexport class InfrastructurePyProject extends AwsCdkPythonApp {\n  constructor(options: InfrastructurePyProjectOptions) {\n    const moduleName = options.moduleName ?? \"infra\";\n\n    super({\n      ...options,\n      cdkVersion: options.cdkVersion ?? \"2.1.0\",\n      sample: false,\n      poetry: true,\n      moduleName,\n      appEntrypoint: \"main.py\",\n      pytest: options.pytest ?? false,\n      version: options.version ?? \"0.0.0\",\n      authorName: options.authorName ?? \"pdkuser\",\n      authorEmail: options.authorEmail ?? \"user@pdk.com\",\n      name: options.name,\n      readme: {\n        contents: fs\n          .readFileSync(\n            path.resolve(\n              __dirname,\n              \"../../../samples/infrastructure/python/README.md\"\n            )\n          )\n          .toString(),\n      },\n    });\n\n    InfrastructureCommands.ensure(this);\n\n    [\"pytest@^7\", \"syrupy@^4\"].forEach((devDep) =>\n      this.addDevDependency(devDep)\n    );\n    [\"aws_pdk@^0\", \"cdk_nag@^2\", \"python@^3.9\"].forEach((dep) =>\n      this.addDependency(dep)\n    );\n\n    const srcDir = path.resolve(\n      __dirname,\n      \"../../../samples/infrastructure/python/src\"\n    );\n    const testDir = path.resolve(\n      __dirname,\n      \"../../../samples/infrastructure/python/test\"\n    );\n\n    const typeSafeApis = [\n      ...(options.typeSafeApis || []),\n      ...(options.typeSafeApi ? [options.typeSafeApi] : []),\n    ];\n    const cloudscapeReactTsWebsites = [\n      ...(options.cloudscapeReactTsWebsites || []),\n      ...(options.cloudscapeReactTsWebsite\n        ? [options.cloudscapeReactTsWebsite]\n        : []),\n    ];\n\n    typeSafeApis.forEach((tsApi) => {\n      if (!tsApi.infrastructure.python) {\n        throw new Error(\n          \"Cannot pass in a Type Safe Api without Python Infrastructure configured!\"\n        );\n      }\n      NxProject.ensure(this).addPythonPoetryDependency(\n        tsApi.infrastructure.python\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      NxProject.ensure(this).addImplicitDependency(csWebsite);\n    });\n\n    const mustacheConfig = {\n      stackName: options.stackName || DEFAULT_STACK_NAME,\n      allowSignup: options.allowSignup ?? false,\n      moduleName,\n      typeSafeApis: this.generateTypeSafeMustacheConfig(\n        moduleName,\n        typeSafeApis\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            moduleName,\n            csWebsite.typeSafeApis\n          ),\n        };\n      }),\n    };\n\n    const tstDir = \"tests\";\n\n    options.sample !== false &&\n      this.emitSampleFiles(srcDir, [this.moduleName], mustacheConfig);\n    options.sample !== false &&\n      this.emitSampleFiles(testDir, [tstDir], mustacheConfig);\n\n    this.testTask.reset(\n      `poetry run pytest ${tstDir}/ \\${CI:-'--snapshot-update'}`\n    );\n  }\n\n  private generateTypeSafeMustacheConfig(\n    moduleName: string,\n    typeSafeApis?: TypeSafeApiProject[]\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.python?.moduleName,\n        moduleName,\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        return this.emitSampleFiles(\n          `${dir}/${f.name}`,\n          [...pathPrefixes, f.name],\n          mustacheConfig\n        );\n      } else if (f.name.endsWith(\"api.py.mustache\")) {\n        mustacheConfig.typeSafeApis.forEach((tsApi: any) => {\n          new SampleFile(\n            this,\n            `${path.join(...pathPrefixes, `${tsApi.apiNameLowercase}.py`)}`,\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.py.mustache\")) {\n        mustacheConfig.cloudscapeReactTsWebsites.forEach((csWebsite: any) => {\n          new SampleFile(\n            this,\n            `${path.join(\n              ...pathPrefixes,\n              `${csWebsite.websiteNameLowercase}.py`\n            )}`,\n            {\n              contents: Mustache.render(\n                fs.readFileSync(`${dir}/${f.name}`).toString(),\n                csWebsite\n              ),\n            }\n          );\n        });\n      } else {\n        const contents = Mustache.render(\n          fs.readFileSync(`${dir}/${f.name}`).toString(),\n          mustacheConfig\n        );\n        return new SampleFile(\n          this,\n          `${path.join(\n            ...(f.name !== \"main.py.mustache\" ? pathPrefixes : []), // emit at the root so package imports work correctly :(\n            f.name.replace(\".mustache\", \"\")\n          )}`,\n          {\n            contents,\n            sourcePath: (!contents && `${dir}/${f.name}`) || undefined,\n          }\n        );\n      }\n    });\n  }\n}\n"]}