Skip to content

Commit 0c800b4

Browse files
committed
Parse deferred templates twice
Currently it is not possible to have a template file.epp ```puppet <%- | Stdlib::Port $port, String[1] $password, | %> port <%= $port %> password <%= $password %> ``` and run ```puppet file{'/tmp/junk': content => stdlib::deferrable_epp('module/file.epp', { 'port' => '1234', pass => Deferred('secrets::get',['mysecret'])}), } ``` since the deferred template substitution will fail: ``` Error: Failed to apply catalog: Evaluation Error: Resource type not found: Stdlib::Port (file: inlined-epp-text, line: 2, column: 3) ``` due to Stdlib::Port not being available on the agent node. This change now parses the EPP twice. The first pass will reduce the template to: ```puppet port = 1234 password <%= $password %> ``` and this simpler template will be passed in deferred mode. Note the original template type for password must accept the intermediate generated value of `<%= $password %>` which is typically case for a secret password.
1 parent d871c4d commit 0c800b4

File tree

3 files changed

+94
-15
lines changed

3 files changed

+94
-15
lines changed

REFERENCE.md

+34-8
Original file line numberDiff line numberDiff line change
@@ -3173,18 +3173,38 @@ Type: Puppet Language
31733173
This function returns either a rendered template or a deferred function to render at runtime.
31743174
If any of the values in the variables hash are deferred, then the template will be deferred.
31753175

3176-
Note: this function requires all parameters to be explicitly passed in. It cannot expect to
3177-
use facts, class variables, and other variables in scope. This is because when deferred, we
3178-
have to explicitly pass the entire scope to the client.
3179-
3180-
#### `stdlib::deferrable_epp(String $template, Hash $variables)`
3176+
Note: In the case where at least some of the values are deferred and preparse is `true` the template
3177+
is parsed twice:
3178+
The first parse will evalute any parameters in the template that do not have deferred values.
3179+
The second parse will run deferred and evaluate only the remaining deferred parameters. Consequently
3180+
any parameters to be deferred must accept a String[1] in original template so as to accept the value
3181+
"<%= $variable_with_deferred_value %>" on the first parse.
3182+
3183+
@param template template location - identical to epp function template location.
3184+
@param variables parameters to pass into the template - some of which may have deferred values.
3185+
@param preparse
3186+
If `true` the epp template will be parsed twice, once normally and then a second time deferred.
3187+
It may be nescessary to set `preparse` `false` when deferred values are somethig other than
3188+
a string
3189+
3190+
#### `stdlib::deferrable_epp(String $template, Hash $variables, Boolean $preparse = true)`
31813191

31823192
This function returns either a rendered template or a deferred function to render at runtime.
31833193
If any of the values in the variables hash are deferred, then the template will be deferred.
31843194

3185-
Note: this function requires all parameters to be explicitly passed in. It cannot expect to
3186-
use facts, class variables, and other variables in scope. This is because when deferred, we
3187-
have to explicitly pass the entire scope to the client.
3195+
Note: In the case where at least some of the values are deferred and preparse is `true` the template
3196+
is parsed twice:
3197+
The first parse will evalute any parameters in the template that do not have deferred values.
3198+
The second parse will run deferred and evaluate only the remaining deferred parameters. Consequently
3199+
any parameters to be deferred must accept a String[1] in original template so as to accept the value
3200+
"<%= $variable_with_deferred_value %>" on the first parse.
3201+
3202+
@param template template location - identical to epp function template location.
3203+
@param variables parameters to pass into the template - some of which may have deferred values.
3204+
@param preparse
3205+
If `true` the epp template will be parsed twice, once normally and then a second time deferred.
3206+
It may be nescessary to set `preparse` `false` when deferred values are somethig other than
3207+
a string
31883208

31893209
Returns: `Variant[String, Sensitive[String], Deferred]`
31903210

@@ -3200,6 +3220,12 @@ Data type: `Hash`
32003220

32013221

32023222

3223+
##### `preparse`
3224+
3225+
Data type: `Boolean`
3226+
3227+
3228+
32033229
### <a name="stdlib--end_with"></a>`stdlib::end_with`
32043230

32053231
Type: Ruby 4.x API

functions/deferrable_epp.pp

+29-5
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,39 @@
11
# This function returns either a rendered template or a deferred function to render at runtime.
22
# If any of the values in the variables hash are deferred, then the template will be deferred.
33
#
4-
# Note: this function requires all parameters to be explicitly passed in. It cannot expect to
5-
# use facts, class variables, and other variables in scope. This is because when deferred, we
6-
# have to explicitly pass the entire scope to the client.
4+
# Note: In the case where at least some of the values are deferred and preparse is `true` the template
5+
# is parsed twice:
6+
# The first parse will evalute any parameters in the template that do not have deferred values.
7+
# The second parse will run deferred and evaluate only the remaining deferred parameters. Consequently
8+
# any parameters to be deferred must accept a String[1] in original template so as to accept the value
9+
# "<%= $variable_with_deferred_value %>" on the first parse.
710
#
8-
function stdlib::deferrable_epp(String $template, Hash $variables) >> Variant[String, Sensitive[String], Deferred] {
11+
# @param template template location - identical to epp function template location.
12+
# @param variables parameters to pass into the template - some of which may have deferred values.
13+
# @param preparse
14+
# If `true` the epp template will be parsed twice, once normally and then a second time deferred.
15+
# It may be nescessary to set `preparse` `false` when deferred values are somethig other than
16+
# a string
17+
#
18+
function stdlib::deferrable_epp(String $template, Hash $variables, Boolean $preparse = true) >> Variant[String, Sensitive[String], Deferred] {
919
if $variables.stdlib::nested_values.any |$value| { $value.is_a(Deferred) } {
20+
if $preparse {
21+
$_variables_escaped = $variables.map | $_var , $_value | {
22+
if $_value.is_a(Deferred) {
23+
{ $_var => "<%= \$${_var} %>" }
24+
} else {
25+
{ $_var => $_value }
26+
}
27+
}.reduce | $_memo, $_kv | { $_memo + $_kv }
28+
29+
$_template = inline_epp(find_template($template).file,$_variables_escaped)
30+
} else {
31+
$_template = find_template($template).file
32+
}
33+
1034
Deferred(
1135
'inline_epp',
12-
[find_template($template).file, $variables],
36+
[$_template, $variables],
1337
)
1438
}
1539
else {

spec/functions/stdlib_deferrable_epp_spec.rb

+31-2
Original file line numberDiff line numberDiff line change
@@ -24,8 +24,37 @@
2424

2525
it {
2626
foo = Puppet::Pops::Types::TypeFactory.deferred.create('join', [1, 2, 3])
27-
# This kind_of matcher requires https://github.com/puppetlabs/rspec-puppet/pull/24
28-
expect(subject).to run.with_params('mymod/template.epp', { 'foo' => foo }) # .and_return(kind_of Puppet::Pops::Types::PuppetObject)
27+
expect(subject).to run.with_params('mymod/template.epp', { 'foo' => foo }).and_return(kind_of(Puppet::Pops::Types::PuppetObject))
28+
}
29+
end
30+
31+
context 'defers rendering with mixed deferred and undeferred input' do
32+
let(:pre_condition) do
33+
<<~END
34+
function epp($str, $data) { fail("should not have invoked epp()") }
35+
function find_template($str) { return "path" }
36+
function file($path) { return "foo: <%= foo %>, bar: <%= bar %>" }
37+
END
38+
end
39+
40+
it {
41+
foo = Puppet::Pops::Types::TypeFactory.deferred.create('join', [1, 2, 3])
42+
expect(subject).to run.with_params('mymod/template.epp', { 'foo' => foo, 'bar' => 'xyz' }).and_return(kind_of(Puppet::Pops::Types::PuppetObject))
43+
}
44+
end
45+
46+
context 'defers rendering with mixed deferred and undeferred input and preparse false' do
47+
let(:pre_condition) do
48+
<<~END
49+
function epp($str, $data) { fail("should not have invoked epp()") }
50+
function find_template($str) { return "path" }
51+
function file($path) { return "foo: <%= foo %>, bar: <%= bar %>" }
52+
END
53+
end
54+
55+
it {
56+
foo = Puppet::Pops::Types::TypeFactory.deferred.create('join', [1, 2, 3])
57+
expect(subject).to run.with_params('mymod/template.epp', { 'foo' => foo, 'bar' => 'xyz' }, false).and_return(kind_of(Puppet::Pops::Types::PuppetObject))
2958
}
3059
end
3160
end

0 commit comments

Comments
 (0)