projen
Version:
CDK for software projects
109 lines • 15.2 kB
JavaScript
;
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 { 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"]}