At first glance, writing tests for your Puppet modules appears to be no more than simply duplicating your manifests in a different language and, for basic “package/file/service” modules, it is.
However, when you start leveling up your modules to include dynamic content from templates, support multiple operating systems or take different actions when passed parameters, these tests become invaluable when adding new functionality to your modules, protecting against regressions when refactoring or upgrading to a new Puppet release.
There are a lot of people confused by the purpose of these tests as they can’t test the result of the manifest on a live system. That is not the point of rspec-puppet.
Rspec-puppet tests are there to test the behaviour of Puppet when it compiles
your manifests into a catalogue of Puppet resources. For example, you might
want to test that your apache::vhost defined type creates a file resource
with a path of /etc/apache2/sites-available/foo when run on a Debian host.
When writing your test cases, you should only test the first level of resources
in your manifest. By this I mean, when testing your ‘webserver’ role class,
you would test for the existence of the apache::vhost types, but not for the
file resources created by them, that’s the job of the tests for
apache::vhost.
Whether you’re testing classes, defined types, hosts or functions the structure of your test file is always the same.
require 'spec_helper'
describe '<name of the thing being tested>' do
# Your tests go in here
endThe important thing is what you name your test file and where you put it. Test
files should always end in _spec.rb (generally, they’re named <thing being
tested>_spec.rb). Class tests should be placed in spec/classes, defined type
tests should go in spec/defines, host tests should be placed in spec/hosts
and function tests should go in spec/functions.
This is not intended to be an RSpec tutorial, just an explanation of how to use the extended functionality that rspec-puppet provides. If you are not familiar with the basics of RSpec, I highly recommend you take some time before continuing to read through the RSpec documentation.
Lets say you’re writing tests for a logrotate::rule type that does two
things:
logrotate::setup class which handles installing logrotatefile resource that drops your logrotate rule into /etc/logrotate.dFirst off, lets create a skeleton spec file for your defined type
(modules/logrotate/spec/defines/rule_spec.rb)
require 'spec_helper'
describe 'logrotate::rule' do
endAs this is a defined type, the first thing we need to do is give it a title
(the string after the { in your manifests).
let(:title) { 'nginx' }Now, lets test that we’re including that logrotate::setup class
it { is_expected.to contain_class('logrotate::setup') }Remember, we don’t want to test what logrotate::setup does, we’ll leave that
to the test cases you’re going to be writing for that class.
At this point, your spec file should look like this
require 'spec_helper'
describe 'logrotate::rule' do
let(:title) { 'nginx' }
it { is_expected.to contain_class('logrotate::setup') }
endOK, on to dealing with that file resource, lets use the title of the
logrotate::rule resource as the name of the file you’re dropping into
/etc/logrotate.d/.
it { is_expected.to contain_file('/etc/logrotate.d/nginx') }As it currently stands, this test is pretty useless as it doesn’t actually
check anything about the file. We can check values of the parameters passed to
the file resource by chaining the with method onto our test and passing it
a hash of expected parameters and values. Lets say we want to set some sane
values: present, owned by root and read only:
it do
is_expected.to contain_file('/etc/logrotate.d/nginx').with({
'ensure' => 'present',
'owner' => 'root',
'group' => 'root',
'mode' => '0444',
})
endYou should now have a spec file that looks like this
require 'spec_helper'
describe 'logrotate::rule' do
let(:title) { 'nginx' }
it { is_expected.to contain_class('logrotate::setup') }
it do
is_expected.to contain_file('/etc/logrotate.d/nginx').with({
'ensure' => 'present',
'owner' => 'root',
'group' => 'root',
'mode' => '0444',
})
end
endWhat about the most important part of the file, its contents? Before we get
to that, we’re going to make your type take a boolean parameter called
compress. If this value is true, a line containing compress should
exist in the file. If this value is false, a line containing nocompress
should exist in the file.
context 'with compress => true' do
let(:params) { {'compress' => true} }
it do
is_expected.to contain_file('/etc/logrotate.d/nginx') \
.with_content(/^\s*compress$/)
end
end
context 'with compress => false' do
let(:params) { {'compress' => false} }
it do
is_expected.to contain_file('/etc/logrotate.d/nginx') \
.with_content(/^\s*nocompress$/)
end
endYou’ll note that we’re now specifying the parameters that should be sent to our
logrotate::rule type by setting params to a hash. Similarly, you can also
specify the value of facts by using let(:facts) { }. The other thing we did
was chain a with_content method onto our test and passed it a regex that the
value should match. You can do this with any other parameter as well, eg
with_ensure, with_owner, with_foobarbaz.
As our type can only handle two possible values for compress, let’s be nice
and make sure that compilation will fail if someone passes something else to
it.
context 'with compress => foo' do
let(:params) { {'compress' => 'foo'} }
it { is_expected.to compile.and_raise_error(/compress must be true or false/) }
endThe final version of your spec file should be:
require 'spec_helper'
describe 'logrotate::rule' do
let(:title) { 'nginx' }
it { is_expected.to contain_class('logrotate::setup') }
it do
is_expected.to contain_file('/etc/logrotate.d/nginx').with({
'ensure' => 'present',
'owner' => 'root',
'group' => 'root',
'mode' => '0444',
})
end
context 'with compress => true' do
let(:params) { {'compress' => true} }
it do
is_expected.to contain_file('/etc/logrotate.d/nginx') \
.with_content(/^\s*compress$/)
end
end
context 'with compress => false' do
let(:params) { {'compress' => false} }
it do
is_expected.to contain_file('/etc/logrotate.d/nginx') \
.with_content(/^\s*nocompress$/)
end
end
context 'with compress => foo' do
let(:params) { {'compress' => 'foo'} }
it do
expect {
is_expected.to contain_file('/etc/logrotate.d/nginx')
}.to raise_error(Puppet::Error, /compress must be true or false/)
end
end
endCongratulations, you’ve just written a set of tests for a defined type without writing a single line of Puppet code. You should now head over to the documentation to learn more.
Now go write the manifests needed to make these tests pass!