假设我在 Recipe 中有一个默认属性:
default.nginx_upstreams = {
'service1' => ['service1.server.com'],
'service2' => ['service2.server.com'],
}
然后它在角色和环境中被修改和覆盖,直到它最终达到我的 Recipe 。在那里,我计算了一些我想要添加到属性中的附加服务。如果我做这样的事情:
node.nginx_upstreams.merge! {'service3' => ['service3.server.com']}
然后,当我尝试在模板中使用该属性时,我得到 undefined method 'each' for nil:NilClass
当我尝试这样做时在我的模板中
<% node.nginx_upstreams.each do |name, servers| %>
另外,我还得到了 WARN: Setting attributes without specifying a precedence is deprecated and will be removed in Chef 11.0
。有用的警告告诉我如何以正常优先级设置属性(显然,使用 node.set["key"] = "value"
,但没有告诉我如何指定默认或覆盖属性。
我可以通过这样做来解决这个问题:
upstreams = node.nginx_upstreams.to_hash
upstreams.merge! {'service3' => ['service3.server.com']}
template "nginx_config" do
variables({:upstreams=>upstreams})
end
但这感觉就像是黑客攻击。我找不到关于 node.set()
的任何文档超越this page ,这也表明您可以在配方中设置普通属性和覆盖属性,但没有说明如何设置。
那么...如何从配方内部正确设置属性(与其他所有内容一起深度合并)? node.set()
是什么意思? call 实际上是这样做的,我可以告诉它我想要合并的优先级吗?
最佳答案
default.nginx_upstreams
与 default[:nginx_upstreams]
和 default['nginx_upstreams']
相同 - 约定是使用 1后两者中。当您进一步使用字符串时,也可以在这里使用它们。
在属性文件中初始化nginx_upstreams
的方式与这样做相同:
default['nginx_upstreams']['service1'] = ['service1.server.com']
default['nginx_upstreams']['service2'] = ['service2.server.com']
并且您不必在此之前初始化 default['nginx_upstreams'] = {}
。这些不是哈希值,而是属性,而且它们要聪明得多。 :)
从配方内部修改属性是这样完成的:
node.default['nginx_upstreams']['service3'] = ['service3.server.com']
如果需要更改优先级,可以在此处使用set
或override
代替default
。省略优先级名称(node['nginx_upstreams']
或 node.nginx_upstreams
)将使用 set
优先级。但这已被弃用并且很快将被删除 - 这就是警告的全部内容。
查看manual page about attributes ,因为一切实际上都在那里。
关于chef-infra - 覆盖配方中的属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14619001/