@aws/pdk
Version:
All documentation is located at: https://aws.github.io/aws-pdk
149 lines • 28.8 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.CloudscapeReactTsWebsiteProject = 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 Mustache = require("mustache");
const projen_1 = require("projen");
const javascript_1 = require("projen/lib/javascript");
const web_1 = require("projen/lib/web");
/**
* Synthesizes a Cloudscape React Typescript Website Project.
*
* @pjid cloudscape-react-ts-website
*/
class CloudscapeReactTsWebsiteProject extends web_1.ReactTypeScriptProject {
constructor(options) {
super({
...options,
defaultReleaseBranch: options.defaultReleaseBranch ?? "main",
name: options.name,
sampleCode: false,
prettier: options.prettier || true,
packageManager: options.parent && options.parent instanceof javascript_1.NodeProject
? options.parent.package.packageManager
: options.packageManager ?? javascript_1.NodePackageManager.PNPM,
readme: {
contents: fs
.readFileSync(path.resolve(__dirname, "../samples/cloudscape-react-ts-website/README.md"))
.toString(),
},
gitignore: [
"public/runtime-config.json",
"public/api.json",
"public/*/api.json",
...(options.gitignore || []),
],
});
this.typeSafeApis = [
...(options.typeSafeApis || []),
...(options.typeSafeApi ? [options.typeSafeApi] : []),
];
this.typeSafeWebSocketApis = options.typeSafeWebSocketApis ?? [];
this.addDeps("@aws-northstar/ui", "@cloudscape-design/components", "@cloudscape-design/board-components", "react-router-dom");
this.testTask.reset();
const lintTask = this.tasks.tryFind("eslint");
lintTask && this.testTask.spawn(lintTask);
this.testTask.exec("react-scripts test --watchAll=false --passWithNoTests");
this.applicationName = options.applicationName ?? "Sample App";
this.allowSignup = options.allowSignup ?? false;
this.publicDir = options.publicDir ?? "public";
const srcDir = path.resolve(__dirname, "../samples/cloudscape-react-ts-website/src");
const publicDir = path.resolve(__dirname, "../samples/cloudscape-react-ts-website/public");
this.typeSafeApis.forEach((typeSafeApi) => {
const hooks = typeSafeApi.library?.typescriptReactQueryHooks;
const libraryHooksPackage = hooks?.package?.packageName;
const libraryHooksPackageVersion = hooks?.package?.manifest.version;
if (!libraryHooksPackage) {
throw new Error("Cannot pass in a Type Safe Api without React Hooks Library configured!");
}
this.addDeps(`${libraryHooksPackage}@${libraryHooksPackageVersion}`);
this.setupApiExplorer(typeSafeApi);
});
this.typeSafeWebSocketApis.forEach((typeSafeWebSocketApi) => {
const hooks = typeSafeWebSocketApi.library?.typescriptWebSocketHooks;
const client = typeSafeWebSocketApi.library?.typescriptWebSocketClient;
if (!hooks || !client) {
throw new Error("Cannot pass in a Type Safe WebSocket Api without React Hooks Library configured");
}
this.addDeps(`${hooks.package.packageName}@${hooks.package.manifest.version}`, `${client.package.packageName}@${client.package.manifest.version}`);
this.setupApiExplorer(typeSafeWebSocketApi);
});
const apis = this.typeSafeApis.map((tsApi, idx) => ({
apiName: tsApi.model.apiName,
isLast: idx === this.typeSafeApis.length - 1,
apiNameSafe: tsApi.model.apiName
?.replace(/[^a-z0-9_]+/gi, "")
.replace(/^[0-9]+/gi, ""),
hooksPackage: tsApi.library?.typescriptReactQueryHooks?.package?.packageName,
}));
const webSocketApis = this.typeSafeWebSocketApis.map((tsApi, idx) => ({
apiName: tsApi.model.apiName,
isLast: idx === this.typeSafeWebSocketApis.length - 1,
apiNameSafe: tsApi.model.apiName
?.replace(/[^a-z0-9_]+/gi, "")
.replace(/^[0-9]+/gi, ""),
hooksPackage: tsApi.library?.typescriptWebSocketHooks?.package?.packageName,
clientPackage: tsApi.library?.typescriptWebSocketClient?.package?.packageName,
}));
const mustacheConfig = {
applicationName: this.applicationName,
hasApis: apis.length + webSocketApis.length > 0,
typeSafeApis: apis,
typeSafeApisReversed: [...apis].reverse(),
typeSafeWebSocketApis: webSocketApis,
typeSafeWebSocketApisReversed: [...webSocketApis].reverse(),
allowSignup: options.allowSignup ?? false,
};
new projen_1.SampleDir(this, this.srcdir, {
files: {
...Object.fromEntries(this.buildSampleDirEntries(srcDir, [], mustacheConfig)),
},
});
new projen_1.SampleDir(this, this.publicDir, {
files: {
...Object.fromEntries(this.buildSampleDirEntries(publicDir, [], mustacheConfig)),
},
});
// Linting is managed as part of the test task already, so disable react-scripts running eslint again
this.tasks.addEnvironment("DISABLE_ESLINT_PLUGIN", "true");
// Relax EsLint and TSC for dev
this.tasks.tryFind("dev")?.env("ESLINT_NO_DEV_ERRORS", "true");
this.tasks.tryFind("dev")?.env("TSC_COMPILE_ON_ERROR", "true");
}
setupApiExplorer(tsApi) {
this.addDevDeps("@types/swagger-ui-react", "@types/uuid");
this.addDeps("swagger-ui-react@5.5.0", "aws4fetch", "uuid", "@aws-crypto/sha256-js", "@aws-sdk/signature-v4", "@aws-sdk/protocol-http");
const targetApiSpecFolder = `public/${tsApi.model.apiName}`;
const targetApiSpecPath = `${targetApiSpecFolder}/api.json`;
this.preCompileTask.exec(`rm -rf ${targetApiSpecFolder}`);
this.preCompileTask.exec(`mkdir -p ${targetApiSpecFolder} && cp ${path.relative(this.outdir, tsApi.model.outdir)}/${tsApi.model.parsedSpecFile} ${targetApiSpecPath}`);
}
buildSampleDirEntries(dir, pathPrefixes = [], mustacheConfig) {
return fs
.readdirSync(dir, { withFileTypes: true })
.filter((f) => mustacheConfig.typeSafeApis.length +
mustacheConfig.typeSafeWebSocketApis.length >
0 || !`${pathPrefixes.join("/")}${f.name}`.includes("ApiExplorer"))
.filter((f) => mustacheConfig.typeSafeApis.length > 0 ||
!`${pathPrefixes.join("/")}${f.name}`.includes("TypeSafeApiClient"))
.filter((f) => mustacheConfig.typeSafeWebSocketApis.length > 0 ||
!`${pathPrefixes.join("/")}${f.name}`.includes("TypeSafeWebSocketApiClient"))
.flatMap((f) => f.isDirectory()
? this.buildSampleDirEntries(`${dir}/${f.name}`, [...pathPrefixes, f.name], mustacheConfig)
: [
[
`${path.join(...pathPrefixes, f.name.replace(".mustache", ""))}`,
Mustache.render(fs.readFileSync(`${dir}/${f.name}`).toString(), mustacheConfig),
],
]);
}
}
exports.CloudscapeReactTsWebsiteProject = CloudscapeReactTsWebsiteProject;
_a = JSII_RTTI_SYMBOL_1;
CloudscapeReactTsWebsiteProject[_a] = { fqn: "@aws/pdk.cloudscape_react_ts_website.CloudscapeReactTsWebsiteProject", version: "0.26.14" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudscape-react-ts-website-project.js","sourceRoot":"","sources":["cloudscape-react-ts-website-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 {\n  TypeSafeApiProject,\n  TypeSafeWebSocketApiProject,\n} from \"@aws/type-safe-api\";\nimport * as Mustache from \"mustache\";\nimport { SampleDir } from \"projen\";\nimport { NodePackageManager, NodeProject } from \"projen/lib/javascript\";\nimport { ReactTypeScriptProject } from \"projen/lib/web\";\nimport { ReactTypeScriptProjectOptions } from \"./react-ts-project-options\";\n\n/**\n * Configuration options for the CloudscapeReactTsWebsiteProject.\n */\nexport interface CloudscapeReactTsWebsiteProjectOptions\n  extends ReactTypeScriptProjectOptions {\n  /**\n   * Name of the application name.\n   *\n   * @default \"Sample App\"\n   */\n  readonly applicationName?: string;\n  /**\n   * Public directory.\n   *\n   * @default \"public\"\n   */\n  readonly publicDir?: string;\n\n  /**\n   * Whether to enable self sign-up.\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   * 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/**\n * Synthesizes a Cloudscape React Typescript Website Project.\n *\n * @pjid cloudscape-react-ts-website\n */\nexport class CloudscapeReactTsWebsiteProject extends ReactTypeScriptProject {\n  public readonly applicationName: string;\n  public readonly allowSignup: boolean;\n  public readonly publicDir: string;\n  public readonly typeSafeApis?: TypeSafeApiProject[];\n  public readonly typeSafeWebSocketApis?: TypeSafeWebSocketApiProject[];\n\n  constructor(options: CloudscapeReactTsWebsiteProjectOptions) {\n    super({\n      ...options,\n      defaultReleaseBranch: options.defaultReleaseBranch ?? \"main\",\n      name: options.name,\n      sampleCode: false,\n      prettier: options.prettier || true,\n      packageManager:\n        options.parent && options.parent instanceof NodeProject\n          ? options.parent.package.packageManager\n          : options.packageManager ?? NodePackageManager.PNPM,\n      readme: {\n        contents: fs\n          .readFileSync(\n            path.resolve(\n              __dirname,\n              \"../samples/cloudscape-react-ts-website/README.md\"\n            )\n          )\n          .toString(),\n      },\n      gitignore: [\n        \"public/runtime-config.json\",\n        \"public/api.json\",\n        \"public/*/api.json\",\n        ...(options.gitignore || []),\n      ],\n    });\n\n    this.typeSafeApis = [\n      ...(options.typeSafeApis || []),\n      ...(options.typeSafeApi ? [options.typeSafeApi] : []),\n    ];\n    this.typeSafeWebSocketApis = options.typeSafeWebSocketApis ?? [];\n\n    this.addDeps(\n      \"@aws-northstar/ui\",\n      \"@cloudscape-design/components\",\n      \"@cloudscape-design/board-components\",\n      \"react-router-dom\"\n    );\n\n    this.testTask.reset();\n    const lintTask = this.tasks.tryFind(\"eslint\");\n    lintTask && this.testTask.spawn(lintTask);\n    this.testTask.exec(\"react-scripts test --watchAll=false --passWithNoTests\");\n\n    this.applicationName = options.applicationName ?? \"Sample App\";\n    this.allowSignup = options.allowSignup ?? false;\n    this.publicDir = options.publicDir ?? \"public\";\n    const srcDir = path.resolve(\n      __dirname,\n      \"../samples/cloudscape-react-ts-website/src\"\n    );\n    const publicDir = path.resolve(\n      __dirname,\n      \"../samples/cloudscape-react-ts-website/public\"\n    );\n\n    this.typeSafeApis.forEach((typeSafeApi) => {\n      const hooks = typeSafeApi.library?.typescriptReactQueryHooks;\n      const libraryHooksPackage = hooks?.package?.packageName;\n      const libraryHooksPackageVersion = hooks?.package?.manifest.version;\n      if (!libraryHooksPackage) {\n        throw new Error(\n          \"Cannot pass in a Type Safe Api without React Hooks Library configured!\"\n        );\n      }\n      this.addDeps(`${libraryHooksPackage}@${libraryHooksPackageVersion}`);\n\n      this.setupApiExplorer(typeSafeApi);\n    });\n\n    this.typeSafeWebSocketApis.forEach((typeSafeWebSocketApi) => {\n      const hooks = typeSafeWebSocketApi.library?.typescriptWebSocketHooks;\n      const client = typeSafeWebSocketApi.library?.typescriptWebSocketClient;\n      if (!hooks || !client) {\n        throw new Error(\n          \"Cannot pass in a Type Safe WebSocket Api without React Hooks Library configured\"\n        );\n      }\n      this.addDeps(\n        `${hooks.package.packageName}@${hooks.package.manifest.version}`,\n        `${client.package.packageName}@${client.package.manifest.version}`\n      );\n\n      this.setupApiExplorer(typeSafeWebSocketApi);\n    });\n\n    const apis = this.typeSafeApis.map((tsApi, idx) => ({\n      apiName: tsApi.model.apiName,\n      isLast: idx === this.typeSafeApis!.length - 1,\n      apiNameSafe: tsApi.model.apiName\n        ?.replace(/[^a-z0-9_]+/gi, \"\")\n        .replace(/^[0-9]+/gi, \"\"),\n      hooksPackage:\n        tsApi.library?.typescriptReactQueryHooks?.package?.packageName,\n    }));\n    const webSocketApis = this.typeSafeWebSocketApis.map((tsApi, idx) => ({\n      apiName: tsApi.model.apiName,\n      isLast: idx === this.typeSafeWebSocketApis!.length - 1,\n      apiNameSafe: tsApi.model.apiName\n        ?.replace(/[^a-z0-9_]+/gi, \"\")\n        .replace(/^[0-9]+/gi, \"\"),\n      hooksPackage:\n        tsApi.library?.typescriptWebSocketHooks?.package?.packageName,\n      clientPackage:\n        tsApi.library?.typescriptWebSocketClient?.package?.packageName,\n    }));\n    const mustacheConfig = {\n      applicationName: this.applicationName,\n      hasApis: apis.length + webSocketApis.length > 0,\n      typeSafeApis: apis,\n      typeSafeApisReversed: [...apis].reverse(),\n      typeSafeWebSocketApis: webSocketApis,\n      typeSafeWebSocketApisReversed: [...webSocketApis].reverse(),\n      allowSignup: options.allowSignup ?? false,\n    };\n\n    new SampleDir(this, this.srcdir, {\n      files: {\n        ...Object.fromEntries(\n          this.buildSampleDirEntries(srcDir, [], mustacheConfig)\n        ),\n      },\n    });\n\n    new SampleDir(this, this.publicDir, {\n      files: {\n        ...Object.fromEntries(\n          this.buildSampleDirEntries(publicDir, [], mustacheConfig)\n        ),\n      },\n    });\n\n    // Linting is managed as part of the test task already, so disable react-scripts running eslint again\n    this.tasks.addEnvironment(\"DISABLE_ESLINT_PLUGIN\", \"true\");\n\n    // Relax EsLint and TSC for dev\n    this.tasks.tryFind(\"dev\")?.env(\"ESLINT_NO_DEV_ERRORS\", \"true\");\n    this.tasks.tryFind(\"dev\")?.env(\"TSC_COMPILE_ON_ERROR\", \"true\");\n  }\n\n  private setupApiExplorer(\n    tsApi: TypeSafeApiProject | TypeSafeWebSocketApiProject\n  ) {\n    this.addDevDeps(\"@types/swagger-ui-react\", \"@types/uuid\");\n    this.addDeps(\n      \"swagger-ui-react@5.5.0\",\n      \"aws4fetch\",\n      \"uuid\",\n      \"@aws-crypto/sha256-js\",\n      \"@aws-sdk/signature-v4\",\n      \"@aws-sdk/protocol-http\"\n    );\n\n    const targetApiSpecFolder = `public/${tsApi.model.apiName}`;\n    const targetApiSpecPath = `${targetApiSpecFolder}/api.json`;\n    this.preCompileTask.exec(`rm -rf ${targetApiSpecFolder}`);\n    this.preCompileTask.exec(\n      `mkdir -p ${targetApiSpecFolder} && cp ${path.relative(\n        this.outdir,\n        tsApi.model.outdir\n      )}/${tsApi.model.parsedSpecFile} ${targetApiSpecPath}`\n    );\n  }\n\n  private buildSampleDirEntries(\n    dir: string,\n    pathPrefixes: string[] = [],\n    mustacheConfig: any\n  ): [string, string][] {\n    return fs\n      .readdirSync(dir, { withFileTypes: true })\n      .filter(\n        (f) =>\n          mustacheConfig.typeSafeApis.length +\n            mustacheConfig.typeSafeWebSocketApis.length >\n            0 || !`${pathPrefixes.join(\"/\")}${f.name}`.includes(\"ApiExplorer\")\n      )\n      .filter(\n        (f) =>\n          mustacheConfig.typeSafeApis.length > 0 ||\n          !`${pathPrefixes.join(\"/\")}${f.name}`.includes(\"TypeSafeApiClient\")\n      )\n      .filter(\n        (f) =>\n          mustacheConfig.typeSafeWebSocketApis.length > 0 ||\n          !`${pathPrefixes.join(\"/\")}${f.name}`.includes(\n            \"TypeSafeWebSocketApiClient\"\n          )\n      )\n      .flatMap((f) =>\n        f.isDirectory()\n          ? this.buildSampleDirEntries(\n              `${dir}/${f.name}`,\n              [...pathPrefixes, f.name],\n              mustacheConfig\n            )\n          : [\n              [\n                `${path.join(\n                  ...pathPrefixes,\n                  f.name.replace(\".mustache\", \"\")\n                )}`,\n                Mustache.render(\n                  fs.readFileSync(`${dir}/${f.name}`).toString(),\n                  mustacheConfig\n                ),\n              ],\n            ]\n      );\n  }\n}\n"]}