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,写下了一些东西。