是否有一个优雅的(相对)解决方案来跨多个定义调用共享变量,这允许每个定义调用将其数据附加到全局变量?在软件中,这将被称为单身人士.
– 杂草 –
Nginx有一个map指令,它指示应该将请求传递给哪个上游服务器池,如下所示:
map $http_host $upstream_pool { hostnames; blog.example.com blog_example_com; repo.example.com repo_example_com; default example_com; }
如您所见,对blog.example.com的任何请求都将传递到blog_example_com上游服务器池(通过proxy_pass).
问题是map指令语法是它只能包含在主http块(Nginx.conf)中,而vhost特定指令(如上游和位置)可以包含在vhost配置的服务器块中.
我的nodes.pp清单看起来像这样:
service-a-1.example.com inherits project_dev { Nginx::vhost { 'mothership': } Nginx::vhost { 'mothership_blog': } Nginx::vhost { 'repo': } }
正如您所看到的,在成功运行木偶之后,我应该在/etc/Nginx/vhost.d/dir中使用3个不同的vhost配置文件.
我遇到的问题是,为了使map指令工作,我需要知道加载了哪些vhost,所以我可以将它们各自的上游id添加到map指令中,我在主配置中定义了:/ etc / http块中的Nginx / Nginx.conf(其中只能有一个).
– 我尝试过的 –
我有一个global.pp文件,它做了一些“bootstrapping”,在这个文件中我添加了一个$singleton =”语法,然后在Nginx :: vhost定义中,我添加了这个语法:
$tpl_upstream_pool_labels = inline_template("<% tpl_upstream_pools.keys.sort.each do |role| %><%= role %>_<%= tpl_domain_primary %>_<%= tpl_domain_top_level %>|<% end %>") $singleton = "${singleton}${tpl_upstream_pool_labels}" notify { "\n--------------------- Nginx::Conf::Vhost::Touch | Timestamp: ${timestamp} | Pool labels: ${singleton} -------------------------\n": }
这应该导致上游ID的管道分隔列表.如前所述,在nodes.pp清单中,我对Nginx :: vhost进行了三次调用,并期望为每次调用附加$singleton全局变量,但不是,它只包含最后一次调用的数据.
$temp_file_upstream_pool_labels_uri = "/tmp/puppet_known_upstreams_${timestamp}.txt" exec { "event_record_known_upstream_${name}" : command => "touch ${temp_file_upstream_pool_labels_uri} && echo ${tpl_upstream_pool_labels} >> ${temp_file_upstream_pool_labels_uri}",provider => 'shell' }
然后在Nginx :: conf :: touch define中,主要配置Nginx.conf将由puppet编写,我试过这个:
$temp_file_upstream_pool_labels_uri = "/tmp/puppet_known_upstreams_${timestamp}.txt" $contents = file($temp_file_upstream_pool_labels_uri)
从理论上讲,这应该将文件的内容加载到$contents变量中.但是当我使用这种方法运行puppet时,我得到一个文件不存在的错误.我确保在考虑所有vhost之后才进行Nginx :: conf :: touch调用,但仍无效.
解决方法
我已经实现了一个可以满足您需求的模式,但我并不为此感到自豪,也不建议使用它.不过,既然你问,那么:
class Nginx::conf { $global = '' file { '/etc/Nginx/Nginx.conf': content => template(...) } }
可以附加类中的变量,至少在2.7.x中.
define Nginx::vhost(...) { include Nginx::conf $Nginx::conf::global += "new content here" Nginx::override_conf_content { "for-vhost-$name": } }
魔法发生在神秘的最后一行,它实例化另一个定义.
define Nginx::override_conf_content() { include Nginx::conf $global = $Nginx::conf::global File<| title == '/etc/Nginx/Nginx.conf' |> { content => template(...) } }
Nginx.conf的content属性将被另一个模板评估结果覆盖,并带有新变量值.
完全披露,我不知道为什么这样做.它可能不应该,并且可能依赖于一个不起眼的bug.它可能会在将来的版本中停止工作.
请注意,我称之为$global而不是$singleton,因为它就是这样.单例可用于实现全局变量的语义,但它们不是同一个东西.
最后,即使我能感受到Puppet 3更新所面临的痛苦,你也应该花时间让它继续下去.我们都可能无法负担更长时间运行2.x.