Ruby 每个方法中的Chef通知
我有一个配方,它在Ruby 每个方法中的Chef通知,ruby,chef-infra,devops,Ruby,Chef Infra,Devops,我有一个配方,它在each方法中迭代一个包含SQL脚本的散列,如果脚本从上一次运行时更改,cookbook\u文件资源通知execute资源运行。 问题是它似乎总是使用散列的最后一个元素运行execute 在属性文件之后 default['sql_scripts_dir'] = 'C:\\DBScripts' default['script_runner']['scripts'] = [ { 'name' => 'test', 'hostname' => 'local' },
each
方法中迭代一个包含SQL脚本的散列,如果脚本从上一次运行时更改,cookbook\u文件
资源通知execute
资源运行。
问题是它似乎总是使用散列的最后一个元素运行execute
在属性文件之后
default['sql_scripts_dir'] = 'C:\\DBScripts'
default['script_runner']['scripts'] = [
{ 'name' => 'test', 'hostname' => 'local' },
{ 'name' => 'test2', 'hostname' => 'local' },
{ 'name' => 'test3', 'hostname' => 'local' },
{ 'name' => 'test4', 'hostname' => 'local' },
]
配方呢
directory node['sql_scripts_dir'] do
recursive true
end
node['script_runner']['scripts'].each do |script|
cookbook_file "#{node['sql_scripts_dir']}\\#{script['name']}.sql" do
source "#{script['name']}.sql"
action :create
notifies :run, 'execute[Create_scripts]', :immediately
end
execute 'Create_scripts' do
command "sqlcmd -S \"#{script['hostname']}\" -i \"#{node['sql_scripts_dir']}\\#{script['name']}.sql\""
action :nothing
end
end
并产生以下输出:
Recipe: test_runner::default
* directory[C:\DBScripts] action create
- create new directory C:\DBScripts
* cookbook_file[C:\DBScripts\test.sql] action create
- create new file C:\DBScripts\test.sql
- update content in file C:\DBScripts\test.sql from none to 8c40f1
--- C:\DBScripts\test.sql 2020-07-30 17:30:30.959220400 +0000
+++ C:\DBScripts/chef-test20200730-1500-11bz3an.sql 2020-07-30 17:30:30.959220400 +0000
@@ -1 +1,2 @@
+select @@version
* execute[Create_scripts] action run
================================================================================
Error executing action `run` on resource 'execute[Create_scripts]'
================================================================================
Mixlib::ShellOut::ShellCommandFailed
------------------------------------
Expected process to exit with [0], but received '1'
---- Begin output of sqlcmd -S "local" -i "C:\DBScripts\test4.sql" ----
STDOUT:
STDERR: Sqlcmd: 'C:\DBScripts\test4.sql': Invalid filename.
---- End output of sqlcmd -S "local" -i "C:\DBScripts\test4.sql" ----
Ran sqlcmd -S "local" -i "C:\DBScripts\test4.sql" returned 1
预期的行为是,配方按顺序运行示例中的4个脚本,而不是仅运行最后一个脚本。完成此任务时,我缺少什么?您正在创建4个几乎相同的资源,所有资源都命名为
执行[Create_scripts]
,当通知从更新的第一个cookbook_文件
资源发出时,它会找到最后一个要通知的资源,并针对test4
(无论哪本食谱\u文件资源更新)
修复方法是使用字符串插值将执行资源的名称更改为唯一,并基于该唯一名称进行通知:
directory node['sql_scripts_dir'] do
recursive true
end
node['script_runner']['scripts'].each do |script|
cookbook_file "#{node['sql_scripts_dir']}\\#{script['name']}.sql" do
source "#{script['name']}.sql"
action :create
notifies :run, "execute[create #{script['name']} scripts]", :immediately
end
execute "create #{script['name']} scripts" do
command "sqlcmd -S \"#{script['hostname']}\" -i \"#{node['sql_scripts_dir']}\\#{script['name']}.sql\""
action :nothing
end
end
请注意,这是旧的CHEF-3694警告消息背后的相同问题的一种表现,其中将发生的情况是,所有四个执行资源将通过“资源克隆”合并到一个资源中,后续资源的属性为“last writer wins”
在Chef 13中,这被更改为删除资源克隆和警告,并且在大多数情况下,在资源集合中有两个名称相同的资源是完全无害的——除非您尝试通知其中一个资源。在这种情况下,资源通知系统应该真正发出警告,而不是默默地选择最后一个匹配的资源(但在通知、订阅、延迟解析和现在的统一模式之间,代码非常复杂,您只希望它在完全正确的条件下启动)。您正在构建一个资源集合,其中包含4个cookbook\u文件资源和4个几乎相同的资源,所有这些资源都命名为
execute[Create\u scripts]
,当通知触发时,它总是找到最后一个,因此它针对test4
运行并同样更改通知。好的,will tryworks就像一个符咒,@lamont请将它转换为一个答案,这样我就可以接受。thxokay,写下了一些东西。