Skip to content

Commit 60fd443

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 60fd443

File tree

4 files changed

+188
-15
lines changed

4 files changed

+188
-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 {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,94 @@
1+
# frozen_string_literal: true
2+
3+
require 'spec_helper_acceptance'
4+
5+
describe 'stdlib::deferable_epp function' do
6+
let(:testfile) { (os[:family] == 'windows') ? 'C:\\test.epp' : '/tmp/test.epp' }
7+
8+
before(:all) do
9+
apply_manifest(<<-MANIFEST)
10+
$_epp = @(EPP)
11+
<%- |
12+
Stdlib::Port $port,
13+
String[1] $password,
14+
| -%>
15+
port=<%= $port %>
16+
password=<%= $password %>"
17+
| EPP
18+
$_testfile = $facts['os']['family'] ? {
19+
'windows' => 'C:\\test.epp',
20+
default => '/tmp/test.epp',
21+
}
22+
23+
file{ $_testfile:
24+
ensure => file,
25+
content => $_epp,
26+
}
27+
MANIFEST
28+
end
29+
30+
before(:each) do
31+
rm_testfile = <<-MANIFEST
32+
$_testfile = $facts['os']['family'] ? {
33+
'windows' => 'C:\\test.epp',
34+
default => '/tmp/test.epp',
35+
}
36+
file { "${_testfile}.rendered":
37+
ensure => absent,
38+
}
39+
MANIFEST
40+
apply_manifest(rm_testfile)
41+
end
42+
43+
context 'with no deferred values' do
44+
let(:pp) do
45+
<<-MANIFEST
46+
$_testfile = $facts['os']['family'] ? {
47+
'windows' => 'C:\\test.epp',
48+
default => '/tmp/test.epp',
49+
}
50+
51+
file{ "${_testfile}.rendered":
52+
ensure => file,
53+
content => stdlib::deferrable_epp(
54+
$_testfile,
55+
{'port' => 1234, 'password' => 'top_secret'}
56+
),
57+
}
58+
MANIFEST
59+
end
60+
61+
it 'applies manifest, generates file' do
62+
idempotent_apply(pp)
63+
expect(file("#{testfile}.rendered")).to be_file
64+
expect(file("#{testfile}.rendered").content).to match(%r{port=1234})
65+
expect(file("#{testfile}.rendered").content).to match(%r{password=top_secret})
66+
end
67+
end
68+
69+
context 'with deferred values' do
70+
let(:pp) do
71+
<<-MANIFEST
72+
$_testfile = $facts['os']['family'] ? {
73+
'windows' => 'C:\\test.epp',
74+
default => '/tmp/test.epp',
75+
}
76+
77+
file{ "${_testfile}.rendered":
78+
ensure => file,
79+
content => stdlib::deferrable_epp(
80+
$_testfile,
81+
{'port' => 1234, 'password' => Deferred('inline_epp',['<%= $secret_password %>',{'secret_password' => 'so_secret'}])},
82+
),
83+
}
84+
MANIFEST
85+
end
86+
87+
it 'applies manifest, generates file' do
88+
idempotent_apply(pp)
89+
expect(file("#{testfile}.rendered")).to be_file
90+
expect(file("#{testfile}.rendered").content).to match(%r{port=1234})
91+
expect(file("#{testfile}.rendered").content).to match(%r{password=so_secret})
92+
end
93+
end
94+
end

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)