From afd9173ad8b2450281b00147d3b071d3bb989656 Mon Sep 17 00:00:00 2001 From: Peter Muessig Date: Wed, 10 Jan 2024 21:15:16 +0100 Subject: [PATCH] feat: support nesting of generators (#141) Sub generators can include nested generators as a static property which are resolved and installed automatically by Easy UI5 and finally executed in a chain after the root generator and the sub generator. ```js export default class extends Generator { static displayName = "Create a new UI5 application"; static nestedGenerators = [ "wdi5", "library:app" ]; ``` Just describe the subgenerator name as you would specify it when using Easy UI5 and if you want to address a dedicated generator in the subgenerator, just use the namespace syntax defining the dedicated generator with the `:`. --- generators/app/index.js | 339 +++++++++++++++++++++++++--------------- package-lock.json | 61 +++++--- package.json | 2 +- 3 files changed, 253 insertions(+), 149 deletions(-) diff --git a/generators/app/index.js b/generators/app/index.js index 6dcb8f1..b4eaa40 100644 --- a/generators/app/index.js +++ b/generators/app/index.js @@ -134,6 +134,10 @@ const generatorOptions = { type: Boolean, description: "Preview the next mode to consume subgenerators from bestofui5.org", }, + skipNested: { + type: Boolean, + description: "Skips the nested generators and runs only the first subgenerator", + }, }; const generatorArgs = { @@ -149,6 +153,7 @@ const generatorArgs = { }, }; +// The Easy UI5 Generator! export default class extends Generator { constructor(args, opts) { super(args, opts, { @@ -239,6 +244,102 @@ export default class extends Generator { return "Easy UI5"; } + async _getGeneratorMetadata({ env, generatorPath }) { + // filter the hidden subgenerators already + // -> subgenerators must be found in env as they are returned by lookup! + const lookupGeneratorMeta = await env.lookup({ localOnly: true, packagePaths: generatorPath }); + const subGenerators = lookupGeneratorMeta.filter((sub) => { + const subGenerator = env.get(sub.namespace); + return !subGenerator.hidden; + }); + return subGenerators; + } + + async _installGenerator({ octokit, generator, generatorPath }) { + // lookup the default path of the generator if not set + if (!generator.branch) { + try { + const repoInfo = await octokit.repos.get({ + owner: generator.org, + repo: generator.name, + }); + generator.branch = repoInfo.data.default_branch; + } catch (e) { + console.error(`Generator "${owner}/${repo}!${dir}${branch ? "#" + branch : ""}" not found! Run with --verbose for details!`); + if (this.options.verbose) { + console.error(e); + } + return; + } + } + // fetch the branch to retrieve the latest commit SHA + let commitSHA; + try { + // determine the commitSHA + const reqBranch = await octokit.repos.getBranch({ + owner: generator.org, + repo: generator.name, + branch: generator.branch, + }); + commitSHA = reqBranch.data.commit.sha; + } catch (ex) { + console.error(chalk.red(`Failed to retrieve the branch "${generator.branch}" for repository "${generator.name}" for "${generator.org}" organization! Run with --verbose for details!`)); + if (this.options.verbose) { + console.error(chalk.red(ex.message)); + } + return; + } + + if (this.options.verbose) { + this.log(`Using commit ${commitSHA} from @${generator.org}/${generator.name}#${generator.branch}!`); + } + const shaMarker = path.join(generatorPath, `.${commitSHA}`); + + if (fs.existsSync(generatorPath) && !this.options.skipUpdate) { + // check if the SHA marker exists to know whether the generator is up-to-date or not + if (this.options.forceUpdate || !fs.existsSync(shaMarker)) { + if (this.options.verbose) { + this.log(`Generator ${chalk.yellow(generator.name)} in "${generatorPath}" is outdated!`); + } + // remove if the SHA marker doesn't exist => outdated! + this._showBusy(` Deleting subgenerator ${chalk.yellow(generator.name)}...`); + fs.rmSync(generatorPath, { recursive: true }); + } + } + + // re-fetch the generator and extract into local plugin folder + if (!fs.existsSync(generatorPath)) { + // unzip the archive + if (this.options.verbose) { + this.log(`Extracting ZIP to "${generatorPath}"...`); + } + this._showBusy(` Downloading subgenerator ${chalk.yellow(generator.name)}...`); + const reqZIPArchive = await octokit.repos.downloadZipballArchive({ + owner: generator.org, + repo: generator.name, + ref: commitSHA, + }); + + this._showBusy(` Extracting subgenerator ${chalk.yellow(generator.name)}...`); + const buffer = Buffer.from(new Uint8Array(reqZIPArchive.data)); + this._unzip(buffer, generatorPath, generator.dir); + + // write the sha marker + fs.writeFileSync(shaMarker, commitSHA); + } + + // run npm install when not embedding the generator (always for self-healing!) + if (!this.options.embed) { + if (this.options.verbose) { + this.log("Installing the subgenerator dependencies..."); + } + this._showBusy(` Preparing ${chalk.yellow(generator.name)}...`); + await this._npmInstall(generatorPath, this.options.pluginsWithDevDeps); + } + + this._clearBusy(true); + } + async prompting() { const home = path.join(__dirname, "..", ".."); const pkgJson = JSON.parse(fs.readFileSync(path.join(home, "package.json"), "utf8")); @@ -369,31 +470,28 @@ export default class extends Generator { // determine the generator to be used let generator; - // try to identify whether concrete generator is defined - if (!generator) { - // determine generator by ${owner}/${repo}(!${dir})? syntax, e.g.: - // > yo easy-ui5 SAP-samples/ui5-typescript-tutorial - // > yo easy-ui5 SAP-samples/ui5-typescript-tutorial#1.0 - // > yo easy-ui5 SAP-samples/ui5-typescript-tutorial\!/generator - // > yo easy-ui5 SAP-samples/ui5-typescript-tutorial\!/generator#1.0 - const reGenerator = /([^\/]+)\/([^\!\#]+)(?:\!([^\#]+))?(?:\#(.+))?/; - const matchGenerator = reGenerator.exec(this.options.generator); - if (matchGenerator) { - // derive and path the generator information from command line - const [owner, repo, dir = "/generator", branch] = matchGenerator.slice(1); - // the plugin path is derived from the owner, repo, dir and branch - const pluginPath = `_/${owner}/${repo}${dir.replace(/[\/\\]/g, "_")}${branch ? `#${branch.replace(/[\/\\]/g, "_")}` : ""}`; - generator = { - org: owner, - name: repo, - branch, - dir, - pluginPath, - }; - // log which generator is being used! - if (this.options.verbose) { - this.log(`Using generator ${chalk.green(`${owner}/${repo}!${dir}${branch ? "#" + branch : ""}`)}`); - } + // determine generator by ${owner}/${repo}(!${dir})? syntax, e.g.: + // > yo easy-ui5 SAP-samples/ui5-typescript-tutorial + // > yo easy-ui5 SAP-samples/ui5-typescript-tutorial#1.0 + // > yo easy-ui5 SAP-samples/ui5-typescript-tutorial\!/generator + // > yo easy-ui5 SAP-samples/ui5-typescript-tutorial\!/generator#1.0 + const reGenerator = /([^\/]+)\/([^\!\#]+)(?:\!([^\#]+))?(?:\#(.+))?/; + const matchGenerator = reGenerator.exec(this.options.generator); + if (matchGenerator) { + // derive and path the generator information from command line + const [owner, repo, dir = "/generator", branch] = matchGenerator.slice(1); + // the plugin path is derived from the owner, repo, dir and branch + const pluginPath = `_/${owner}/${repo}${dir.replace(/[\/\\]/g, "_")}${branch ? `#${branch.replace(/[\/\\]/g, "_")}` : ""}`; + generator = { + org: owner, + name: repo, + branch, + dir, + pluginPath, + }; + // log which generator is being used! + if (this.options.verbose) { + this.log(`Using generator ${chalk.green(`${owner}/${repo}!${dir}${branch ? "#" + branch : ""}`)}`); } } @@ -527,85 +625,10 @@ export default class extends Generator { } } - let generatorPath = path.join(pluginsHome, generator.pluginPath || generator.name); + // install the generator if not running in offline mode + const generatorPath = path.join(pluginsHome, generator.pluginPath || generator.name); if (!this.options.offline) { - // lookup the default path of the generator if not set - if (!generator.branch) { - try { - const repoInfo = await octokit.repos.get({ - owner: generator.org, - repo: generator.name, - }); - generator.branch = repoInfo.data.default_branch; - } catch (e) { - console.error(`Generator "${owner}/${repo}!${dir}${branch ? "#" + branch : ""}" not found! Run with --verbose for details!`); - if (this.options.verbose) { - console.error(e); - } - return; - } - } - // fetch the branch to retrieve the latest commit SHA - let commitSHA; - try { - // determine the commitSHA - const reqBranch = await octokit.repos.getBranch({ - owner: generator.org, - repo: generator.name, - branch: generator.branch, - }); - commitSHA = reqBranch.data.commit.sha; - } catch (ex) { - console.error(chalk.red(`Failed to retrieve the branch "${generator.branch}" for repository "${generator.name}" for "${generator.org}" organization! Run with --verbose for details!`)); - if (this.options.verbose) { - console.error(chalk.red(ex.message)); - } - return; - } - - if (this.options.verbose) { - this.log(`Using commit ${commitSHA} from @${generator.org}/${generator.name}#${generator.branch}!`); - } - const shaMarker = path.join(generatorPath, `.${commitSHA}`); - - if (fs.existsSync(generatorPath) && !this.options.skipUpdate) { - // check if the SHA marker exists to know whether the generator is up-to-date or not - if (this.options.forceUpdate || !fs.existsSync(shaMarker)) { - if (this.options.verbose) { - this.log(`Generator ${chalk.yellow(generator.name)} in "${generatorPath}" is outdated!`); - } - // remove if the SHA marker doesn't exist => outdated! - this._showBusy(` Deleting subgenerator ${chalk.yellow(generator.name)}...`); - fs.rmSync(generatorPath, { recursive: true }); - } - } - - // re-fetch the generator and extract into local plugin folder - if (!fs.existsSync(generatorPath)) { - // unzip the archive - if (this.options.verbose) { - this.log(`Extracting ZIP to "${generatorPath}"...`); - } - this._showBusy(` Downloading subgenerator ${chalk.yellow(generator.name)}...`); - const reqZIPArchive = await octokit.repos.downloadZipballArchive({ - owner: generator.org, - repo: generator.name, - ref: commitSHA, - }); - - this._showBusy(` Extracting subgenerator ${chalk.yellow(generator.name)}...`); - const buffer = Buffer.from(new Uint8Array(reqZIPArchive.data)); - this._unzip(buffer, generatorPath, generator.dir); - - // write the sha marker - fs.writeFileSync(shaMarker, commitSHA); - } - - // only when embedding we clear the busy state as otherwise - // the npm install will immediately again show the busy state - if (this.options.embed) { - this._clearBusy(true); - } + await this._installGenerator({ octokit, generator, generatorPath }); } // do not execute the plugin generator during the setup/embed mode @@ -613,14 +636,6 @@ export default class extends Generator { // filter the local options and the help command const opts = Object.keys(this._options).filter((optionName) => !(generatorOptions.hasOwnProperty(optionName) || optionName === "help")); - // run npm install (always for self-healing!) - if (this.options.verbose) { - this.log("Installing the subgenerator dependencies..."); - } - this._showBusy(` Preparing ${chalk.yellow(generator.name)}...`); - await this._npmInstall(generatorPath, this.options.pluginsWithDevDeps); - this._clearBusy(true); - // create the env for the plugin generator let env = this.env; // in case of Yeoman UI the env is injected! if (!env) { @@ -628,19 +643,20 @@ export default class extends Generator { env = yeoman.createEnv(this.args, opts); } - // helper to derive the subcommand - function deriveSubcommand(namespace) { - const match = namespace.match(/[^:]+:(.+)/); - return match ? match[1] : namespace; + // read the generator metadata + let subGenerators = await this._getGeneratorMetadata({ env, generatorPath }); + + // helper to derive the generator from the namespace + function deriveGenerator(namespace, defaultValue) { + const match = namespace.match(/([^:]+):.+/); + return match ? match[1] : defaultValue === undefined ? namespace : defaultValue; } - // filter the hidden subgenerators already - // -> subgenerators must be found in env as they are returned by lookup! - const lookupGeneratorMeta = await env.lookup({ localOnly: true, packagePaths: generatorPath }); - let subGenerators = lookupGeneratorMeta.filter((sub) => { - const subGenerator = env.get(sub.namespace); - return !subGenerator.hidden; - }); + // helper to derive the subcommand from the namespace + function deriveSubcommand(namespace, defaultValue) { + const match = namespace.match(/^[^:]+:(.+)$/); + return match ? match[1] : defaultValue === undefined ? namespace : defaultValue; + } // list the available subgenerators in the console (as help) if (this.options.list) { @@ -726,16 +742,79 @@ export default class extends Generator { ).subGenerator; } - if (this.options.verbose) { - this.log(`Calling ${chalk.red(subGenerator)}...\n \\_ in "${generatorPath}"`); + // determine the list of subgenerators to be executed + const subGensToRun = [subGenerator]; + + // method to resolve nested generators (only once!) + const resolved = []; + const resolveNestedGenerator = async (generatorToResolve) => { + const constructor = await env.get(generatorToResolve); + await Promise.all( + constructor.nestedGenerators?.map(async (nestedGenerator) => { + const theNestedGenerator = deriveGenerator(nestedGenerator); + if (resolved.indexOf(theNestedGenerator) === -1) { + resolved.push(theNestedGenerator); + const nestedGeneratorInfo = availGenerators.find((repo) => repo.subGeneratorName === theNestedGenerator); + const nestedGeneratorPath = path.join(pluginsHome, nestedGeneratorInfo.pluginPath || nestedGeneratorInfo.name); + await this._installGenerator({ octokit, generator: nestedGeneratorInfo, generatorPath: nestedGeneratorPath }); + const nestedGens = await this._getGeneratorMetadata({ env, generatorPath: nestedGeneratorPath }); + const subcommand = deriveSubcommand(nestedGenerator, ""); + const theNestedGen = nestedGens.filter((nested) => { + const nestedSubcommand = deriveSubcommand(nested.namespace, ""); + return subcommand ? nestedSubcommand === subcommand : !nestedSubcommand; + })?.[0]; + if (theNestedGen) { + subGensToRun.push(theNestedGen.namespace); + await resolveNestedGenerator(theNestedGen.namespace); + } else { + this.log(`The nested generator "${nestedGeneratorInfo.org}/${nestedGeneratorInfo.name}" has no subgenerator "${subcommand || "default"}"! Ignoring execution...`); + } + } + }) || [] + ); + }; + + // only resolve nested generators when they should not be skipped + if (!this.options.skipNested) { + await resolveNestedGenerator(subGenerator); } - // finally, run the subgenerator - env.run(subGenerator, { - verbose: this.options.verbose, - embedded: true, - destinationRoot: this.destinationRoot(), - }); + // intercept the environments runGenerator method to determine + // and forward the destinationRoot between the generator executions + const runGenerator = env.runGenerator; + let cwd; + env.runGenerator = async function (gen) { + if (cwd) { + // apply the cwd to the next gen + gen.destinationRoot(cwd); + } + return runGenerator.apply(this, arguments).then((retval) => { + // store the cwd from the current gen + cwd = gen.destinationRoot(); + return retval; + }); + }; + + // chain the execution of the generators + let chain = Promise.resolve(); + for (const subGen of subGensToRun) { + chain = chain.then( + function () { + // we need to use env.run and not composeWith + // to ensure that subgenerators can have different + // dependencies than the root generator + return env.run(subGen, { + verbose: this.options.verbose, + embedded: true, + destinationRoot: this.destinationRoot(), + }); + }.bind(this) + ); + } + + if (this.options.verbose) { + this.log(`Running generators in "${generatorPath}"...`); + } } else { this.log(`The generator ${chalk.red(this.options.generator)} has no visible subgenerators!`); } diff --git a/package-lock.json b/package-lock.json index c795d8c..6d95344 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "generator-easy-ui5", - "version": "3.7.0", + "version": "3.8.0", "lockfileVersion": 2, "requires": true, "packages": { "": { "name": "generator-easy-ui5", - "version": "3.7.0", + "version": "3.8.0", "license": "Apache-2.0", "dependencies": { "@octokit/plugin-throttling": "^8.1.3", @@ -2341,9 +2341,9 @@ } }, "node_modules/cli-spinners": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.7.0.tgz", - "integrity": "sha512-qu3pN8Y3qHNgE2AFweciB1IfMnmZ/fsNTEE+NOFjmGB2F/7rLhnhzppvpCnN4FovtP26k8lHyy9ptEbNwWFLzw==", + "version": "2.9.2", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.9.2.tgz", + "integrity": "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==", "engines": { "node": ">=6" }, @@ -10693,18 +10693,18 @@ } }, "node_modules/yargs": { - "version": "17.5.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.5.1.tgz", - "integrity": "sha512-t6YAJcxDkNX7NFYiVtKvWUz8l+PaKTLiL63mJYWR2GnHq2gjEWISzsLp9wg3aY36dY1j+gfIEL3pIF+XlJJfbA==", + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", "dev": true, "dependencies": { - "cliui": "^7.0.2", + "cliui": "^8.0.1", "escalade": "^3.1.1", "get-caller-file": "^2.0.5", "require-directory": "^2.1.1", "string-width": "^4.2.3", "y18n": "^5.0.5", - "yargs-parser": "^21.0.0" + "yargs-parser": "^21.1.1" }, "engines": { "node": ">=12" @@ -10758,6 +10758,20 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/yargs/node_modules/cliui": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dev": true, + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^7.0.0" + }, + "engines": { + "node": ">=12" + } + }, "node_modules/yargs/node_modules/yargs-parser": { "version": "21.1.1", "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", @@ -13969,9 +13983,9 @@ } }, "cli-spinners": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.7.0.tgz", - "integrity": "sha512-qu3pN8Y3qHNgE2AFweciB1IfMnmZ/fsNTEE+NOFjmGB2F/7rLhnhzppvpCnN4FovtP26k8lHyy9ptEbNwWFLzw==" + "version": "2.9.2", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.9.2.tgz", + "integrity": "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==" }, "cli-table": { "version": "0.3.11", @@ -20148,20 +20162,31 @@ "dev": true }, "yargs": { - "version": "17.5.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.5.1.tgz", - "integrity": "sha512-t6YAJcxDkNX7NFYiVtKvWUz8l+PaKTLiL63mJYWR2GnHq2gjEWISzsLp9wg3aY36dY1j+gfIEL3pIF+XlJJfbA==", + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", "dev": true, "requires": { - "cliui": "^7.0.2", + "cliui": "^8.0.1", "escalade": "^3.1.1", "get-caller-file": "^2.0.5", "require-directory": "^2.1.1", "string-width": "^4.2.3", "y18n": "^5.0.5", - "yargs-parser": "^21.0.0" + "yargs-parser": "^21.1.1" }, "dependencies": { + "cliui": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-8.0.1.tgz", + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dev": true, + "requires": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.1", + "wrap-ansi": "^7.0.0" + } + }, "yargs-parser": { "version": "21.1.1", "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.1.1.tgz", diff --git a/package.json b/package.json index e58a79d..ea7dcdc 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "generator-easy-ui5", - "version": "3.7.0", + "version": "3.8.0", "description": "Generator for UI5-based project", "main": "generators/app/index.js", "type": "module",