Hi, Sorry Henrik for my late answer.
On 12/06/2015 01:54, Henrik Lindberg wrote: > Cool, you are an early user, we expect to add more to this, and the road > may be a bit bumpy at the beginning :-) Generally, I'm not a "early user" but currently I need to create a new Puppet infra from scratch and it seemed to me more logical to choose directly Puppet 4. >> 1. About the "data binding" in a modules, if I understand well the >> `./mymodule/lib/puppet/functions/mymodule/data.rb` function replaces >> completely the "params" pattern, so that the "params" pattern is >> completely useless in a module using the Puppet 4 API. Is is correct? >> > > It is an alternative to using the "params" pattern, yes. Now, since it > is possible to write the "mymodule::data" function in the puppet > language it should also be much easier and the code should look very > similar to the puppet logic for the "params" pattern. Ok, I see but if the data() function is a function in the puppet language, should I change the name of the file? Should I take `./mymodule/lib/puppet/functions/mymodule/data.pp` (.pp extension) instead of `./mymodule/lib/puppet/functions/mymodule/data.rb`? In this case, should I modify the content of the file `./mymodule/lib/puppet/bindings/mymodule/default.rb`? In fact, when I want to just manipulate data, I think I prefer the Ruby language because the fact that variables are immutable in the Puppet language causes me problems sometimes (just when I want to manipulate data, to merge 2 hashes, make a look in a hash to retrieve a specific value etc. etc). > (One more related tip is to take a look at the deep case match feature > when writing logic that have many if/then/else and nested such expressions). > >> 2. Always about the "data binding" in a modules, is it correct to say >> that the `data.rb` function replaces too the default values in the >> declaration of a class? For instance with: >> >> class mymodule ( >> $param1 = 'default1', >> $param2 = 'default2', >> ) { >> ... >> } >> >> I can remove the default values in the class and put it in the `data.rb` >> function. Is it correct? >> > Yes you can remove the defaults if you are sure you have values for them > in the data function - this because they will automatically get values. > > A reason to keep them is for documentation purposes; they may help > increase the understanding of how to use the class. Ok, I see. >> 3. In a module, if I have 2 public classes foo.pp and bar.pp, is it >> possible to have 2 different `data.rb` functions, one for foo.pp and one >> for bar.pp? Or maybe I can have only one `data.rb` function which >> provides simultaneously the default values for foo.pp *and* bar.pp? >> > > There is only one function that is called by the "data in modules" > framework. There is nothing stopping you from implementing it by calling > other functions and composing the result. Thus, modularizing the design. > > Say something like: > > function mymodule::data() { > mymodule::data::classa() + mymodule::data::classb() > } > > function mymodule::data::classa() { > $prefix = "mymodule::data::classa" > { "${prefix}::param1" => valuea1, > "${prefix}::param2" => valuea2, > } > } > function mymodule::data::classb() { > $prefix = "mymodule::data::classb" > { "${prefix}::param1" => valueb } > } Ok, I see. >> 4. In a custom function with the Puppet 4 API, in the `dispatch` method, >> is it possible to provide more complex types than 'Array' or 'String', >> for instance is it possible to provide types like "an array of strings" >> or "an non empty array of non empty string"? >> > Yes, types have parameters for things like that; min, max lenghts, > contain data types. There is a rich type system. > (See the documentation, or my blog posts on the topic). > > Array[String[1]] # array of non empty strings, array may be empty > Array[String] # array of strings (possibly empty string), -"- > Array[String[1], 1] # non empty array of non empty strings > > Read more here: > http://puppet-on-the-edge.blogspot.se/2014/02/the-puppet-type-system-blog-posts.html Ah ok. So I can use the same syntax that I can use in the parameters of a puppet class declaration. That's cool and avoid some painful checking in the code. Cool. :) >> 5. In a custom function with the Puppet 4 API, is it possible to get the >> value of a variable in a module or of a fact? With the Puppet 3 API, >> it's possible with `lookupvar('xxx')` but it works no longer with the >> Puppet 4 API. >> > > Access to the content of the calling scope is a bad idea for general > purpose code - it is far better to have pure functions (they process > what they are given as arguments and return a value). Ok, I understand well the "function" paradigm but if it's just in order to *read* a fact, for instance to just read the 'lsbdistcodename' fact and put it in a variable, is it really a bad idea? Ok I can add a parameter to my function `fct(a, b, lsbdistcodename)` but if I can avoid this and just have `fct(a, b)` and just get the 'lsbdistcodename' fact value in the function (just for reading), where is the problem in this specific case? > It is possible to get either the closure_scope (the scope where the > function is defined), or the calling_scope. It is however required to > write a so called "system function", this because the API of the system > functions is more avanced, and that we may have reasons to change it in > the future. It is specified though (see language specification, an look > at the implementation of the more advanced functions (like the iterative > functions; each etc.) > > Note that closure_scope is enough to lookup fully qualified variables > (and global variables). You invoke lookupvar on the closure_scope. > > e.g. if you need a variable say $::osfamily > > closure_scope.lookupvar('::osfamily') Ok, that will be perfect for me. :) In my mind, I just want to use facter or global variables, just for reading, in the body of the custom function and typically use the arguments of the function for the other types of variables. > If you really, really need the calling_scope, this shows you how: > https://github.com/puppetlabs/puppet-specifications/blob/master/language/func-api.md#calling-scope-support > > >> 6. In a custom function with Puppet 4 API, what is the difference between: >> >> call_function(:fail, "Error, blabla blabla") >> >> and >> >> call_function("fail", "Error, blabla blabla") ? >> > > You should use the string form 'fail', not the symbol form (:fail). > There is no gain using the symbol form since the search for the function > will need to split up the name based on namespace anyway. Ok. It's recorded. ;) > I also noted that you posted questions on my blog - I responded there as > well: > http://puppet-on-the-edge.blogspot.se/2015/01/puppet-40-data-in-modules-and.html?showComment=1434066591043#c2269583057836628025 Yes, sorry again for the duplication. > You asked if it was a good idea to call hiera from within a data > function. And my response there was: > > "There is a new function named lookup that combines the behavior of all > hiera functions, it is also aware of data in modules. It is a bad idea > to call hiera from within a data function because your default data is > then no longer default and your module has a dependency on hiera. Ok, I see, the `data()` function is just for "standard" default values. > Instead, you let users simply override values either in their > environment, or in their global hiera. When you lookup a key > "class_a::param1", if it is defined in hiera, it wins over the default > data provided in the module. Maybe I'm wrong (and if yes I'would be very interested to listen arguments) but I don't really like this pattern of "class_a::param1"-keys in hiera. I prefer organizing my data in hiera in separated logical themes: - a "network" theme (ie an hash entry in hiera), - a "snmp" theme (an hash entry in hiera too), - a "ntp" theme (an hash entry in hiera too) - etc. so that my data in hiera don't necessary match with the parameters of my modules (no "class_a::param1" key in hiera). In fact I do not want to have hiera data which are stuck to the structure of the parameters of the modules, I prefer organize the structure of my hiera data independently of the parameters of modules. I find that, with the pattern of "class_a::param1" keys, sometimes (often) the same data is necessary for a "class_a" and for a "class_b" so that "class_a::data_foo" and "class_b::data_foo" should have a duplicated value. I know that interpolations is possible in hiera but I prefer to organize my data by separated logical themes. Of course, then, when a module retrieves data from different "themes" in hiera, I need to build a new hash which will match with the structure of the parameters of the module and I often need to use a ruby functions to build the new hash which is a merge of data from the "network" theme and from the "snmp" theme etc. etc. or sometimes the construction of the new hash is more complex than a simple merge etc. Am I wrong? Should I use the "class_a::param1"-keys pattern instead and should I use hiera interpolation when I have the same value in "class_a::param1" and in "class_b::param1" (to avoid duplication)? In fact, I have absolutely no certainties on this point. > You can use the "lookup" function though - > it always looks up using both hiera and data in modules. Say your > module_a depends on module_b, and you want some default values in > module_a to use data configured for module_b - then you can use the > lookup function in your module_a::data() function. You mean: I can use `lookup('module_b::param1')` in the module_a::data() function. Is it correct? > It will get the > correct (possibly overridden value) in the user's configuration. (If you > use hiera to do the same, it will not ever lookup inside the "data in > modules/environment" since it is a singleton global implementation > across all environments and modules. Ok, I think I have understood. But, in the example you describe here, you are always in the "class_a::param1"-keys pattern. Correct? > Alternatively, if you module_a is > designed to work with module_b, and both are using data functions you > can call module_b's data function directly (as a function), and pick up > values that you use a defaults in your module_a. This way you get the > defaults from module_b (not overridden by data in the environment, nor > in hiera). This relies on module_b::data() function being considered > part of module_b's API." Ok, if I understand well with lookup function I allow overriding but with module_b::data() this is not the case. > I should have added that you need to be careful not to create circular > lookups (both with hiera and with data in modules). Yes, for instance with a `lookup('class_a::param1')` in the class_a::data() function if I understand well. Sorry but I add a new question. Could the way explained below be a good practice? For a module: 1. in `./mymodule/lib/puppet/functions/mymodule/data.rb` I defined the default "standard" values of the parameters of mymodule/init.pp class. 2. And I create a class `./mymodule/profile.pp` where I do something like that: --------------------------------------- class mymodule::profile { # In this class I use hiera lookups. $h1 = hiera_hash('nagios') $h2 = hiera_hash('snmp') $foo = hiera('foo') $bar = hiera('bar') # I use a internal function of mymodule to merge and maybe # to do some more complex operations in order to build a new # hash which is suitable for the public class of mymodule. $h = ::mymodule::internal_function($h1, $h2) class { '::mymodule': foo => $foo, bar => $bar, var => $h, } } --------------------------------------- In other words, I create in the module a specific public class ::mymodule::profile which retrieves data from hiera in a way which matches well with the organization of my hiera data. Could it be a good practice? > Hope my answers will help you out Oh yes. Thx a lot Henrik for your explanations which are very helpful for me. ;) François Lafont -- You received this message because you are subscribed to the Google Groups "Puppet Users" group. To unsubscribe from this group and stop receiving emails from it, send an email to puppet-users+unsubscr...@googlegroups.com. To view this discussion on the web visit https://groups.google.com/d/msgid/puppet-users/557F8F7E.3020803%40gmail.com. For more options, visit https://groups.google.com/d/optout.