在厨师食谱中做一点干燥的最佳方法是什么?即只是打破一点Ruby代码,所以我不会一遍又一遍地复制它。
以下失败当然是:
NoMethodError: undefined method `connect_root' for Chef::Resource::RubyBlock
我可能在一个配方中有多个ruby_blocks,因为它们做了不同的事情,并且需要使用不同的not_if块作为truley idempotent。
def connect_root(root_password)
m = Mysql.new("localhost", "root", root_password)
begin
yield m
ensure
m.close
end
end
ruby_block "set readonly" do
block do
connect_root node[:mysql][:server_root_password] do |connection|
command = 'SET GLOBAL read_only = ON'
Chef::Log.info "#{command}"
connection.query(command)
end
end
not_if do
ro = nil
connect_root node[:mysql][:server_root_password] do |connection|
connection.query("SELECT @@read_only as ro") {|r| r.each_hash {|h|
ro = h['ro']
} }
end
ro
end
end
答案 0 :(得分:23)
正如您已经想到的那样,您无法在食谱中定义功能。为此提供了库。您应该在食谱中的 libraries 文件夹中创建一个文件(例如 mysql_helper.rb ),其中包含以下内容:
module MysqlHelper
def self.connect_root( root_password )
m = Mysql.new("localhost", "root", root_password)
begin
yield m
ensure
m.close
end
end
end
必须是一个模块,而不是一个类。另请注意,我们将其定义为static(使用self.method_name)。然后,您将能够使用模块名称和方法名称在配方中使用此模块中定义的函数:
MysqlHelper.connect_root node[:mysql][:server_root_password] do |connection|
[...]
end
答案 1 :(得分:1)
为了记录,我刚刚创建了一个包含以下内容的库。但这对于一个文件中的DRY来说似乎有些过分。我也无法弄清楚如何让模块使用任何其他命名空间来工作。
class Chef
class Resource
def connect_root(root_password)
...