Creating a blueprint

A JHipster blueprint is a Yeoman generator that is composed from a specific JHipster sub-generator to extend the functionality of that sub-generator. The blueprint can override any defined getter of the sub generator and provide its own templates and functionality.

JHipster blueprints are listed on the JHipster marketplace with the jhipster-blueprint label.

This allows to create third-party blueprints that can override a specific part of JHipster, say for example only the client side templates.

Using a blueprint

To use a blueprint, run the below command

jhipster --blueprints <blueprint name>

Example

The JHipster Kotlin blueprint replaces most of the server side Java code with equivalent Kotlin code.

It is our official blueprint that showcases how you can create your own blueprint.

The JHipster Sample Blueprint shows how a client sub-generator can be overridden.

Or, you can use the JHipster blueprint generator to help you to initialize your blueprint.

To use the JHipster blueprint generator run the following commands

npm install -g generator-jhipster-blueprint

mkdir my-blueprint && cd my-blueprint

yo jhipster-blueprint

Choose the sub-generators that you would like to override while answering the questions.

Basic rules for a JHipster blueprint

A JHipster blueprint:

  • is an NPM package, and is a Yeoman generator.
  • follows an extension of the Yeoman rules listed at http://yeoman.io/generators/ and can be installed, used and updated using the yo command. Instead of being prefixed by generator-, it is prefixed by generator-jhipster-, and instead of having just the yeoman-generator keyword, it must have 2 keywords, yeoman-generator and jhipster-blueprint.
  • A blueprint can only extend the following sub-generators (under the generators folder)
    • app
    • common
    • client
    • server
    • entity
    • entity-client
    • entity-server
    • entity-i18n
    • languages
    • spring-controller
    • spring-service
    • heroku
    • ci-cd
    • cypress
    • page
    • entities

Import the generator-jhipster

A JHipster blueprint must have generator-jhipster as a dependency and should import the appropriate sub generator to override it.

    const chalk = require('chalk');
    const ClientGenerator = require('generator-jhipster/generators/client');
    ...

    module.exports = class extends ClientGenerator {
        constructor(args, opts) {
            super(args, Object.assign({ fromBlueprint: true }, opts)); // fromBlueprint variable is important

            const jhContext = this.jhipsterContext = this.options.jhipsterContext;

            if (!jhContext) {
                this.error(`This is a JHipster blueprint and should be used only like ${chalk.yellow('jhipster --blueprint helloworld')}`);
            }

            this.configOptions = jhContext.configOptions || {};
        }

        get initializing() {
            // Here we are not overriding this phase and hence its being handled by JHipster
            return super._initializing();
        }

        // other phases of the sub generator
    }

Any method beginning with _ can be reused from the superclass that is being extended, for example ClientGenerator in the example above.

Each JHipster sub-generator is made of multiple yeoman phases, each phase is a getter, get initializing for example. A blueprint can customize one or more phases of the sub-generator that it is overriding.

There are multiple ways to customize a phase from JHipster.

1) Let JHipster handle a phase, blueprint doesn’t override anything.

    get initializing() {
        return super._initializing();
    }

2) Override the entire phase, this is when the blueprint takes control of a phase.

    get initializing() {
        return {
            myCustomInitPhaseStep() {
                // Do all your stuff here
            },
            myAnotherCustomInitPhaseStep(){
                // Do all your stuff here
            }
        };
    }

3) Partially override a phase, this is when the blueprint gets the phase from JHipster and customizes it.

    get initializing() {
        const phaseFromJHipster = super._initializing();
        const myCustomPhaseSteps = {
            displayLogo() {
                // override the displayLogo method from the _initializing phase of JHipster
            },
            myCustomInitPhaseStep() {
                // Do all your stuff here
            },
        }
        return { ...phaseFromJHipster, ...myCustomPhaseSteps };
    }

4) Decorate a phase, this is when the blueprint runs custom steps before or after the phase coming from JHipster.

    // Run the blueprint steps before and/or after any parent steps
    get initializing() {
        const customPrePhaseSteps = {
            myCustomPreInitStep() {
                // Stuff to do BEFORE the JHipster steps
            }
        };
        const customPostPhaseSteps = {
            myCustomPostInitStep() {
                // Stuff to do AFTER the JHipster steps
            }
        };
        return {
            ...customPrePhaseSteps,
            ...super._initializing(),
            ...customPostPhaseSteps
        };
    }

You can also access to JHipster’s variables and functions directly from a Blueprint.

Available variables and functions

Variables from configuration:

You can access to configuration in .yo-rc.json which will consist of both the JHipster config and your blueprint config.

Global variables:

You can use constants in generator-constants:

    const javaDir = `${jhipsterConstants.SERVER_MAIN_SRC_DIR + this.packageFolder}/`;
    const resourceDir = jhipsterConstants.SERVER_MAIN_RES_DIR;
    const webappDir = jhipsterConstants.CLIENT_MAIN_SRC_DIR;

Functions:

You can use all functions in generator-base:

    this.angularAppName = this.getAngularAppName(); // get the Angular application name.
    this.printJHipsterLogo(); // to print the JHipster logo

Note: The functions in generator-base.js and variables in generator-constants.js are part of public API and hence will follow semver versioning. But other files like generator-base-private.js, utils.js etc will not follow semver versioning and might break method signature across minor versions.

Running local Blueprint version for development

During development of blueprint, please note the below steps. they are very important.

  1. Link your blueprint globally

Note: If you do not want to link the blueprint(step 3) to each project being created, use NPM instead of Yarn as yeoman doesn’t seem to fetch globally linked Yarn modules. On the other hand, this means you have to use NPM in all the below steps as well.

cd generator-jhipster-my-blueprint
npm link
  1. Link a development version of JHipster to your blueprint (optional: required only if you want to use a non-released JHipster version, like the main branch or your own custom fork)
cd generator-jhipster
npm link

cd generator-jhipster-my-blueprint
npm link generator-jhipster
  1. Create a new folder for the app to be generated, and run JHipster ignoring JHipster dependencies (otherwise a released version will be installed each time npm install/ci is called)
mkdir my-app && cd my-app

jhipster --blueprints my-blueprint --skip-jhipster-dependencies
  1. Once the blueprint/generator-jhipster was released re-add the jhipster dependencies for reproducibility
jhipster --no-skip-jhipster-dependencies

Registering a blueprint to the JHipster marketplace

To have your blueprint available in the JHipster marketplace, you need to make sure you have the two keyword yeoman-generator and jhipster-blueprint in your published npm package.json. If you find any entry in the marketplace which is not a JHipster module or blueprint, you can help to blacklist it by adding it to the blacklistedModules section of the modules-config.json file by doing a Pull Request to the jhipster/jhipster.github.io project.

Once you publish your blueprint to NPM, your blueprint will become available in our marketplace.