/hierarchy

Processes yaml files based on a configuration hierarchy and outputs a single file

Primary LanguageGoApache License 2.0Apache-2.0

Hierarchy

Build Status Go Report Card codecov Join the chat at https://gitter.im/KohlsTechnology/hierarchy

Project Deprication and Archival

As of March 18, 2022 this project is no longer maintained. This repository is being archived(marked as read only).

Installation

Hierarchy is a simple utility to merge a set of yaml or json files, based on a defined hierarchy. It is inspired by Hiera from Puppet.

You can either download a binary file from GitHub releases or compile from source. Note: vendor/ files are intentionally included in GitHub repo to ensure deprecation of dependent packages do not cause service to break.

Documentation

The main goal of Hierarchy is to prevent the duplication of configuration data and allow for fine-grained control over the GitOps process. The output YAML can be used to generate a values.yml file for GitOps tools, like Eunomia, Helm, and others.

Usage

You can control the behavior of Hierarchy either through command line options or environment variables. The latter is especially helpful if you are running it inside a container.

Command Line Flag Environment Variable Default Description
-f, --file HIERARCHY_FILE hierarchy.lst Name of the hierarchy file.
-b, --base HIERARCHY_BASE ./ Base path.
-o, --output HIERARCHY_OUTPUT ./output.yaml Path and name of the output file.
-i, --filter HIERARCHY_FILTER (.yaml|.yml|.json)$ Regex for allowed file extension(s) of files being merged.
--output-no-variables HIERARCHY_OUTPUT_NO_VARIABLES false Do not find and replace environment variables in output file.
--fail.missinghierarchy HIERARCHY_FAIL_MISSING_HIERARCHY false Fail if a hierarchy file is not found, otherwise merge all files in base folder.
--fail.missingpath HIERARCHY_FAIL_MISSING_PATH false Fail if a directory in the hierarchy is missing.
--fail.missingvariable HIERARCHY_FAIL_MISSING_VARIABLE false Fail if an environment variable defined in the final yaml is not found.
-d, --debug HIERARCHY_DEBUG false Print debug output.
--trace HIERARCHY_TRACE false Prints a diff after processing each file. This generates A LOT of output.
-V, --version Print version and build information, then exit.

Merging

The Hierarchy utility processes the YAML structure as a deep merge, with the exception of lists. Lists are completely overwritten; therefore, it is important to keep that in mind when using them.

Hierarchy

The hierarchy is defined in the file hierarchy.lst. This is a simple text file that lists one include folder per line and supports comments prefixed with #. The directories listed can be relative or absolute (try to avoid) paths. You can have directories included that are higher or lower in the structure to control their precedence. You can look at examples here.

If the file hierarchy.lst is not found in the base path, then Hierarchy will merge all files found in the base directory that match the filter criteria. The execution will fail if the base path is not found.

Example content

../defaults    #this is the first ... lowest priority
../marketing   #this is the second
../development #this is the third ... highest priority

In this case, it will load all yaml files from ../defaults, then merge it with everything in ../marketing, and lastly merge it with everything in ../development. You can also use the relative path ./, which means that it will also load variables defined in contextDir directly (same folder level as hierarchy.lst). You can insert ./ in any desired order in the hierarchy.lst, thus determining its priority.

Example

Let's assume you have multiple applications that get deployed to different cloud providers. This application also has development, QA, and production environments. You can specify the exact priority (order) the configuration files are merged.

defaults             # all applications will have this
└── cloud            # settings specific to a cloud provider
  └── environment    # settings specific to the environment level (e.g. development or production)
    └── application  # settings specific to an application

In order to generate the final configuration for an application running in the development environment on "cloud A", the hierarchy.lst could look like the below example.

# Hierarchy file for application "demo" running in "cloud A".
# location `..../applications/demo/dev/clouda1.hierarchy.lst
../../defaults
../clouds/A
./environments/dev
./

Environment variables in the hierarchy

Hierarchy allows the use of environment variables to make it even more flexible. The variables must: be in the format ${NAME}, only consist of letters, numbers, and underscores, and start with a letter. The environment variable names will be converted to upper case to avoid ambiguity. If an environment variable is not found, the program will error out to avoid generating the wrong data.

Example

# Hierarchy file for application "demo" running in "cloud A".
# location `..../applications/demo/clouda1.hierarchy.lst
../../defaults
../clouds/A
./environments/${ENVIRONMENT}
./

Developing

See CONTRIBUTING.md for details.

Dependencies

Go 1.17+

Compiling From Source

make build

Testing

To run the tests, simply execute:

make test

Releasing

This project is using goreleaser. GitHub release creation is automated using Travis CI. New releases are automatically created when new tags are pushed to the repo.

$ TAG=v0.0.2 make tag

How to manually create a release without relying on Travis CI.

$ TAG=v0.0.2 make tag
$ GITHUB_TOKEN=xxx make clean release

License

See LICENSE for details.

Code of Conduct

See CODE_OF_CONDUCT.md for details.