In the first post of this series about Facter, we introduced its basic features and we talked about Core facts, the ones shipped directly with Facter and available whenever we use Puppet.
In this post we will see more details about facts that we can write in Ruby by ourselves for whatever purpose we may have.
Custom facts
Custom facts are shipped with Puppet modules, if we use already Puppet it’s likely we are already using some custom fact present in one of the public modules we might be using.
They are written in Ruby language and have to be placed in the directory lib/facter
of a module.
The simplest example of a fact is one that just executes a command and shows its output:
Facter.add('connected_users') do
setcode do
Facter::Core::Execution.execute('/usr/bin/who | wc -l')
end
end
This fact is called collected_users
, this is the argument passed to the Facter.add
method.
Whatever code we want to run in order to calculate the value of this fact, has to stay inside the setcode
statement. In this case we just want to run a shell command, and we use the dedicated Facter::Core::Execution.execute
method for this (which takes care of wrapping our command as needed). The argument passed here is the command we want to execute (/usr/bin/who | wc -l
).
The output of this command is the value of our collected_users fact.
In order to ship and use this fact, we have to place the above piece of Ruby code in a file called lib/facter/collected_users.rb
of a module.
If we have such a module in the $modulepath
of our Puppet Server (for example the directory /etc/puppetlabs/code/modules
for modules available to all Puppet environments, or /etc/puppetlabs/code/environments/production/modules
for modules available for the, default, production Puppet environment), this fact would be automatically copied to each client, before running the catalog request. This means that the fact is immediately available to the client and can be used straight on in our Puppet manifests.
This automatic copy of each custom fact (and other extensions placed in the lib
directory) of a module is called pluginsync and we actually see it happening when we run Puppet, with an output like:
Info: Using configured environment 'production'
Info: Retrieving pluginfacts
Notice: /File[/opt/puppetlabs/puppet/cache/lib/facter/connected_users.rb]/ensure: defined content as '{md5}d4cfb32bbc71e8d738004e584b0ac8bf'
[...]
which is telling us that the connected_users.rb
file present in the lib/facter
directory of one of our modules has been copied to the /opt/puppetlabs/puppet/cache/lib/facter/
directory of the client.
Note that custom facts are not visible when we run the facter
command from the local cli. We need to specify -p
(--puppet
) argument, or, recommended starting from Facter 3, use instead the puppet facts
command:
root@client:~# facter connected_users
root@client:~# facter -p connected_users
1
root@puppet:~# puppet facts | grep connected_users
"connected_users": "1",
Facts confinement
What’s wrong about the fact we just wrote, if we work in a multi OS environment?
It runs a shell command which is available under Linux / Unix, but this is what happens when we run it under Windows:
Info: Retrieving plugin
Notice: /File[C:/ProgramData/PuppetLabs/puppet/cache/lib/facter/connected_users.rb]/ensure: defined content as '{md5}d4cfb32bbc71e8d738004e584b0ac8bf'
Info: Loading facts
Error: Facter: error while resolving custom fact "connected_users": execution of command "/usr/bin/who | wc -l" failed: command not found.
and we don’t want unnecessary (in this case not blocking) errors, right?
We can use the confine
statement, which restricts the execution of the fact only on systems that match the given condition, based on another fact.
For example, to confine our connected_users
fact to run only on Linux we can use the kernel
core fact:
Facter.add('connected_users') do
**confine :kernel => 'Linux'**
setcode do
Facter::Core::Execution.execute('/usr/bin/who | wc -l')
end
end
Facts within Facts
We can refer to and use the values of other facts inside our custom facts code.
For example we can assign to a local variable, the value of a fact with:
k = Facter.value(:kernel)
Facts precedence
We can have multiple Facter.add
statements with the same name, they can have different confinements so that we can resolve the possible values of a fact using different code and logic.
We can also have different entries for the same confinement group, in this case we need a way to decide what’s the actual value to use for a fact, and here comes handy the concept of Facts’s weight which defines the wight to give to a give, if we have valid values for two different fact names, the one with higher weight “wins” and provides the relevant value.
A good example of usage if confine
and has_weight
is the code of the core fact virtual
in Ruby (so in Facter version 2, as the equivalent in version 3 is written in C++). This can be seen here.
Note that we have, in the virtual.rb
file, multiple blocks like:
Facter.add("virtual") do
confine :kernel => 'XXX
confine: :XXX => 'XXX'
has_weight XXX
end
their combination is used to provide the final value of the virtual fact for different OS and different Hypervisors.
Structured facts and aggregated resolutions
In the fist part of this series about What you need to know about Facter, we have seen that since Facter version 2, with new acceptable data types (Integer, Float, TrueClass, FalseClass, NilClass, String, Array, Hash. ) as values, Structured facts were introduced.
At the same time, aggregated resolutions have been introduced.
This is a way to compose the content of a structured fact but executing different chunks of code.
A fact with aggregated resolution may look like:
Facter.add(:connected_users, :type => :aggregate) do
chunk(:a_name) do
[..] # Ruby code that get some information from the system
end
chunk(:another_name) do
[..]
end
aggregate do | chunks |
# Optional aggregate block that massages the output ProgramData
end
end
A glorious example of structured fact using aggregated resolution is the Ruby version of the $::os
fact. Note that in this case different chunks are defined and simply merged in the final result, without the use of the aggregate
method.
Conclusions
We have seen the basics to create custom facts in Ruby. Give a look on Puppet docs site to the Overview of custom facts with examples, and the Custom facts walkthrough, and check the version 2 branch of facter for, probably, the most advanced examples (being also the core facts) of fact written in Ruby.
Next week we are going to see a much simpler way to create custom facts: external facts.
If you don’t know much about them, prepare to have nice surprises.
Alessandro Franceschi