hygen
is the simple, fast, and scalable code generator that lives in your project.
Features
Quick Start
Hygen can be used to supercharge your workflow with Redux, React Native, Express and more, by allowing you avoid manual work and generate, add, inject and perform custom operations on your codebase.
If you're on macOS and have Homebrew:
$ brew tap jondot/tap
$ brew install hygen
If you have node.js installed, you can install globally with npm
(or Yarn):
$ npm i -g hygen
If you like a no-strings-attached approach, you can use npx
without installing globally:
$ npx hygen ...
For other platforms, see releases.
Initialize hygen
in your project (do this once per project):
$ cd your-project
$ hygen init self
Build your first generator, called mygen
:
$ hygen generator new --name mygen
Loaded templates: _templates
added: _templates/mygen/new/hello.ejs.t
Now you can use it:
$ hygen mygen new
Loaded templates: _templates
added: app/hello.js
You've generated content into the current working directory in app/
. To see how the generator is built, look at _templates
(which you should check-in to your project from now on, by the way).
You can build a generator that uses an interactive prompt to fill in a variable:
$ hygen generator with-prompt --name mygen
Loaded templates: _templates
added: _templates/mygen/with-prompt/hello.ejs.t
added: _templates/mygen/with-prompt/prompt.js
Done! Now let's use mygen
:
$ hygen mygen with-prompt
? What's your message? hello
Loaded templates: _templates
added: app/hello.js
What's Next?
Go to the documentation to get to know the rest of Hygen and generators.
If you're in a hurry:
- To learn how to edit generator templates, look here
- To see how to use generators look here
- Take a look at the ecosystem and tooling built around Hygen.
A Different Kind of a Generator
hygen
is a scalable generator. It has developer and team ergonomics as first priority.
It avoids "blessed" or dedicated projects that codifies code generation, which before you know it, become a product you build, needs testing, CI, separate pull request reviews, and ultimately sucks up your time and becomes this super hated thing no one likes to touch anymore.
Plus, since they are not the product you are building they become notoriously hard to reason about.
Scratch Your Own Itch
Because hygen
templates live in your project, it cuts the time from having an itch for generating code (Redux, anyone?) in your current
project to code generated with it and others benefiting from it.
Template Locality
hygen
picks up a local _templates
folder
at any folder level of your project you're working from.
This is important because:
- Templates are project-local. A git clone of the project fetches all generators as well.
- Different generators can be tucked in different parts of the project, making it contextual.
- Template locality is scalable; different teams can maintain different generators.
- When you change your code, you can make changes in the template and pack in the same commit, to be reviewed and merged in the same PR (as opposed to installing different "plugins" or different templates from out-of-repo places).
And yet, if you don't like project-local templates:
- You can have a global
_templates
folder (maybe a central git repo you maintain?) by populating an environment variableHYGEN_TMPLS
- You can build a custom generator of your own with
hygen
at its core, and pack your own templates with it.
Folder Structure is Command Structure
The templates folder structure maps directly to the command structure:
$ hygen worker new --name jobrunner
For this command, hygen worker new
maps to _templates/worker/new
and all files within worker/new
are rendered.
Template parameters are given with --flag VALUE
, as many as you'd like. In this example we've set a parameter named name
to the value jobrunner
.
Subcommands
A subcommand is a file inside a your folder structure. So if the structure is this:
_templates/
worker/
new/
index.html.ejs
setup.html.ejs
And you only want setup
, you can run:
$ hygen worker new:setup
You can also use the subcommand as a regular expression so, these will do the same:
$ hygen worker new:se
$ hygen worker new:se.*
Frontmatter for Decluttering
Here's how a template looks like (in our example, index.ejs.t
). Templates bodies are ejs:
---
to: app/workers/<%=name%>.js
---
class <%= h.capitalize(name) %> {
work(){
// your code here!
}
}
The first part of the template is a front matter, idea borrowed from Markdown, this is the metadata part of a hygen
template and is part of the reason of why your templates will feel more lightweight and flexible.
All frontmatter metadata are also run through the template engine so feel free to use variables in the frontmatter as you wish.
There's one required metadata variable: to
.
to
points to where this file will be placed (folders are created as needed).
Addition, Injection, and More
By default templates are 'added' to your project as a new target file, but you can also choose to inject a template into an existing target file.
For this to work, you need to use inject: true
with the accompanied inject-specific props.
---
to: package.json
inject: true
after: dependencies
skip_if: react-native-fs
---
"react-native-fs":"*",
This template will add the react-native-fs
dependency into a package.json
file, but it will not add it twice (see skip_if
).
Here are the available mutually-exclusive options for where to inject at:
before | after
- a regular expression / text to locate. The inject line will appear before or after the located line.prepend | append
- add a line to start or end of file respectively.line_at
- add a line at this exact line number.
You can guard against double injection:
skip_if
- a regular expression / text. If exists injection is skipped.
Also you can insert or remove empty line to injection body. That feature very useful if your editor or formatter automatically insert blank line at the end of file on save:
eof_last
- if falsy - trim blank line from the end of injection body, if truthy - insert it.
Build Your Own
hygen
is highly embeddable. You should be able to use it by simply listing it
as a dependency and having this kind of workflow in your binary.
const { runner } = require('hygen')
const Logger = require('hygen/lib/logger')
const path = require('path')
const defaultTemplates = path.join(__dirname, 'templates')
runner(process.argv.slice(2), {
templates: defaultTemplates,
cwd: process.cwd(),
logger: new Logger(console.log.bind(console)),
createPrompter: () => require('inquirer'),
exec: (action, body) => {
const opts = body && body.length > 0 ? { input: body } : {}
return require('execa').shell(action, opts)
},
debug: !!process.env.DEBUG
})
Development
The Hygen codebase has a functional style where possible. This is because naturally, it feeds parameters and spits out files. Try to keep it that way.
Running hygen
locally, rigged to your current codebase (note the additional --
to allow passing flags)
$ yarn hygen -- mailer new --name foobar
Running tests in watch mode:
$ yarn watch
Metaverse Testing
The easiest way to make an impact is to use the built-in metaverse
tests suite, and then add the tests here.
The idea here is to copy templates from any project that use hygen
and to test that it works at all times. This keeps tabs on the hygen universe / ecosystem (nicknamed metaverse) and makes sure this test suite breaks before hygen
clients break.
Internal Testing
Start Up Speed Testing
Many generators become painfully slow to use as the thing you want to generate grow (because, real life),
This is why hygen
takes speed as a first class citizen, and sports a dedicated start-up timing suite:
$ yarn test:require
In addition, thought is given to which dependencies to take in, how their file structure fan out and what kind of disk access (due to require
) would hygen
ultimately have when we run it. This is recorded with every test run.
Bundling a single file was evaluated (and the infrastructure is still there, using webpack
) and wasn't faster than what we have right now.
Contributing
Fork, implement, add tests, pull request, get my everlasting thanks and a respectable place here :).
Thanks:
To all Contributors - you make this happen, thanks!
Copyright
Copyright (c) 2018 Dotan Nahum @jondot. See LICENSE for further details.