projen
Version:
CDK for software projects
338 lines • 45.7 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.discover = discover;
exports.readManifest = readManifest;
exports.resolveProjectType = resolveProjectType;
exports.toProjectType = toProjectType;
exports.readJsiiManifest = readJsiiManifest;
const fs = require("fs");
const path = require("path");
const zlib_1 = require("zlib");
const case_1 = require("case");
const PROJEN_MODULE_ROOT = path.join(__dirname, "..");
const PROJECT_BASE_FQN = "projen.Project";
/**
* Returns a list of project types exported the modules defined in `moduleDirs`.
* This list will always also include the built-in projen project types.
* Modules without a .jsii manifest are skipped.
*
* @param moduleDirs A list of npm module directories
*/
function discover(...moduleDirs) {
const jsii = discoverJsiiTypes(...moduleDirs);
const result = new Array();
for (const fqn of Object.keys(jsii)) {
if (isProjectType(jsii, fqn)) {
const p = toProjectType(jsii, fqn);
result.push(p);
}
}
return result.sort((r1, r2) => r1.pjid.localeCompare(r2.pjid));
}
function readManifest(dir) {
const jsiiFile = path.join(dir, ".jsii");
if (!fs.existsSync(jsiiFile)) {
return undefined;
} // no jsii manifest
let manifest = JSON.parse(fs.readFileSync(jsiiFile, "utf-8"));
if (manifest.schema === "jsii/file-redirect") {
const compressedFile = path.join(dir, manifest.filename);
if (!fs.existsSync(compressedFile)) {
throw new Error(`${compressedFile} does not exist.`);
}
switch (manifest.compression) {
case "gzip":
manifest = JSON.parse((0, zlib_1.unzipSync)(fs.readFileSync(compressedFile)).toString());
break;
default:
throw new Error(`Unsupported compression format: ${manifest.compression}`);
}
}
return manifest;
}
/**
* Resolve all jsii types from @modulesDirs.
* When a jsii module is found it will recusively list the types from the dependant module as well
*
* @param moduleDirs
* @returns
*/
function discoverJsiiTypes(...moduleDirs) {
const jsii = {};
const discoveredManifests = [];
const discoverJsii = (dir) => {
const manifest = readManifest(dir);
if (!manifest) {
return;
}
if (discoveredManifests.includes(manifest.fingerprint)) {
return;
}
discoveredManifests.push(manifest.fingerprint);
for (const [fqn, type] of Object.entries(manifest.types)) {
jsii[fqn] = {
...type,
};
}
// Also search recursively in nested project dependencies. If the requested module is an external module
// this will also end-up in the projen module and add the projen types
if (manifest.dependencies) {
for (const dependency of Object.keys(manifest.dependencies)) {
const nestedDependencyFolder = path.dirname(require.resolve(`${dependency}/package.json`, {
paths: [dir],
}));
if (fs.existsSync(nestedDependencyFolder)) {
discoverJsii(nestedDependencyFolder);
}
}
}
};
// read all .jsii manifests from all requested modules and merge
// them all into a single map of fqn->type.
for (const dir of [...moduleDirs, PROJEN_MODULE_ROOT]) {
discoverJsii(dir);
// Read from scoped packages
if (dir.includes("@") && fs.lstatSync(dir).isDirectory()) {
const childDirs = fs.readdirSync(dir).map((file) => path.join(dir, file));
for (const child of childDirs) {
discoverJsii(child);
}
}
}
return jsii;
}
function resolveProjectType(projectFqn) {
let [moduleName] = projectFqn.split(".");
if (moduleName === "projen") {
moduleName = PROJEN_MODULE_ROOT;
}
// try picking the manifest. We only need the base folder but this is directly a nice check if we request from a valid jsii package
const jsiiManifestFile = require.resolve(`${moduleName}/.jsii`, {
paths: [process.cwd()],
});
const moduleFolder = path.dirname(jsiiManifestFile);
// Read all jsii types that can be loaded from this project type
const jsii = discoverJsiiTypes(moduleFolder);
return toProjectType(jsii, projectFqn);
}
function toProjectType(jsii, fqn) {
if (!isProjectType(jsii, fqn)) {
throw new Error(`Fully qualified name "${fqn}" is not a valid project type.`);
}
const typeinfo = jsii[fqn];
// projen.web.ReactProject -> web.ReactProject
const typename = fqn.substring(fqn.indexOf(".") + 1);
// projen.web.ReactProject -> web
// projen.Project -> projen
const readmeFileName = typename.includes(".")
? typename.split(".", 1)[0]
: typeinfo.assembly;
// * [java](https://projen.io/docs/api/java#javaproject-) - Java project.
const docsurl = `https://projen.io/docs/api/${readmeFileName}#${typename
.substring(typename.indexOf(".") + 1)
.toLowerCase()}-`;
let pjid = typeinfo.docs?.custom?.pjid ?? (0, case_1.snake)(typename).replace(/_project$/, "");
return {
moduleName: typeinfo.assembly,
typename,
pjid,
fqn,
options: discoverOptions(jsii, fqn),
docs: typeinfo.docs?.summary,
docsurl,
};
}
function readJsiiManifest(jsiiFqn) {
let [moduleName] = jsiiFqn.split(".");
if (moduleName === "projen") {
moduleName = PROJEN_MODULE_ROOT;
}
const jsiiManifestFile = require.resolve(`${moduleName}/.jsii`);
return JSON.parse(fs.readFileSync(jsiiManifestFile, "utf-8"));
}
function discoverOptions(jsii, fqn) {
const options = {};
const params = jsii[fqn]?.initializer?.parameters ?? [];
const optionsParam = params[0];
const optionsTypeFqn = optionsParam?.type?.fqn;
if (params.length > 1 ||
(params.length === 1 && optionsParam?.name !== "options")) {
throw new Error(`constructor for project ${fqn} must have a single "options" argument of a struct type. got ${JSON.stringify(params)}`);
}
addOptions(optionsTypeFqn);
const opts = Object.values(options);
return opts.sort((a, b) => a.name.localeCompare(b.name));
function addOptions(ofqn, basePath = [], optional = false) {
if (!ofqn) {
return;
}
const struct = jsii[ofqn];
if (!struct) {
throw new Error(`unable to find options type ${ofqn} for project ${fqn}`);
}
for (const prop of struct.properties ?? []) {
const propPath = [...basePath, prop.name];
// protect against double-booking
if (prop.name in options) {
throw new Error(`duplicate option "${prop.name}" in ${fqn} (already declared in ${options[prop.name].parent})`);
}
let jsiiKind;
if (prop.type?.fqn) {
jsiiKind = jsii[prop.type?.fqn].kind; // e.g. 'class', 'interface', 'enum'
}
const isOptional = optional || prop.optional;
const defaultValue = sanitizeValue(prop.docs?.default);
const pjnew = sanitizeValue(prop.docs?.custom?.pjnew);
// if this is a mandatory option and we have a default value,
// or the option is tagged to be rendered with an initial value,
// the value has to be JSON-parsable to the correct type
const initialValue = getInitialValue(defaultValue, pjnew, isOptional);
if (initialValue) {
checkDefaultIsParsable(prop.name, initialValue, prop.type);
}
options[prop.name] = filterUndefined({
path: propPath,
parent: struct.name,
name: prop.name,
fqn: prop.type?.fqn,
docs: prop.docs.summary,
simpleType: prop.type ? getSimpleTypeName(prop.type) : "unknown",
fullType: prop.type,
kind: jsiiKind,
jsonLike: prop.type ? isJsonLike(jsii, prop.type) : undefined,
switch: propPath.map((p) => (0, case_1.snake)(p).replace(/_/g, "-")).join("-"),
default: defaultValue,
initialValue: initialValue,
optional: isOptional,
featured: prop.docs?.custom?.featured === "true",
deprecated: prop.docs.stability === "deprecated" ? true : undefined,
});
}
for (const ifc of struct.interfaces ?? []) {
addOptions(ifc);
}
}
}
function getInitialValue(defaultValue, pjnew, isOptional = false) {
if (pjnew) {
return pjnew;
}
if (!isOptional) {
return defaultValue;
}
return undefined;
}
function sanitizeValue(val) {
if (val === "undefined") {
return undefined;
}
return val;
}
function getSimpleTypeName(type) {
if (type?.primitive) {
return type.primitive; // e.g. 'string', 'boolean', 'number'
}
else if (type?.fqn) {
return type.fqn.split(".").pop(); // projen.NodeProjectOptions -> NodeProjectOptions
}
else {
// any other types such as collection types
return "unknown";
}
}
/**
* Whether a value of this type is serializable into JSON.
*/
function isJsonLike(jsii, type) {
if (type.primitive) {
// string, boolean, number, any
return true;
}
else if (type.fqn) {
const kind = jsii[type.fqn].kind;
if (["interface", "enum"].includes(kind)) {
// not 'class'
return true;
}
}
else if (type.collection) {
return isJsonLike(jsii, type.collection.elementtype);
}
return false;
}
function filterUndefined(obj) {
const ret = {};
for (const [k, v] of Object.entries(obj)) {
if (v !== undefined) {
ret[k] = v;
}
}
return ret;
}
function isProjectType(jsii, fqn) {
const type = jsii[fqn];
if (!type) {
throw new Error(`Could not find project type with fqn "${fqn}" in .jsii file.`);
}
if (type.kind !== "class") {
return false;
}
if (type.abstract) {
return false;
}
if (type.docs?.deprecated) {
return false;
}
let curr = type;
while (true) {
if (curr.fqn === PROJECT_BASE_FQN) {
return true;
}
if (!curr.base) {
return false;
}
curr = jsii[curr.base];
if (!curr) {
return false;
}
}
}
function isPrimitiveArray({ collection }) {
return Boolean(collection?.kind === "array" && collection?.elementtype.primitive);
}
function isPrimitiveOrPrimitiveArray(type) {
return Boolean(type?.primitive || isPrimitiveArray(type));
}
function checkDefaultIsParsable(prop, value, type) {
if (!(type && isPrimitiveOrPrimitiveArray(type))) {
throw new Error(`required option "${prop}" with a @default must use primitive types (string, number and boolean) or a primitive array. type found is: ${JSON.stringify(type)}`);
}
// macros are pass-through
if (value.startsWith("$")) {
return;
}
try {
const parsed = JSON.parse(value);
// Primitive type
if (typeof parsed === type.primitive) {
return;
}
// Primitive array
if (Array.isArray(parsed) && isPrimitiveArray(type)) {
// but empty (which is okay)
if (parsed.length === 0) {
return;
}
// if first element matches the type, assume it's correct
if (typeof parsed[0] === type?.collection?.elementtype.primitive) {
return;
}
}
// Parsed value does not match type
throw new Error(`cannot parse @default value for mandatory option ${prop} as a ${type}: ${parsed}`);
}
catch (e) {
throw new Error(`unable to JSON.parse() value "${value}" specified as @default for mandatory option "${prop}": ${e.message}`);
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"inventory.js","sourceRoot":"","sources":["../src/inventory.ts"],"names":[],"mappings":";;AA+FA,4BAaC;AAED,oCA4BC;AAiED,gDAeC;AAED,sCAkCC;AAED,4CAQC;AAxQD,yBAAyB;AACzB,6BAA6B;AAC7B,+BAAiC;AACjC,+BAA6B;AAE7B,MAAM,kBAAkB,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;AACtD,MAAM,gBAAgB,GAAG,gBAAgB,CAAC;AAkF1C;;;;;;GAMG;AACH,SAAgB,QAAQ,CAAC,GAAG,UAAoB;IAC9C,MAAM,IAAI,GAAG,iBAAiB,CAAC,GAAG,UAAU,CAAC,CAAC;IAE9C,MAAM,MAAM,GAAG,IAAI,KAAK,EAAe,CAAC;IAExC,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;QACpC,IAAI,aAAa,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC;YAC7B,MAAM,CAAC,GAAG,aAAa,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;YACnC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACjB,CAAC;IACH,CAAC;IAED,OAAO,MAAM,CAAC,IAAI,CAAC,CAAC,EAAE,EAAE,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC;AACjE,CAAC;AAED,SAAgB,YAAY,CAAC,GAAW;IACtC,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,OAAO,CAAC,CAAC;IACzC,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC7B,OAAO,SAAS,CAAC;IACnB,CAAC,CAAC,mBAAmB;IACrB,IAAI,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC,CAAC;IAE9D,IAAI,QAAQ,CAAC,MAAM,KAAK,oBAAoB,EAAE,CAAC;QAC7C,MAAM,cAAc,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,QAAQ,CAAC,QAAQ,CAAC,CAAC;QAEzD,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,cAAc,CAAC,EAAE,CAAC;YACnC,MAAM,IAAI,KAAK,CAAC,GAAG,cAAc,kBAAkB,CAAC,CAAC;QACvD,CAAC;QAED,QAAQ,QAAQ,CAAC,WAAW,EAAE,CAAC;YAC7B,KAAK,MAAM;gBACT,QAAQ,GAAG,IAAI,CAAC,KAAK,CACnB,IAAA,gBAAS,EAAC,EAAE,CAAC,YAAY,CAAC,cAAc,CAAC,CAAC,CAAC,QAAQ,EAAE,CACtD,CAAC;gBACF,MAAM;YACR;gBACE,MAAM,IAAI,KAAK,CACb,mCAAmC,QAAQ,CAAC,WAAW,EAAE,CAC1D,CAAC;QACN,CAAC;IACH,CAAC;IAED,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED;;;;;;GAMG;AACH,SAAS,iBAAiB,CAAC,GAAG,UAAoB;IAChD,MAAM,IAAI,GAAc,EAAE,CAAC;IAC3B,MAAM,mBAAmB,GAAkB,EAAE,CAAC;IAE9C,MAAM,YAAY,GAAG,CAAC,GAAW,EAAE,EAAE;QACnC,MAAM,QAAQ,GAAG,YAAY,CAAC,GAAG,CAAC,CAAC;QAEnC,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,OAAO;QACT,CAAC;QAED,IAAI,mBAAmB,CAAC,QAAQ,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;YACvD,OAAO;QACT,CAAC;QACD,mBAAmB,CAAC,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;QAE/C,KAAK,MAAM,CAAC,GAAG,EAAE,IAAI,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,KAAkB,CAAC,EAAE,CAAC;YACtE,IAAI,CAAC,GAAG,CAAC,GAAG;gBACV,GAAG,IAAI;aACR,CAAC;QACJ,CAAC;QAED,wGAAwG;QACxG,sEAAsE;QACtE,IAAI,QAAQ,CAAC,YAAY,EAAE,CAAC;YAC1B,KAAK,MAAM,UAAU,IAAI,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,CAAC,EAAE,CAAC;gBAC5D,MAAM,sBAAsB,GAAG,IAAI,CAAC,OAAO,CACzC,OAAO,CAAC,OAAO,CAAC,GAAG,UAAU,eAAe,EAAE;oBAC5C,KAAK,EAAE,CAAC,GAAG,CAAC;iBACb,CAAC,CACH,CAAC;gBAEF,IAAI,EAAE,CAAC,UAAU,CAAC,sBAAsB,CAAC,EAAE,CAAC;oBAC1C,YAAY,CAAC,sBAAsB,CAAC,CAAC;gBACvC,CAAC;YACH,CAAC;QACH,CAAC;IACH,CAAC,CAAC;IAEF,gEAAgE;IAChE,2CAA2C;IAC3C,KAAK,MAAM,GAAG,IAAI,CAAC,GAAG,UAAU,EAAE,kBAAkB,CAAC,EAAE,CAAC;QACtD,YAAY,CAAC,GAAG,CAAC,CAAC;QAElB,4BAA4B;QAC5B,IAAI,GAAG,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,WAAW,EAAE,EAAE,CAAC;YACzD,MAAM,SAAS,GAAG,EAAE,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC,CAAC;YAC1E,KAAK,MAAM,KAAK,IAAI,SAAS,EAAE,CAAC;gBAC9B,YAAY,CAAC,KAAK,CAAC,CAAC;YACtB,CAAC;QACH,CAAC;IACH,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED,SAAgB,kBAAkB,CAAC,UAAkB;IACnD,IAAI,CAAC,UAAU,CAAC,GAAG,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACzC,IAAI,UAAU,KAAK,QAAQ,EAAE,CAAC;QAC5B,UAAU,GAAG,kBAAkB,CAAC;IAClC,CAAC;IAED,mIAAmI;IACnI,MAAM,gBAAgB,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,UAAU,QAAQ,EAAE;QAC9D,KAAK,EAAE,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC;KACvB,CAAC,CAAC;IACH,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,CAAC;IAEpD,gEAAgE;IAChE,MAAM,IAAI,GAAG,iBAAiB,CAAC,YAAY,CAAC,CAAC;IAC7C,OAAO,aAAa,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;AACzC,CAAC;AAED,SAAgB,aAAa,CAAC,IAAe,EAAE,GAAW;IACxD,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC;QAC9B,MAAM,IAAI,KAAK,CACb,yBAAyB,GAAG,gCAAgC,CAC7D,CAAC;IACJ,CAAC;IAED,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC;IAE3B,8CAA8C;IAC9C,MAAM,QAAQ,GAAG,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;IAErD,iCAAiC;IACjC,2BAA2B;IAC3B,MAAM,cAAc,GAAG,QAAQ,CAAC,QAAQ,CAAC,GAAG,CAAC;QAC3C,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QAC3B,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC;IAEtB,yEAAyE;IAEzE,MAAM,OAAO,GAAG,8BAA8B,cAAc,IAAI,QAAQ;SACrE,SAAS,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;SACpC,WAAW,EAAE,GAAG,CAAC;IACpB,IAAI,IAAI,GACN,QAAQ,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,IAAI,IAAA,YAAK,EAAC,QAAQ,CAAC,CAAC,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;IAC1E,OAAO;QACL,UAAU,EAAE,QAAQ,CAAC,QAAQ;QAC7B,QAAQ;QACR,IAAI;QACJ,GAAG;QACH,OAAO,EAAE,eAAe,CAAC,IAAI,EAAE,GAAG,CAAC;QACnC,IAAI,EAAE,QAAQ,CAAC,IAAI,EAAE,OAAO;QAC5B,OAAO;KACO,CAAC;AACnB,CAAC;AAED,SAAgB,gBAAgB,CAAC,OAAe;IAC9C,IAAI,CAAC,UAAU,CAAC,GAAG,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACtC,IAAI,UAAU,KAAK,QAAQ,EAAE,CAAC;QAC5B,UAAU,GAAG,kBAAkB,CAAC;IAClC,CAAC;IAED,MAAM,gBAAgB,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,UAAU,QAAQ,CAAC,CAAC;IAChE,OAAO,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,gBAAgB,EAAE,OAAO,CAAC,CAAC,CAAC;AAChE,CAAC;AAED,SAAS,eAAe,CAAC,IAAe,EAAE,GAAW;IACnD,MAAM,OAAO,GAAsC,EAAE,CAAC;IACtD,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,WAAW,EAAE,UAAU,IAAI,EAAE,CAAC;IACxD,MAAM,YAAY,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;IAC/B,MAAM,cAAc,GAAG,YAAY,EAAE,IAAI,EAAE,GAAG,CAAC;IAE/C,IACE,MAAM,CAAC,MAAM,GAAG,CAAC;QACjB,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,IAAI,YAAY,EAAE,IAAI,KAAK,SAAS,CAAC,EACzD,CAAC;QACD,MAAM,IAAI,KAAK,CACb,2BAA2B,GAAG,gEAAgE,IAAI,CAAC,SAAS,CAC1G,MAAM,CACP,EAAE,CACJ,CAAC;IACJ,CAAC;IAED,UAAU,CAAC,cAAc,CAAC,CAAC;IAE3B,MAAM,IAAI,GAAG,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IAEpC,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;IAEzD,SAAS,UAAU,CACjB,IAAa,EACb,WAAqB,EAAE,EACvB,QAAQ,GAAG,KAAK;QAEhB,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,OAAO;QACT,CAAC;QAED,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC;QAC1B,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,MAAM,IAAI,KAAK,CAAC,+BAA+B,IAAI,gBAAgB,GAAG,EAAE,CAAC,CAAC;QAC5E,CAAC;QAED,KAAK,MAAM,IAAI,IAAI,MAAM,CAAC,UAAU,IAAI,EAAE,EAAE,CAAC;YAC3C,MAAM,QAAQ,GAAG,CAAC,GAAG,QAAQ,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;YAE1C,iCAAiC;YACjC,IAAI,IAAI,CAAC,IAAI,IAAI,OAAO,EAAE,CAAC;gBACzB,MAAM,IAAI,KAAK,CACb,qBAAqB,IAAI,CAAC,IAAI,QAAQ,GAAG,yBACvC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,MACrB,GAAG,CACJ,CAAC;YACJ,CAAC;YAED,IAAI,QAAQ,CAAC;YACb,IAAI,IAAI,CAAC,IAAI,EAAE,GAAG,EAAE,CAAC;gBACnB,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC,oCAAoC;YAC5E,CAAC;YAED,MAAM,UAAU,GAAG,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC;YAC7C,MAAM,YAAY,GAAG,aAAa,CAAC,IAAI,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;YACvD,MAAM,KAAK,GAAG,aAAa,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,CAAC,CAAC;YAEtD,6DAA6D;YAC7D,gEAAgE;YAChE,wDAAwD;YACxD,MAAM,YAAY,GAAG,eAAe,CAAC,YAAY,EAAE,KAAK,EAAE,UAAU,CAAC,CAAC;YACtE,IAAI,YAAY,EAAE,CAAC;gBACjB,sBAAsB,CAAC,IAAI,CAAC,IAAI,EAAE,YAAY,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC;YAC7D,CAAC;YAED,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,eAAe,CAAC;gBACnC,IAAI,EAAE,QAAQ;gBACd,MAAM,EAAE,MAAM,CAAC,IAAI;gBACnB,IAAI,EAAE,IAAI,CAAC,IAAI;gBACf,GAAG,EAAE,IAAI,CAAC,IAAI,EAAE,GAAG;gBACnB,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,OAAO;gBACvB,UAAU,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS;gBAChE,QAAQ,EAAE,IAAI,CAAC,IAAI;gBACnB,IAAI,EAAE,QAAQ;gBACd,QAAQ,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS;gBAC7D,MAAM,EAAE,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAA,YAAK,EAAC,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC;gBAClE,OAAO,EAAE,YAAY;gBACrB,YAAY,EAAE,YAAY;gBAC1B,QAAQ,EAAE,UAAU;gBACpB,QAAQ,EAAE,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,KAAK,MAAM;gBAChD,UAAU,EAAE,IAAI,CAAC,IAAI,CAAC,SAAS,KAAK,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS;aACpE,CAAC,CAAC;QACL,CAAC;QAED,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,UAAU,IAAI,EAAE,EAAE,CAAC;YAC1C,UAAU,CAAC,GAAG,CAAC,CAAC;QAClB,CAAC;IACH,CAAC;AACH,CAAC;AAED,SAAS,eAAe,CACtB,YAAqB,EACrB,KAAc,EACd,aAAsB,KAAK;IAE3B,IAAI,KAAK,EAAE,CAAC;QACV,OAAO,KAAK,CAAC;IACf,CAAC;IAED,IAAI,CAAC,UAAU,EAAE,CAAC;QAChB,OAAO,YAAY,CAAC;IACtB,CAAC;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,SAAS,aAAa,CAAC,GAAY;IACjC,IAAI,GAAG,KAAK,WAAW,EAAE,CAAC;QACxB,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,OAAO,GAAG,CAAC;AACb,CAAC;AAED,SAAS,iBAAiB,CAAC,IAAsB;IAC/C,IAAI,IAAI,EAAE,SAAS,EAAE,CAAC;QACpB,OAAO,IAAI,CAAC,SAAS,CAAC,CAAC,qCAAqC;IAC9D,CAAC;SAAM,IAAI,IAAI,EAAE,GAAG,EAAE,CAAC;QACrB,OAAO,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAG,CAAC,CAAC,kDAAkD;IACvF,CAAC;SAAM,CAAC;QACN,2CAA2C;QAC3C,OAAO,SAAS,CAAC;IACnB,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,UAAU,CAAC,IAAe,EAAE,IAAsB;IACzD,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;QACnB,+BAA+B;QAC/B,OAAO,IAAI,CAAC;IACd,CAAC;SAAM,IAAI,IAAI,CAAC,GAAG,EAAE,CAAC;QACpB,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC;QACjC,IAAI,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;YACzC,cAAc;YACd,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;SAAM,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;QAC3B,OAAO,UAAU,CAAC,IAAI,EAAE,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;IACvD,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,SAAS,eAAe,CAAC,GAAQ;IAC/B,MAAM,GAAG,GAAQ,EAAE,CAAC;IACpB,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;QACzC,IAAI,CAAC,KAAK,SAAS,EAAE,CAAC;YACpB,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;QACb,CAAC;IACH,CAAC;IACD,OAAO,GAAG,CAAC;AACb,CAAC;AAED,SAAS,aAAa,CAAC,IAAe,EAAE,GAAW;IACjD,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC;IAEvB,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,MAAM,IAAI,KAAK,CACb,yCAAyC,GAAG,mBAAmB,CAChE,CAAC;IACJ,CAAC;IAED,IAAI,IAAI,CAAC,IAAI,KAAK,OAAO,EAAE,CAAC;QAC1B,OAAO,KAAK,CAAC;IACf,CAAC;IACD,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;QAClB,OAAO,KAAK,CAAC;IACf,CAAC;IAED,IAAI,IAAI,CAAC,IAAI,EAAE,UAAU,EAAE,CAAC;QAC1B,OAAO,KAAK,CAAC;IACf,CAAC;IAED,IAAI,IAAI,GAAG,IAAI,CAAC;IAChB,OAAO,IAAI,EAAE,CAAC;QACZ,IAAI,IAAI,CAAC,GAAG,KAAK,gBAAgB,EAAE,CAAC;YAClC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;YACf,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACvB,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;AACH,CAAC;AAED,SAAS,gBAAgB,CAAC,EAAE,UAAU,EAAoB;IACxD,OAAO,OAAO,CACZ,UAAU,EAAE,IAAI,KAAK,OAAO,IAAI,UAAU,EAAE,WAAW,CAAC,SAAS,CAClE,CAAC;AACJ,CAAC;AAED,SAAS,2BAA2B,CAAC,IAAsB;IACzD,OAAO,OAAO,CAAC,IAAI,EAAE,SAAS,IAAI,gBAAgB,CAAC,IAAI,CAAC,CAAC,CAAC;AAC5D,CAAC;AAED,SAAS,sBAAsB,CAC7B,IAAY,EACZ,KAAa,EACb,IAAuB;IAEvB,IAAI,CAAC,CAAC,IAAI,IAAI,2BAA2B,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC;QACjD,MAAM,IAAI,KAAK,CACb,oBAAoB,IAAI,gHAAgH,IAAI,CAAC,SAAS,CACpJ,IAAI,CACL,EAAE,CACJ,CAAC;IACJ,CAAC;IAED,0BAA0B;IAC1B,IAAI,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;QAC1B,OAAO;IACT,CAAC;IAED,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAEjC,iBAAiB;QACjB,IAAI,OAAO,MAAM,KAAK,IAAI,CAAC,SAAS,EAAE,CAAC;YACrC,OAAO;QACT,CAAC;QAED,kBAAkB;QAClB,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,gBAAgB,CAAC,IAAI,CAAC,EAAE,CAAC;YACpD,4BAA4B;YAC5B,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACxB,OAAO;YACT,CAAC;YAED,yDAAyD;YACzD,IAAI,OAAO,MAAM,CAAC,CAAC,CAAC,KAAK,IAAI,EAAE,UAAU,EAAE,WAAW,CAAC,SAAS,EAAE,CAAC;gBACjE,OAAO;YACT,CAAC;QACH,CAAC;QAED,mCAAmC;QACnC,MAAM,IAAI,KAAK,CACb,oDAAoD,IAAI,SAAS,IAAI,KAAK,MAAM,EAAE,CACnF,CAAC;IACJ,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACX,MAAM,IAAI,KAAK,CACb,iCAAiC,KAAK,iDAAiD,IAAI,MACxF,CAAS,CAAC,OACb,EAAE,CACH,CAAC;IACJ,CAAC;AACH,CAAC","sourcesContent":["import * as fs from \"fs\";\nimport * as path from \"path\";\nimport { unzipSync } from \"zlib\";\nimport { snake } from \"case\";\n\nconst PROJEN_MODULE_ROOT = path.join(__dirname, \"..\");\nconst PROJECT_BASE_FQN = \"projen.Project\";\n\ntype JsiiTypes = { [name: string]: JsiiType };\n\nexport interface ProjectOption {\n  path: string[];\n  name: string;\n  fqn?: string;\n  switch: string;\n  /** Simple type name, e.g. \"string\", \"boolean\", \"number\", \"EslintOptions\", \"MyEnum\". Collections are \"unknown\" */\n  simpleType: string;\n  /** Full JSII type, e.g. { primitive: \"string\" } or { collection: { elementtype: { primitive: 'string' }, kind: 'map' } } */\n  fullType: JsiiPropertyType;\n  kind?: \"class\" | \"enum\" | \"interface\";\n  jsonLike?: boolean;\n  parent: string;\n  docs?: string;\n  default?: string;\n  /**\n   * The value that will be used at initial project creation\n   */\n  initialValue?: string;\n  optional?: boolean;\n  deprecated?: boolean;\n  featured?: boolean;\n}\n\nexport interface ProjectType {\n  moduleName: string;\n  pjid: string;\n  fqn: string;\n  typename: string;\n  options: ProjectOption[];\n  docs?: string;\n  docsurl: string;\n}\n\ninterface JsiiType {\n  name: string;\n  assembly: string;\n  kind: string;\n  abstract?: boolean;\n  base?: string;\n  fqn: string;\n  interfaces?: string[];\n  initializer?: {\n    parameters?: Array<{\n      name: string;\n      type?: { fqn?: string };\n    }>;\n  };\n  properties?: Array<{\n    name: string;\n    docs: {\n      summary?: string;\n      default?: string;\n      deprecated?: string;\n      stability?: string;\n      custom?: { [name: string]: string };\n    };\n    optional?: boolean;\n    type?: JsiiPropertyType;\n  }>;\n  docs?: {\n    summary?: string;\n    deprecated?: string;\n    custom?: {\n      pjid?: string;\n      pjnew?: string;\n    };\n  };\n}\n\nexport interface JsiiPropertyType {\n  primitive?: string;\n  fqn?: string;\n  collection?: {\n    elementtype: JsiiPropertyType;\n    kind: string;\n  };\n}\n\n/**\n * Returns a list of project types exported the modules defined in `moduleDirs`.\n * This list will always also include the built-in projen project types.\n * Modules without a .jsii manifest are skipped.\n *\n * @param moduleDirs A list of npm module directories\n */\nexport function discover(...moduleDirs: string[]): ProjectType[] {\n  const jsii = discoverJsiiTypes(...moduleDirs);\n\n  const result = new Array<ProjectType>();\n\n  for (const fqn of Object.keys(jsii)) {\n    if (isProjectType(jsii, fqn)) {\n      const p = toProjectType(jsii, fqn);\n      result.push(p);\n    }\n  }\n\n  return result.sort((r1, r2) => r1.pjid.localeCompare(r2.pjid));\n}\n\nexport function readManifest(dir: string) {\n  const jsiiFile = path.join(dir, \".jsii\");\n  if (!fs.existsSync(jsiiFile)) {\n    return undefined;\n  } // no jsii manifest\n  let manifest = JSON.parse(fs.readFileSync(jsiiFile, \"utf-8\"));\n\n  if (manifest.schema === \"jsii/file-redirect\") {\n    const compressedFile = path.join(dir, manifest.filename);\n\n    if (!fs.existsSync(compressedFile)) {\n      throw new Error(`${compressedFile} does not exist.`);\n    }\n\n    switch (manifest.compression) {\n      case \"gzip\":\n        manifest = JSON.parse(\n          unzipSync(fs.readFileSync(compressedFile)).toString()\n        );\n        break;\n      default:\n        throw new Error(\n          `Unsupported compression format: ${manifest.compression}`\n        );\n    }\n  }\n\n  return manifest;\n}\n\n/**\n * Resolve all jsii types from @modulesDirs.\n * When a jsii module is found it will recusively list the types from the dependant module as well\n *\n * @param moduleDirs\n * @returns\n */\nfunction discoverJsiiTypes(...moduleDirs: string[]) {\n  const jsii: JsiiTypes = {};\n  const discoveredManifests: Array<string> = [];\n\n  const discoverJsii = (dir: string) => {\n    const manifest = readManifest(dir);\n\n    if (!manifest) {\n      return;\n    }\n\n    if (discoveredManifests.includes(manifest.fingerprint)) {\n      return;\n    }\n    discoveredManifests.push(manifest.fingerprint);\n\n    for (const [fqn, type] of Object.entries(manifest.types as JsiiTypes)) {\n      jsii[fqn] = {\n        ...type,\n      };\n    }\n\n    // Also search recursively in nested project dependencies. If the requested module is an external module\n    // this will also end-up in the projen module and add the projen types\n    if (manifest.dependencies) {\n      for (const dependency of Object.keys(manifest.dependencies)) {\n        const nestedDependencyFolder = path.dirname(\n          require.resolve(`${dependency}/package.json`, {\n            paths: [dir],\n          })\n        );\n\n        if (fs.existsSync(nestedDependencyFolder)) {\n          discoverJsii(nestedDependencyFolder);\n        }\n      }\n    }\n  };\n\n  // read all .jsii manifests from all requested modules and merge\n  // them all into a single map of fqn->type.\n  for (const dir of [...moduleDirs, PROJEN_MODULE_ROOT]) {\n    discoverJsii(dir);\n\n    // Read from scoped packages\n    if (dir.includes(\"@\") && fs.lstatSync(dir).isDirectory()) {\n      const childDirs = fs.readdirSync(dir).map((file) => path.join(dir, file));\n      for (const child of childDirs) {\n        discoverJsii(child);\n      }\n    }\n  }\n\n  return jsii;\n}\n\nexport function resolveProjectType(projectFqn: string): ProjectType {\n  let [moduleName] = projectFqn.split(\".\");\n  if (moduleName === \"projen\") {\n    moduleName = PROJEN_MODULE_ROOT;\n  }\n\n  // try picking the manifest. We only need the base folder but this is directly a nice check if we request from a valid jsii package\n  const jsiiManifestFile = require.resolve(`${moduleName}/.jsii`, {\n    paths: [process.cwd()],\n  });\n  const moduleFolder = path.dirname(jsiiManifestFile);\n\n  // Read all jsii types that can be loaded from this project type\n  const jsii = discoverJsiiTypes(moduleFolder);\n  return toProjectType(jsii, projectFqn);\n}\n\nexport function toProjectType(jsii: JsiiTypes, fqn: string): ProjectType {\n  if (!isProjectType(jsii, fqn)) {\n    throw new Error(\n      `Fully qualified name \"${fqn}\" is not a valid project type.`\n    );\n  }\n\n  const typeinfo = jsii[fqn];\n\n  // projen.web.ReactProject -> web.ReactProject\n  const typename = fqn.substring(fqn.indexOf(\".\") + 1);\n\n  // projen.web.ReactProject -> web\n  // projen.Project -> projen\n  const readmeFileName = typename.includes(\".\")\n    ? typename.split(\".\", 1)[0]\n    : typeinfo.assembly;\n\n  // * [java](https://projen.io/docs/api/java#javaproject-) - Java project.\n\n  const docsurl = `https://projen.io/docs/api/${readmeFileName}#${typename\n    .substring(typename.indexOf(\".\") + 1)\n    .toLowerCase()}-`;\n  let pjid =\n    typeinfo.docs?.custom?.pjid ?? snake(typename).replace(/_project$/, \"\");\n  return {\n    moduleName: typeinfo.assembly,\n    typename,\n    pjid,\n    fqn,\n    options: discoverOptions(jsii, fqn),\n    docs: typeinfo.docs?.summary,\n    docsurl,\n  } as ProjectType;\n}\n\nexport function readJsiiManifest(jsiiFqn: string): any {\n  let [moduleName] = jsiiFqn.split(\".\");\n  if (moduleName === \"projen\") {\n    moduleName = PROJEN_MODULE_ROOT;\n  }\n\n  const jsiiManifestFile = require.resolve(`${moduleName}/.jsii`);\n  return JSON.parse(fs.readFileSync(jsiiManifestFile, \"utf-8\"));\n}\n\nfunction discoverOptions(jsii: JsiiTypes, fqn: string): ProjectOption[] {\n  const options: { [name: string]: ProjectOption } = {};\n  const params = jsii[fqn]?.initializer?.parameters ?? [];\n  const optionsParam = params[0];\n  const optionsTypeFqn = optionsParam?.type?.fqn;\n\n  if (\n    params.length > 1 ||\n    (params.length === 1 && optionsParam?.name !== \"options\")\n  ) {\n    throw new Error(\n      `constructor for project ${fqn} must have a single \"options\" argument of a struct type. got ${JSON.stringify(\n        params\n      )}`\n    );\n  }\n\n  addOptions(optionsTypeFqn);\n\n  const opts = Object.values(options);\n\n  return opts.sort((a, b) => a.name.localeCompare(b.name));\n\n  function addOptions(\n    ofqn?: string,\n    basePath: string[] = [],\n    optional = false\n  ) {\n    if (!ofqn) {\n      return;\n    }\n\n    const struct = jsii[ofqn];\n    if (!struct) {\n      throw new Error(`unable to find options type ${ofqn} for project ${fqn}`);\n    }\n\n    for (const prop of struct.properties ?? []) {\n      const propPath = [...basePath, prop.name];\n\n      // protect against double-booking\n      if (prop.name in options) {\n        throw new Error(\n          `duplicate option \"${prop.name}\" in ${fqn} (already declared in ${\n            options[prop.name].parent\n          })`\n        );\n      }\n\n      let jsiiKind;\n      if (prop.type?.fqn) {\n        jsiiKind = jsii[prop.type?.fqn].kind; // e.g. 'class', 'interface', 'enum'\n      }\n\n      const isOptional = optional || prop.optional;\n      const defaultValue = sanitizeValue(prop.docs?.default);\n      const pjnew = sanitizeValue(prop.docs?.custom?.pjnew);\n\n      // if this is a mandatory option and we have a default value,\n      // or the option is tagged to be rendered with an initial value,\n      // the value has to be JSON-parsable to the correct type\n      const initialValue = getInitialValue(defaultValue, pjnew, isOptional);\n      if (initialValue) {\n        checkDefaultIsParsable(prop.name, initialValue, prop.type);\n      }\n\n      options[prop.name] = filterUndefined({\n        path: propPath,\n        parent: struct.name,\n        name: prop.name,\n        fqn: prop.type?.fqn,\n        docs: prop.docs.summary,\n        simpleType: prop.type ? getSimpleTypeName(prop.type) : \"unknown\",\n        fullType: prop.type,\n        kind: jsiiKind,\n        jsonLike: prop.type ? isJsonLike(jsii, prop.type) : undefined,\n        switch: propPath.map((p) => snake(p).replace(/_/g, \"-\")).join(\"-\"),\n        default: defaultValue,\n        initialValue: initialValue,\n        optional: isOptional,\n        featured: prop.docs?.custom?.featured === \"true\",\n        deprecated: prop.docs.stability === \"deprecated\" ? true : undefined,\n      });\n    }\n\n    for (const ifc of struct.interfaces ?? []) {\n      addOptions(ifc);\n    }\n  }\n}\n\nfunction getInitialValue(\n  defaultValue?: string,\n  pjnew?: string,\n  isOptional: boolean = false\n) {\n  if (pjnew) {\n    return pjnew;\n  }\n\n  if (!isOptional) {\n    return defaultValue;\n  }\n\n  return undefined;\n}\n\nfunction sanitizeValue(val?: string) {\n  if (val === \"undefined\") {\n    return undefined;\n  }\n\n  return val;\n}\n\nfunction getSimpleTypeName(type: JsiiPropertyType): string {\n  if (type?.primitive) {\n    return type.primitive; // e.g. 'string', 'boolean', 'number'\n  } else if (type?.fqn) {\n    return type.fqn.split(\".\").pop()!; // projen.NodeProjectOptions -> NodeProjectOptions\n  } else {\n    // any other types such as collection types\n    return \"unknown\";\n  }\n}\n\n/**\n * Whether a value of this type is serializable into JSON.\n */\nfunction isJsonLike(jsii: JsiiTypes, type: JsiiPropertyType): boolean {\n  if (type.primitive) {\n    // string, boolean, number, any\n    return true;\n  } else if (type.fqn) {\n    const kind = jsii[type.fqn].kind;\n    if ([\"interface\", \"enum\"].includes(kind)) {\n      // not 'class'\n      return true;\n    }\n  } else if (type.collection) {\n    return isJsonLike(jsii, type.collection.elementtype);\n  }\n  return false;\n}\n\nfunction filterUndefined(obj: any) {\n  const ret: any = {};\n  for (const [k, v] of Object.entries(obj)) {\n    if (v !== undefined) {\n      ret[k] = v;\n    }\n  }\n  return ret;\n}\n\nfunction isProjectType(jsii: JsiiTypes, fqn: string) {\n  const type = jsii[fqn];\n\n  if (!type) {\n    throw new Error(\n      `Could not find project type with fqn \"${fqn}\" in  .jsii file.`\n    );\n  }\n\n  if (type.kind !== \"class\") {\n    return false;\n  }\n  if (type.abstract) {\n    return false;\n  }\n\n  if (type.docs?.deprecated) {\n    return false;\n  }\n\n  let curr = type;\n  while (true) {\n    if (curr.fqn === PROJECT_BASE_FQN) {\n      return true;\n    }\n\n    if (!curr.base) {\n      return false;\n    }\n\n    curr = jsii[curr.base];\n    if (!curr) {\n      return false;\n    }\n  }\n}\n\nfunction isPrimitiveArray({ collection }: JsiiPropertyType) {\n  return Boolean(\n    collection?.kind === \"array\" && collection?.elementtype.primitive\n  );\n}\n\nfunction isPrimitiveOrPrimitiveArray(type: JsiiPropertyType) {\n  return Boolean(type?.primitive || isPrimitiveArray(type));\n}\n\nfunction checkDefaultIsParsable(\n  prop: string,\n  value: string,\n  type?: JsiiPropertyType\n) {\n  if (!(type && isPrimitiveOrPrimitiveArray(type))) {\n    throw new Error(\n      `required option \"${prop}\" with a @default must use primitive types (string, number and boolean) or a primitive array. type found is: ${JSON.stringify(\n        type\n      )}`\n    );\n  }\n\n  // macros are pass-through\n  if (value.startsWith(\"$\")) {\n    return;\n  }\n\n  try {\n    const parsed = JSON.parse(value);\n\n    // Primitive type\n    if (typeof parsed === type.primitive) {\n      return;\n    }\n\n    // Primitive array\n    if (Array.isArray(parsed) && isPrimitiveArray(type)) {\n      // but empty (which is okay)\n      if (parsed.length === 0) {\n        return;\n      }\n\n      // if first element matches the type, assume it's correct\n      if (typeof parsed[0] === type?.collection?.elementtype.primitive) {\n        return;\n      }\n    }\n\n    // Parsed value does not match type\n    throw new Error(\n      `cannot parse @default value for mandatory option ${prop} as a ${type}: ${parsed}`\n    );\n  } catch (e) {\n    throw new Error(\n      `unable to JSON.parse() value \"${value}\" specified as @default for mandatory option \"${prop}\": ${\n        (e as any).message\n      }`\n    );\n  }\n}\n"]}