UNPKG

projen

Version:

CDK for software projects

109 lines • 15.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.synthSnapshot = synthSnapshot; exports.directorySnapshot = directorySnapshot; const fs = require("fs"); const os = require("os"); const path = require("path"); const JSONC = require("comment-json"); const comment_json_1 = require("comment-json"); const glob = require("fast-glob"); const json_1 = require("../json"); /** * Creates a snapshot of the files generated by a project. Ignores any non-text * files so that the snapshots are human readable. */ function synthSnapshot(project, options = {}) { // defensive: verify that "outdir" is actually in a temporary directory if (!path.resolve(project.outdir).startsWith(os.tmpdir()) && !project.outdir.includes("project-temp-dir")) { throw new Error("Trying to capture a snapshot of a project outside of tmpdir, which implies this test might corrupt an existing project"); } const synthed = Symbol.for("synthed"); if (synthed in project) { throw new Error("duplicate synth()"); } project[synthed] = true; const ENV_PROJEN_DISABLE_POST = process.env.PROJEN_DISABLE_POST; try { process.env.PROJEN_DISABLE_POST = "true"; project.synth(); const ignoreExts = ["png", "ico"]; return directorySnapshot(project.outdir, { ...options, excludeGlobs: ignoreExts.map((ext) => `**/*.${ext}`), supportJsonComments: project.files.some( // At least one json file in project supports comments (file) => file instanceof json_1.JsonFile && file.supportsComments), }); } finally { fs.rmSync(project.outdir, { force: true, recursive: true }); // values assigned to process.env.XYZ are automatically converted to strings if (ENV_PROJEN_DISABLE_POST === undefined) { delete process.env.PROJEN_DISABLE_POST; } else { process.env.PROJEN_DISABLE_POST = ENV_PROJEN_DISABLE_POST; } } } function isJsonLikeFile(filePath) { const file = filePath.toLowerCase(); return (file.endsWith(".json") || file.endsWith(".json5") || file.endsWith(".jsonc")); } function directorySnapshot(root, options = {}) { const output = {}; const files = glob.sync("**", { ignore: [".git/**", ...(options.excludeGlobs ?? [])], cwd: root, onlyFiles: true, followSymbolicLinks: false, dot: true, }); // returns relative file paths with POSIX separators const parseJson = options.parseJson ?? true; for (const file of files) { const filePath = path.join(root, file); let content; if (!options.onlyFileNames) { content = fs.readFileSync(filePath, "utf-8"); if (parseJson && isJsonLikeFile(filePath)) { content = cleanCommentArrays(JSONC.parse(content, undefined, !options.supportJsonComments)); } } else { content = true; } output[file] = content; } return output; } /** * Converts type "CommentArray" back to regular JS "Array" * if there are no comments stored in it. * Prevents strict checks from failing. */ function cleanCommentArrays(obj) { if (Array.isArray(obj) || isCommentArrayWithoutComments(obj)) { return Array.from(obj).map(cleanCommentArrays); } if (obj instanceof Object) { for (const p of Object.keys(obj)) { if (isCommentArrayWithoutComments(obj[p])) { obj[p] = Array.from(obj[p]).map(cleanCommentArrays); } else if (obj[p] instanceof Object) { obj[p] = cleanCommentArrays(obj[p]); } } } return obj; } /** * Checks if a "CommentArray" has no comments stored in it. */ function isCommentArrayWithoutComments(obj) { return (obj instanceof comment_json_1.CommentArray && Object.getOwnPropertySymbols(obj).length === 0); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"synth.js","sourceRoot":"","sources":["../../src/util/synth.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from \"fs\";\nimport * as os from \"os\";\nimport * as path from \"path\";\nimport * as JSONC from \"comment-json\";\nimport { CommentArray } from \"comment-json\";\nimport * as glob from \"fast-glob\";\nimport { JsonFile } from \"../json\";\nimport type { Project } from \"../project\";\n\n/**\n * Options for the Snapshot synthesis\n */\nexport interface SnapshotOptions {\n  /**\n   * Parse .json files as a JS object for improved inspection.\n   * This will fail if the contents are invalid JSON.\n   *\n   * @default true parse .json files into an object\n   */\n  readonly parseJson?: boolean;\n}\n\nexport interface SynthOutput {\n  [filePath: string]: any;\n}\n\n/**\n * Creates a snapshot of the files generated by a project. Ignores any non-text\n * files so that the snapshots are human readable.\n */\nexport function synthSnapshot(\n  project: Project,\n  options: SnapshotOptions = {},\n): SynthOutput {\n  // defensive: verify that \"outdir\" is actually in a temporary directory\n  if (\n    !path.resolve(project.outdir).startsWith(os.tmpdir()) &&\n    !project.outdir.includes(\"project-temp-dir\")\n  ) {\n    throw new Error(\n      \"Trying to capture a snapshot of a project outside of tmpdir, which implies this test might corrupt an existing project\",\n    );\n  }\n\n  const synthed = Symbol.for(\"synthed\");\n  if (synthed in project) {\n    throw new Error(\"duplicate synth()\");\n  }\n\n  (project as any)[synthed] = true;\n\n  const ENV_PROJEN_DISABLE_POST = process.env.PROJEN_DISABLE_POST;\n  try {\n    process.env.PROJEN_DISABLE_POST = \"true\";\n    project.synth();\n    const ignoreExts = [\"png\", \"ico\"];\n    return directorySnapshot(project.outdir, {\n      ...options,\n      excludeGlobs: ignoreExts.map((ext) => `**/*.${ext}`),\n      supportJsonComments: project.files.some(\n        // At least one json file in project supports comments\n        (file) => file instanceof JsonFile && file.supportsComments,\n      ),\n    });\n  } finally {\n    fs.rmSync(project.outdir, { force: true, recursive: true });\n\n    // values assigned to process.env.XYZ are automatically converted to strings\n    if (ENV_PROJEN_DISABLE_POST === undefined) {\n      delete process.env.PROJEN_DISABLE_POST;\n    } else {\n      process.env.PROJEN_DISABLE_POST = ENV_PROJEN_DISABLE_POST;\n    }\n  }\n}\n\nexport interface DirectorySnapshotOptions extends SnapshotOptions {\n  /**\n   * Globs of files to exclude.\n   * @default [] include all files\n   */\n  readonly excludeGlobs?: string[];\n\n  /**\n   * Only snapshot the names of files and not their contents.\n   * The value for a path will be `true` if it exists.\n   *\n   * @default false include file content\n   */\n  readonly onlyFileNames?: boolean;\n\n  /**\n   * Parses files with different parser, supporting comments\n   * inside .json files.\n   * @default false\n   */\n  readonly supportJsonComments?: boolean;\n}\n\nfunction isJsonLikeFile(filePath: string): boolean {\n  const file = filePath.toLowerCase();\n  return (\n    file.endsWith(\".json\") || file.endsWith(\".json5\") || file.endsWith(\".jsonc\")\n  );\n}\n\nexport function directorySnapshot(\n  root: string,\n  options: DirectorySnapshotOptions = {},\n) {\n  const output: SynthOutput = {};\n\n  const files = glob.sync(\"**\", {\n    ignore: [\".git/**\", ...(options.excludeGlobs ?? [])],\n    cwd: root,\n    onlyFiles: true,\n    followSymbolicLinks: false,\n    dot: true,\n  }); // returns relative file paths with POSIX separators\n\n  const parseJson = options.parseJson ?? true;\n\n  for (const file of files) {\n    const filePath = path.join(root, file);\n\n    let content;\n    if (!options.onlyFileNames) {\n      content = fs.readFileSync(filePath, \"utf-8\");\n      if (parseJson && isJsonLikeFile(filePath)) {\n        content = cleanCommentArrays(\n          JSONC.parse(content, undefined, !options.supportJsonComments),\n        );\n      }\n    } else {\n      content = true;\n    }\n\n    output[file] = content;\n  }\n\n  return output;\n}\n\n/**\n * Converts type \"CommentArray\" back to regular JS \"Array\"\n * if there are no comments stored in it.\n * Prevents strict checks from failing.\n */\nfunction cleanCommentArrays(obj: any): typeof obj {\n  if (Array.isArray(obj) || isCommentArrayWithoutComments(obj)) {\n    return Array.from(obj).map(cleanCommentArrays);\n  }\n\n  if (obj instanceof Object) {\n    for (const p of Object.keys(obj)) {\n      if (isCommentArrayWithoutComments(obj[p])) {\n        obj[p] = Array.from(obj[p]).map(cleanCommentArrays);\n      } else if (obj[p] instanceof Object) {\n        obj[p] = cleanCommentArrays(obj[p]);\n      }\n    }\n  }\n\n  return obj;\n}\n\n/**\n * Checks if a \"CommentArray\" has no comments stored in it.\n */\nfunction isCommentArrayWithoutComments(obj: any): boolean {\n  return (\n    obj instanceof CommentArray &&\n    Object.getOwnPropertySymbols(obj).length === 0\n  );\n}\n"]}