UNPKG

generator-begcode

Version:

Spring Boot + Angular/React/Vue in one handy generator

270 lines (269 loc) 13.7 kB
import fs from 'fs'; import path, { dirname } from 'path'; import { fileURLToPath } from 'url'; import didYouMean from 'didyoumean'; import chalk from 'chalk'; import logo from './logo.mjs'; import EnvironmentBuilder from './environment-builder.mjs'; import SUB_GENERATORS from './commands.mjs'; import JHipsterCommand from './jhipster-command.mjs'; import { CLI_NAME, logger, getCommand, done } from './utils.mjs'; import { packageJson } from '../lib/index.js'; import { packageNameToNamespace } from '../generators/base/support/index.js'; import command from '../generators/base/command.js'; import { GENERATOR_APP, GENERATOR_BOOTSTRAP, GENERATOR_JDL } from '../generators/generator-list.js'; import { extractArgumentsFromConfigs } from '../generators/base/internal/command.js'; const __filename = fileURLToPath(import.meta.url); const __dirname = dirname(__filename); const { version: JHIPSTER_VERSION } = packageJson; const JHIPSTER_NS = CLI_NAME; const moreInfo = `\n For more info visit ${chalk.blue('https://www.begcode.com')}\n`; export const printJHipsterLogo = () => { console.log(); console.log(logo); }; const buildAllDependencies = async (generatorNames, { env, blueprintNamespaces }) => { const allDependencies = {}; const registerDependency = async ({ namespace, blueprintNamespace }) => { const meta = await env.getGeneratorMeta(namespace.includes(':') ? namespace : `${JHIPSTER_NS}:${namespace}`); if (meta) { allDependencies[namespace] = { meta, blueprintNamespace }; } return meta?.importModule(); }; const lookupDependencyOptions = async ({ namespace, blueprintNamespace }) => { const lookupGeneratorAndImports = async ({ namespace, blueprintNamespace }) => { const module = await registerDependency({ namespace, blueprintNamespace }); if (module?.command?.import) { for (const generator of module?.command?.import ?? []) { await lookupDependencyOptions({ namespace: generator, blueprintNamespace }); } } return module?.command?.override; }; let overridden = false; if (!namespace.includes(':')) { for (const nextBlueprint of blueprintNamespaces) { const blueprintSubGenerator = `${nextBlueprint}:${namespace}`; if (!allDependencies[blueprintSubGenerator]) { if (await lookupGeneratorAndImports({ namespace: blueprintSubGenerator, blueprintNamespace: nextBlueprint })) { overridden = true; } } } } if (!overridden && !allDependencies[namespace]) { await lookupGeneratorAndImports({ namespace, blueprintNamespace }); } }; for (const generatorName of generatorNames) { await lookupDependencyOptions({ namespace: generatorName }); } return allDependencies; }; const addCommandGeneratorOptions = async (command, generatorMeta, { root, blueprintOptionDescription, info } = {}) => { const generatorModule = await generatorMeta.importModule(); if (generatorModule.command) { const { options, configs } = generatorModule.command; if (options) { command.addJHipsterOptions(options, blueprintOptionDescription); } if (configs) { command.addJHipsterConfigs(configs, blueprintOptionDescription); } } try { if (root || !generatorModule.command || generatorModule.command.loadGeneratorOptions) { const generator = await generatorMeta.instantiateHelp(); command.addGeneratorOptions(generator._options, blueprintOptionDescription); } } catch (error) { if (!info) { throw error; } logger.verboseInfo(`${info}, error: ${error}`); } }; const addCommandRootGeneratorOptions = async (command, generatorMeta, { usage = true } = {}) => { const generatorModule = await generatorMeta.importModule(); if (generatorModule.command) { command.addJHipsterArguments(generatorModule.command.arguments ?? extractArgumentsFromConfigs(generatorModule.command.configs)); } else { const generator = await generatorMeta.instantiateHelp(); command.addGeneratorArguments(generator._arguments); } if (usage) { const usagePath = path.resolve(path.dirname(generatorMeta.resolved), 'USAGE'); if (fs.existsSync(usagePath)) { command.addHelpText('after', `\n${fs.readFileSync(usagePath, 'utf8')}`); } } }; export const createProgram = ({ executableName = CLI_NAME, executableVersion } = {}) => { return (new JHipsterCommand() .name(executableName) .storeOptionsAsProperties(false) .version(executableVersion ? `${executableVersion} (generator-begcode ${JHIPSTER_VERSION})` : JHIPSTER_VERSION) .addHelpText('after', moreInfo) .option('--blueprints <value>', 'A comma separated list of one or more generator blueprints to use for the sub generators, e.g. --blueprints kotlin,vuejs') .option('--force-insight', 'Force insight') .option('--no-insight', 'Disable insight') .option('--force', 'Override every file', false) .option('--dry-run', 'Print conflicts', false) .option('--whitespace', 'Whitespace changes will not trigger conflicts', false) .option('--bail', 'Fail on first conflict', false) .option('--install-path', 'Show jhipster install path', false) .option('--skip-regenerate', "Don't regenerate identical files", false) .option('--skip-yo-resolve', 'Ignore .yo-resolve files', false) .addJHipsterOptions(command.options) .addJHipsterConfigs(command.configs)); }; const rejectExtraArgs = ({ program, command, extraArgs }) => { const first = extraArgs[0]; if (command.name() !== GENERATOR_APP) { logger.fatal(`${chalk.yellow(command.name())} command doesn't take ${chalk.yellow(first)} argument. See '${chalk.white(`${program.name()} ${command.name()} --help`)}'.`); } const availableCommands = program.commands.map(c => c._name); const suggestion = didYouMean(first, availableCommands); if (suggestion) { logger.verboseInfo(`Did you mean ${chalk.yellow(suggestion)}?`); } const message = `${chalk.yellow(first)} is not a known command. See '${chalk.white(`${CLI_NAME} --help`)}'.`; logger.fatal(message); }; export const buildCommands = async ({ program, commands = {}, envBuilder, env, loadCommand, defaultCommand = GENERATOR_APP, entrypointGenerator, printLogo = printJHipsterLogo, printBlueprintLogo = () => { }, createEnvBuilder, }) => { Object.entries(commands).forEach(([cmdName, opts]) => { const { desc, blueprint, argument, options: commandOptions, alias, help: commandHelp, cliOnly, removed, useOptions = {} } = opts; program .command(cmdName, '', { isDefault: cmdName === defaultCommand, hidden: Boolean(removed) }) .description(desc + (blueprint ? chalk.yellow(` (blueprint: ${blueprint})`) : '')) .addCommandArguments(argument) .addCommandOptions(commandOptions) .addHelpText('after', commandHelp) .addAlias(alias) .excessArgumentsCallback(function (receivedArgs) { rejectExtraArgs({ program, command: this, extraArgs: receivedArgs }); }) .lazyBuildCommand(async function (operands) { logger.debug(`cmd: lazyBuildCommand ${cmdName} ${operands}`); if (removed) { logger.fatal(removed); return; } const command = this; if (cmdName === 'run') { command.usage(`${operands} [options]`); operands = Array.isArray(operands) ? operands : [operands]; command.generatorNamespaces = operands.map(namespace => `${namespace.startsWith(JHIPSTER_NS) ? '' : `${JHIPSTER_NS}-`}${namespace}`); await envBuilder.lookupGenerators(command.generatorNamespaces.map(namespace => `generator-${namespace.split(':')[0]}`)); await Promise.all(command.generatorNamespaces.map(async (namespace) => { const generatorMeta = env.getGeneratorMeta(namespace.includes(':') ? namespace : `${JHIPSTER_NS}:${namespace}`); if (!generatorMeta) { logger.fatal(chalk.red(`\nGenerator ${namespace} not found.\n`)); } await addCommandRootGeneratorOptions(command, generatorMeta, { usage: command.generatorNamespaces.length === 1 }); await addCommandGeneratorOptions(command, generatorMeta, { root: true }); })); return; } if (!cliOnly) { const generator = blueprint ? `${packageNameToNamespace(blueprint)}:${cmdName}` : cmdName; const generatorMeta = env.getGeneratorMeta(generator.includes(':') ? generator : `${JHIPSTER_NS}:${generator}`); if (!generatorMeta) { return; } await addCommandRootGeneratorOptions(command, generatorMeta); const boostrapGen = [GENERATOR_BOOTSTRAP, generator]; if (cmdName === GENERATOR_JDL) { boostrapGen.push(entrypointGenerator ?? GENERATOR_APP); } const allDependencies = await buildAllDependencies(boostrapGen, { env, blueprintNamespaces: envBuilder.getBlueprintsNamespaces(), }); for (const [metaName, { meta: generatorMeta, blueprintNamespace }] of Object.entries(allDependencies)) { if (blueprintNamespace) { const blueprintOptionDescription = chalk.yellow(` (blueprint option: ${blueprintNamespace.replace(/^jhipster-/, '')})`); await addCommandGeneratorOptions(command, generatorMeta, { blueprintOptionDescription, info: `Error parsing options for generator ${metaName}`, }); } else { await addCommandGeneratorOptions(command, generatorMeta, { root: metaName === generator }); } } } command.addHelpText('after', moreInfo); }) .action(async (...everything) => { logger.debug('cmd: action'); const command = everything.pop(); const cmdOptions = everything.pop(); const args = everything; const options = { ...program.opts(), ...cmdOptions, ...useOptions, commandName: cmdName, entrypointGenerator, blueprints: envBuilder.getBlueprintsOption(), positionalArguments: args, }; if (options.installPath) { console.log(path.dirname(__dirname)); return Promise.resolve(); } printLogo(); printBlueprintLogo(); if (cliOnly) { logger.debug('Executing CLI only script'); const cliOnlyCommand = await loadCommand(cmdName); return cliOnlyCommand instanceof Function ? cliOnlyCommand(args, options, env, envBuilder, createEnvBuilder) : Promise.reject(new Error(`Command ${cmdName} is not a function.`)); } if (cmdName === 'run') { return Promise.all(command.generatorNamespaces.map(generator => env.run(generator, options))).then(results => done(results.find(result => result)), errors => done(errors.find(error => error))); } if (cmdName === 'upgrade') { options.programName = program.name(); options.createEnvBuilder = createEnvBuilder; } const namespace = blueprint ? `${packageNameToNamespace(blueprint)}:${cmdName}` : `${JHIPSTER_NS}:${cmdName}`; const generatorCommand = getCommand(namespace, args, opts); return env.run(generatorCommand, options).then(done, done); }); }); }; export const buildJHipster = async ({ executableName, executableVersion, program = createProgram({ executableName, executableVersion }), blueprints, lookups, createEnvBuilder, envBuilder, commands, printLogo, printBlueprintLogo, devBlueprintPath, env, loadCommand = async (key) => { const { default: command } = await import(`./${key}.mjs`); return command; }, defaultCommand, entrypointGenerator, } = {}) => { createEnvBuilder = createEnvBuilder ?? (async (options) => EnvironmentBuilder.create(options).prepare({ blueprints, lookups, devBlueprintPath })); envBuilder = envBuilder ?? (await createEnvBuilder()); env = env ?? envBuilder.getEnvironment(); commands = commands ?? { ...SUB_GENERATORS, ...(await envBuilder.getBlueprintCommands()) }; await buildCommands({ program, commands, envBuilder, env, loadCommand, defaultCommand, entrypointGenerator, printLogo, printBlueprintLogo, createEnvBuilder, }); return program; }; export const runJHipster = async (args = {}) => { const { argv = process.argv } = args; const jhipsterProgram = await buildJHipster(args); return jhipsterProgram.parseAsync(argv); }; export { done };