To build the chef-dk, we use the omnibus system. Go to the omnibus README to find out how to build!
A standard bundle install
in the root directory and in the omnibus/
directory will get you started. You can run unit tests via:
bundle exec chef
bundle exec rspec spec/
You can run the chef verify command on your local dev environment if you have the ChefDK binary installed. Note that you will likely need the current version of ChefDK, not the last released version. You can get the current version here.
Once installed, you can run the chef verify
command by pointing at the location
of the root the latest current version on your local disk by running:
bin/chef verify git --omnibus-dir <path_to_root_of_current_version>
If you used the defaults while installing, <path_to_root_of_current_version>
is
simply /opt/chefdk
.
If you want to change our constraints (change which packages and versions we accept in the chef-dk), there are several places to do so:
- To add or remove a package from the chef-dk, or update its version, edit Gemfile.
- To change the version of binary packages, edit omnibus_overrides.rb.
- To add new packages to the chef-dk, edit omnibus/config/projects/chefdk.rb.
Once you've made any changes you want, you have to update the lockfiles that actually drive the build:
- To add any new gems, and update the chef-dk's dependencies to the very latest versions
available, run
rake dependencies:force_update
. - To update the chef-dk's dependencies (changing as
little as possible), run
rake dependencies:update
. the rake task handles both the windows and non-windows lockfiles and updates them in sync. - To update the gems used by omnibus (generally when changing the
definitions in
omnibus-software
), runrake dependencies:update_omnibus_gemfile_lock
.
The ChefDK is an amalgam of many components. These components update all the time, necessitating new builds. This is an overview of the process of versioning, building and releasing the ChefDK.
The ChefDK is distributed as packages for Debian, RHEL, Ubuntu, Windows and macOS. It includes a large number of components from various sources, and these are versioned and maintained separately from the chef-dk project, which bundles them all together conveniently for the user.
These packages go through several milestones:
master
: When code is checked in to master, the patch version of chef-dk is bumped (e.g. 0.9.10 -> 0.9.11) and a build is kicked off automatically to create and test the packages in Chef's Jenkins cluster.unstable
: When a package is built, it enters the unstable channel. When all packages for all OS's have successfully built, the test phase is kicked off in Jenkins across all supported OS.current
: If the packages pass all the tests on all supported OS's, it is promoted as a unit tocurrent
, and is available to install from the current channel on https://downloads.chef.io/ or by usingmixlib-install
.stable
: Once a month, Chef will mark a release as stable. When this happens, it is manually promoted to stable and an announcement is sent to the list. It can be reached at https://downloads.chef.io or installed usingmixlib-install
Whenever a change is merged to either the master
or chefdk-1
branches, the expeditor
bot updates the changelog, and increments the
patch version of the branch. Expeditor is documented thoroughly,
and is used by many Chef projects.
Expeditor is controlled by a number of Pull Request labels:
Expeditor: Exclude From Changelog
: This PR is not a software change and should not be included in the ChangeLogExpeditor: Skip Build
: This PR doesn't warrant a new build of ChefDK.Expeditor: Skip Version Bump
: This PR doesn't change any behaviour of ChefDK, and so does not warrant a new patch version. Typically, skipping a version bump will also imply one should skip running a build, too.Expeditor: Skip All
: All of the above.
The chef-dk has two sorts of component: ruby components like berkshelf
and test-kitchen
, and binary components like openssl
and even ruby
itself.
In general, you can find all chef-dk desired versions in the Gemfile and omnibus_overrides.rb files. The Gemfile.lock is the locked version of the Gemfile. omnibus. build and test Gemfiles and Berksfile version the toolset we use to build and test.
The versions of binary components (as well as rubygems and bundler, which can't be versioned in a Gemfile) are stored in omnibus_overrides. These are updated as required.
These have software definitions either in omnibus/config/software or, more often, in the omnibus-software project.
Most of the actual front-facing software in the chef-dk is composed of
ruby projects. berkshelf, test-kitchen and even chef itself are made of
ruby gems. Chef uses the typical ruby way of controlling rubygems
versions, the Gemfile
. Specifically, the Gemfile
at the top of the
chef-dk repository governs the version of every single gem we install
into the chef-dk package. It's a one-stop shop.
See Updating Dependencies for further information.
Of special mention is the software we use to build omnibus itself. There are two distinct bits of code that control the versions of compilers, make, git, and other tools we use to build.
First, the Jenkins machines that run the build are configured entirely
by the opscode-ci cookbook
cookbook. They install most of the tools we use via build-essentials
,
and standardize the build environment so we can tear down and bring up
builders at will. These machines are kept alive long-running, are
periodically updated by Chef to the latest opscode-ci, omnibus and
build-essentials cookbooks.
Second, the version of omnibus we use to build the chef-dk is governed
by omnibus/Gemfile
. When software definitions or the omnibus framework
is updated, this is the file that drives whether we pick it up.
The omnibus tooling versions are locked down with
omnibus/Gemfile.lock
, and can be updated by running rake dependencies
.
chef-dk is tested by the chef-acceptance framework,
which contains suites that are run on the Jenkins test machines. The
definitions of the tests are in the acceptance
directory. The version
of chef-acceptance and test-kitchen, are governed by
acceptance/Gemfile
.
The test tooling versions are locked down with
acceptance/Gemfile.lock
, which can be updated by running rake dependencies
.
The actual ChefDK build process is done with Omnibus, and has several general steps:
bundle install
fromchef-dk/Gemfile.lock
- Reinstall any gems that came from git or path using
rake install
- appbundle chef, chef-dk, test-kitchen and berkshelf
- Put miscellaneous powershell scripts and cleanup
The build is kicked off in Jenkins by running this on the machine (which
is already the correct OS and already has the correct dependencies,
loaded by the omnibus
cookbook):
load-omnibus-toolchain.bat
cd chef-dk/omnibus
bundle install
bundle exec omnibus build chefdk
This causes the chefdk project definition to load, which runs the chef-dk-complete software definition, the primary software definition driving the whole build process. The reason we embed it all in a software definiton instead of the project is to take advantage of omnibus caching: omnibus will invalidate the entire project (and recompile ruby, openssl, and everything else) if you change anything at all in the project file. Not so with a software definition.
The primary build definition that installs the many ChefDK rubygems is
software/chef-dk.rb
. This has
dependencies on any binary libraries, ruby, rubygems and bundler. It has
a lot of steps, so it uses a
library to help reuse code and
make it manageable to look at.
What it does:
- Depends on software defs for pre-cached gems (see "Gems and Caching" below).
- Installs all gems from the bundle:
- Reinstalls any gems that were installed via path:
- Creates
/opt/chefdk/Gemfile
and/opt/chefdk/Gemfile.lock
with the gems that were installed in the build.
Some gems take a super long time to install (particularly native-compiled ones such as nokogiri and dep-selector-libgecode) and do not change version very often. In order to avoid doing this work every time, we take advantage of omnibus caching by separating out these gems into their own software definitions. chef-dk-gem-dep-selector-libgecode for example.
Each of these gems uses the config/files/chef-dk-gem/build-chef-dk-gem
library to define itself. The name of the software definition itself
indicates the .
We only create software definitions for long-running gems. Everything
else is just installed in the
chef-dk software definition in a
big bundle install
catchall.
Most gems we just install in the single chef-dk
software definition.
The first thing
After the gems are installed, we appbundle them in chef-dk-appbundle. This creates binstubs that use the bundle to pin the software .
During the process of appbundling, we update the gem's Gemfile
to
include the locks in the top level /opt/chefdk/Gemfile.lock
, so we can
guarantee they will never pick up things outside the build. We then run
bundle lock
to update the gem's Gemfile.lock
, and bundle check
to
ensure all the gems are actually installed. The appbundler then uses
these pins.
Finally, the chef-dk does several more steps including installing powershell scripts and shortcuts, and removing extra documentation to keep the build slim.