neat
Neat is a CLI tool and a collection of the neatest repository templates to improve your repos.
neat
Neat is a CLI tool and a collection of the neatest repository templates to boost your repos.
Table of Contents
- 💾 Installation
- 🔥 CLI usage
- 🤘 Creating a neat repo
- 💚 Contributing
- 💡 Todo
- 💬 Support
- 📜 License
- ⭕ About Olivr
💾 Installation
Install with:
-
NPM
npm install -g neat -
Yarn
yarn global add neat -
Or use directly with NPX
npx neat
🔥 CLI usage
Neat is very powerful, as it allows you to enhance existing repos or create new repos by:
- downloading files
- asking questions
- replacing strings with other strings, files or command outputs
- running pre-defined commands or scripts
The default behaviour is to just download files, but Neat repos can specify a .neat.yml
which can virtually do anything when a user "neats" a repo.
Neat is safe to run in an existing directory because it will never change existing files unless you use the force flags. That said, always inspect the configuration file before neating a repo to spot anything that could be harmful to your system or your files (potential threats will be highlighted).
There is a collection of neat repos in neat-repos.json (See contributing if you want to submit yours). Although we look at the repo at the time of adding it to the list, we cannot vouch for any changes committed after that.
⚠️ As a general rule (not just for Neat), you should never execute a remote file without prior verification because it could have been tampered with malicious code. As such, it is always recommended to execute remote files in a controlled environment such as a remote CI environment or a local docker container to contain eventual damage.
Use a "registered" repo
Use a repo name from the neat-repos.json
Download files in the current working directory (without overwriting existing files):
neat repo
By default the master
branch is used, if you prefer another branch/tag, you can use the @ notation:
neat repo@emoji
Use any GitHub repo
Download files in the current working directory (without overwriting existing files):
neat your/repo
By default the master
branch is used, if you prefer another branch/tag, you can use the @ notation:
neat your/repo@v2
Some repos may have a
.neat.yml
config file to change their behaviour when they are "neated" but this file is optional.
Specify a target folder
Download files in my-project
(without overwriting existing files):
neat repo my-project
This is usually used for creating a new repo using a neat template
Inspect a repo configuration
You can easily inspect a configuration file before neating repo by running:
neat inspect repo
Options
--force-download
The default behaviour is to skip processing remote files that already exist locally.
Using this flag will overwrite all local files with their remote counterparts.
neat repo --force-download
--force-inject
The default behaviour is to skip injecting local files that existed already before running Neat.
Using this flag will force injecting those files.
neat repo --force-inject
This flag is quite commonly used, its default behavior is a safety measure to prevent modification of existing files and is inline with the philosophy of Neat not being invasive on an existing repo unless specifically told.
-f, --force
If you wish to use both --force-download
and --force-inject
, you can use the alias --force
or -f
Note on future versions of Neat: As new features will be added to Neat, using
--force
will always be the same as combining all the--force-*
flags. Keep that in mind if you use Neat in automation or if you're a heavy neat user.
So, in the current version, running this:
neat repo --force
Is the same as running this:
neat repo --force-download --force-inject
-s, --silent
Don't ask for any user input.
If this flag is used Neat will skip any questions asked by a template and will use answers from NEAT_ASK_*
environment variables
It is best used in CI environments.
export NEAT_ASK_PROJECT_NAME="My project"export NEAT_ASK_SECURITY_CONTACT="me@example.com"export NEAT_ASK_SUPPORT_URL="https://stackoverflow.com" neat repo --silent
-e, --except
Filter out remote files from processing by passing a regular expression.
Example: Process all but markdown files
neat repo --except "\.md$"
The regular expression is run as case insensitive.
-o, --only
Filter remote files to process by passing a regular expression.
Example: Process only markdown files
neat repo --only "\.md$"
The regular expression is run as case insensitive.
-d, --debug
Outputs debug information to help debug something gone wrong. This is very helpful to debug a Neat config file that doesn't work as expected.
neat repo --debug
Example use case
Your organization, maintains a "default" repo which contains files to be used when creating other repositories:
docs/SECURITY.mdLICENSEREADME.md
When you create a new repo you can use
neat organization/default new-repo
Or if you already worked in a repo and you forgot to create it from the default repo, you could run from within your repo folder, it will just add new files that are not present in your local folder and will not overwrite any files
neat organization/default
Now, let's say you want to:
- Make sure your repo's security policy is always up to date with your organization's latest security policy
- Add files from a repo template you created for your favourite framework on your personal GitHub account
- Add to your repo any new files added in the organization's default repo
- Add any generic files that you didn't create already from the neatest repo
You could run the following (or add it in your CI pipeline, package.json, pre-commit hook, etc.)
neat organization/default -f -o docs/SECURITY.mdneat mygithub/favourite-frameworkneat organization/defaultneat oss
This is non invasive: it will not overwrite your files except for docs/SECURITY.md
🤘 Creating a neat repo
Each Neat repo can contain a .neat.yml
configuration file which specifies what to do when someone "neats" the repo.
There is a neat template to help you make a neat template.
Tips and advice
Examples
- You can find configuration examples in the examples folder
- For more real-world examples, you can look at the configuration files of the neat-repos.json
Debugging
You can easily inspect a local configuration file by providing a local directory instead of a repo:
neat inspect ./localrepo
The path must start with "." to be recognized as a local path.
When your Neat repo doesn't work as expected, you can use the --debug command if inspect
is not of enough help, however you have to push your repo to GitHub first.
neat your/repo --debug
Note that inspect
is meant to help final users to inspect a repo. Commands are always highlighted in red and scripts in yellow, so the colors DO NOT indicate errors in your configuration.
If there is an error in your configuration:
- You will see an error if the YAML file is malformed
- If the YAML is wellformed, but Neat cannot understand some configuration, it will ignore it and this configuration will not appear when inspecting.
Composability
In order to improve composability between neat templates, respect these guidelines:
- Make use of injections and before/after hooks.
- Choose your injection IDs/patterns wisely so that you don't overwrite another template's content. For example, if two templates both use the ID 'description', the only description remaining will be the last one that was run. A good practice is to precede all your IDs with your template name.
- If your repo is composing well with another repo, say it in your Readme.
Pre-run
Pre-run commands are run on the local machine before any files are processed.
They can be commands to be run as is on the local machine or JavaScript (using Node.js).
pre-run: - script: console.log('I am javascript') - echo 'I am ran at the begining, before any file is downloaded'
The variable fs is available on all Javascript commands so you don't have to require the filesystem module.
Javascript is always preferred because of its cross-os compatibility.
If you plan for other people to neat your repo, you should make sure these commands can run on any OS, or tell otherwise in your README.
Symbolic links
Sometimes, it is necessary for one file's content to be another file's content. Any specified symbolic links are created before downloading any files.
For example, the most common use case is that you have a readme explaining how to use your Neat template. But you don't want this readme to be the default readme once your user has neated your template. Instead, you prepared another readme.
symlink: - README.md: README.tpl.md
Essentially, in this example:
- The content of README.md will be replaced with the content of README.tpl.md
- README.tpl.md will be ignored from processing
- Neat will process README.md like any other file, including replacements, injections or even ignoring it.
Ask questions
When someone neats your repository, you can ask him/her some questions and then use those values to replace strings or run any other arbitrary command.
A question is structured as follows:
ask: - id: project_name description: What is your project name? default: My project
id
is the only required fielddescription
is the actual question asked to the user. If it is not set, Neat will use theid
by replacing underscores by spaces (eg. Project name)default
is used to provide default values and determine the question type
Neat supports three question types: input, choice and multiple choice that are deducted based on the provided value for default
Input
If no default value is specified or if the default value is a string
ask: - id: project_name description: What is your project name?
Choice
If the default value is a list of strings
ask: - id: ci description: What is your preferred CI? default: [Travis, Circle CI, Github Actions]
Multiple choice
If the default value is a list of key/value pairs
true
means this choice is checked by defaultfalse
means this choice is unchecked by default
ask: - id: options description: What options do you want? default: - "Code Coverage": true - "PR template": false - "Issue templates": true
Replacements
Neat can search the added files and replace certain strings with answers to questions in the added files.
For each question, you can specify if Neat has to make a replacement by adding replace: true
ask: - id: project_name description: What is your project name? replace: true
This will have the effect of searching all added files and replacing the question ID in mustache style {{project_name}}
with the value of the answer (eg. My project)
For the multiple choice question type, the answer is a string of comma-space-separated values (eg. PR template, Issue templates)
You can change the pattern format or filter which files to search and replace.
If these replacement options are not enough for your use case, you can make use of the post-run commands to do pretty much anything you like
Replacement pattern
You can specify which pattern to replace. By default, it will search and replace mustache variables: {{%s}}
.
Example: Replace HTML comments
ask: - id: project_name description: What is your project name? replace: truereplace_pattern: "<!-- %s -->"
Replacement filter
You can specify which files to run replacements on. By default, it will search and replace in all added files.
Example: Make replacements only in markdown and text files
ask: - id: project_name description: What is your project name? replace: truereplace_filter: \.(md|txt)$
Pre-download
Pre-download commands are run on the local machine before any files are processed but just after asking questions, so you can have access to the answers environment variables.
ask: - id: project_namepre-download: - script: console.log(process.env.NEAT_ASK_PROJECT_NAME)
Just like pre-run commands, they can accept system commands or Javascript.
Inject files
You can specify a list of files or command outputs to be "injected" into specific files.
This could be used for example to ensure certain chunks of text are included in a readme, even if the local folder of a user already has a readme when he neats your repo.
You can specify either a file
, a command
(system command) or an url
as the source.
This is an example config for the xyz
template:
inject: - id: xyz-hello command: echo "hello world" target: README.md - id: xyz-support file: readme/support.md target: docs/CONTRIBUTING.md - id: xyz-google url: https://google.com target: google.html
If the target file does not exist, it will be created.
Any replacements will also be applied to each injected chunk (including replacement filter).
Be sure to read our note on cross-OS compatibility if you plan to use a command as the source.
Injection pattern
The default pattern is <!-- id -->
with id
being the value of the corresponding id. For example, the id xyz-hello
in the example above would inject:
<!-- xyz-hello --> hello world <!-- xyz-hello -->
Neat will find either:
- Two occurences of the pattern with or without any text in between
- One occurence of the pattern
If it cannot find this pattern in the target file, Neat will add it at the bottom of the file (unless you specified a before/after pattern).
You can customize the replacement pattern:
inject: - id: xyz-hello command: echo "hello world" target: README.md pattern: "<!-- hello-docs -->"
In this example, the following will be injected in README.md
<!-- hello-docs --> hello world <!-- hello-docs -->
Several targets
This example will find the pattern <!-- xyz-hello -->
and replace it with hello world
in both targets
inject: - id: xyz-hello command: echo "hello world" target: [docs/CONTRIBUTING.md, README.md]
if and if not
You can tweak what you inject based on these conditional parameters:
- the target file does not exist (
no-file
) - the target file exists and no occurence of the pattern is found (
no-pattern
) - the target file exists and one occurence of the pattern is found (
single-pattern
) - the target file exists and two occurences of the pattern are found (
double-pattern
)
You can include an if
or ifnot
declaration per injection. If both are specified, ifnot
is ignored.
The if/ifnot declaration can either be one conditional parameter or an array of them.
If if/ifnot are not specified, it is the same as adding all of the conditional parameters in if if: [no-file, no-pattern, single-pattern, double-pattern]
The example below will inject the content of Google.com in the file search.html
only if the search.html
file doesn't exist.
If search.html
exists already, it will inject the content of Bing.com
inject: - id: google if: no-file url: https://google.com target: search.html pattern: "<!-- search-engine -->" - id: bing ifnot: [no-file] url: https://bing.com target: search.html pattern: "<!-- search-engine -->"
This other example will produce exactly the same result:
inject: - id: google ifnot: [no-pattern, single-pattern, double-pattern] url: https://google.com target: search.html pattern: "<!-- search-engine -->" - id: bing if: [no-pattern, single-pattern, double-pattern] url: https://bing.com target: search.html pattern: "<!-- search-engine -->"
The example below will inject the content of Google.com in the file search.html
only if the search.html
file doesn't exist, otherwise it will inject the content of Bing.com except if search.html
already includes two occurences of the pattern.
So, essentially, no injection will happen if a double pattern is found.
inject: - id: google if: no-file url: https://google.com target: search.html pattern: "<!-- search-engine -->" - id: bing if: [no-pattern, single-pattern] url: https://bing.com target: search.html pattern: "<!-- search-engine -->"
Position
If the pattern is not found in the target file, the injection is inserted at the end of the file by default.
If you want to insert it in another position, you can use before
or after
.
For example, the hooks provided by the neat repo can be used as values for before/after.
This example config for the xyz
template will find <!-- project-usage -->
and inject just after:
inject: - id: xyz-hello command: echo "hello world" target: README.md after: <!-- project-usage -->
As a result, when a user runs:
neat reponeat xyz
The project usage part of the Readme will become:
<!-- project-usage --> <!-- xyz-hello --> hello world <!-- xyz-hello -->
Note that before/after are only used when no occurence of the pattern (in this case <!-- xyz-hello -->
) are found.
If the before/after pattern is not found neither, the injection will be appended at the end of the file.
Wrap
The default behaviour is to wrap the injected content with the pattern. You can change this by specifying the strings added before
and after
the content to inject.
For example, the following only changes the behaviour of what is added before the injected content
inject: - id: xyz-hello command: echo "hello world" target: README.md wrap: before: <!-- before-hello -->
<!-- before-hello --> hello world <!-- xyz-hello -->
The following does not add the pattern after the injected content
inject: - id: xyz-hello command: echo "hello world" target: README.md wrap: before: <!-- before-hello --> after: false
<!-- before-hello --> hello world
The following changes the whole wrapping
inject: - id: xyz-hello command: echo "hello world" target: README.md wrap: <!-- wrap-hello -->
<!-- wrap-hello --> hello world <!-- wrap-hello -->
The following disables wrapping
inject: - id: xyz-hello command: echo "hello world" target: README.md wrap: false
hello world
Inject used in conjuction with ignore
For example, if your neat repo is using a README.md to describe what it can do and how to use it but you don't want this README.md to be downloaded when someone neats it, you can ignore it.
-
Using
ignore
in conjunction withinject
:ignore: [README.md]inject:- id: xyz-hellocommand: echo "hello world"target: README.mdAs expected,
README.md
will not be downloaded. However:- If Neat is run in a folder that doesn't contain a
README.md
already, it will create it and inject thexyz-hello
section. - If Neat is run in a folder that already contains a
README.md
, it will either replace or append the pattern (as described in Injection pattern)
- If Neat is run in a folder that doesn't contain a
-
If the source file for the injection is ignored:
ignore: [readme/support.md]inject:- id: xyz-supportfile: readme/support.mdtarget: README.mdAs expected,
readme/support.md
will not be downloaded. However:- If Neat is run in a folder that doesn't contain a
readme/support.md
already, it will inject intoREADME.md
the content ofreadme/support.md
that will be fetched remotely from your neat repo. - If Neat is run in a folder that already contains a
readme/support.md
, it will use that content as the content to inject intoREADME.md
- If Neat is run in a folder that doesn't contain a
Ignore files
You can specify a list of relative paths to files that should be ignored (aka never downloaded)
You could use it to provide a documentation for your template in a README.md
but you don't want this to be included when people neat your template:
ignore: [README.md]
Note:
.neat.yml
is always ignored by default, the only way not to include it is to use a symbolic link
Post-run
Post-run commands are run on the local machine after files are processed.
post-run: - script: console.log('I am run at the end')
Just like pre-run commands, they can accept system commands or Javascript.
Files environment variables
Post-run commands can access several environment variables. These variables will never include files and directories filtered out using the --only or --except flags because they are simply not processed by Neat.
The most used environment variable is NEAT_ADDED_FILES
because it contains a lit of files that were effectively added in the repo.
Environment variable | Description |
---|---|
NEAT_ALL_FILES_DIRS | Space-separated list of all files and directories that were processed, whether they were added or skipped. |
NEAT_ADDED_FILES_DIRS | Space-separated list of files and directories that were added. |
NEAT_SKIPPED_FILES_DIRS | Space-separated list of files and directories that were skipped. |
NEAT_ALL_FILES | Space-separated list of all files that were processed, whether they were added or skipped. |
NEAT_ADDED_FILES | Space-separated list of files that were added. |
NEAT_SKIPPED_FILES | Space-separated list of files that were skipped. |
NEAT_ALL_DIRS | Space-separated list of all directories that were processed, whether they were added or skipped. |
NEAT_ADDED_DIRS | Space-separated list of directories that were added. |
NEAT_SKIPPED_DIRS | Space-separated list of directories that were skipped. |
Answers environment variables
In addition, if some questions were asked, their answers are available as environment variables constructed with the question ID in uppercase.
Examples:
-
Will produce the environment variable
NEAT_ASK_PROJECT_NAME
whose value will be a string of the user's answer (eg. My project)- id: project_name -
Will produce the environment variable
NEAT_ASK_CI
whose value will be a string of the user's answer (eg. Circle CI)- id: cidefault: [Travis, Circle CI, Github Actions] -
Will produce the environment variable
NEAT_ASK_OPTIONS
whose value will be a string of comma-separated answers (eg. PR template, Issue templates)- id: optionsdefault:- "Code Coverage": true- "PR template": false- "Issue templates": true
A note on cross-OS compatibility
When running commands on the user machine, you can either use system commands or Javascript (Node.js). We recommend using Javascript as much as possible because it is the most cross-os compatible way to execute commands. Note that the variable fs is available so you could run filesystem commands directly.
pre-run: - script: console.log(fs.readFileSync('README.md', 'utf8'))
If your script becomes quite long, you can add it as an external script file:
pre-run: - node script.js
If using a script file, be sure to bundle any dependencies (Webpack could help). Also, we recommend cleaning up your script by removing it from the filesystem so you don't leave any configuration artifacts behind.
For example:
const fs = ; // Run commandsconsole; // Remove this scriptfs;
If you are running other programs or system-specific commands in your Neat configuration file, make sure to tell it in your Readme in a requirements section for example so as to not disappoint other users of your template.
💚 Contributing
Add your repo to the registered repo list
If you created a neat repo you're proud of, please add it to the registered repos list:
-
Fork this repo
-
Add your repo to neat-repos.json
- The syntax is
"name": "repo/path"
. As a result, runningneat name
will fetchrepo/path
- Insert your repo in alphabetical order
- Only use strings, numbers and dashes (-) in the neat name. It must not start or end with a dash
- The syntax is
Top five ways to contribute
⭐ Star this repo: it's quick and goes a long way! 🔝
Please see the docs/CONTRIBUTING.md for more information.
Project specifics
Install dependencies:
yarn install
Run tests:
yarn test
Run neat:
./bin/run
Use your code as the neat version you use on your system:
yarn link
For maintainers
We follow Semantic versioning and make use of Yarn version to manage new versions.
Patch
When you make backwards compatible bug fixes:
yarn version --patch
New feature
When you add functionality in a backwards compatible manner:
yarn version --minor
Major version
When you make incompatible API changes:
yarn version --major
💡 Todo
- Overwrite remote config with local config (Add a .neat.yml locally with for example
remote: [repo: [inject: [...]]]
and it will ignore any injections from the remote) - GitHub action running on a schedule to perform automated verification of pre/post run commands in list of neatest repos and add the SHA of the latest commit to
neatest-repos.json
- When neating a repo, verify which SHA is used and display a warning if it has not been verified yet
- Provide a Docker image with Neat already installed to easily run it in a containerized environment
💬 Support
Join Olivr on Keybase 🔐
Or you can use our Reddit community
📜 License
This project is licensed under the Apache 2.0 License - see the LICENSE file for details
⭕ About Olivr
Olivr is an AI co-founder for your startup.