- Overview
- Motivation
- Operating Principle and Examples
- Relationship Metaparameters and Resource References
- Default Parameters
- Function Aliases
- Advanced Usage
- Crazy Advanced Usage
- Legacy (pre-1.0.0) Support
- Reference
A puppet swiss army knife, bridging the gap between code and configuration, making ad hoc modifications a bit more elegant.
Most modern puppet deployments are composed of puppetlabs/r10k, puppetlabs/hiera and an external node classifier. Although the design is powerful and versioned, it doesn't leave much room for the reality of ad hoc configurations. Often times puppet users find themselves adding a module, or editing an existing module, just to create a file or add a single package, which leads to an r10k push to every relevant environment. The addition of a single missing resource can result in several commits, new repositories and a bunch of individual changes to production environments.
Resource Tree aims to drastically reduce the complexity of ad hoc configurations in puppet by providing a simple mechanism to define puppet resources, and relationships between those resources, entirely in hieradata.
Of course Resource Tree's capabilities extend far beyond defining individual resources, enabling users to do terrible blasphemous things to puppet. Therefore it is highly advisable to keep Resource Tree configurations short and sweet, and avoid writing collections which would be better suited to a module.
Resource Tree is ideal for building simple collections of puppet resources, both user defined and built-ins, which have logical relationships to each other.
A trivial example of such a resource collection would be an index.html
running on a standard apache server with a docroot of /var/www/html
, requiring a total of three file
resources, assuming that /var
is a given. Written briefly:
File['/var/www']->File['/var/www/html']->File['/var/www/html/index.html']
Where the aformentioned file
resource objects would be written in puppet code like so:
file {
'/var/www':
ensure => 'directory';
'/var/www/html':
ensure => 'directory';
'/var/www/html/index.html':
content => 'hello world!',
group => 'apache',
owner => 'apache';
}
Resource Tree provides a method of building the same collection of file
resources entirely in yaml, without creating a separate module:
resource_tree::collections:
'httpd_index_file': # collection
file: # resource type
'/var/www': # resource name
ensure: 'directory' # resource parameter
'/var/www/html': # resource name
ensure: 'directory' # resource parameter
'/var/www/html/index.html': # resource name
content: 'hello world!' # resource parameter
group: 'apache' # resource parameter
owner: 'apache' # resource parameter
The collection would only be applied to a node if resource_tree::apply
, an array, contains the value httpd_index_file
in the local hiera scope. This allows the author to have a shared set of Resource Tree collections, but only apply the desired collections to a given node. For instance:
resource_tree::apply:
- httpd_index_file
Each resource in the tree may optionally contain an rt_resources
parameter, where any included resources implicitly require those closer to the root of the tree. For instance in the following example an apache configuration file and service implicitly require the httpd package:
resource_tree::collections:
apache:
package:
'httpd':
ensure: 'installed'
rt_resources:
service:
'httpd':
ensure: 'running'
file:
'/etc/httpd/conf.d/status.load':
ensure: 'present'
owner: 'apache'
group: 'apache'
content: 'LoadModule status_module "modules/mod_status.so"'
notify:
service: 'httpd'
Relationships between resources in the tree can also be explicitly stated using the require
parameter. For instance the previous example could be rewritten with explicit relationships:
resource_tree::collections:
apache:
package:
'httpd':
ensure: 'installed'
service:
'httpd':
ensure: 'running'
require: 'Package[httpd]'
file:
'/etc/httpd/conf.d/status.load':
ensure: 'present'
owner: 'apache'
group: 'apache'
content: 'LoadModule status_module "modules/mod_status.so"'
require: 'Package[httpd]'
notify: 'Service[httpd]'
Resource Tree supports the four puppet relationship metaparameters: before, require, notify and subscribe. Since it would be difficult and overly verbose to write out serialized resource references in YAML, Resource Tree provides three styles in which resource references can be expressed as strings.
resource_tree::collections:
'resource_ref_styles':
file:
'/tmp/example.txt':
# Puppet style resource ref string
require: 'Service[puppet]'
# Resource Tree style ref string
before: 'package-puppet'
# Hash of type:title
notify:
service: 'puppet'
In addition to individual strings, and hashes, users may provide arrays of references, even mixing different styles for each element of the array. For more information, refer to the documentation for the resref
function.
Although users are free to implement all styles, it is strongly encouraged to adhere to a single style for any individual resource collection to enhance readability.
Users may provide default parameters for resources declared via Resource Tree. This can greatly reduce repetition of parameters when declaring many resources of the same type.
In puppet we would define default parameters for Package
resources like so:
Package {
provider => 'yum'
}
In Resource Tree we would declare it similarly:
resource_tree::default_params:
package:
provider: 'yum'
Note: Unfortunately relationship metaparameters are not yet supported as default parameters.
Traditionally calling puppet functions from a template requires a fairly ugly syntax. Resource Tree instead provides four common functions with their traditional puppet syntax: hiera
, hiera_array
, hiera_hash
and inline_template
.
This capability may be extended to other functions in future releases.
Resource Tree provides a number of advanced features so collections can be built dynamically. Any individual resource definition, resource collection, or collection of child resources which is a string will be evaluated as ruby code, eg.
resource_tree::collections:
'resource_tree_advanced_examples':
file:
'/tmp/thetime.txt': |
# Create a file containing just the time
{ 'content' => Time.now.to_s }
'/tmp/test':
ensure: 'directory'
rt_resources: |
# Create five files in /tmp/test
{ 'file' =>
Hash[(1..5).map {|n|
[
"/tmp/test/test-file-#{n}",
{ "content" => rand(500).to_s }
]
}]
}
host: |
# Add five host entries
Hash[(1..5).map {|n|
[
"test-node-0#{n}",
{"ip" => "192.168.1.#{n}", "ensure" => "present" }
]
}]
Additionally individual resource parameters can be evaluated by prefixing them with rt_eval::
.
A few intrepid users have discovered interesting ways to combine Resource Tree and ruby to dynamically create resources, including querying puppetdb to find members for proxies/load-balancers. Here are a few tricks that tip the scales for what Resource Tree can do.
resource_tree::collections:
hiera_include:
class: |
Hash[(hiera_array('classes') - ['resource_tree']).map {|c| [c, {}] }]
resource_tree::collections:
haproxy_webservers:
haproxy::balancermember: |
member_ip_by_cert = scope.function_query_facts(["environment='%{environment}' and app='webserver'", [:ipaddress]])
{
"webservers" => {
"listening_service" => 'webservers',
"ports" => '80',
"server_names" => member_ip_by_cert.map{|k,v| k.split(".")[0] },
"ipaddresses" => member_ip_by_cert.map{|k,v| v["ipaddress"] },
"options" => [ 'check' ]
}
}
config_file_template: |
<% config = scope.function_hiera(['configs', {}]).fetch(@name[5..-1], {}) -%>
foo=<%= config["foo"] %>
biz=<%= config["biz"] %>
configs:
'/etc/example/default.properties':
foo: 'boz'
biz: 'bar'
'/etc/example/example.properties':
foo: 'bar'
biz: 'boz'
resource_tree::collections:
apply_configs:
file: |
Hash[hiera('configs').map {|k,v|
[
"#{k}",
{
'ensure' => 'present',
'owner' => 'root',
'group' => 'root',
'content' => 'rt_eval::inline_template(hiera("config_file_template"))'
}
]
}]
Prior to version 1.0.0, Resource Tree created Placeholder
resources which were related to other objects in the tree. This strategy helped avoid the need to generate resource references from strings. Unfortunately this had the caveat of preventing users from referencing resources outside the tree which didn't have Placeholder resources.
Resource Tree now has elaborate capabilities to generate native puppet resource references. During this transition, legacy relationship metaparameters rt_requires
and rt_notify
were replaced with more conventional require
and notify
.
To ease migration from older versions of Resource Tree, the old relationship metaparameters are still supported, however NO Placeholder
resources will be created. Furthermore users may choose to use either the old or new relationship metaparameters, but not both. In the event that both old and new relationship metaparameters are used, only the new ones will be honored.
Converts all arguments to an array of puppet resource references. Each argument can be a string/dict or array of string/dict.
Examples:
resref('File[foo]') => [File[foo]]
resref(['File[foo]','File[bar]'], 'Package[biz]') => [File[foo], File[bar], Package[biz]]
resref({'file' => ["foo", "bar"]}) => [File[foo], File[bar]]
resref(['file-foo', {'file' => 'bar'}]) => [File[foo], File[bar]]
Type: rvalue.