summaryrefslogtreecommitdiff
path: root/spec
diff options
context:
space:
mode:
authorlamont-opscode <lamont@opscode.com>2012-12-21 10:52:42 -0800
committerlamont-opscode <lamont@opscode.com>2012-12-21 10:52:42 -0800
commiteed92c71356f44afc03c575ca5aadff8023cb334 (patch)
treece77d52b4c14ab85b558f120d481d0fa78dcc533 /spec
parentf70530ee2ba698ffa06f8cefd360fac3d4b6ef01 (diff)
parent217cf85876651185b33fb574de7256e5a499d499 (diff)
downloadchef-eed92c71356f44afc03c575ca5aadff8023cb334.tar.gz
Merge pull request #562 from lamont-opscode/lcg/OC-4660-registry
Lcg/oc 4660 registry
Diffstat (limited to 'spec')
-rw-r--r--spec/functional/dsl/registry_helper_spec.rb63
-rw-r--r--spec/functional/resource/registry_spec.rb576
-rw-r--r--spec/functional/win32/registry_helper_spec.rb632
-rw-r--r--spec/spec_helper.rb2
-rw-r--r--spec/support/platform_helpers.rb10
-rw-r--r--spec/unit/dsl/regsitry_helper_spec.rb55
-rw-r--r--spec/unit/provider/registry_key_spec.rb269
-rw-r--r--spec/unit/registry_helper_spec.rb374
-rw-r--r--spec/unit/resource/registry_key_spec.rb171
9 files changed, 2152 insertions, 0 deletions
diff --git a/spec/functional/dsl/registry_helper_spec.rb b/spec/functional/dsl/registry_helper_spec.rb
new file mode 100644
index 0000000000..452c4c2799
--- /dev/null
+++ b/spec/functional/dsl/registry_helper_spec.rb
@@ -0,0 +1,63 @@
+#
+# Author:: Prajakta Purohit (<prajakta@opscode.com>)
+# Copyright:: Copyright (c) 2011 Opscode, Inc.
+# License:: Apache License, Version 2.0
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+require "chef/dsl/registry_helper"
+require "spec_helper"
+
+describe Chef::Resource::RegistryKey, :windows_only do
+
+ before (:all) do
+ ::Win32::Registry::HKEY_CURRENT_USER.create "Software\\Root"
+ ::Win32::Registry::HKEY_CURRENT_USER.create "Software\\Root\\Branch"
+ ::Win32::Registry::HKEY_CURRENT_USER.open('Software\\Root', Win32::Registry::KEY_ALL_ACCESS) do |reg|
+ reg['RootType1', Win32::Registry::REG_SZ] = 'fibrous'
+ reg.write('Roots', Win32::Registry::REG_MULTI_SZ, ["strong roots", "healthy tree"])
+ end
+
+ events = Chef::EventDispatch::Dispatcher.new
+ node = Chef::Node.new
+ ohai = Ohai::System.new
+ ohai.all_plugins
+ node.consume_external_attrs(ohai.data,{})
+ run_context = Chef::RunContext.new(node, {}, events)
+ @resource = Chef::Resource.new("foo", run_context)
+ end
+
+ context "tests registry dsl" do
+ it "returns true if registry_key_exists" do
+ @resource.registry_key_exists?("HKCU\\Software\\Root").should == true
+ end
+ it "returns true if registry has specified value" do
+ values = @resource.registry_get_values("HKCU\\Software\\Root")
+ values.include?({:name=>"RootType1",:type=>:string,:data=>"fibrous"}).should == true
+ end
+ it "returns true if specified registry_has_subkey" do
+ @resource.registry_has_subkeys?("HKCU\\Software\\Root").should == true
+ end
+ it "returns true if specified key has specified subkey" do
+ subkeys = @resource.registry_get_subkeys("HKCU\\Software\\Root")
+ subkeys.include?("Branch").should == true
+ end
+ it "returns true if registry_value_exists" do
+ @resource.registry_value_exists?("HKCU\\Software\\Root", {:name=>"RootType1", :type=>:string, :data=>"fibrous"}).should == true
+ end
+ it "returns true if data_value_exists" do
+ @resource.registry_data_exists?("HKCU\\Software\\Root", {:name=>"RootType1", :type=>:string, :data=>"fibrous"}).should == true
+ end
+ end
+end
diff --git a/spec/functional/resource/registry_spec.rb b/spec/functional/resource/registry_spec.rb
new file mode 100644
index 0000000000..2ef359273d
--- /dev/null
+++ b/spec/functional/resource/registry_spec.rb
@@ -0,0 +1,576 @@
+#
+# Author:: Prajakta Purohit (<prajakta@opscode.com>)
+# Author:: Lamont Granquist (<lamont@opscode.com>)
+# Copyright:: Copyright (c) 2011 Opscode, Inc.
+# License:: Apache License, Version 2.0
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+require "chef/win32/registry"
+require "chef/resource_reporter"
+require "spec_helper"
+
+describe Chef::Resource::RegistryKey, :unix_only do
+ before(:all) do
+ events = Chef::EventDispatch::Dispatcher.new
+ node = Chef::Node.new
+ ohai = Ohai::System.new
+ ohai.all_plugins
+ node.consume_external_attrs(ohai.data,{})
+ run_context = Chef::RunContext.new(node, {}, events)
+ @resource = Chef::Resource::RegistryKey.new("HKCU\\Software", run_context)
+ end
+ context "when load_current_resource is run on a non-windows node" do
+ it "throws an exception because you don't have a windows registry (derp)" do
+ @resource.key("HKCU\\Software\\Opscode")
+ @resource.values([{:name=>"Color", :type=>:string, :data=>"Orange"}])
+ lambda{@resource.run_action(:create)}.should raise_error(Chef::Exceptions::Win32NotWindows)
+ end
+ end
+end
+
+describe Chef::Resource::RegistryKey, :windows_only do
+
+ # parent and key must be single keys, not paths
+ let(:parent) { 'Opscode' }
+ let(:child) { 'Whatever' }
+ let(:key_parent) { "SOFTWARE\\" + parent }
+ let(:key_child) { "SOFTWARE\\" + parent + "\\" + child }
+ # must be under HKLM\SOFTWARE for WOW64 redirection to work
+ let(:reg_parent) { "HKLM\\" + key_parent }
+ let(:reg_child) { "HKLM\\" + key_child }
+ let(:hive_class) { ::Win32::Registry::HKEY_LOCAL_MACHINE }
+ let(:resource_name) { "This is the name of my Resource" }
+
+ def clean_registry
+ # clean 64-bit space on WOW64
+ begin
+ hive_class.open(key_parent, Win32::Registry::KEY_WRITE | 0x0100) do |reg|
+ reg.delete_key(child, true)
+ end
+ rescue
+ end
+ # clean 32-bit space on WOW64
+ begin
+ hive_class.open(key_parent, Win32::Registry::KEY_WRITE | 0x0200) do |reg|
+ reg.delete_key(child, true)
+ end
+ rescue
+ end
+ end
+
+ def reset_registry
+ clean_registry
+ hive_class.create(key_parent, Win32::Registry::KEY_WRITE | 0x0100)
+ hive_class.create(key_parent, Win32::Registry::KEY_WRITE | 0x0200)
+ end
+
+ def create_deletable_keys
+ # create them both 32-bit and 64-bit
+ [ 0x0100, 0x0200 ].each do |flag|
+ hive_class.create(key_parent + '\Opscode', Win32::Registry::KEY_WRITE | flag)
+ hive_class.open(key_parent + '\Opscode', Win32::Registry::KEY_ALL_ACCESS | flag) do |reg|
+ reg["Color", Win32::Registry::REG_SZ] = "Orange"
+ reg.write("Opscode", Win32::Registry::REG_MULTI_SZ, ["Seattle", "Washington"])
+ reg["AKA", Win32::Registry::REG_SZ] = "OC"
+ end
+ hive_class.create(key_parent + '\ReportKey', Win32::Registry::KEY_WRITE | flag)
+ hive_class.open(key_parent + '\ReportKey', Win32::Registry::KEY_ALL_ACCESS | flag) do |reg|
+ reg["ReportVal4", Win32::Registry::REG_SZ] = "report4"
+ reg["ReportVal5", Win32::Registry::REG_SZ] = "report5"
+ end
+ hive_class.create(key_parent + '\OpscodeWhyRun', Win32::Registry::KEY_WRITE | flag)
+ hive_class.open(key_parent + '\OpscodeWhyRun', Win32::Registry::KEY_ALL_ACCESS | flag) do |reg|
+ reg["BriskWalk", Win32::Registry::REG_SZ] = "is good for health"
+ end
+ end
+ end
+
+ before(:all) do
+ @events = Chef::EventDispatch::Dispatcher.new
+ @node = Chef::Node.new
+ ohai = Ohai::System.new
+ ohai.all_plugins
+ @node.consume_external_attrs(ohai.data,{})
+ @run_context = Chef::RunContext.new(@node, {}, @events)
+
+ @new_resource = Chef::Resource::RegistryKey.new(resource_name, @run_context)
+ @registry = Chef::Win32::Registry.new(@run_context)
+
+ @current_whyrun = Chef::Config[:why_run]
+
+ reset_registry
+ end
+
+ #Reporting setup
+ before do
+ @node.name("windowsbox")
+ @rest_client = mock("Chef::REST (mock)")
+ @rest_client.stub!(:create_url).and_return("reports/nodes/windowsbox/runs/ABC123");
+ @rest_client.stub!(:raw_http_request).and_return({"result"=>"ok"});
+ @rest_client.stub!(:post_rest).and_return({"uri"=>"https://example.com/reports/nodes/windowsbox/runs/ABC123"});
+
+ @resource_reporter = Chef::ResourceReporter.new(@rest_client)
+ @events.register(@resource_reporter)
+ @resource_reporter.node_load_completed(@node, :expanded_run_list, :config)
+
+ @new_resource.cookbook_name = "monkey"
+ @cookbook_version = mock("Cookbook::Version", :version => "1.2.3")
+ @new_resource.stub!(:cookbook_version).and_return(@cookbook_version)
+ end
+
+ after (:all) do
+ clean_registry
+ end
+
+ context "when action is create" do
+ before (:all) do
+ reset_registry
+ end
+ it "creates registry key, value if the key is missing" do
+ @new_resource.key(reg_child)
+ @new_resource.values([{:name=>"Color", :type=>:string, :data=>"Orange"}])
+ @new_resource.run_action(:create)
+
+ @registry.key_exists?(reg_child).should == true
+ @registry.data_exists?(reg_child, {:name=>"Color", :type=>:string, :data=>"Orange"}).should == true
+ end
+
+ it "does not create the key if it already exists with same value, type and data" do
+ @new_resource.key(reg_child)
+ @new_resource.values([{:name=>"Color", :type=>:string, :data=>"Orange"}])
+ @new_resource.run_action(:create)
+
+ @registry.key_exists?(reg_child).should == true
+ @registry.data_exists?(reg_child, {:name=>"Color", :type=>:string, :data=>"Orange"}).should == true
+ end
+
+ it "creates a value if it does not exist" do
+ @new_resource.key(reg_child)
+ @new_resource.values([{:name=>"Mango", :type=>:string, :data=>"Yellow"}])
+ @new_resource.run_action(:create)
+
+ @registry.data_exists?(reg_child, {:name=>"Mango", :type=>:string, :data=>"Yellow"}).should == true
+ end
+
+ it "modifies the data if the key and value exist and type matches" do
+ @new_resource.key(reg_child)
+ @new_resource.values([{:name=>"Color", :type=>:string, :data=>"Not just Orange - OpscodeOrange!"}])
+ @new_resource.run_action(:create)
+
+ @registry.data_exists?(reg_child, {:name=>"Color", :type=>:string, :data=>"Not just Orange - OpscodeOrange!"}).should == true
+ end
+
+ it "modifys the type if the key and value exist and the type does not match" do
+ @new_resource.key(reg_child)
+ @new_resource.values([{:name=>"Color", :type=>:multi_string, :data=>["Not just Orange - OpscodeOrange!"]}])
+ @new_resource.run_action(:create)
+
+ @registry.data_exists?(reg_child, {:name=>"Color", :type=>:multi_string, :data=>["Not just Orange - OpscodeOrange!"]}).should == true
+ end
+
+ it "creates subkey if parent exists" do
+ @new_resource.key(reg_child + '\OpscodeTest')
+ @new_resource.values([{:name=>"Chef", :type=>:multi_string, :data=>["OpscodeOrange", "Rules"]}])
+ @new_resource.recursive(false)
+ @new_resource.run_action(:create)
+
+ @registry.key_exists?(reg_child + '\OpscodeTest').should == true
+ @registry.value_exists?(reg_child + '\OpscodeTest', {:name=>"Chef", :type=>:multi_string, :data=>["OpscodeOrange", "Rules"]}).should == true
+ end
+
+ it "gives error if action create and parent does not exist and recursive is set to false" do
+ @new_resource.key(reg_child + '\Missing1\Missing2')
+ @new_resource.values([{:name=>"OC", :type=>:string, :data=>"MissingData"}])
+ @new_resource.recursive(false)
+ lambda{@new_resource.run_action(:create)}.should raise_error(Chef::Exceptions::Win32RegNoRecursive)
+ end
+
+ it "creates missing keys if action create and parent does not exist and recursive is set to true" do
+ @new_resource.key(reg_child + '\Missing1\Missing2')
+ @new_resource.values([{:name=>"OC", :type=>:string, :data=>"MissingData"}])
+ @new_resource.recursive(true)
+ @new_resource.run_action(:create)
+
+ @registry.key_exists?(reg_child + '\Missing1\Missing2').should == true
+ @registry.value_exists?(reg_child + '\Missing1\Missing2', {:name=>"OC", :type=>:string, :data=>"MissingData"}).should == true
+ end
+
+ it "creates key with multiple value as specified" do
+ @new_resource.key(reg_child)
+ @new_resource.values([{:name=>"one", :type=>:string, :data=>"1"},{:name=>"two", :type=>:string, :data=>"2"},{:name=>"three", :type=>:string, :data=>"3"}])
+ @new_resource.recursive(true)
+ @new_resource.run_action(:create)
+
+ @new_resource.values.each do |value|
+ @registry.value_exists?(reg_child, value).should == true
+ end
+ end
+
+ context "when running on 64-bit server", :windows64_only do
+ before(:all) do
+ reset_registry
+ end
+ after(:all) do
+ @new_resource.architecture(:machine)
+ @registry.architecture = :machine
+ end
+ it "creates a key in a 32-bit registry that is not viewable in 64-bit" do
+ @new_resource.key(reg_child + '\Atraxi' )
+ @new_resource.values([{:name=>"OC", :type=>:string, :data=>"Data"}])
+ @new_resource.recursive(true)
+ @new_resource.architecture(:i386)
+ @new_resource.run_action(:create)
+ @registry.architecture = :i386
+ @registry.data_exists?(reg_child + '\Atraxi', {:name=>"OC", :type=>:string, :data=>"Data"}).should == true
+ @registry.architecture = :x86_64
+ @registry.key_exists?(reg_child + '\Atraxi').should == false
+ end
+ end
+
+ it "prepares the reporting data for action :create" do
+ @new_resource.key(reg_child + '\Ood')
+ @new_resource.values([{:name=>"ReportingVal1", :type=>:string, :data=>"report1"},{:name=>"ReportingVal2", :type=>:string, :data=>"report2"}])
+ @new_resource.recursive(true)
+ @new_resource.run_action(:create)
+ @report = @resource_reporter.prepare_run_data
+
+ @report["action"].should == "end"
+ @report["resources"][0]["type"].should == "registry_key"
+ @report["resources"][0]["name"].should == resource_name
+ @report["resources"][0]["id"].should == reg_child + '\Ood'
+ @report["resources"][0]["after"][:values].should == [{:name=>"ReportingVal1", :type=>:string, :data=>"report1"},
+ {:name=>"ReportingVal2", :type=>:string, :data=>"report2"}]
+ @report["resources"][0]["before"][:values].should == []
+ @report["resources"][0]["result"].should == "create"
+ @report["status"].should == "success"
+ @report["total_res_count"].should == "1"
+ end
+
+ context "while running in whyrun mode" do
+ before (:all) do
+ Chef::Config[:why_run] = true
+ end
+ after (:all) do
+ Chef::Config[:why_run] = @current_whyrun
+ end
+
+ it "does not throw an exception if the keys do not exist but recursive is set to false" do
+ @new_resource.key(reg_child + '\Slitheen\Raxicoricofallapatorius')
+ @new_resource.values([{:name=>"BriskWalk",:type=>:string,:data=>"is good for health"}])
+ @new_resource.recursive(false)
+ lambda{@new_resource.run_action(:create)}.should_not raise_error
+ @registry.key_exists?(reg_child + '\Slitheen').should == false
+ @registry.key_exists?(reg_child + '\Slitheen\Raxicoricofallapatorius').should == false
+ end
+ it "does not create key if the action is create" do
+ @new_resource.key(reg_child + '\Slitheen')
+ @new_resource.values([{:name=>"BriskWalk",:type=>:string,:data=>"is good for health"}])
+ @new_resource.recursive(false)
+ @new_resource.run_action(:create)
+ @registry.key_exists?(reg_child + '\Slitheen').should == false
+ end
+ end
+ end
+
+ context "when action is create_if_missing" do
+ before (:all) do
+ reset_registry
+ end
+
+ it "creates registry key, value if the key is missing" do
+ @new_resource.key(reg_child)
+ @new_resource.values([{:name=>"Color", :type=>:string, :data=>"Orange"}])
+ @new_resource.run_action(:create_if_missing)
+
+ @registry.key_exists?(reg_parent).should == true
+ @registry.key_exists?(reg_child).should == true
+ @registry.data_exists?(reg_child, {:name=>"Color", :type=>:string, :data=>"Orange"}).should == true
+ end
+
+ it "does not create the key if it already exists with same value, type and data" do
+ @new_resource.key(reg_child)
+ @new_resource.values([{:name=>"Color", :type=>:string, :data=>"Orange"}])
+ @new_resource.run_action(:create_if_missing)
+
+ @registry.key_exists?(reg_child).should == true
+ @registry.data_exists?(reg_child, {:name=>"Color", :type=>:string, :data=>"Orange"}).should == true
+ end
+
+ it "creates a value if it does not exist" do
+ @new_resource.key(reg_child)
+ @new_resource.values([{:name=>"Mango", :type=>:string, :data=>"Yellow"}])
+ @new_resource.run_action(:create_if_missing)
+
+ @registry.data_exists?(reg_child, {:name=>"Mango", :type=>:string, :data=>"Yellow"}).should == true
+ end
+
+ it "creates subkey if parent exists" do
+ @new_resource.key(reg_child + '\Pyrovile')
+ @new_resource.values([{:name=>"Chef", :type=>:multi_string, :data=>["OpscodeOrange", "Rules"]}])
+ @new_resource.recursive(false)
+ @new_resource.run_action(:create_if_missing)
+
+ @registry.key_exists?(reg_child + '\Pyrovile').should == true
+ @registry.value_exists?(reg_child + '\Pyrovile', {:name=>"Chef", :type=>:multi_string, :data=>["OpscodeOrange", "Rules"]}).should == true
+ end
+
+ it "gives error if action create and parent does not exist and recursive is set to false" do
+ @new_resource.key(reg_child + '\Sontaran\Sontar')
+ @new_resource.values([{:name=>"OC", :type=>:string, :data=>"MissingData"}])
+ @new_resource.recursive(false)
+ lambda{@new_resource.run_action(:create_if_missing)}.should raise_error(Chef::Exceptions::Win32RegNoRecursive)
+ end
+
+ it "creates missing keys if action create and parent does not exist and recursive is set to true" do
+ @new_resource.key(reg_child + '\Sontaran\Sontar')
+ @new_resource.values([{:name=>"OC", :type=>:string, :data=>"MissingData"}])
+ @new_resource.recursive(true)
+ @new_resource.run_action(:create_if_missing)
+
+ @registry.key_exists?(reg_child + '\Sontaran\Sontar').should == true
+ @registry.value_exists?(reg_child + '\Sontaran\Sontar', {:name=>"OC", :type=>:string, :data=>"MissingData"}).should == true
+ end
+
+ it "creates key with multiple value as specified" do
+ @new_resource.key(reg_child + '\Adipose')
+ @new_resource.values([{:name=>"one", :type=>:string, :data=>"1"},{:name=>"two", :type=>:string, :data=>"2"},{:name=>"three", :type=>:string, :data=>"3"}])
+ @new_resource.recursive(true)
+ @new_resource.run_action(:create_if_missing)
+
+ @new_resource.values.each do |value|
+ @registry.value_exists?(reg_child + '\Adipose', value).should == true
+ end
+ end
+
+ it "prepares the reporting data for :create_if_missing" do
+ @new_resource.key(reg_child + '\Judoon')
+ @new_resource.values([{:name=>"ReportingVal3", :type=>:string, :data=>"report3"}])
+ @new_resource.recursive(true)
+ @new_resource.run_action(:create_if_missing)
+ @report = @resource_reporter.prepare_run_data
+
+ @report["action"].should == "end"
+ @report["resources"][0]["type"].should == "registry_key"
+ @report["resources"][0]["name"].should == resource_name
+ @report["resources"][0]["id"].should == reg_child + '\Judoon'
+ @report["resources"][0]["after"][:values].should == [{:name=>"ReportingVal3", :type=>:string, :data=>"report3"}]
+ @report["resources"][0]["before"][:values].should == []
+ @report["resources"][0]["result"].should == "create_if_missing"
+ @report["status"].should == "success"
+ @report["total_res_count"].should == "1"
+ end
+
+ context "while running in whyrun mode" do
+ before (:all) do
+ Chef::Config[:why_run] = true
+ end
+ after (:all) do
+ Chef::Config[:why_run] = @current_whyrun
+ end
+
+ it "does not throw an exception if the keys do not exist but recursive is set to false" do
+ @new_resource.key(reg_child + '\Zygons\Zygor')
+ @new_resource.values([{:name=>"BriskWalk",:type=>:string,:data=>"is good for health"}])
+ @new_resource.recursive(false)
+ lambda{@new_resource.run_action(:create_if_missing)}.should_not raise_error
+ @registry.key_exists?(reg_child + '\Zygons').should == false
+ @registry.key_exists?(reg_child + '\Zygons\Zygor').should == false
+ end
+ it "does nothing if the action is create_if_missing" do
+ @new_resource.key(reg_child + '\Zygons')
+ @new_resource.values([{:name=>"BriskWalk",:type=>:string,:data=>"is good for health"}])
+ @new_resource.recursive(false)
+ @new_resource.run_action(:create_if_missing)
+ @registry.key_exists?(reg_child + '\Zygons').should == false
+ end
+ end
+ end
+
+ context "when the action is delete" do
+ before(:all) do
+ reset_registry
+ create_deletable_keys
+ end
+
+ it "takes no action if the specified key path does not exist in the system" do
+ @registry.key_exists?(reg_parent + '\Osirian').should == false
+
+ @new_resource.key(reg_parent+ '\Osirian')
+ @new_resource.recursive(false)
+ @new_resource.run_action(:delete)
+
+ @registry.key_exists?(reg_parent + '\Osirian').should == false
+ end
+
+ it "takes no action if the key exists but the value does not" do
+ @registry.data_exists?(reg_parent + '\Opscode', {:name=>"Color", :type=>:string, :data=>"Orange"}).should == true
+
+ @new_resource.key(reg_parent + '\Opscode')
+ @new_resource.values([{:name=>"LooksLike", :type=>:multi_string, :data=>["SeattleGrey", "OCOrange"]}])
+ @new_resource.recursive(false)
+ @new_resource.run_action(:delete)
+
+ @registry.data_exists?(reg_parent + '\Opscode', {:name=>"Color", :type=>:string, :data=>"Orange"}).should == true
+ end
+
+ it "deletes only specified values under a key path" do
+ @new_resource.key(reg_parent + '\Opscode')
+ @new_resource.values([{:name=>"Opscode", :type=>:multi_string, :data=>["Seattle", "Washington"]}, {:name=>"AKA", :type=>:string, :data=>"OC"}])
+ @new_resource.recursive(false)
+ @new_resource.run_action(:delete)
+
+ @registry.data_exists?(reg_parent + '\Opscode', {:name=>"Color", :type=>:string, :data=>"Orange"}).should == true
+ @registry.value_exists?(reg_parent + '\Opscode', {:name=>"AKA", :type=>:string, :data=>"OC"}).should == false
+ @registry.value_exists?(reg_parent + '\Opscode', {:name=>"Opscode", :type=>:multi_string, :data=>["Seattle", "Washington"]}).should == false
+ end
+
+ it "it deletes the values with the same name irrespective of it type and data" do
+ @new_resource.key(reg_parent + '\Opscode')
+ @new_resource.values([{:name=>"Color", :type=>:multi_string, :data=>["Black", "Orange"]}])
+ @new_resource.recursive(false)
+ @new_resource.run_action(:delete)
+
+ @registry.value_exists?(reg_parent + '\Opscode', {:name=>"Color", :type=>:string, :data=>"Orange"}).should == false
+ end
+
+ it "prepares the reporting data for action :delete" do
+ @new_resource.key(reg_parent + '\ReportKey')
+ @new_resource.values([{:name=>"ReportVal4", :type=>:string, :data=>"report4"},{:name=>"ReportVal5", :type=>:string, :data=>"report5"}])
+ @new_resource.recursive(true)
+ @new_resource.run_action(:delete)
+
+ @report = @resource_reporter.prepare_run_data
+
+ @registry.value_exists?(reg_parent + '\ReportKey', [{:name=>"ReportVal4", :type=>:string, :data=>"report4"},{:name=>"ReportVal5", :type=>:string, :data=>"report5"}]).should == false
+
+ @report["action"].should == "end"
+ @report["resources"].count.should == 1
+ @report["resources"][0]["type"].should == "registry_key"
+ @report["resources"][0]["name"].should == resource_name
+ @report["resources"][0]["id"].should == reg_parent + '\ReportKey'
+ @report["resources"][0]["before"][:values].should == [{:name=>"ReportVal4", :type=>:string, :data=>"report4"},
+ {:name=>"ReportVal5", :type=>:string, :data=>"report5"}]
+ #Not testing for after values to match since after -> new_resource values.
+ @report["resources"][0]["result"].should == "delete"
+ @report["status"].should == "success"
+ @report["total_res_count"].should == "1"
+ end
+
+ context "while running in whyrun mode" do
+ before (:all) do
+ Chef::Config[:why_run] = true
+ end
+ after (:all) do
+ Chef::Config[:why_run] = @current_whyrun
+ end
+ it "does nothing if the action is delete" do
+ @new_resource.key(reg_parent + '\OpscodeWhyRun')
+ @new_resource.values([{:name=>"BriskWalk",:type=>:string,:data=>"is good for health"}])
+ @new_resource.recursive(false)
+ @new_resource.run_action(:delete)
+
+ @registry.key_exists?(reg_parent + '\OpscodeWhyRun').should == true
+ end
+ end
+ end
+
+ context "when the action is delete_key" do
+ before (:all) do
+ reset_registry
+ create_deletable_keys
+ end
+
+ it "takes no action if the specified key path does not exist in the system" do
+ @registry.key_exists?(reg_parent + '\Osirian').should == false
+
+ @new_resource.key(reg_parent + '\Osirian')
+ @new_resource.recursive(false)
+ @new_resource.run_action(:delete_key)
+
+ @registry.key_exists?(reg_parent + '\Osirian').should == false
+ end
+
+ it "deletes key if it has no subkeys and recursive == false" do
+ @new_resource.key(reg_parent + '\OpscodeTest')
+ @new_resource.recursive(false)
+ @new_resource.run_action(:delete_key)
+
+ @registry.key_exists?(reg_parent + '\OpscodeTest').should == false
+ end
+
+ it "raises an exception if the the key has subkeys and recursive == false" do
+ @new_resource.key(reg_parent)
+ @new_resource.recursive(false)
+ lambda{@new_resource.run_action(:delete_key)}.should raise_error(Chef::Exceptions::Win32RegNoRecursive)
+ end
+
+ it "ignores the values under a key" do
+ @new_resource.key(reg_parent + '\OpscodeIgnoredValues')
+ #@new_resource.values([{:name=>"DontExist", :type=>:string, :data=>"These will be ignored anyways"}])
+ @new_resource.recursive(true)
+ @new_resource.run_action(:delete_key)
+ end
+
+ it "deletes the key if it has subkeys and recursive == true" do
+ @new_resource.key(reg_parent + '\Opscode')
+ @new_resource.recursive(true)
+ @new_resource.run_action(:delete_key)
+
+ @registry.key_exists?(reg_parent + '\Opscode').should == false
+ end
+
+ it "prepares the reporting data for action :delete_key" do
+ @new_resource.key(reg_parent + '\ReportKey')
+ @new_resource.recursive(true)
+ @new_resource.run_action(:delete_key)
+
+ @report = @resource_reporter.prepare_run_data
+ @report["action"].should == "end"
+ @report["resources"][0]["type"].should == "registry_key"
+ @report["resources"][0]["name"].should == resource_name
+ @report["resources"][0]["id"].should == reg_parent + '\ReportKey'
+ #Not testing for before or after values to match since
+ #after -> new_resource.values and
+ #before -> current_resource.values
+ @report["resources"][0]["result"].should == "delete_key"
+ @report["status"].should == "success"
+ @report["total_res_count"].should == "1"
+ end
+ context "while running in whyrun mode" do
+ before (:all) do
+ Chef::Config[:why_run] = true
+ end
+ after (:all) do
+ Chef::Config[:why_run] = @current_whyrun
+ end
+
+ it "does not throw an exception if the key has subkeys but recursive is set to false" do
+ @new_resource.key(reg_parent + '\OpscodeWhyRun')
+ @new_resource.values([{:name=>"BriskWalk",:type=>:string,:data=>"is good for health"}])
+ @new_resource.recursive(false)
+ @new_resource.run_action(:delete_key)
+ @new_resource.should_not raise_error(ArgumentError)
+ end
+ it "does nothing if the action is delete_key" do
+ @new_resource.key(reg_parent + '\OpscodeWhyRun')
+ @new_resource.values([{:name=>"BriskWalk",:type=>:string,:data=>"is good for health"}])
+ @new_resource.recursive(false)
+ @new_resource.run_action(:delete_key)
+
+ @registry.key_exists?(reg_parent + '\OpscodeWhyRun').should == true
+ end
+ end
+ end
+end
diff --git a/spec/functional/win32/registry_helper_spec.rb b/spec/functional/win32/registry_helper_spec.rb
new file mode 100644
index 0000000000..830d6f4777
--- /dev/null
+++ b/spec/functional/win32/registry_helper_spec.rb
@@ -0,0 +1,632 @@
+#
+# Author:: Prajakta Purohit (<prajakta@opscode.com>)
+# Author:: Lamont Granquist (<lamont@opscode.com>)
+# Copyright:: Copyright (c) 2012 Opscode, Inc.
+# License:: Apache License, Version 2.0
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+require 'spec_helper'
+require 'chef/win32/registry'
+
+describe Chef::Resource::RegistryKey, :unix_only do
+ before(:all) do
+ events = Chef::EventDispatch::Dispatcher.new
+ node = Chef::Node.new
+ ohai = Ohai::System.new
+ ohai.all_plugins
+ node.consume_external_attrs(ohai.data,{})
+ run_context = Chef::RunContext.new(node, {}, events)
+ @resource = Chef::Resource::RegistryKey.new("HKCU\\Software", run_context)
+ end
+ context "when load_current_resource is run on a non-windows node" do
+ it "throws an exception because you don't have a windows registry (derp)" do
+ @resource.key("HKCU\\Software\\Opscode")
+ @resource.values([{:name=>"Color", :type=>:string, :data=>"Orange"}])
+ lambda{@resource.run_action(:create)}.should raise_error(Chef::Exceptions::Win32NotWindows)
+ end
+ end
+end
+
+describe 'Chef::Win32::Registry', :windows_only do
+
+ before(:all) do
+ #Create a registry item
+ ::Win32::Registry::HKEY_CURRENT_USER.create "Software\\Root"
+ ::Win32::Registry::HKEY_CURRENT_USER.create "Software\\Root\\Branch"
+ ::Win32::Registry::HKEY_CURRENT_USER.create "Software\\Root\\Branch\\Flower"
+ ::Win32::Registry::HKEY_CURRENT_USER.open('Software\\Root', Win32::Registry::KEY_ALL_ACCESS) do |reg|
+ reg['RootType1', Win32::Registry::REG_SZ] = 'fibrous'
+ reg.write('Roots', Win32::Registry::REG_MULTI_SZ, ["strong roots", "healthy tree"])
+ end
+ ::Win32::Registry::HKEY_CURRENT_USER.open('Software\\Root\\Branch', Win32::Registry::KEY_ALL_ACCESS) do |reg|
+ reg['Strong', Win32::Registry::REG_SZ] = 'bird nest'
+ end
+ ::Win32::Registry::HKEY_CURRENT_USER.open('Software\\Root\\Branch\\Flower', Win32::Registry::KEY_ALL_ACCESS) do |reg|
+ reg['Petals', Win32::Registry::REG_MULTI_SZ] = ["Pink", "Delicate"]
+ end
+
+ #Create the node with ohai data
+ events = Chef::EventDispatch::Dispatcher.new
+ @node = Chef::Node.new
+ ohai = Ohai::System.new
+ ohai.all_plugins
+ @node.consume_external_attrs(ohai.data,{})
+ @run_context = Chef::RunContext.new(@node, {}, events)
+
+ #Create a registry object that has access ot the node previously created
+ @registry = Chef::Win32::Registry.new(@run_context)
+ end
+
+ #Delete what is left of the registry key-values previously created
+ after(:all) do
+ ::Win32::Registry::HKEY_CURRENT_USER.open("Software") do |reg|
+ reg.delete_key("Root", true)
+ end
+ end
+
+ # Server Versions
+ # it "succeeds if server versiion is 2003R2, 2008, 2008R2, 2012" do
+ # end
+ # it "falis if the server versions are anything else" do
+ # end
+
+ describe "hive_exists?" do
+ it "returns true if the hive exists" do
+ @registry.hive_exists?("HKCU\\Software\\Root").should == true
+ end
+
+ it "returns false if the hive does not exist" do
+ hive = @registry.hive_exists?("LYRU\\Software\\Root").should == false
+ end
+ end
+
+ describe "key_exists?" do
+ it "returns true if the key path exists" do
+ @registry.key_exists?("HKCU\\Software\\Root\\Branch\\Flower").should == true
+ end
+
+ it "returns false if the key path does not exist" do
+ @registry.key_exists?("HKCU\\Software\\Branch\\Flower").should == false
+ end
+
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.key_exists?("JKLM\\Software\\Branch\\Flower")}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ end
+
+ describe "key_exists!" do
+ it "returns true if the key path exists" do
+ @registry.key_exists!("HKCU\\Software\\Root\\Branch\\Flower").should == true
+ end
+
+ it "throws an exception if the key path does not exist" do
+ lambda {@registry.key_exists!("HKCU\\Software\\Branch\\Flower")}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.key_exists!("JKLM\\Software\\Branch\\Flower")}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ end
+
+ describe "value_exists?" do
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.value_exists?("JKLM\\Software\\Branch\\Flower", {:name=>"Petals"})}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ it "throws an exception if the key does not exist" do
+ lambda {@registry.value_exists?("HKCU\\Software\\Branch\\Flower", {:name=>"Petals"})}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ it "returns true if the value exists" do
+ @registry.value_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals"}).should == true
+ end
+ it "returns false if the value does not exist" do
+ @registry.value_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"FOOBAR"}).should == false
+ end
+ end
+
+ describe "value_exists!" do
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.value_exists!("JKLM\\Software\\Branch\\Flower", {:name=>"Petals"})}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ it "throws an exception if the key does not exist" do
+ lambda {@registry.value_exists!("HKCU\\Software\\Branch\\Flower", {:name=>"Petals"})}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ it "returns true if the value exists" do
+ @registry.value_exists!("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals"}).should == true
+ end
+ it "throws an exception if the value does not exist" do
+ lambda {@registry.value_exists!("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"FOOBAR"})}.should raise_error(Chef::Exceptions::Win32RegValueMissing)
+ end
+ end
+
+ describe "data_exists?" do
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.data_exists?("JKLM\\Software\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Pink", "Delicate"]})}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ it "throws an exception if the key does not exist" do
+ lambda {@registry.data_exists?("HKCU\\Software\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Pink", "Delicate"]})}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ it "returns true if all the data matches" do
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Pink", "Delicate"]}).should == true
+ end
+ it "returns false if the name does not exist" do
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"slateP", :type=>:multi_string, :data=>["Pink", "Delicate"]}).should == false
+ end
+ it "returns false if the types do not match" do
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:string, :data=>"Pink"}).should == false
+ end
+ it "returns false if the data does not match" do
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Mauve", "Delicate"]}).should == false
+ end
+ end
+
+ describe "data_exists!" do
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.data_exists!("JKLM\\Software\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Pink", "Delicate"]})}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ it "throws an exception if the key does not exist" do
+ lambda {@registry.data_exists!("HKCU\\Software\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Pink", "Delicate"]})}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ it "returns true if all the data matches" do
+ @registry.data_exists!("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Pink", "Delicate"]}).should == true
+ end
+ it "throws an exception if the name does not exist" do
+ lambda {@registry.data_exists!("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"slateP", :type=>:multi_string, :data=>["Pink", "Delicate"]})}.should raise_error(Chef::Exceptions::Win32RegDataMissing)
+ end
+ it "throws an exception if the types do not match" do
+ lambda {@registry.data_exists!("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:string, :data=>"Pink"})}.should raise_error(Chef::Exceptions::Win32RegDataMissing)
+ end
+ it "throws an exception if the data does not match" do
+ lambda {@registry.data_exists!("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Mauve", "Delicate"]})}.should raise_error(Chef::Exceptions::Win32RegDataMissing)
+ end
+ end
+
+ describe "get_values" do
+ it "returns all values for a key if it exists" do
+ values = @registry.get_values("HKCU\\Software\\Root")
+ values.should be_an_instance_of Array
+ values.should == [{:name=>"RootType1", :type=>:string, :data=>"fibrous"},
+ {:name=>"Roots", :type=>:multi_string, :data=>["strong roots", "healthy tree"]}]
+ end
+
+ it "throws an exception if the key does not exist" do
+ lambda {@registry.get_values("HKCU\\Software\\Branch\\Flower")}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.get_values("JKLM\\Software\\Branch\\Flower")}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ end
+
+ describe "set_value" do
+ it "updates a value if the key, value exist and type matches and value different" do
+ @registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Yellow", "Changed Color"]}).should == true
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Yellow", "Changed Color"]}).should == true
+ end
+
+ it "updates a value if the type does match and the values are different" do
+ @registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:string, :data=>"Yellow"}).should == true
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:string, :data=>"Yellow"}).should == true
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Yellow", "Changed Color"]}).should == false
+ end
+
+ it "creates a value if key exists and value does not" do
+ @registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Stamen", :type=>:multi_string, :data=>["Yellow", "Changed Color"]}).should == true
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Stamen", :type=>:multi_string, :data=>["Yellow", "Changed Color"]}).should == true
+ end
+
+ it "does nothing if data,type and name parameters for the value are same" do
+ @registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Stamen", :type=>:multi_string, :data=>["Yellow", "Changed Color"]}).should == false
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"Stamen", :type=>:multi_string, :data=>["Yellow", "Changed Color"]}).should == true
+ end
+
+ it "throws an exception if the key does not exist" do
+ lambda {@registry.set_value("HKCU\\Software\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Yellow", "Changed Color"]})}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.set_value("JKLM\\Software\\Root\\Branch\\Flower", {:name=>"Petals", :type=>:multi_string, :data=>["Yellow", "Changed Color"]})}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+
+ # we are validating that the data gets .to_i called on it when type is a :dword
+
+ it "casts an integer string given as a dword into an integer" do
+ @registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldBe32767", :type=>:dword, :data=>"32767"}).should == true
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldBe32767", :type=>:dword, :data=>32767}).should == true
+ end
+
+ it "casts a nonsense string given as a dword into zero" do
+ @registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldBeZero", :type=>:dword, :data=>"whatdoesthisdo"}).should == true
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldBeZero", :type=>:dword, :data=>0}).should == true
+ end
+
+ it "throws an exception when trying to cast an array to an int for a dword" do
+ lambda {@registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldThrow", :type=>:dword, :data=>["one","two"]})}.should raise_error
+ end
+
+ # we are validating that the data gets .to_s called on it when type is a :string
+
+ it "stores the string representation of an array into a string if you pass it an array" do
+ @registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldBePainful", :type=>:string, :data=>["one","two"]}).should == true
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldBePainful", :type=>:string, :data=>'["one", "two"]'}).should == true
+ end
+
+ it "stores the string representation of a number into a string if you pass it an number" do
+ @registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldBe65535", :type=>:string, :data=>65535}).should == true
+ @registry.data_exists?("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldBe65535", :type=>:string, :data=>"65535"}).should == true
+ end
+
+ # we are validating that the data gets .to_a called on it when type is a :multi_string
+
+ it "throws an exception when a multi-string is passed a number" do
+ lambda {@registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldThrow", :type=>:multi_string, :data=>65535})}.should raise_error
+ end
+
+ it "throws an exception when a multi-string is passed a string" do
+ lambda {@registry.set_value("HKCU\\Software\\Root\\Branch\\Flower", {:name=>"ShouldBeWat", :type=>:multi_string, :data=>"foo"})}.should raise_error
+ end
+ end
+
+ describe "create_key" do
+ before(:all) do
+ ::Win32::Registry::HKEY_CURRENT_USER.open("Software\\Root") do |reg|
+ begin
+ reg.delete_key("Trunk", true)
+ rescue
+ end
+ end
+ end
+
+ it "throws an exception if the path has missing keys but recursive set to false" do
+ lambda {@registry.create_key("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker", false)}.should raise_error(Chef::Exceptions::Win32RegNoRecursive)
+ @registry.key_exists?("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker").should == false
+ end
+
+ it "creates the key_path if the keys were missing but recursive was set to true" do
+ @registry.create_key("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker", true).should == true
+ @registry.key_exists?("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker").should == true
+ end
+
+ it "does nothing if the key already exists" do
+ @registry.create_key("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker", false).should == true
+ @registry.key_exists?("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker").should == true
+ end
+
+ it "throws an exception of the hive does not exist" do
+ lambda {@registry.create_key("JKLM\\Software\\Root\\Trunk\\Peck\\Woodpecker", false)}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ end
+
+ describe "delete_value" do
+ before(:all) do
+ ::Win32::Registry::HKEY_CURRENT_USER.create "Software\\Root\\Trunk\\Peck\\Woodpecker"
+ ::Win32::Registry::HKEY_CURRENT_USER.open('Software\\Root\\Trunk\\Peck\\Woodpecker', Win32::Registry::KEY_ALL_ACCESS) do |reg|
+ reg['Peter', Win32::Registry::REG_SZ] = 'Tiny'
+ end
+ end
+
+ it "deletes values if the value exists" do
+ @registry.delete_value("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker", {:name=>"Peter", :type=>:string, :data=>"Tiny"}).should == true
+ @registry.value_exists?("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker", {:name=>"Peter", :type=>:string, :data=>"Tiny"}).should == false
+ end
+
+ it "does nothing if value does not exist" do
+ @registry.delete_value("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker", {:name=>"Peter", :type=>:string, :data=>"Tiny"}).should == true
+ @registry.value_exists?("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker", {:name=>"Peter", :type=>:string, :data=>"Tiny"}).should == false
+ end
+
+ it "throws an exception if the key does not exist?" do
+ lambda {@registry.delete_value("HKCU\\Software\\Trunk\\Peck\\Woodpecker", {:name=>"Peter", :type=>:string, :data=>"Tiny"})}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.delete_value("JKLM\\Software\\Root\\Trunk\\Peck\\Woodpecker", {:name=>"Peter", :type=>:string, :data=>"Tiny"})}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ end
+
+ describe "delete_key" do
+ before (:all) do
+ ::Win32::Registry::HKEY_CURRENT_USER.create "Software\\Root\\Branch\\Fruit"
+ ::Win32::Registry::HKEY_CURRENT_USER.open('Software\\Root\\Branch\\Fruit', Win32::Registry::KEY_ALL_ACCESS) do |reg|
+ reg['Apple', Win32::Registry::REG_MULTI_SZ] = ["Red", "Juicy"]
+ end
+ ::Win32::Registry::HKEY_CURRENT_USER.create "Software\\Root\\Trunk\\Peck\\Woodpecker"
+ ::Win32::Registry::HKEY_CURRENT_USER.open('Software\\Root\\Trunk\\Peck\\Woodpecker', Win32::Registry::KEY_ALL_ACCESS) do |reg|
+ reg['Peter', Win32::Registry::REG_SZ] = 'Tiny'
+ end
+ end
+
+ it "deletes a key if it has no subkeys" do
+ @registry.delete_key("HKCU\\Software\\Root\\Branch\\Fruit", false).should == true
+ @registry.key_exists?("HKCU\\Software\\Root\\Branch\\Fruit").should == false
+ end
+
+ it "throws an exception if key to delete has subkeys and recursive is false" do
+ lambda { @registry.delete_key("HKCU\\Software\\Root\\Trunk", false) }.should raise_error(Chef::Exceptions::Win32RegNoRecursive)
+ @registry.key_exists?("HKCU\\Software\\Root\\Trunk\\Peck\\Woodpecker").should == true
+ end
+
+ it "deletes a key if it has subkeys and recursive true" do
+ @registry.delete_key("HKCU\\Software\\Root\\Trunk", true).should == true
+ @registry.key_exists?("HKCU\\Software\\Root\\Trunk").should == false
+ end
+
+ it "does nothing if the key does not exist" do
+ @registry.delete_key("HKCU\\Software\\Root\\Trunk", true).should == true
+ @registry.key_exists?("HKCU\\Software\\Root\\Trunk").should == false
+ end
+
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.delete_key("JKLM\\Software\\Root\\Branch\\Flower", false)}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ end
+
+ describe "has_subkeys?" do
+ before(:all) do
+ ::Win32::Registry::HKEY_CURRENT_USER.create "Software\\Root\\Trunk"
+ ::Win32::Registry::HKEY_CURRENT_USER.open("Software\\Root\\Trunk") do |reg|
+ begin
+ reg.delete_key("Red", true)
+ rescue
+ end
+ end
+ end
+
+ it "throws an exception if the hive was missing" do
+ lambda {@registry.has_subkeys?("LMNO\\Software\\Root")}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+
+ it "throws an exception if the key is missing" do
+ lambda {@registry.has_subkeys?("HKCU\\Software\\Root\\Trunk\\Red")}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+
+ it "returns true if the key has subkeys" do
+ @registry.has_subkeys?("HKCU\\Software\\Root").should == true
+ end
+
+ it "returns false if the key has no subkeys" do
+ ::Win32::Registry::HKEY_CURRENT_USER.create "Software\\Root\\Trunk\\Red"
+ @registry.has_subkeys?("HKCU\\Software\\Root\\Trunk\\Red").should == false
+ end
+ end
+
+ describe "get_subkeys" do
+ it "throws an exception if the key is missing" do
+ lambda {@registry.get_subkeys("HKCU\\Software\\Trunk\\Red")}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ it "throws an exception if the hive does not exist" do
+ lambda {@registry.get_subkeys("JKLM\\Software\\Root")}.should raise_error(Chef::Exceptions::Win32RegHiveMissing)
+ end
+ it "returns the array of subkeys for a given key" do
+ subkeys = @registry.get_subkeys("HKCU\\Software\\Root")
+ reg_subkeys = []
+ ::Win32::Registry::HKEY_CURRENT_USER.open("Software\\Root", Win32::Registry::KEY_ALL_ACCESS) do |reg|
+ reg.each_key{|name| reg_subkeys << name}
+ end
+ reg_subkeys.should == subkeys
+ end
+ end
+
+ describe "architecture" do
+ describe "on 32-bit" do
+ before(:all) do
+ @saved_kernel_machine = @node.automatic_attrs[:kernel][:machine]
+ @node.automatic_attrs[:kernel][:machine] = :i386
+ end
+
+ after(:all) do
+ @node.automatic_attrs[:kernel][:machine] = @saved_kernel_machine
+ end
+
+ context "registry constructor" do
+ it "throws an exception if requested architecture is 64bit but running on 32bit" do
+ lambda {Chef::Win32::Registry.new(@run_context, :x86_64)}.should raise_error(Chef::Exceptions::Win32RegArchitectureIncorrect)
+ end
+
+ it "can correctly set the requested architecture to 32-bit" do
+ @r = Chef::Win32::Registry.new(@run_context, :i386)
+ @r.architecture.should == :i386
+ @r.registry_system_architecture.should == 0x0200
+ end
+
+ it "can correctly set the requested architecture to :machine" do
+ @r = Chef::Win32::Registry.new(@run_context, :machine)
+ @r.architecture.should == :machine
+ @r.registry_system_architecture.should == 0x0200
+ end
+ end
+
+ context "architecture setter" do
+ it "throws an exception if requested architecture is 64bit but running on 32bit" do
+ lambda {@registry.architecture = :x86_64}.should raise_error(Chef::Exceptions::Win32RegArchitectureIncorrect)
+ end
+
+ it "sets the requested architecture to :machine if passed :machine" do
+ @registry.architecture = :machine
+ @registry.architecture.should == :machine
+ @registry.registry_system_architecture.should == 0x0200
+ end
+
+ it "sets the requested architecture to 32-bit if passed i386 as a string" do
+ @registry.architecture = :i386
+ @registry.architecture.should == :i386
+ @registry.registry_system_architecture.should == 0x0200
+ end
+ end
+ end
+
+ describe "on 64-bit" do
+ before(:all) do
+ @saved_kernel_machine = @node.automatic_attrs[:kernel][:machine]
+ @node.automatic_attrs[:kernel][:machine] = :x86_64
+ end
+
+ after(:all) do
+ @node.automatic_attrs[:kernel][:machine] = @saved_kernel_machine
+ end
+
+ context "registry constructor" do
+ it "can correctly set the requested architecture to 32-bit" do
+ @r = Chef::Win32::Registry.new(@run_context, :i386)
+ @r.architecture.should == :i386
+ @r.registry_system_architecture.should == 0x0200
+ end
+
+ it "can correctly set the requested architecture to 64-bit" do
+ @r = Chef::Win32::Registry.new(@run_context, :x86_64)
+ @r.architecture.should == :x86_64
+ @r.registry_system_architecture.should == 0x0100
+ end
+
+ it "can correctly set the requested architecture to :machine" do
+ @r = Chef::Win32::Registry.new(@run_context, :machine)
+ @r.architecture.should == :machine
+ @r.registry_system_architecture.should == 0x0100
+ end
+ end
+
+ context "architecture setter" do
+ it "sets the requested architecture to 64-bit if passed 64-bit" do
+ @registry.architecture = :x86_64
+ @registry.architecture.should == :x86_64
+ @registry.registry_system_architecture.should == 0x0100
+ end
+
+ it "sets the requested architecture to :machine if passed :machine" do
+ @registry.architecture = :machine
+ @registry.architecture.should == :machine
+ @registry.registry_system_architecture.should == 0x0100
+ end
+
+ it "sets the requested architecture to 32-bit if passed 32-bit" do
+ @registry.architecture = :i386
+ @registry.architecture.should == :i386
+ @registry.registry_system_architecture.should == 0x0200
+ end
+ end
+ end
+
+ describe "when running on an actual 64-bit server", :windows64_only do
+ before(:all) do
+ begin
+ ::Win32::Registry::HKEY_LOCAL_MACHINE.open("Software\\Root", ::Win32::Registry::KEY_ALL_ACCESS | 0x0100) do |reg|
+ reg.delete_key("Trunk", true)
+ end
+ rescue
+ end
+ begin
+ ::Win32::Registry::HKEY_LOCAL_MACHINE.open("Software\\Root", ::Win32::Registry::KEY_ALL_ACCESS | 0x0200) do |reg|
+ reg.delete_key("Trunk", true)
+ end
+ rescue
+ end
+ # 64-bit
+ ::Win32::Registry::HKEY_LOCAL_MACHINE.create("Software\\Root\\Mauve", ::Win32::Registry::KEY_ALL_ACCESS | 0x0100)
+ ::Win32::Registry::HKEY_LOCAL_MACHINE.open('Software\\Root\\Mauve', Win32::Registry::KEY_ALL_ACCESS | 0x0100) do |reg|
+ reg['Alert', Win32::Registry::REG_SZ] = 'Universal'
+ end
+ # 32-bit
+ ::Win32::Registry::HKEY_LOCAL_MACHINE.create("Software\\Root\\Poosh", ::Win32::Registry::KEY_ALL_ACCESS | 0x0200)
+ ::Win32::Registry::HKEY_LOCAL_MACHINE.open('Software\\Root\\Poosh', Win32::Registry::KEY_ALL_ACCESS | 0x0200) do |reg|
+ reg['Status', Win32::Registry::REG_SZ] = 'Lost'
+ end
+ end
+
+ after(:all) do
+ ::Win32::Registry::HKEY_LOCAL_MACHINE.open("Software\\Root", ::Win32::Registry::KEY_ALL_ACCESS | 0x0100) do |reg|
+ reg.delete_key("Trunk", true)
+ end
+ ::Win32::Registry::HKEY_LOCAL_MACHINE.open("Software\\Root", ::Win32::Registry::KEY_ALL_ACCESS | 0x0200) do |reg|
+ reg.delete_key("Trunk", true)
+ end
+ end
+
+ describe "key_exists?" do
+ it "does not find 64-bit keys in the 32-bit registry" do
+ @registry.architecture=:i386
+ @registry.key_exists?("HKLM\\Software\\Root\\Mauve").should == false
+ end
+ it "finds 32-bit keys in the 32-bit registry" do
+ @registry.architecture=:i386
+ @registry.key_exists?("HKLM\\Software\\Root\\Poosh").should == true
+ end
+ it "does not find 32-bit keys in the 64-bit registry" do
+ @registry.architecture=:x86_64
+ @registry.key_exists?("HKLM\\Software\\Root\\Mauve").should == true
+ end
+ it "finds 64-bit keys in the 64-bit registry" do
+ @registry.architecture=:x86_64
+ @registry.key_exists?("HKLM\\Software\\Root\\Poosh").should == false
+ end
+ end
+
+ describe "value_exists?" do
+ it "does not find 64-bit values in the 32-bit registry" do
+ @registry.architecture=:i386
+ lambda{@registry.value_exists?("HKLM\\Software\\Root\\Mauve", {:name=>"Alert"})}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ it "finds 32-bit values in the 32-bit registry" do
+ @registry.architecture=:i386
+ @registry.value_exists?("HKLM\\Software\\Root\\Poosh", {:name=>"Status"}).should == true
+ end
+ it "does not find 32-bit values in the 64-bit registry" do
+ @registry.architecture=:x86_64
+ @registry.value_exists?("HKLM\\Software\\Root\\Mauve", {:name=>"Alert"}).should == true
+ end
+ it "finds 64-bit values in the 64-bit registry" do
+ @registry.architecture=:x86_64
+ lambda{@registry.value_exists?("HKLM\\Software\\Root\\Poosh", {:name=>"Status"})}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ end
+
+ describe "data_exists?" do
+ it "does not find 64-bit keys in the 32-bit registry" do
+ @registry.architecture=:i386
+ lambda{@registry.data_exists?("HKLM\\Software\\Root\\Mauve", {:name=>"Alert", :type=>:string, :data=>"Universal"})}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ it "finds 32-bit keys in the 32-bit registry" do
+ @registry.architecture=:i386
+ @registry.data_exists?("HKLM\\Software\\Root\\Poosh", {:name=>"Status", :type=>:string, :data=>"Lost"}).should == true
+ end
+ it "does not find 32-bit keys in the 64-bit registry" do
+ @registry.architecture=:x86_64
+ @registry.data_exists?("HKLM\\Software\\Root\\Mauve", {:name=>"Alert", :type=>:string, :data=>"Universal"}).should == true
+ end
+ it "finds 64-bit keys in the 64-bit registry" do
+ @registry.architecture=:x86_64
+ lambda{@registry.data_exists?("HKLM\\Software\\Root\\Poosh", {:name=>"Status", :type=>:string, :data=>"Lost"})}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ end
+
+ describe "create_key" do
+ it "can create a 32-bit only registry key" do
+ @registry.architecture = :i386
+ @registry.create_key("HKLM\\Software\\Root\\Trunk\\Red", true).should == true
+ @registry.key_exists?("HKLM\\Software\\Root\\Trunk\\Red").should == true
+ @registry.architecture = :x86_64
+ @registry.key_exists?("HKLM\\Software\\Root\\Trunk\\Red").should == false
+ end
+
+ it "can create a 64-bit only registry key" do
+ @registry.architecture = :x86_64
+ @registry.create_key("HKLM\\Software\\Root\\Trunk\\Blue", true).should == true
+ @registry.key_exists?("HKLM\\Software\\Root\\Trunk\\Blue").should == true
+ @registry.architecture = :i386
+ @registry.key_exists?("HKLM\\Software\\Root\\Trunk\\Blue").should == false
+ end
+ end
+
+ end
+ end
+end
diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb
index 8d64997d50..3a5ca22868 100644
--- a/spec/spec_helper.rb
+++ b/spec/spec_helper.rb
@@ -71,6 +71,8 @@ RSpec.configure do |config|
# Add jruby filters here
config.filter_run_excluding :windows_only => true unless windows?
config.filter_run_excluding :not_supported_on_win2k3 => true if windows_win2k3?
+ config.filter_run_excluding :windows64_only => true unless windows64?
+ config.filter_run_excluding :windows32_only => true unless windows32?
config.filter_run_excluding :unix_only => true unless unix?
config.filter_run_excluding :ruby_18_only => true unless ruby_18?
config.filter_run_excluding :ruby_19_only => true unless ruby_19?
diff --git a/spec/support/platform_helpers.rb b/spec/support/platform_helpers.rb
index fcb825319c..df94d991cc 100644
--- a/spec/support/platform_helpers.rb
+++ b/spec/support/platform_helpers.rb
@@ -18,6 +18,16 @@ def windows_win2k3?
(host.version && host.version.start_with?("5.2"))
end
+# detects if the hardware is 64-bit (evaluates to true in "WOW64" mode in a 32-bit app on a 64-bit system)
+def windows64?
+ windows? && ( ENV['PROCESSOR_ARCHITECTURE'] == 'AMD64' || ENV['PROCESSOR_ARCHITEW6432'] == 'AMD64' )
+end
+
+# detects if the hardware is 32-bit
+def windows32?
+ windows? && !windows64?
+end
+
# def jruby?
def unix?
diff --git a/spec/unit/dsl/regsitry_helper_spec.rb b/spec/unit/dsl/regsitry_helper_spec.rb
new file mode 100644
index 0000000000..7fe08e310f
--- /dev/null
+++ b/spec/unit/dsl/regsitry_helper_spec.rb
@@ -0,0 +1,55 @@
+#
+# Author:: Prajakta Purohit (<prajakta@opscode.com>)
+# Copyright:: Copyright (c) 2011 Opscode, Inc.
+# License:: Apache License, Version 2.0
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+require "chef/dsl/registry_helper"
+require "spec_helper"
+
+describe Chef::Resource::RegistryKey do
+
+ before (:all) do
+ events = Chef::EventDispatch::Dispatcher.new
+ node = Chef::Node.new
+ ohai = Ohai::System.new
+ ohai.all_plugins
+ node.consume_external_attrs(ohai.data,{})
+ run_context = Chef::RunContext.new(node, {}, events)
+ @resource = Chef::Resource::new("foo", run_context)
+ end
+
+ context "tests registry dsl" do
+ it "resource can access registry_helper method registry_key_exists" do
+ @resource.respond_to?('registry_key_exists?').should == true
+ end
+ it "resource can access registry_helper method registry_get_values" do
+ @resource.respond_to?('registry_get_values').should == true
+ end
+ it "resource can access registry_helper method registry_has_subkey" do
+ @resource.respond_to?('registry_has_subkeys?').should == true
+ end
+ it "resource can access registry_helper method registry_get_subkeys" do
+ @resource.respond_to?('registry_get_subkeys').should == true
+ end
+ it "resource can access registry_helper method registry_value_exists" do
+ @resource.respond_to?('registry_value_exists?').should == true
+ end
+ it "resource can access registry_helper method data_value_exists" do
+ @resource.respond_to?('registry_data_exists?').should == true
+ end
+ end
+end
+
diff --git a/spec/unit/provider/registry_key_spec.rb b/spec/unit/provider/registry_key_spec.rb
new file mode 100644
index 0000000000..2f6f8179e6
--- /dev/null
+++ b/spec/unit/provider/registry_key_spec.rb
@@ -0,0 +1,269 @@
+#
+# Author:: Lamont Granquist (lamont@opscode.com)
+# Copyright:: Copyright (c) 2012 Opscode, Inc.
+# License:: Apache License, Version 2.0
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+require 'spec_helper'
+
+describe Chef::Provider::RegistryKey do
+
+ let(:testval1) { { :name => "one", :type => :string, :data => "1" } }
+ let(:testval1_wrong_type) { { :name => "one", :type => :multi_string, :data => "1" } }
+ let(:testval1_wrong_data) { { :name => "one", :type => :string, :data => "2" } }
+ let(:testval2) { { :name => "two", :type => :string, :data => "2" } }
+ let(:testkey1) { 'HKLM\Software\Opscode\Testing' }
+
+ before(:each) do
+ @node = Chef::Node.new
+ @events = Chef::EventDispatch::Dispatcher.new
+ @run_context = Chef::RunContext.new(@node, {}, @events)
+
+ @new_resource = Chef::Resource::RegistryKey.new("windows is fun", @run_context)
+ @new_resource.key testkey1
+ @new_resource.values( testval1 )
+ @new_resource.recursive false
+
+ @provider = Chef::Provider::RegistryKey.new(@new_resource, @run_context)
+
+ @provider.stub!(:running_on_windows!).and_return(true)
+ @double_registry = double(Chef::Win32::Registry)
+ @provider.stub!(:registry).and_return(@double_registry)
+ end
+
+ describe "when first created" do
+ end
+
+ describe "executing load_current_resource" do
+ describe "when the key exists" do
+ before(:each) do
+ @double_registry.should_receive(:key_exists?).with(testkey1).and_return(true)
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval2 )
+ @provider.load_current_resource
+ end
+
+ it "should set the key of the current resource to the key of the new resource" do
+ @provider.current_resource.key.should == @new_resource.key
+ end
+
+ it "should set the architecture of the current resource to the architecture of the new resource" do
+ @provider.current_resource.architecture.should == @new_resource.architecture
+ end
+
+ it "should set the recursive flag of the current resource to the recursive flag of the new resource" do
+ @provider.current_resource.recursive.should == @new_resource.recursive
+ end
+
+ it "should set the values of the current resource to the values it got from the registry" do
+ @provider.current_resource.values.should == [ testval2 ]
+ end
+ end
+
+ describe "when the key does not exist" do
+ before(:each) do
+ @double_registry.should_receive(:key_exists?).with(testkey1).and_return(false)
+ @provider.load_current_resource
+ end
+
+ it "should set the values in the current resource to empty array" do
+ @provider.current_resource.values.should == []
+ end
+ end
+ end
+
+ describe "action_create" do
+ context "when the key exists" do
+ before(:each) do
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(true)
+ end
+ it "should do nothing if the key and the value both exist" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1 )
+ @double_registry.should_not_receive(:set_value)
+ @provider.load_current_resource
+ @provider.action_create
+ end
+ it "should create the value if the key exists but the value does not" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval2 )
+ @double_registry.should_receive(:set_value).with(testkey1, testval1)
+ @provider.load_current_resource
+ @provider.action_create
+ end
+ it "should set the value if the key exists but the data does not match" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1_wrong_data )
+ @double_registry.should_receive(:set_value).with(testkey1, testval1)
+ @provider.load_current_resource
+ @provider.action_create
+ end
+ it "should set the value if the key exists but the type does not match" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1_wrong_type )
+ @double_registry.should_receive(:set_value).with(testkey1, testval1)
+ @provider.load_current_resource
+ @provider.action_create
+ end
+ end
+ context "when the key exists and the values in the new resource are empty" do
+ it "when a value is in the key, it should do nothing" do
+ @provider.new_resource.values([])
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(true)
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1 )
+ @double_registry.should_not_receive(:create_key)
+ @double_registry.should_not_receive(:set_value)
+ @provider.load_current_resource
+ @provider.action_create
+ end
+ it "when no value is in the key, it should do nothing" do
+ @provider.new_resource.values([])
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(true)
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( nil )
+ @double_registry.should_not_receive(:create_key)
+ @double_registry.should_not_receive(:set_value)
+ @provider.load_current_resource
+ @provider.action_create
+ end
+ end
+ context "when the key does not exist" do
+ before(:each) do
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(false)
+ end
+ it "should create the key and the value" do
+ @double_registry.should_receive(:create_key).with(testkey1, false)
+ @double_registry.should_receive(:set_value).with(testkey1, testval1)
+ @provider.load_current_resource
+ @provider.action_create
+ end
+ end
+ context "when the key does not exist and the values in the new resource are empty" do
+ it "should create the key" do
+ @new_resource.values([])
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(false)
+ @double_registry.should_receive(:create_key).with(testkey1, false)
+ @double_registry.should_not_receive(:set_value)
+ @provider.load_current_resource
+ @provider.action_create
+ end
+ end
+ end
+
+ describe "action_create_if_missing" do
+ context "when the key exists" do
+ before(:each) do
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(true)
+ end
+ it "should do nothing if the key and the value both exist" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1 )
+ @double_registry.should_not_receive(:set_value)
+ @provider.load_current_resource
+ @provider.action_create_if_missing
+ end
+ it "should create the value if the key exists but the value does not" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval2 )
+ @double_registry.should_receive(:set_value).with(testkey1, testval1)
+ @provider.load_current_resource
+ @provider.action_create_if_missing
+ end
+ it "should not set the value if the key exists but the data does not match" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1_wrong_data )
+ @double_registry.should_not_receive(:set_value)
+ @provider.load_current_resource
+ @provider.action_create_if_missing
+ end
+ it "should not set the value if the key exists but the type does not match" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1_wrong_type )
+ @double_registry.should_not_receive(:set_value)
+ @provider.load_current_resource
+ @provider.action_create_if_missing
+ end
+ end
+ context "when the key does not exist" do
+ before(:each) do
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(false)
+ end
+ it "should create the key and the value" do
+ @double_registry.should_receive(:create_key).with(testkey1, false)
+ @double_registry.should_receive(:set_value).with(testkey1, testval1)
+ @provider.load_current_resource
+ @provider.action_create_if_missing
+ end
+ end
+ end
+
+ describe "action_delete" do
+ context "when the key exists" do
+ before(:each) do
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(true)
+ end
+ it "deletes the value when the value exists" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1 )
+ @double_registry.should_receive(:delete_value).with(testkey1, testval1)
+ @provider.load_current_resource
+ @provider.action_delete
+ end
+ it "deletes the value when the value exists, but the type is wrong" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1_wrong_type )
+ @double_registry.should_receive(:delete_value).with(testkey1, testval1)
+ @provider.load_current_resource
+ @provider.action_delete
+ end
+ it "deletes the value when the value exists, but the data is wrong" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1_wrong_data )
+ @double_registry.should_receive(:delete_value).with(testkey1, testval1)
+ @provider.load_current_resource
+ @provider.action_delete
+ end
+ it "does not delete the value when the value does not exist" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval2 )
+ @double_registry.should_not_receive(:delete_value)
+ @provider.load_current_resource
+ @provider.action_delete
+ end
+ end
+ context "when the key does not exist" do
+ before(:each) do
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(false)
+ end
+ it "does nothing" do
+ @double_registry.should_not_receive(:delete_value)
+ @provider.load_current_resource
+ @provider.action_delete
+ end
+ end
+ end
+
+ describe "action_delete_key" do
+ context "when the key exists" do
+ before(:each) do
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(true)
+ end
+ it "deletes the key" do
+ @double_registry.should_receive(:get_values).with(testkey1).and_return( testval1 )
+ @double_registry.should_receive(:delete_key).with(testkey1, false)
+ @provider.load_current_resource
+ @provider.action_delete_key
+ end
+ end
+ context "when the key does not exist" do
+ before(:each) do
+ @double_registry.should_receive(:key_exists?).twice.with(testkey1).and_return(false)
+ end
+ it "does nothing" do
+ @double_registry.should_not_receive(:delete_key)
+ @provider.load_current_resource
+ @provider.action_delete_key
+ end
+ end
+ end
+
+end
+
diff --git a/spec/unit/registry_helper_spec.rb b/spec/unit/registry_helper_spec.rb
new file mode 100644
index 0000000000..a83895ba20
--- /dev/null
+++ b/spec/unit/registry_helper_spec.rb
@@ -0,0 +1,374 @@
+#
+# Author:: Prajakta Purohit (prajakta@opscode.com)
+# Copyright:: Copyright (c) 2012 Opscode, Inc.
+# License:: Apache License, Version 2.0
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+require 'spec_helper'
+
+describe Chef::Provider::RegistryKey do
+
+ let(:value1) { { :name => "one", :type => :string, :data => "1" } }
+ let(:key_path) { 'HKCU\Software\OpscodeNumbers' }
+ let(:key) { 'Software\OpscodeNumbers' }
+ let(:key_parent) { 'Software' }
+ let(:key_to_delete) { 'OpscodeNumbers' }
+ let(:sub_key) {'OpscodePrimes'}
+ let(:missing_key_path) {'HKCU\Software'}
+
+ before(:each) do
+ Chef::Win32::Registry.any_instance.stub(:machine_architecture).and_return(:x86_64)
+ @registry = Chef::Win32::Registry.new()
+
+ #Making the values for registry constants available on unix
+ Object.send(:remove_const, 'Win32') if defined?(Win32)
+ Win32 = Module.new
+ Win32::Registry = Class.new
+ Win32::Registry::KEY_SET_VALUE = 0x0002
+ Win32::Registry::KEY_QUERY_VALUE = 0x0001
+ Win32::Registry::KEY_WRITE = 0x00020000 | 0x0002 | 0x0004
+ Win32::Registry::KEY_READ = 0x00020000 | 0x0001 | 0x0008 | 0x0010
+
+ Win32::Registry::Error = Class.new(RuntimeError)
+
+ @hive_mock = mock("::Win32::Registry::HKEY_CURRENT_USER")
+ @reg_mock = mock("reg")
+ end
+
+ describe "get_values" do
+ it "gets all values for a key if the key exists" do
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @reg_mock.should_receive(:map)
+ @registry.get_values(key_path)
+ end
+
+ it "throws an exception if key does not exist" do
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @registry.should_receive(:key_exists!).with(key_path).and_raise(Chef::Exceptions::Win32RegKeyMissing)
+ lambda{@registry.get_values(key_path)}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ end
+
+ describe "set_value" do
+ it "does nothing if key and hive and value exist" do
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @registry.should_receive(:value_exists?).with(key_path, value1).and_return(true)
+ @registry.should_receive(:data_exists?).with(key_path, value1).and_return(true)
+ @registry.set_value(key_path, value1)
+ end
+
+ it "updates value if key and hive and value exist, but data is different" do
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @registry.should_receive(:value_exists?).with(key_path, value1).and_return(true)
+ @registry.should_receive(:data_exists?).with(key_path, value1).and_return(false)
+ @hive_mock.should_receive(:open).with(key, Win32::Registry::KEY_SET_VALUE | ::Win32::Registry::KEY_QUERY_VALUE | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @registry.should_receive(:get_type_from_name).with(:string).and_return(1)
+ @reg_mock.should_receive(:write).with("one", 1, "1")
+ @registry.set_value(key_path, value1)
+ end
+
+ it "creates value if the key exists and the value does not exist" do
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @registry.should_receive(:value_exists?).with(key_path, value1).and_return(false)
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_SET_VALUE | ::Win32::Registry::KEY_QUERY_VALUE | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @registry.should_receive(:get_type_from_name).with(:string).and_return(1)
+ @reg_mock.should_receive(:write).with("one", 1, "1")
+ @registry.set_value(key_path, value1)
+ end
+
+ it "should raise an exception if the key does not exist" do
+ @registry.should_receive(:key_exists!).with(key_path).and_raise(Chef::Exceptions::Win32RegKeyMissing)
+ lambda {@registry.set_value(key_path, value1)}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ end
+
+ describe "delete_value" do
+ it "deletes value if value exists" do
+ @registry.should_receive(:value_exists?).with(key_path, value1).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_SET_VALUE | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @reg_mock.should_receive(:delete_value).with("one").and_return(true)
+ @registry.delete_value(key_path, value1)
+ end
+
+ it "raises an exception if the key does not exist" do
+ @registry.should_receive(:value_exists?).with(key_path, value1).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_raise(Chef::Exceptions::Win32RegKeyMissing)
+ @registry.delete_value(key_path, value1)
+ end
+
+ it "does nothing if the value does not exist" do
+ @registry.should_receive(:value_exists?).with(key_path, value1).and_return(false)
+ @registry.delete_value(key_path, value1)
+ end
+ end
+
+ describe "create_key" do
+ it "creates key if intermediate keys are missing and recursive is set to true" do
+ @registry.should_receive(:keys_missing?).with(key_path).and_return(true)
+ @registry.should_receive(:create_missing).with(key_path)
+ @registry.should_receive(:key_exists?).with(key_path).and_return(false)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:create).with(key, ::Win32::Registry::KEY_WRITE | @registry.registry_system_architecture)
+ @registry.create_key(key_path, true)
+ end
+
+ it "raises an exception if intermediate keys are missing and recursive is set to false" do
+ @registry.should_receive(:keys_missing?).with(key_path).and_return(true)
+ lambda{@registry.create_key(key_path, false)}.should raise_error(Chef::Exceptions::Win32RegNoRecursive)
+ end
+
+ it "does nothing if the key exists" do
+ @registry.should_receive(:keys_missing?).with(key_path).and_return(true)
+ @registry.should_receive(:create_missing).with(key_path)
+ @registry.should_receive(:key_exists?).with(key_path).and_return(true)
+ @registry.create_key(key_path, true)
+ end
+
+ it "create key if intermediate keys not missing and recursive is set to false" do
+ @registry.should_receive(:keys_missing?).with(key_path).and_return(false)
+ @registry.should_receive(:key_exists?).with(key_path).and_return(false)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:create).with(key, ::Win32::Registry::KEY_WRITE | @registry.registry_system_architecture)
+ @registry.create_key(key_path, false)
+ end
+
+ it "create key if intermediate keys not missing and recursive is set to true" do
+ @registry.should_receive(:keys_missing?).with(key_path).and_return(false)
+ @registry.should_receive(:key_exists?).with(key_path).and_return(false)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:create).with(key, ::Win32::Registry::KEY_WRITE | @registry.registry_system_architecture)
+ @registry.create_key(key_path, true)
+ end
+ end
+
+ describe "delete_key" do
+ it "deletes key if it has subkeys and recursive is set to true" do
+ @registry.should_receive(:key_exists?).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @registry.should_receive(:has_subkeys?).with(key_path).and_return(true)
+ @hive_mock.should_receive(:open).with(key_parent, ::Win32::Registry::KEY_WRITE | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @reg_mock.should_receive(:delete_key).with(key_to_delete, true)
+ @registry.delete_key(key_path, true)
+ end
+
+ it "raises an exception if it has subkeys but recursive is set to false" do
+ @registry.should_receive(:key_exists?).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @registry.should_receive(:has_subkeys?).with(key_path).and_return(true)
+ lambda{@registry.delete_key(key_path, false)}.should raise_error(Chef::Exceptions::Win32RegNoRecursive)
+ end
+
+ it "deletes key if the key exists and has no subkeys" do
+ @registry.should_receive(:key_exists?).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @registry.should_receive(:has_subkeys?).with(key_path).and_return(false)
+ @hive_mock.should_receive(:open).with(key_parent, ::Win32::Registry::KEY_WRITE | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @reg_mock.should_receive(:delete_key).with(key_to_delete)
+ @registry.delete_key(key_path, true)
+ end
+ end
+
+ describe "key_exists?" do
+ it "returns true if key_exists" do
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @registry.key_exists?(key_path).should == true
+ end
+
+ it "returns false if key does not exist" do
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_raise(::Win32::Registry::Error)
+ @registry.key_exists?(key_path).should == false
+ end
+ end
+
+ describe "key_exists!" do
+ it "throws an exception if the key_parent does not exist" do
+ @registry.should_receive(:key_exists?).with(key_path).and_return(false)
+ lambda{@registry.key_exists!(key_path)}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ end
+
+ describe "hive_exists?" do
+ it "returns true if the hive exists" do
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @registry.hive_exists?(key_path) == true
+ end
+
+ it "returns false if the hive does not exist" do
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_raise(Chef::Exceptions::Win32RegHiveMissing)
+ @registry.hive_exists?(key_path) == false
+ end
+ end
+
+ describe "has_subkeys?" do
+ it "returns true if the key has subkeys" do
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @reg_mock.should_receive(:each_key).and_yield(key)
+ @registry.has_subkeys?(key_path) == true
+ end
+
+ it "returns false if the key does not have subkeys" do
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @reg_mock.should_receive(:each_key).and_return(no_args())
+ @registry.has_subkeys?(key_path).should == false
+ end
+
+ it "throws an exception if the key does not exist" do
+ @registry.should_receive(:key_exists!).with(key_path).and_raise(Chef::Exceptions::Win32RegKeyMissing)
+ lambda {@registry.set_value(key_path, value1)}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+ end
+
+ describe "get_subkeys" do
+ it "returns the subkeys if they exist" do
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @reg_mock.should_receive(:each_key).and_yield(sub_key)
+ @registry.get_subkeys(key_path)
+ end
+ end
+
+ describe "value_exists?" do
+ it "throws an exception if the key does not exist" do
+ @registry.should_receive(:key_exists!).with(key_path).and_raise(Chef::Exceptions::Win32RegKeyMissing)
+ lambda {@registry.value_exists?(key_path, value1)}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+
+ it "returns true if the value exists" do
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @reg_mock.should_receive(:any?).and_yield("one")
+ @registry.value_exists?(key_path, value1) == true
+ end
+
+ it "returns false if the value does not exist" do
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @reg_mock.should_receive(:any?).and_yield(no_args())
+ @registry.value_exists?(key_path, value1) == false
+ end
+ end
+
+ describe "data_exists?" do
+ it "throws an exception if the key does not exist" do
+ @registry.should_receive(:key_exists!).with(key_path).and_raise(Chef::Exceptions::Win32RegKeyMissing)
+ lambda {@registry.data_exists?(key_path, value1)}.should raise_error(Chef::Exceptions::Win32RegKeyMissing)
+ end
+
+ it "returns true if the data exists" do
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @registry.should_receive(:get_type_from_name).with(:string).and_return(1)
+ @reg_mock.should_receive(:each).with(no_args()).and_yield("one", 1, "1")
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @registry.data_exists?(key_path, value1).should == true
+ end
+
+ it "returns false if the data does not exist" do
+ @registry.should_receive(:key_exists!).with(key_path).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @registry.should_receive(:get_type_from_name).with(:string).and_return(1)
+ @reg_mock.should_receive(:each).with(no_args()).and_yield("one", 1, "2")
+ @registry.data_exists?(key_path, value1).should == false
+ end
+ end
+
+ describe "value_exists!" do
+ it "does nothing if the value exists" do
+ @registry.should_receive(:value_exists?).with(key_path, value1).and_return(true)
+ @registry.value_exists!(key_path, value1)
+ end
+
+ it "throws an exception if the value does not exist" do
+ @registry.should_receive(:value_exists?).with(key_path, value1).and_return(false)
+ lambda{@registry.value_exists!(key_path, value1)}.should raise_error(Chef::Exceptions::Win32RegValueMissing)
+ end
+ end
+
+ describe "data_exists!" do
+ it "does nothing if the data exists" do
+ @registry.should_receive(:data_exists?).with(key_path, value1).and_return(true)
+ @registry.data_exists!(key_path, value1)
+ end
+
+ it "throws an exception if the data does not exist" do
+ @registry.should_receive(:data_exists?).with(key_path, value1).and_return(false)
+ lambda{@registry.data_exists!(key_path, value1)}.should raise_error(Chef::Exceptions::Win32RegDataMissing)
+ end
+ end
+
+ describe "type_matches?" do
+ it "returns true if type matches" do
+ @registry.should_receive(:value_exists!).with(key_path, value1).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @registry.should_receive(:get_type_from_name).with(:string).and_return(1)
+ @reg_mock.should_receive(:each).and_yield("one", 1)
+ @registry.type_matches?(key_path, value1).should == true
+ end
+
+ it "returns false if type does not match" do
+ @registry.should_receive(:value_exists!).with(key_path, value1).and_return(true)
+ @registry.should_receive(:get_hive_and_key).with(key_path).and_return([@hive_mock, key])
+ @hive_mock.should_receive(:open).with(key, ::Win32::Registry::KEY_READ | @registry.registry_system_architecture).and_yield(@reg_mock)
+ @reg_mock.should_receive(:each).and_yield("two", 2)
+ @registry.type_matches?(key_path, value1).should == false
+ end
+
+ it "throws an exception if value does not exist" do
+ @registry.should_receive(:value_exists?).with(key_path, value1).and_return(false)
+ lambda{@registry.type_matches?(key_path, value1)}.should raise_error(Chef::Exceptions::Win32RegValueMissing)
+ end
+ end
+
+ describe "type_matches!" do
+ it "does nothing if the type_matches" do
+ @registry.should_receive(:type_matches?).with(key_path, value1).and_return(true)
+ @registry.type_matches!(key_path, value1)
+ end
+
+ it "throws an exception if the type does not match" do
+ @registry.should_receive(:type_matches?).with(key_path, value1).and_return(false)
+ lambda{@registry.type_matches!(key_path, value1)}.should raise_error(Chef::Exceptions::Win32RegTypesMismatch)
+ end
+ end
+
+ describe "keys_missing?" do
+ it "returns true if the keys are missing" do
+ @registry.should_receive(:key_exists?).with(missing_key_path).and_return(false)
+ @registry.keys_missing?(key_path).should == true
+ end
+
+ it "returns false if no keys in the path are missing" do
+ @registry.should_receive(:key_exists?).with(missing_key_path).and_return(true)
+ @registry.keys_missing?(key_path).should == false
+ end
+ end
+end
diff --git a/spec/unit/resource/registry_key_spec.rb b/spec/unit/resource/registry_key_spec.rb
new file mode 100644
index 0000000000..9d7680de0c
--- /dev/null
+++ b/spec/unit/resource/registry_key_spec.rb
@@ -0,0 +1,171 @@
+#
+# Author:: Lamont Granquist (<lamont@opscode.com>)
+# Copyright:: Copyright (c) 2012 OpsCode, Inc.
+# License:: Apache License, Version 2.0
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+require 'spec_helper'
+
+describe Chef::Resource::RegistryKey, "initialize" do
+ before(:each) do
+ @resource = Chef::Resource::RegistryKey.new('HKCU\Software\Raxicoricofallapatorius')
+ end
+
+ it "should create a new Chef::Resource::RegistryKey" do
+ @resource.should be_a_kind_of(Chef::Resource)
+ @resource.should be_a_kind_of(Chef::Resource::RegistryKey)
+ end
+
+ it "should set the resource_name to :registry_key" do
+ @resource.resource_name.should eql(:registry_key)
+ end
+
+ it "should set the key equal to the argument to initialize" do
+ @resource.key.should eql('HKCU\Software\Raxicoricofallapatorius')
+ end
+
+ it "should default recursive to false" do
+ @resource.recursive.should eql(false)
+ end
+
+ it "should default architecture to :machine" do
+ @resource.architecture.should eql(:machine)
+ end
+
+ it "should set action to :create" do
+ @resource.action.should eql(:create)
+ end
+
+ %w{create create_if_missing delete delete_key}.each do |action|
+ it "should allow action #{action}" do
+ @resource.allowed_actions.detect { |a| a == action.to_sym }.should eql(action.to_sym)
+ end
+ end
+end
+
+describe Chef::Resource::RegistryKey, "key" do
+ before(:each) do
+ @resource = Chef::Resource::RegistryKey.new('HKCU\Software\Raxicoricofallapatorius')
+ end
+
+ it "should allow a string" do
+ @resource.key 'HKCU\Software\Poosh'
+ @resource.key.should eql('HKCU\Software\Poosh')
+ end
+
+ it "should not allow an integer" do
+ lambda { @resource.send(:key, 100) }.should raise_error(ArgumentError)
+ end
+
+ it "should not allow a hash" do
+ lambda { @resource.send(:key, { :sonic => "screwdriver" }) }.should raise_error(ArgumentError)
+ end
+end
+
+describe Chef::Resource::RegistryKey, "values" do
+ before(:each) do
+ @resource = Chef::Resource::RegistryKey.new('HKCU\Software\Raxicoricofallapatorius')
+ end
+
+ it "should allow a single proper hash of registry values" do
+ @resource.values( { :name => 'poosh', :type => :string, :data => 'carmen' } )
+ @resource.values.should eql([ { :name => 'poosh', :type => :string, :data => 'carmen' } ])
+ end
+
+ it "should allow an array of proper hashes of registry values" do
+ @resource.values [ { :name => 'poosh', :type => :string, :data => 'carmen' } ]
+ @resource.values.should eql([ { :name => 'poosh', :type => :string, :data => 'carmen' } ])
+ end
+
+ it "should throw an exception if the name field is missing" do
+ lambda { @resource.values [ { :type => :string, :data => 'carmen' } ] }.should raise_error(ArgumentError)
+ end
+
+ it "should throw an exception if the type field is missing" do
+ lambda { @resource.values [ { :name => 'poosh', :data => 'carmen' } ] }.should raise_error(ArgumentError)
+ end
+
+ it "should throw an exception if the data field is missing" do
+ lambda { @resource.values [ { :name => 'poosh', :type => :string } ] }.should raise_error(ArgumentError)
+ end
+
+ it "should throw an exception if extra fields are present" do
+ lambda { @resource.values [ { :name => 'poosh', :type => :string, :data => 'carmen', :screwdriver => 'sonic' } ] }.should raise_error(ArgumentError)
+ end
+
+ it "should not allow a string" do
+ lambda { @resource.send(:values, 'souffle') }.should raise_error(ArgumentError)
+ end
+
+ it "should not allow an integer" do
+ lambda { @resource.send(:values, 100) }.should raise_error(ArgumentError)
+ end
+end
+
+describe Chef::Resource::RegistryKey, "recursive" do
+ before(:each) do
+ @resource = Chef::Resource::RegistryKey.new('HKCU\Software\Raxicoricofallapatorius')
+ end
+
+ it "should allow a boolean" do
+ @resource.recursive(true)
+ @resource.recursive.should eql(true)
+ end
+
+ it "should not allow a hash" do
+ lambda { @resource.recursive({:sonic => :screwdriver}) }.should raise_error(ArgumentError)
+ end
+
+ it "should not allow an array" do
+ lambda { @resource.recursive([:nose, :chin]) }.should raise_error(ArgumentError)
+ end
+
+ it "should not allow a string" do
+ lambda { @resource.recursive('souffle') }.should raise_error(ArgumentError)
+ end
+
+ it "should not allow an integer" do
+ lambda { @resource.recursive(100) }.should raise_error(ArgumentError)
+ end
+end
+
+describe Chef::Resource::RegistryKey, "architecture" do
+ before(:each) do
+ @resource = Chef::Resource::RegistryKey.new('HKCU\Software\Raxicoricofallapatorius')
+ end
+
+ [ :i386, :x86_64, :machine ].each do |arch|
+ it "should allow #{arch} as a symbol" do
+ @resource.architecture(arch)
+ @resource.architecture.should eql(arch)
+ end
+ end
+
+ it "should not allow a hash" do
+ lambda { @resource.architecture({:sonic => :screwdriver}) }.should raise_error(ArgumentError)
+ end
+
+ it "should not allow an array" do
+ lambda { @resource.architecture([:nose, :chin]) }.should raise_error(ArgumentError)
+ end
+
+ it "should not allow a string" do
+ lambda { @resource.architecture('souffle') }.should raise_error(ArgumentError)
+ end
+
+ it "should not allow an integer" do
+ lambda { @resource.architecture(100) }.should raise_error(ArgumentError)
+ end
+end