diff options
Diffstat (limited to 'chef/spec')
90 files changed, 5816 insertions, 0 deletions
diff --git a/chef/spec/chef_server/controllers/log/merb_test.log b/chef/spec/chef_server/controllers/log/merb_test.log new file mode 100644 index 0000000000..86fbe4b9cb --- /dev/null +++ b/chef/spec/chef_server/controllers/log/merb_test.log @@ -0,0 +1,4 @@ +Tue, 27 May 2008 00:52:34 GMT ~ info ~ Logfile created + ~ Not Using Sessions + ~ Not Using Sessions + ~ Not Using Sessions diff --git a/chef/spec/chef_server/controllers/nodes_spec.rb b/chef/spec/chef_server/controllers/nodes_spec.rb new file mode 100644 index 0000000000..65c2df301a --- /dev/null +++ b/chef/spec/chef_server/controllers/nodes_spec.rb @@ -0,0 +1,232 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.join(File.dirname(__FILE__), "..", 'spec_helper.rb') +# +# describe Nodes, "index action" do +# it "should get a list of all the nodes" do +# Chef::Node.should_receive(:list).and_return(["one"]) +# dispatch_to(Nodes, :index) do |c| +# c.stub!(:display) +# end +# end +# +# it "should send a list of nodes to display" do +# Chef::Node.stub!(:list).and_return(["one"]) +# dispatch_to(Nodes, :index) do |c| +# c.should_receive(:display).with(["one"]) +# end +# end +# end +# +# describe Nodes, "show action" do +# it "should load a node from the filestore based on the id" do +# node = stub("Node", :null_object => true) +# Chef::Node.should_receive(:load).with("bond").once.and_return(node) +# dispatch_to(Nodes, :show, { :id => "bond" }) do |c| +# c.should_receive(:display).with(node).once.and_return(true) +# end +# end +# +# it "should return 200 on a well formed request" do +# node = stub("Node", :null_object => true) +# Chef::Node.should_receive(:load).with("bond").once.and_return(node) +# controller = dispatch_to(Nodes, :show, { :id => "bond" }) do |c| +# c.stub!(:display) +# end +# controller.status.should eql(200) +# end +# +# it "should raise a BadRequest if the id is not found" do +# Chef::Node.should_receive(:load).with("bond").once.and_raise(RuntimeError) +# lambda { +# dispatch_to(Nodes, :show, { :id => "bond" }) +# }.should raise_error(Merb::ControllerExceptions::BadRequest) +# end +# end +# +# describe Nodes, "create action" do +# it "should create a node from an inflated object" do +# mnode = mock("Node", :null_object => true) +# mnode.stub!(:name).and_return("bond") +# mnode.should_receive(:save).once.and_return(true) +# controller = dispatch_to(Nodes, :create) do |c| +# c.stub!(:params).and_return({ "inflated_object" => mnode }) +# c.stub!(:session).and_return({ +# :openid => 'http://localhost/openid/server/node/bond', +# :level => :node, +# :node_name => "bond", +# }) +# c.stub!(:display) +# end +# controller.status.should eql(202) +# end +# +# it "should raise an exception if it cannot inflate an object" do +# lambda { +# dispatch_to(Nodes, :create) do |c| +# c.stub!(:params).and_return({ }) +# end +# }.should raise_error(Merb::Controller::BadRequest) +# end +# end +# +# describe Nodes, "update action" do +# it "should update a node from an inflated object" do +# mnode = mock("Node", :null_object => true) +# mnode.stub!(:name).and_return("one") +# Chef::FileStore.should_receive(:store).with("node", "one", mnode).once.and_return(true) +# controller = dispatch_to(Nodes, :update, { :id => "one" }) do |c| +# c.stub!(:session).and_return({ +# :openid => 'http://localhost/openid/server/node/one', +# :level => :node, +# :node_name => "one", +# }) +# c.stub!(:params).and_return({ "inflated_object" => mnode }) +# c.stub!(:display) +# end +# controller.status.should eql(202) +# end +# +# it "should raise an exception if it cannot inflate an object" do +# lambda { dispatch_to(Nodes, :update) }.should raise_error(Merb::Controller::BadRequest) +# end +# end +# +# describe Nodes, "destroy action" do +# def do_destroy +# dispatch_to(Nodes, :destroy, { :id => "one" }) do |c| +# c.stub!(:display) +# end +# end +# +# it "should load the node it's about to destroy from the filestore" do +# mnode = stub("Node", :null_object => true) +# Chef::FileStore.should_receive(:load).with("node", "one").once.and_return(mnode) +# Chef::FileStore.stub!(:delete) +# do_destroy +# end +# +# it "should raise an exception if it cannot find the node to destroy" do +# Chef::FileStore.should_receive(:load).with("node", "one").once.and_raise(RuntimeError) +# lambda { do_destroy }.should raise_error(Merb::Controller::BadRequest) +# end +# +# it "should remove the node from the filestore" do +# mnode = stub("Node", :null_object => true) +# Chef::FileStore.stub!(:load).with("node", "one").and_return(mnode) +# Chef::FileStore.should_receive(:delete).with("node", "one") +# do_destroy +# end +# +# it "should remove the node from the search index" do +# mnode = stub("Node", :null_object => true) +# Chef::FileStore.stub!(:load).with("node", "one").and_return(mnode) +# Chef::FileStore.stub!(:delete) +# do_destroy +# end +# +# it "should return the node it just deleted" do +# mnode = stub("Node", :null_object => true) +# Chef::FileStore.stub!(:load).with("node", "one").and_return(mnode) +# Chef::FileStore.stub!(:delete) +# dispatch_to(Nodes, :destroy, { :id => "one" }) do |c| +# c.should_receive(:display).once.with(mnode) +# end +# end +# +# it "should return a status of 202" do +# mnode = stub("Node", :null_object => true) +# Chef::FileStore.stub!(:load).with("node", "one").and_return(mnode) +# Chef::FileStore.stub!(:delete) +# controller = do_destroy +# controller.status.should eql(202) +# end +# end +# +# describe Nodes, "compile action" do +# before(:each) do +# @compile = stub("Compile", :null_object => true) +# @node = stub("Node", :null_object => true) +# @node.stub!(:[]).and_return(true) +# @node.stub!(:[]=).and_return(true) +# @node.stub!(:recipes).and_return([]) +# @compile.stub!(:load_definitions).and_return(true) +# @compile.stub!(:load_recipes).and_return(true) +# @compile.stub!(:collection).and_return([]) +# @compile.stub!(:node, @node) +# @compile.stub!(:load_node).and_return(true) +# @stored_node = stub("Stored Node", :null_object => true) +# end +# +# def do_compile +# Chef::FileStore.stub!(:store).and_return(true) +# Chef::FileStore.stub!(:load).and_return(@stored_node) +# Chef::Compile.stub!(:new).and_return(@compile) +# dispatch_to(Nodes, :compile, { :id => "one" }) do |c| +# c.stub!(:display) +# end +# end +# +# it "should load the node from the node resource" do +# @compile.should_receive(:load_node).with("one").and_return(true) +# do_compile +# end +# +# it "should merge the data with the currently stored node" do +# node1 = Chef::Node.new +# node1.name "adam" +# node1.music "crowe" +# node1.recipes << "monkey" +# @compile.stub!(:node).and_return(node1) +# @stored_node = Chef::Node.new +# @stored_node.name "adam" +# @stored_node.music "crown" +# @stored_node.woot "woot" +# @stored_node.recipes << "monkeysoup" +# do_compile +# node1.name.should eql("adam") +# node1.music.should eql("crown") +# node1.woot.should eql("woot") +# node1.recipes.should eql([ "monkey", "monkeysoup" ]) +# end +# +# it "should load definitions" do +# @compile.should_receive(:load_definitions) +# do_compile +# end +# +# it "should load recipes" do +# @compile.should_receive(:load_recipes) +# do_compile +# end +# +# it "should display the collection and node object" do +# Chef::FileStore.stub!(:load).and_return(@stored_node) +# Chef::Compile.stub!(:new).and_return(@compile) +# dispatch_to(Nodes, :compile, { :id => "one" }) do |c| +# c.should_receive(:display).with({ :collection => [], :node => nil }) +# end +# end +# +# it "should return 200" do +# controller = do_compile +# controller.status.should eql(200) +# end +# +# end
\ No newline at end of file diff --git a/chef/spec/chef_server/controllers/openid_consumer_spec.rb b/chef/spec/chef_server/controllers/openid_consumer_spec.rb new file mode 100644 index 0000000000..0156fd8d49 --- /dev/null +++ b/chef/spec/chef_server/controllers/openid_consumer_spec.rb @@ -0,0 +1,47 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.join(File.dirname(__FILE__), "..", 'spec_helper.rb') +# +# describe OpenidConsumer, "check_valid_openid_provider method" do +# it "should confirm the openid provider if the openid_providers config is nil" do +# Chef::Config[:openid_providers] = nil +# c = OpenidConsumer.new(true) +# c.send(:check_valid_openid_provider, "monkeyid").should eql(true) +# end +# +# it "should return true if the openid provider is in openid_providers list" do +# Chef::Config[:openid_providers] = [ 'monkeyid' ] +# c = OpenidConsumer.new(true) +# c.send(:check_valid_openid_provider, "monkeyid").should eql(true) +# end +# +# it "should return true if the openid provider is in openid_providers list with http://" do +# Chef::Config[:openid_providers] = [ 'monkeyid' ] +# c = OpenidConsumer.new(true) +# c.send(:check_valid_openid_provider, "http://monkeyid").should eql(true) +# end +# +# it "should raise an exception if the openid provider is not in openid_providers list" do +# Chef::Config[:openid_providers] = [ 'monkeyid' ] +# c = OpenidConsumer.new(true) +# lambda { +# c.send(:check_valid_openid_provider, "monkey") +# }.should raise_error(Merb::Controller::Unauthorized) +# end +# end
\ No newline at end of file diff --git a/chef/spec/chef_server/controllers/openid_register_spec.rb b/chef/spec/chef_server/controllers/openid_register_spec.rb new file mode 100644 index 0000000000..c883715fc6 --- /dev/null +++ b/chef/spec/chef_server/controllers/openid_register_spec.rb @@ -0,0 +1,111 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.join(File.dirname(__FILE__), "..", 'spec_helper.rb') +# +# describe OpenidRegister, "index action" do +# it "should get a list of all registered nodes" do +# Chef::OpenIDRegistration.should_receive(:list).with(true).and_return(["one"]) +# dispatch_to(OpenidRegister, :index) do |c| +# c.stub!(:display) +# end +# end +# end +# +# describe OpenidRegister, "show action" do +# it "should raise a 404 if the nodes registration is not found" do +# Chef::OpenIDRegistration.should_receive(:load).with("foo").and_raise(RuntimeError) +# lambda { +# dispatch_to(OpenidRegister, :show, { :id => "foo" }) +# }.should raise_error(Merb::ControllerExceptions::NotFound) +# end +# +# it "should call display on the node registration" do +# Chef::OpenIDRegistration.stub!(:load).and_return(true) +# dispatch_to(OpenidRegister, :show, { :id => "foo" }) do |c| +# c.should_receive(:display).with(true) +# end +# end +# end +# +# describe OpenidRegister, "create action" do +# def do_create +# dispatch_to(OpenidRegister, :create, { :id => "foo", :password => "beck" }) do |c| +# c.stub!(:display) +# end +# end +# +# it "should require an id to register" do +# lambda { +# dispatch_to(OpenidRegister, :create, { :password => "beck" }) +# }.should raise_error(Merb::ControllerExceptions::BadRequest) +# end +# +# it "should require a password to register" do +# lambda { +# dispatch_to(OpenidRegister, :create, { :id => "foo" }) +# }.should raise_error(Merb::ControllerExceptions::BadRequest) +# end +# +# it "should return 400 if a node is already registered" do +# Chef::OpenIDRegistration.should_receive(:has_key?).with("foo").and_return(true) +# lambda { +# dispatch_to(OpenidRegister, :create, { :id => "foo", :password => "beck" }) +# }.should raise_error(Merb::ControllerExceptions::BadRequest) +# end +# +# it "should store the registration in a new Chef::OpenIDRegistration" do +# mock_reg = mock("Chef::OpenIDRegistration", :null_object => true) +# mock_reg.should_receive(:name=).with("foo").and_return(true) +# mock_reg.should_receive(:set_password).with("beck").and_return(true) +# mock_reg.should_receive(:save).and_return(true) +# Chef::OpenIDRegistration.stub!(:has_key?).and_return(false) +# Chef::OpenIDRegistration.should_receive(:new).and_return(mock_reg) +# do_create +# end +# end +# +# describe OpenidRegister, "update action" do +# it "should raise a 400 error" do +# lambda { +# dispatch_to(OpenidRegister, :update) +# } +# end +# end +# +# describe OpenidRegister, "destroy action" do +# def do_destroy +# dispatch_to(OpenidRegister, :destroy, { :id => "foo" }) do |c| +# c.stub!(:display) +# end +# end +# +# it "should return 400 if it cannot find the registration" do +# Chef::OpenIDRegistration.should_receive(:load).and_raise(ArgumentError) +# lambda { +# do_destroy +# }.should raise_error(Merb::ControllerExceptions::BadRequest) +# end +# +# it "should delete the registration from the store" do +# mock_reg = mock("OpenIDRegistration") +# mock_reg.should_receive(:destroy).and_return(true) +# Chef::OpenIDRegistration.should_receive(:load).with("foo").and_return(mock_reg) +# do_destroy +# end +# end
\ No newline at end of file diff --git a/chef/spec/chef_server/log/merb_test.log b/chef/spec/chef_server/log/merb_test.log new file mode 100644 index 0000000000..b2aa15df9a --- /dev/null +++ b/chef/spec/chef_server/log/merb_test.log @@ -0,0 +1,4 @@ +Mon, 16 Jun 2008 02:08:02 GMT ~ info ~ Logfile created + ~ Not Using Sessions + ~ Not Using Sessions + ~ Not Using Sessions diff --git a/chef/spec/chef_server/spec.opts b/chef/spec/chef_server/spec.opts new file mode 100644 index 0000000000..3a2ff550a2 --- /dev/null +++ b/chef/spec/chef_server/spec.opts @@ -0,0 +1,4 @@ +--color +--loadby +mtime + diff --git a/chef/spec/chef_server/spec_helper.rb b/chef/spec/chef_server/spec_helper.rb new file mode 100644 index 0000000000..90c1e0af7a --- /dev/null +++ b/chef/spec/chef_server/spec_helper.rb @@ -0,0 +1,16 @@ +require 'rubygems' +require 'merb-core' +require 'spec' # Satiates Autotest and anyone else not using the Rake tasks + +Merb.start_environment( + :testing => true, + :adapter => 'runner', + :environment => ENV['MERB_ENV'] || 'test', + :init_file => File.join(File.dirname(__FILE__), "..", "..", "lib", "chef_server", "init.rb") +) + +Spec::Runner.configure do |config| + config.include(Merb::Test::ViewHelper) + config.include(Merb::Test::RouteHelper) + config.include(Merb::Test::ControllerHelper) +end diff --git a/chef/spec/data/bad-config.rb b/chef/spec/data/bad-config.rb new file mode 100644 index 0000000000..5477a69366 --- /dev/null +++ b/chef/spec/data/bad-config.rb @@ -0,0 +1 @@ +monkey_soup("tastes nice")
\ No newline at end of file diff --git a/chef/spec/data/compile/cookbooks/test/attributes/george.rb b/chef/spec/data/compile/cookbooks/test/attributes/george.rb new file mode 100644 index 0000000000..5df9567761 --- /dev/null +++ b/chef/spec/data/compile/cookbooks/test/attributes/george.rb @@ -0,0 +1 @@ +george "washington"
\ No newline at end of file diff --git a/chef/spec/data/compile/cookbooks/test/definitions/new_cat.rb b/chef/spec/data/compile/cookbooks/test/definitions/new_cat.rb new file mode 100644 index 0000000000..a49b53348c --- /dev/null +++ b/chef/spec/data/compile/cookbooks/test/definitions/new_cat.rb @@ -0,0 +1,5 @@ +define :new_cat, :is_pretty => true do + cat "#{params[:name]}" do + pretty_kitty params[:is_pretty] + end +end diff --git a/chef/spec/data/compile/cookbooks/test/recipes/default.rb b/chef/spec/data/compile/cookbooks/test/recipes/default.rb new file mode 100644 index 0000000000..d769dc826d --- /dev/null +++ b/chef/spec/data/compile/cookbooks/test/recipes/default.rb @@ -0,0 +1,5 @@ + +cat "einstein" do + pretty_kitty true +end + diff --git a/chef/spec/data/compile/cookbooks/test/recipes/one.rb b/chef/spec/data/compile/cookbooks/test/recipes/one.rb new file mode 100644 index 0000000000..7795cc1d4a --- /dev/null +++ b/chef/spec/data/compile/cookbooks/test/recipes/one.rb @@ -0,0 +1,7 @@ +cat "loulou" do + pretty_kitty true +end + +new_cat "birthday" do + pretty_kitty false +end diff --git a/chef/spec/data/compile/cookbooks/test/recipes/two.rb b/chef/spec/data/compile/cookbooks/test/recipes/two.rb new file mode 100644 index 0000000000..01def1b2ac --- /dev/null +++ b/chef/spec/data/compile/cookbooks/test/recipes/two.rb @@ -0,0 +1,7 @@ +cat "peanut" do + pretty_kitty true +end + +new_cat "fat peanut" do + pretty_kitty false +end diff --git a/chef/spec/data/compile/nodes/compile.rb b/chef/spec/data/compile/nodes/compile.rb new file mode 100644 index 0000000000..84d52bb1cf --- /dev/null +++ b/chef/spec/data/compile/nodes/compile.rb @@ -0,0 +1,5 @@ +## +# Nodes should have a unique name +## +name "compile" +recipes "test", "test::one", "test::two" diff --git a/chef/spec/data/config.rb b/chef/spec/data/config.rb new file mode 100644 index 0000000000..0b3340ce57 --- /dev/null +++ b/chef/spec/data/config.rb @@ -0,0 +1,6 @@ +# +# Sample Chef Config File +# + +cookbook_path "/etc/chef/cookbook", "/etc/chef/site-cookbook" + diff --git a/chef/spec/data/cookbooks/openldap/attributes/default.rb b/chef/spec/data/cookbooks/openldap/attributes/default.rb new file mode 100644 index 0000000000..204ae9ed77 --- /dev/null +++ b/chef/spec/data/cookbooks/openldap/attributes/default.rb @@ -0,0 +1,15 @@ +chef_env ||= nil +case chef_env +when "prod" + ldap_server "ops1prod" + ldap_basedn "dc=hjksolutions,dc=com" + ldap_replication_password "yes" +when "corp" + ldap_server "ops1prod" + ldap_basedn "dc=hjksolutions,dc=com" + ldap_replication_password "yougotit" +else + ldap_server "ops1prod" + ldap_basedn "dc=hjksolutions,dc=com" + ldap_replication_password "forsure" +end diff --git a/chef/spec/data/cookbooks/openldap/attributes/smokey.rb b/chef/spec/data/cookbooks/openldap/attributes/smokey.rb new file mode 100644 index 0000000000..63f5b56c7f --- /dev/null +++ b/chef/spec/data/cookbooks/openldap/attributes/smokey.rb @@ -0,0 +1 @@ +smokey "robinson"
\ No newline at end of file diff --git a/chef/spec/data/cookbooks/openldap/definitions/client.rb b/chef/spec/data/cookbooks/openldap/definitions/client.rb new file mode 100644 index 0000000000..ac81831d11 --- /dev/null +++ b/chef/spec/data/cookbooks/openldap/definitions/client.rb @@ -0,0 +1,5 @@ +define :openldap_client, :mothra => "a big monster" do + cat "#{params[:name]}" do + pretty_kitty true + end +end diff --git a/chef/spec/data/cookbooks/openldap/definitions/server.rb b/chef/spec/data/cookbooks/openldap/definitions/server.rb new file mode 100644 index 0000000000..2df437aa84 --- /dev/null +++ b/chef/spec/data/cookbooks/openldap/definitions/server.rb @@ -0,0 +1,5 @@ +define :openldap_server, :mothra => "a big monster" do + cat "#{params[:name]}" do + pretty_kitty true + end +end diff --git a/chef/spec/data/cookbooks/openldap/ignore b/chef/spec/data/cookbooks/openldap/ignore new file mode 100644 index 0000000000..e96f4e7df4 --- /dev/null +++ b/chef/spec/data/cookbooks/openldap/ignore @@ -0,0 +1,6 @@ +# +# The ignore file allows you to skip files in cookbooks with the same name that appear +# later in the search path. +# + +recipes/ignoreme.rb
\ No newline at end of file diff --git a/chef/spec/data/cookbooks/openldap/recipes/default.rb b/chef/spec/data/cookbooks/openldap/recipes/default.rb new file mode 100644 index 0000000000..0ac8a9bb4b --- /dev/null +++ b/chef/spec/data/cookbooks/openldap/recipes/default.rb @@ -0,0 +1,3 @@ +cat "blanket" do + pretty_kitty true +end diff --git a/chef/spec/data/cookbooks/openldap/recipes/gigantor.rb b/chef/spec/data/cookbooks/openldap/recipes/gigantor.rb new file mode 100644 index 0000000000..b450eca7cd --- /dev/null +++ b/chef/spec/data/cookbooks/openldap/recipes/gigantor.rb @@ -0,0 +1,3 @@ +cat "blanket" do + pretty_kitty false +end diff --git a/chef/spec/data/cookbooks/openldap/recipes/one.rb b/chef/spec/data/cookbooks/openldap/recipes/one.rb new file mode 100644 index 0000000000..e1c3cff92e --- /dev/null +++ b/chef/spec/data/cookbooks/openldap/recipes/one.rb @@ -0,0 +1,15 @@ +## +# Nodes should have a unique name +## +name "test.example.com default" + +## +# Nodes can set arbitrary arguments +## +sunshine "in" +something "else" + +## +# Nodes should have recipes +## +recipes "operations-master", "operations-monitoring" diff --git a/chef/spec/data/cookbooks/openldap/templates/default/test.erb b/chef/spec/data/cookbooks/openldap/templates/default/test.erb new file mode 100644 index 0000000000..f39fa7da89 --- /dev/null +++ b/chef/spec/data/cookbooks/openldap/templates/default/test.erb @@ -0,0 +1 @@ +We could be diving for pearls! diff --git a/chef/spec/data/definitions/test.rb b/chef/spec/data/definitions/test.rb new file mode 100644 index 0000000000..b0d0effc27 --- /dev/null +++ b/chef/spec/data/definitions/test.rb @@ -0,0 +1,5 @@ +define :rico_suave, :rich => "smooth" do + zen_master "test" do + something "#{params[:rich]}" + end +end
\ No newline at end of file diff --git a/chef/spec/data/kitchen/openldap/attributes/default.rb b/chef/spec/data/kitchen/openldap/attributes/default.rb new file mode 100644 index 0000000000..d208959475 --- /dev/null +++ b/chef/spec/data/kitchen/openldap/attributes/default.rb @@ -0,0 +1,3 @@ +# +# Nothing to see here, move along +# diff --git a/chef/spec/data/kitchen/openldap/attributes/robinson.rb b/chef/spec/data/kitchen/openldap/attributes/robinson.rb new file mode 100644 index 0000000000..9d6b44d464 --- /dev/null +++ b/chef/spec/data/kitchen/openldap/attributes/robinson.rb @@ -0,0 +1,3 @@ +# +# Smokey lives here +#
\ No newline at end of file diff --git a/chef/spec/data/kitchen/openldap/definitions/client.rb b/chef/spec/data/kitchen/openldap/definitions/client.rb new file mode 100644 index 0000000000..d4c2263b54 --- /dev/null +++ b/chef/spec/data/kitchen/openldap/definitions/client.rb @@ -0,0 +1,3 @@ +# +# A sad client +# diff --git a/chef/spec/data/kitchen/openldap/definitions/drewbarrymore.rb b/chef/spec/data/kitchen/openldap/definitions/drewbarrymore.rb new file mode 100644 index 0000000000..510f0c35da --- /dev/null +++ b/chef/spec/data/kitchen/openldap/definitions/drewbarrymore.rb @@ -0,0 +1,3 @@ +# +# Was in people magazine this month... +#
\ No newline at end of file diff --git a/chef/spec/data/kitchen/openldap/recipes/gigantor.rb b/chef/spec/data/kitchen/openldap/recipes/gigantor.rb new file mode 100644 index 0000000000..70a41960eb --- /dev/null +++ b/chef/spec/data/kitchen/openldap/recipes/gigantor.rb @@ -0,0 +1,3 @@ +cat "blanket" do + pretty_kitty true +end
\ No newline at end of file diff --git a/chef/spec/data/kitchen/openldap/recipes/ignoreme.rb b/chef/spec/data/kitchen/openldap/recipes/ignoreme.rb new file mode 100644 index 0000000000..15095986c6 --- /dev/null +++ b/chef/spec/data/kitchen/openldap/recipes/ignoreme.rb @@ -0,0 +1,3 @@ +# +# this file will never be seen +#
\ No newline at end of file diff --git a/chef/spec/data/kitchen/openldap/recipes/woot.rb b/chef/spec/data/kitchen/openldap/recipes/woot.rb new file mode 100644 index 0000000000..44893dae36 --- /dev/null +++ b/chef/spec/data/kitchen/openldap/recipes/woot.rb @@ -0,0 +1,3 @@ +# +# Such a funny word.. +# diff --git a/chef/spec/data/nodes/default.rb b/chef/spec/data/nodes/default.rb new file mode 100644 index 0000000000..e1c3cff92e --- /dev/null +++ b/chef/spec/data/nodes/default.rb @@ -0,0 +1,15 @@ +## +# Nodes should have a unique name +## +name "test.example.com default" + +## +# Nodes can set arbitrary arguments +## +sunshine "in" +something "else" + +## +# Nodes should have recipes +## +recipes "operations-master", "operations-monitoring" diff --git a/chef/spec/data/nodes/test.example.com.rb b/chef/spec/data/nodes/test.example.com.rb new file mode 100644 index 0000000000..9c374395bf --- /dev/null +++ b/chef/spec/data/nodes/test.example.com.rb @@ -0,0 +1,15 @@ +## +# Nodes should have a unique name +## +name "test.example.com" + +## +# Nodes can set arbitrary arguments +## +sunshine "in" +something "else" + +## +# Nodes should have recipes +## +recipes "operations-master", "operations-monitoring" diff --git a/chef/spec/data/nodes/test.rb b/chef/spec/data/nodes/test.rb new file mode 100644 index 0000000000..d968816d60 --- /dev/null +++ b/chef/spec/data/nodes/test.rb @@ -0,0 +1,15 @@ +## +# Nodes should have a unique name +## +name "test.example.com short" + +## +# Nodes can set arbitrary arguments +## +sunshine "in" +something "else" + +## +# Nodes should have recipes +## +recipes "operations-master", "operations-monitoring" diff --git a/chef/spec/data/recipes/test.rb b/chef/spec/data/recipes/test.rb new file mode 100644 index 0000000000..1c94b917f0 --- /dev/null +++ b/chef/spec/data/recipes/test.rb @@ -0,0 +1,7 @@ + +file "/etc/nsswitch.conf" do + action "create" + owner "root" + group "root" + mode 0644 +end diff --git a/chef/spec/data/seattle.txt b/chef/spec/data/seattle.txt new file mode 100644 index 0000000000..19f6290939 --- /dev/null +++ b/chef/spec/data/seattle.txt @@ -0,0 +1 @@ +Seattle is a great town. Next time you visit, you should buy me a beer.
\ No newline at end of file diff --git a/chef/spec/lib/chef/provider/easy.rb b/chef/spec/lib/chef/provider/easy.rb new file mode 100644 index 0000000000..b637efa768 --- /dev/null +++ b/chef/spec/lib/chef/provider/easy.rb @@ -0,0 +1,37 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# License:: GNU General Public License version 2 or later +# +# This program and entire repository is free software; you can +# redistribute it and/or modify it under the terms of the GNU +# General Public License as published by the Free Software +# Foundation; either version 2 of the License, or any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +# + +class Chef + class Provider + class Easy < Chef::Provider + def load_current_resource + true + end + + def action_sell + true + end + + def action_buy + true + end + end + end +end
\ No newline at end of file diff --git a/chef/spec/lib/chef/provider/snakeoil.rb b/chef/spec/lib/chef/provider/snakeoil.rb new file mode 100644 index 0000000000..0486402686 --- /dev/null +++ b/chef/spec/lib/chef/provider/snakeoil.rb @@ -0,0 +1,37 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# License:: GNU General Public License version 2 or later +# +# This program and entire repository is free software; you can +# redistribute it and/or modify it under the terms of the GNU +# General Public License as published by the Free Software +# Foundation; either version 2 of the License, or any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +# + +class Chef + class Provider + class SnakeOil < Chef::Provider + def load_current_resource + true + end + + def action_sell + true + end + + def action_buy + true + end + end + end +end
\ No newline at end of file diff --git a/chef/spec/lib/chef/resource/cat.rb b/chef/spec/lib/chef/resource/cat.rb new file mode 100644 index 0000000000..e651c44512 --- /dev/null +++ b/chef/spec/lib/chef/resource/cat.rb @@ -0,0 +1,42 @@ +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# License:: GNU General Public License version 2 or later +# +# This program and entire repository is free software; you can +# redistribute it and/or modify it under the terms of the GNU +# General Public License as published by the Free Software +# Foundation; either version 2 of the License, or any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +# + +class Chef + class Resource + class Cat < Chef::Resource + + attr_accessor :action + + def initialize(name, collection=nil, node=nil) + @resource_name = :cat + super(name, collection, node) + @action = "sell" + end + + def pretty_kitty(arg=nil) + set_if_args(@pretty_kitty, arg) do + case arg + when true, false + @pretty_kitty = arg + end + end + end + end + end +end
\ No newline at end of file diff --git a/chef/spec/lib/chef/resource/zen_master.rb b/chef/spec/lib/chef/resource/zen_master.rb new file mode 100644 index 0000000000..8b4225fbd4 --- /dev/null +++ b/chef/spec/lib/chef/resource/zen_master.rb @@ -0,0 +1,44 @@ +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# License:: GNU General Public License version 2 or later +# +# This program and entire repository is free software; you can +# redistribute it and/or modify it under the terms of the GNU +# General Public License as published by the Free Software +# Foundation; either version 2 of the License, or any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +# + +class Chef + class Resource + class ZenMaster < Chef::Resource + attr_reader :peace + + def initialize(name, collection=nil, node=nil) + @resource_name = :zen_master + super(name, collection, node) + end + + def peace(tf) + @peace = tf + end + + def something(arg=nil) + set_if_args(@something, arg) do + case arg + when true, false + @something = arg + end + end + end + end + end +end
\ No newline at end of file diff --git a/chef/spec/rcov.opts b/chef/spec/rcov.opts new file mode 100644 index 0000000000..484626ea9c --- /dev/null +++ b/chef/spec/rcov.opts @@ -0,0 +1,2 @@ +--exclude +spec,bin,/Library/Ruby diff --git a/chef/spec/spec.opts b/chef/spec/spec.opts new file mode 100644 index 0000000000..c9c9b4ddf0 --- /dev/null +++ b/chef/spec/spec.opts @@ -0,0 +1,3 @@ +--color +--loadby +mtime diff --git a/chef/spec/spec_helper.rb b/chef/spec/spec_helper.rb new file mode 100644 index 0000000000..9120b2876c --- /dev/null +++ b/chef/spec/spec_helper.rb @@ -0,0 +1,23 @@ +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# License:: GNU General Public License version 2 or later +# +# This program and entire repository is free software; you can +# redistribute it and/or modify it under the terms of the GNU +# General Public License as published by the Free Software +# Foundation; either version 2 of the License, or any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +# + +require File.join(File.dirname(__FILE__), "..", "lib", "chef") +Dir[File.join(File.dirname(__FILE__), 'lib', '**', '*.rb')].sort.each { |lib| require lib } + +Chef::Config.log_level(:error) diff --git a/chef/spec/unit/chef_spec.rb b/chef/spec/unit/chef_spec.rb new file mode 100644 index 0000000000..7d2af6f88d --- /dev/null +++ b/chef/spec/unit/chef_spec.rb @@ -0,0 +1,25 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef do + it "should have a version defined" do + Chef::VERSION.should match(/(\d+)\.(\d+)\.(\d+)/) + end +end diff --git a/chef/spec/unit/client_spec.rb b/chef/spec/unit/client_spec.rb new file mode 100644 index 0000000000..a2732f28c3 --- /dev/null +++ b/chef/spec/unit/client_spec.rb @@ -0,0 +1,61 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Client, "initialize" do + it "should create a new Chef::Client object" do + Chef::Client.new.should be_kind_of(Chef::Client) + end +end + +describe Chef::Client, "build_node" do + before(:each) do + @mock_facter_fqdn = mock("Facter FQDN") + @mock_facter_fqdn.stub!(:value).and_return("foo.bar.com") + @mock_facter_hostname = mock("Facter Hostname") + @mock_facter_hostname.stub!(:value).and_return("foo") + Facter.stub!(:[]).with("fqdn").and_return(@mock_facter_fqdn) + Facter.stub!(:[]).with("hostname").and_return(@mock_facter_hostname) + Facter.stub!(:each).and_return(true) + @client = Chef::Client.new + end + + it "should set the name equal to the FQDN" do + @client.build_node + @client.node.name.should eql("foo.bar.com") + end + + it "should set the name equal to the hostname if FQDN is not available" do + @mock_facter_fqdn.stub!(:value).and_return(nil) + @client.build_node + @client.node.name.should eql("foo") + end +end + +describe Chef::Client, "register" do + before(:each) do + @client = Chef::Client.new + end + + it "should check to see if it's already registered" + + it "should create a new passphrase if not registered" + + it "should create a new registration if it has not registered" +end
\ No newline at end of file diff --git a/chef/spec/unit/compile_spec.rb b/chef/spec/unit/compile_spec.rb new file mode 100644 index 0000000000..c7f76a4a58 --- /dev/null +++ b/chef/spec/unit/compile_spec.rb @@ -0,0 +1,71 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Compile do + before(:each) do + Chef::Config.node_path(File.join(File.dirname(__FILE__), "..", "data", "compile", "nodes")) + Chef::Config.cookbook_path(File.join(File.dirname(__FILE__), "..", "data", "compile", "cookbooks")) + @compile = Chef::Compile.new + end + + it "should create a new Chef::Compile" do + @compile.should be_a_kind_of(Chef::Compile) + end + + it "should have a Chef::CookbookLoader" do + @compile.cookbook_loader.should be_a_kind_of(Chef::CookbookLoader) + end + + it "should have a Chef::ResourceCollection" do + @compile.collection.should be_a_kind_of(Chef::ResourceCollection) + end + + it "should have a hash of Definitions" do + @compile.definitions.should be_a_kind_of(Hash) + end + + it "should load a node by name" do + node = Chef::Node.new + Chef::Node.stub!(:load).and_return(node) + lambda { + @compile.load_node("compile") + }.should_not raise_error + @compile.node.name.should == "compile" + end + + it "should load all the definitions" do + lambda { @compile.load_definitions }.should_not raise_error + @compile.definitions.should have_key(:new_cat) + end + + it "should load all the recipes specified for this node" do + node = Chef::Node.new + Chef::Node.stub!(:load).and_return(node) + @compile.load_node("compile") + @compile.load_definitions + lambda { @compile.load_recipes }.should_not raise_error + @compile.collection[0].to_s.should == "cat[einstein]" + @compile.collection[1].to_s.should == "cat[loulou]" + @compile.collection[2].to_s.should == "cat[birthday]" + @compile.collection[3].to_s.should == "cat[peanut]" + @compile.collection[4].to_s.should == "cat[fat peanut]" + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/config_spec.rb b/chef/spec/unit/config_spec.rb new file mode 100644 index 0000000000..51e33401d8 --- /dev/null +++ b/chef/spec/unit/config_spec.rb @@ -0,0 +1,91 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Config do + + it "should load a .rb file in context" do + lambda { + Chef::Config.from_file(File.join(File.dirname(__FILE__), "..", "data", "config.rb")) + }.should_not raise_error + end + + it "should raise an ArgumentError with an explanation if you try and set a non-existent variable" do + lambda { + Chef::Config.from_file(File.join(File.dirname(__FILE__), "..", "data", "bad-config.rb")) + }.should raise_error(ArgumentError) + end + + it "should raise an IOError if it can't find the file" do + lambda { + Chef::Config.from_file("/tmp/timmytimmytimmy") + }.should raise_error(IOError) + end + + it "should have a default cookbook_path" do + Chef::Config.cookbook_path.should be_kind_of(Array) + end + + it "should allow you to set a cookbook_path with a string" do + Chef::Config.cookbook_path("/etc/chef/cookbook") + Chef::Config.cookbook_path.should eql("/etc/chef/cookbook") + end + + it "should allow you to set a cookbook_path with multiple strings" do + Chef::Config.cookbook_path("/etc/chef/cookbook", "/etc/chef/upstream-cookbooks") + Chef::Config.cookbook_path.should eql([ + "/etc/chef/cookbook", + "/etc/chef/upstream-cookbooks" + ]) + end + + it "should allow you to set a cookbook_path with an array" do + Chef::Config.cookbook_path ["one", "two"] + Chef::Config.cookbook_path.should eql(["one", "two"]) + end + + it "should allow you to reference a value by index" do + Chef::Config[:cookbook_path].should be_kind_of(Array) + end + + it "should allow you to set a value by index" do + Chef::Config[:cookbook_path] = "one" + Chef::Config[:cookbook_path].should == "one" + end + + it "should allow you to set config values with a block" do + Chef::Config.configure do |c| + c[:cookbook_path] = "monkey_rabbit" + c[:otherthing] = "boo" + end + Chef::Config.cookbook_path.should == "monkey_rabbit" + Chef::Config.otherthing.should == "boo" + end + + it "should raise an ArgumentError if you access a config option that does not exist" do + lambda { Chef::Config[:snob_hobbery] }.should raise_error(ArgumentError) + end + + it "should return true or false with has_key?" do + Chef::Config.has_key?(:monkey).should eql(false) + Chef::Config[:monkey] = "gotcha" + Chef::Config.has_key?(:monkey).should eql(true) + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/cookbook_loader_spec.rb b/chef/spec/unit/cookbook_loader_spec.rb new file mode 100644 index 0000000000..501329777b --- /dev/null +++ b/chef/spec/unit/cookbook_loader_spec.rb @@ -0,0 +1,117 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::CookbookLoader do + before(:each) do + Chef::Config.cookbook_path [ + File.join(File.dirname(__FILE__), "..", "data", "cookbooks"), + File.join(File.dirname(__FILE__), "..", "data", "kitchen") + ] + @cl = Chef::CookbookLoader.new() + end + + it "should be a Chef::CookbookLoader object" do + @cl.should be_kind_of(Chef::CookbookLoader) + end + + it "should return cookbook objects with []" do + @cl[:openldap].should be_a_kind_of(Chef::Cookbook) + end + + it "should raise an exception if it cannot find a cookbook with []" do + lambda { @cl[:monkeypoop] }.should raise_error(ArgumentError) + end + + it "should allow you to look up available cookbooks with [] and a symbol" do + @cl[:openldap].name.should eql(:openldap) + end + + it "should allow you to look up available cookbooks with [] and a string" do + @cl["openldap"].name.should eql(:openldap) + end + + it "should allow you to iterate over cookbooks with each" do + seen = Hash.new + @cl.each do |cb| + seen[cb.name] = true + end + seen.should have_key(:openldap) + seen.should have_key(:apache2) + end + + it "should find all the cookbooks in the cookbook path" do + Chef::Config.cookbook_path << File.join(File.dirname(__FILE__), "..", "data", "hidden-cookbooks") + @cl.load_cookbooks + @cl.detect { |cb| cb.name == :openldap }.should_not eql(nil) + @cl.detect { |cb| cb.name == :apache2 }.should_not eql(nil) + end + + it "should allow you to override an attribute file via cookbook_path" do + @cl[:openldap].attribute_files.detect { |f| + f =~ /cookbooks\/openldap\/attributes\/default.rb/ + }.should_not eql(nil) + @cl[:openldap].attribute_files.detect { |f| + f =~ /kitchen\/openldap\/attributes\/default.rb/ + }.should eql(nil) + end + + it "should load different attribute files from deeper paths" do + @cl[:openldap].attribute_files.detect { |f| + f =~ /kitchen\/openldap\/attributes\/robinson.rb/ + }.should_not eql(nil) + end + + it "should allow you to override a definition file via cookbook_path" do + @cl[:openldap].definition_files.detect { |f| + f =~ /cookbooks\/openldap\/definitions\/client.rb/ + }.should_not eql(nil) + @cl[:openldap].definition_files.detect { |f| + f =~ /kitchen\/openldap\/definitions\/client.rb/ + }.should eql(nil) + end + + it "should load definition files from deeper paths" do + @cl[:openldap].definition_files.detect { |f| + f =~ /kitchen\/openldap\/definitions\/drewbarrymore.rb/ + }.should_not eql(nil) + end + + it "should allow you to override a recipe file via cookbook_path" do + @cl[:openldap].recipe_files.detect { |f| + f =~ /cookbooks\/openldap\/recipes\/gigantor.rb/ + }.should_not eql(nil) + @cl[:openldap].recipe_files.detect { |f| + f =~ /kitchen\/openldap\/recipes\/gigantor.rb/ + }.should eql(nil) + end + + it "should load recipe files from deeper paths" do + @cl[:openldap].recipe_files.detect { |f| + f =~ /kitchen\/openldap\/recipes\/woot.rb/ + }.should_not eql(nil) + end + + it "should allow you to have an 'ignore' file, which skips loading files in later cookbooks" do + @cl[:openldap].recipe_files.detect { |f| + f =~ /kitchen\/openldap\/recipes\/ignoreme.rb/ + }.should eql(nil) + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/cookbook_spec.rb b/chef/spec/unit/cookbook_spec.rb new file mode 100644 index 0000000000..85b4040771 --- /dev/null +++ b/chef/spec/unit/cookbook_spec.rb @@ -0,0 +1,142 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Cookbook do + COOKBOOK_PATH = File.join(File.dirname(__FILE__), "..", "data", "cookbooks", "openldap") + + before(:each) do + @cookbook = Chef::Cookbook.new("openldap") + end + + it "should be a Chef::Cookbook object" do + @cookbook.should be_kind_of(Chef::Cookbook) + end + + it "should have a name" do + @cookbook.name.should eql("openldap") + end + + it "should have a list of attribute files" do + @cookbook.attribute_files.should be_kind_of(Array) + end + + it "should allow you to set the list of attribute files" do + @cookbook.attribute_files = [ "one", "two" ] + @cookbook.attribute_files.should eql(["one", "two"]) + end + + it "should allow you to load all the attributes" do + node = Chef::Node.new + node.name "Julia Child" + node.chef_env false + @cookbook.attribute_files = Dir[File.join(COOKBOOK_PATH, "attributes", "**", "*.rb")] + node = @cookbook.load_attributes(node) + node.ldap_server.should eql("ops1prod") + node.ldap_basedn.should eql("dc=hjksolutions,dc=com") + node.ldap_replication_password.should eql("forsure") + node.smokey.should eql("robinson") + end + + it "should raise an ArgumentError if you don't pass a node object to load_attributes" do + lambda { @cookbook.load_attributes("snake oil") }.should raise_error(ArgumentError) + end + + it "should have a list of definition files" do + @cookbook.definition_files.should be_a_kind_of(Array) + end + + it "should allow you to set the list of definition files" do + @cookbook.definition_files = [ "one", "two" ] + @cookbook.definition_files.should eql(["one", "two"]) + end + + it "should allow you to load all the definitions, returning a hash of ResourceDefinitions by name" do + @cookbook.definition_files = Dir[File.join(COOKBOOK_PATH, "definitions", "**", "*.rb")] + defs = @cookbook.load_definitions + defs.has_key?(:openldap_server).should eql(true) + defs[:openldap_server].should be_a_kind_of(Chef::ResourceDefinition) + defs.has_key?(:openldap_client).should eql(true) + defs[:openldap_client].should be_a_kind_of(Chef::ResourceDefinition) + end + + it "should have a list of recipe files" do + @cookbook.recipe_files.should be_a_kind_of(Array) + end + + it "should allow you to set the list of recipe files" do + @cookbook.recipe_files = [ "one", "two" ] + @cookbook.recipe_files.should eql(["one", "two"]) + end + + it "should have a list of recipes by name" do + @cookbook.recipe_files = [ "one", "two" ] + @cookbook.recipes.detect { |r| r == "openldap::one" }.should eql("openldap::one") + @cookbook.recipes.detect { |r| r == "openldap::two" }.should eql("openldap::two") + end + + it "should take a file /path.rb, and use the filename minus rb as a recipe name" do + @cookbook.recipe_files = [ "/something/one.rb", "/otherthing/two.rb" ] + @cookbook.recipes.detect { |r| r == "openldap::one" }.should eql("openldap::one") + @cookbook.recipes.detect { |r| r == "openldap::two" }.should eql("openldap::two") + end + + it "should take a file path.rb, and use the filename minus rb as a recipe name" do + @cookbook.recipe_files = [ "one.rb", "two.rb" ] + @cookbook.recipes.detect { |r| r == "openldap::one" }.should eql("openldap::one") + @cookbook.recipes.detect { |r| r == "openldap::two" }.should eql("openldap::two") + end + + it "should allow you to test for a recipe with recipe?" do + @cookbook.recipe_files = [ "one", "two" ] + @cookbook.recipe?("one").should eql(true) + @cookbook.recipe?("shanghai").should eql(false) + end + + it "should allow you to test for a recipe? with a fq recipe name" do + @cookbook.recipe_files = [ "one", "two" ] + @cookbook.recipe?("openldap::one").should eql(true) + @cookbook.recipe?("shanghai::city").should eql(false) + end + + it "should allow you to run a recipe by name via load_recipe" do + @cookbook.recipe_files = Dir[File.join(COOKBOOK_PATH, "recipes", "**", "*.rb")] + node = Chef::Node.new + node.name "Julia Child" + recipe = @cookbook.load_recipe("openldap::gigantor", node) + recipe.recipe_name.should eql("gigantor") + recipe.cookbook_name.should eql("openldap") + recipe.collection[0].name.should eql("blanket") + end + + it "should raise an ArgumentException if you try to load a bad recipe name" do + node = Chef::Node.new + node.name "Julia Child" + lambda { @cookbook.load_recipe("smackdown", node) }.should raise_error(ArgumentError) + end + + it "should load the attributes if it has not already when a recipe is loaded" do + @cookbook.attribute_files = Dir[File.join(COOKBOOK_PATH, "attributes", "smokey.rb")] + @cookbook.recipe_files = Dir[File.join(COOKBOOK_PATH, "recipes", "**", "*.rb")] + node = Chef::Node.new + node.name "Julia Child" + recipe = @cookbook.load_recipe("openldap::gigantor", node) + node.smokey.should == "robinson" + end +end
\ No newline at end of file diff --git a/chef/spec/unit/couchdb_spec.rb b/chef/spec/unit/couchdb_spec.rb new file mode 100644 index 0000000000..621439df51 --- /dev/null +++ b/chef/spec/unit/couchdb_spec.rb @@ -0,0 +1,212 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::CouchDB, "new" do + it "should create a new Chef::REST object from the default url" do + Chef::Config[:couchdb_url] = "http://monkey" + Chef::REST.should_receive(:new).with("http://monkey").and_return(true) + Chef::CouchDB.new + end + + it "should create a new Chef::REST object from a provided url" do + Chef::REST.should_receive(:new).with("http://monkeypants").and_return(true) + Chef::CouchDB.new("http://monkeypants") + end +end + +describe Chef::CouchDB, "create_db" do + before(:each) do + @mock_rest = mock("Chef::REST", :null_object => true) + @mock_rest.stub!(:get_rest).and_return([ "chef" ]) + @mock_rest.stub!(:put_rest).and_return(true) + Chef::REST.stub!(:new).and_return(@mock_rest) + end + + def do_create_db + couch = Chef::CouchDB.new + couch.create_db + end + + it "should get a list of current databases" do + @mock_rest.should_receive(:get_rest).and_return(["chef"]) + do_create_db + end + + it "should create the chef database if it does not exist" do + @mock_rest.stub!(:get_rest).and_return([]) + @mock_rest.should_receive(:put_rest).with("chef", {}).and_return(true) + do_create_db + end + + it "should not create the chef database if it does exist" do + @mock_rest.stub!(:get_rest).and_return(["chef"]) + @mock_rest.should_not_receive(:put_rest) + do_create_db + end + + it "should return 'chef'" do + do_create_db.should eql("chef") + end +end + +describe Chef::CouchDB, "create_design_document" do + before(:each) do + @mock_rest = mock("Chef::REST", :null_object => true) + @mock_design = { + "version" => 1, + "_rev" => 1 + } + @mock_data = { + "version" => 1, + "language" => "javascript", + "views" => { + "all" => { + "map" => <<-EOJS + function(doc) { + if (doc.chef_type == "node") { + emit(doc.name, doc); + } + } + EOJS + }, + } + } + @mock_rest.stub!(:get_rest).and_return(@mock_design) + @mock_rest.stub!(:put_rest).and_return(true) + Chef::REST.stub!(:new).and_return(@mock_rest) + end + + def do_create_design_document + couchdb = Chef::CouchDB.new + couchdb.create_design_document("bob", @mock_data) + end + + it "should fetch the existing design document" do + @mock_rest.should_receive(:get_rest).with("chef/_design%2Fbob") + do_create_design_document + end + + it "should populate the _rev in the new design if the versions dont match" do + @mock_data["version"] = 2 + do_create_design_document + @mock_data["_rev"].should eql(1) + end + + it "should create the view if it requires updating" do + @mock_data["version"] = 2 + @mock_rest.should_receive(:put_rest).with("chef/_design%2Fbob", @mock_data) + do_create_design_document + end + + it "should not create the view if it does not require updating" do + @mock_data["version"] = 1 + @mock_rest.should_not_receive(:put_rest) + do_create_design_document + end +end + +describe Chef::CouchDB, "store" do + it "should put the object into couchdb" do + @mock_rest = mock("Chef::REST", :null_object => true) + @mock_rest.should_receive(:put_rest).with("chef/node_bob", {}).and_return(true) + Chef::REST.stub!(:new).and_return(@mock_rest) + Chef::CouchDB.new.store("node", "bob", {}) + end +end + +describe Chef::CouchDB, "load" do + it "should load the object from couchdb" do + @mock_rest = mock("Chef::REST", :null_object => true) + @mock_rest.should_receive(:get_rest).with("chef/node_bob").and_return(true) + Chef::REST.stub!(:new).and_return(@mock_rest) + Chef::CouchDB.new.load("node", "bob").should eql(true) + end +end + +describe Chef::CouchDB, "delete" do + before(:each) do + @mock_current = { + "version" => 1, + "_rev" => 1 + } + @mock_rest = mock("Chef::REST", :null_object => true) + @mock_rest.stub!(:get_rest).and_return(@mock_current) + @mock_rest.stub!(:delete_rest).and_return(true) + Chef::REST.stub!(:new).and_return(@mock_rest) + end + + def do_delete(rev=nil) + Chef::REST.stub!(:new).and_return(@mock_rest) + Chef::CouchDB.new.delete("node", "bob", rev) + end + + it "should remove the object from couchdb with a specific revision" do + @mock_rest.should_receive(:delete_rest).with("chef/node_bob?rev=1") + do_delete(1) + end + + it "should remove the object from couchdb based on the couchdb_rev of the current obj" do + mock_real = mock("Inflated Object") + mock_real.stub!(:respond_to?).and_return(true) + mock_real.stub!(:couchdb_rev).and_return(2) + @mock_rest.should_receive(:get_rest).with("chef/node_bob").and_return(mock_real) + @mock_rest.should_receive(:delete_rest).with("chef/node_bob?rev=2") + do_delete + end + + it "should remove the object from couchdb based on the current objects rev" do + @mock_rest.should_receive(:delete_rest).with("chef/node_bob?rev=1") + do_delete + end +end + +describe Chef::CouchDB, "list" do + before(:each) do + @mock_rest = mock("Chef::REST", :null_object => true) + Chef::REST.stub!(:new).and_return(@mock_rest) + end + + it "should get the view for all objects if inflate is true" do + @mock_rest.should_receive(:get_rest).with("chef/_view/node/all").and_return(true) + Chef::CouchDB.new.list("node", true) + end + + it "should get the view for just the object id's if inflate is false" do + @mock_rest.should_receive(:get_rest).with("chef/_view/node/all_id").and_return(true) + Chef::CouchDB.new.list("node", false) + end +end + +describe Chef::CouchDB, "has_key?" do + before(:each) do + @mock_rest = mock("Chef::REST", :null_object => true) + Chef::REST.stub!(:new).and_return(@mock_rest) + end + + it "should return true if the object exists" do + @mock_rest.should_receive(:get_rest).and_return(true) + Chef::CouchDB.new.has_key?("node", "bob").should eql(true) + end + + it "should return false if the object does not exist" do + @mock_rest.should_receive(:get_rest).and_raise(ArgumentError) + Chef::CouchDB.new.has_key?("node", "bob").should eql(false) + end +end diff --git a/chef/spec/unit/file_cache_spec.rb b/chef/spec/unit/file_cache_spec.rb new file mode 100644 index 0000000000..c84eea376e --- /dev/null +++ b/chef/spec/unit/file_cache_spec.rb @@ -0,0 +1,124 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::FileCache, "store method" do + before(:each) do + Chef::Config[:file_cache_path] = "/tmp/foo" + Dir.stub!(:mkdir).and_return(true) + File.stub!(:directory?).and_return(true) + @io = mock("IO", { :print => true, :close => true }) + File.stub!(:open).and_return(@io) + end + + it "should create the directories leading up to bang" do + File.stub!(:directory?).and_return(false) + Dir.should_receive(:mkdir).with("/tmp").and_return(true) + Dir.should_receive(:mkdir).with("/tmp/foo").and_return(true) + Dir.should_receive(:mkdir).with("/tmp/foo/whiz").and_return(true) + Dir.should_not_receive(:mkdir).with("/tmp/foo/whiz/bang").and_return(true) + Chef::FileCache.store("whiz/bang", "I found a poop") + end + + it "should create a file at /tmp/foo/whiz/bang" do + File.should_receive(:open).with("/tmp/foo/whiz/bang", "w").and_return(@io) + Chef::FileCache.store("whiz/bang", "I found a poop") + end + + it "should print the contents to the file" do + @io.should_receive(:print).with("I found a poop") + Chef::FileCache.store("whiz/bang", "I found a poop") + end + + it "should close the file" do + @io.should_receive(:close) + Chef::FileCache.store("whiz/bang", "I found a poop") + end + +end + +describe Chef::FileCache, "load method" do + before(:each) do + Chef::Config[:file_cache_path] = "/tmp/foo" + Dir.stub!(:mkdir).and_return(true) + File.stub!(:directory?).and_return(true) + File.stub!(:exists?).and_return(true) + File.stub!(:read).and_return("I found a poop") + end + + it "should find the full path to whiz/bang" do + File.should_receive(:read).with("/tmp/foo/whiz/bang").and_return(true) + Chef::FileCache.load('whiz/bang') + end + + it "should raise a Chef::Exception::FileNotFound if the file doesn't exist" do + File.stub!(:exists?).and_return(false) + lambda { Chef::FileCache.load('whiz/bang') }.should raise_error(Chef::Exception::FileNotFound) + end +end + +describe Chef::FileCache, "delete method" do + before(:each) do + Chef::Config[:file_cache_path] = "/tmp/foo" + Dir.stub!(:mkdir).and_return(true) + File.stub!(:directory?).and_return(true) + File.stub!(:exists?).and_return(true) + File.stub!(:unlink).and_return(true) + end + + it "should unlink the full path to whiz/bang" do + File.should_receive(:unlink).with("/tmp/foo/whiz/bang").and_return(true) + Chef::FileCache.delete("whiz/bang") + end + +end + +describe Chef::FileCache, "list method" do + before(:each) do + Chef::Config[:file_cache_path] = "/tmp/foo" + Dir.stub!(:[]).and_return(["/tmp/foo/whiz/bang", "/tmp/foo/snappy/patter"]) + File.stub!(:file?).and_return(true) + end + + it "should return the relative paths" do + Chef::FileCache.list.should eql([ "whiz/bang", "snappy/patter" ]) + end +end + +describe Chef::FileCache, "has_key? method" do + before(:each) do + Chef::Config[:file_cache_path] = "/tmp/foo" + end + + it "should check the full path to the file" do + File.should_receive(:exists?).with("/tmp/foo/whiz/bang") + Chef::FileCache.has_key?("whiz/bang") + end + + it "should return true if the file exists" do + File.stub!(:exists?).and_return(true) + Chef::FileCache.has_key?("whiz/bang").should eql(true) + end + + it "should return false if the file does not exist" do + File.stub!(:exists?).and_return(false) + Chef::FileCache.has_key?("whiz/bang").should eql(false) + end +end + diff --git a/chef/spec/unit/file_store_spec.rb b/chef/spec/unit/file_store_spec.rb new file mode 100644 index 0000000000..8addeb9e31 --- /dev/null +++ b/chef/spec/unit/file_store_spec.rb @@ -0,0 +1,105 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +class Fakestore + attr_accessor :name + + def to_json(*a) + { :name => @name }.to_json(*a) + end + + def self.json_create(o) + new_fakestore = new + new_fakestore.name = o[:name] + new_fakestore + end +end + +describe Chef::FileStore do + before(:each) do + Chef::Config[:file_store_path] = "/tmp/chef-test" + @fakestore = Fakestore.new + @fakestore.name = "Landslide" + @fakestore_digest = "a56a428bddac69e505731708ba206da0bb75e8de883bb4d5ef6be9b327da556a" + end + + it "should return a path to a file given a type and key" do + Dir.stub!(:mkdir).and_return(true) + File.stub!(:directory?).and_return(true) + path = Chef::FileStore.create_store_path("fakestore", @fakestore.name) + path.should eql("/tmp/chef-test/fakestore/a/56a/Landslide") + end + + it "should create directories for the path if needed" do + File.stub!(:directory?).and_return(false) + Dir.should_receive(:mkdir).exactly(4).times.and_return(true) + Chef::FileStore.create_store_path("fakestore", @fakestore.name) + end + + it "should store an object with a type and key" do + Chef::FileStore.should_receive(:create_store_path).with("fakestore", @fakestore.name).and_return("/monkey") + File.stub!(:directory?).and_return(true) + ioobj = mock("IO", :null_object => true) + ioobj.should_receive(:puts).with(@fakestore.to_json) + ioobj.should_receive(:close).once.and_return(true) + File.should_receive(:open).with("/monkey", "w").and_return(ioobj) + Chef::FileStore.store("fakestore", @fakestore.name, @fakestore) + end + + it "should load an object from the store with type and key" do + Chef::FileStore.should_receive(:create_store_path).with("fakestore", @fakestore.name).and_return("/monkey") + File.stub!(:exists?).and_return(true) + IO.should_receive(:read).once.and_return(true) + JSON.should_receive(:parse).and_return(true) + Chef::FileStore.load("fakestore", @fakestore.name) + end + + it "should through an exception if it cannot load a file from the store" do + Chef::FileStore.should_receive(:create_store_path).and_return("/tmp") + File.stub!(:exists?).and_return(false) + lambda { Chef::FileStore.load("fakestore", @fakestore.name) }.should raise_error(RuntimeError) + end + + it "should delete a file from the store if it exists" do + Chef::FileStore.should_receive(:create_store_path).with("node", "nothing").and_return("/tmp/foolio") + File.stub!(:exists?).and_return(true) + File.should_receive(:unlink).with("/tmp/foolio").and_return(1) + Chef::FileStore.delete("node", "nothing") + end + + it "should list all the keys of a particular type" do + Dir.should_receive(:[]).with("/tmp/chef-test/node/**/*").and_return(["pool"]) + File.should_receive(:file?).with("pool").and_return(true) + Chef::FileStore.list("node").should eql(["pool"]) + end + + it "should return all the documents of a particular type with list and inflate" do + Dir.stub!(:[]).and_return(["/foo/pool"]) + File.stub!(:file?).and_return(true) + Chef::FileStore.should_receive(:load).with("node", "pool").and_return("monkey") + Chef::FileStore.list("node", true).should eql(["monkey"]) + end + + it "should let you test whether a key doesnt exist for an object type with has_key?" do + Dir.should_receive(:[]).with("/tmp/chef-test/node/**/*").and_return(["pool"]) + File.should_receive(:file?).with("pool").and_return(true) + Chef::FileStore.has_key?("node", "snake").should eql(false) + end +end
\ No newline at end of file diff --git a/chef/spec/unit/log/formatter_spec.rb b/chef/spec/unit/log/formatter_spec.rb new file mode 100644 index 0000000000..e0d8bfd773 --- /dev/null +++ b/chef/spec/unit/log/formatter_spec.rb @@ -0,0 +1,51 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 'time' +require File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Log::Formatter do + before(:each) do + @formatter = Chef::Log::Formatter.new + end + + it "should print raw strings with msg2str(string)" do + @formatter.msg2str("nuthin new").should == "nuthin new" + end + + it "should format exceptions properly with msg2str(e)" do + e = IOError.new("legendary roots crew") + @formatter.msg2str(e).should == "legendary roots crew (IOError)\n" + end + + it "should format random objects via inspect with msg2str(Object)" do + @formatter.msg2str([ "black thought", "?uestlove" ]).should == '["black thought", "?uestlove"]' + end + + it "should return a formatted string with call" do + time = Time.new + Chef::Log::Formatter.show_time = true + @formatter.call("monkey", time, "test", "mos def").should == "[#{time.rfc2822}] monkey: mos def\n" + end + + it "should allow you to turn the time on and off in the output" do + Chef::Log::Formatter.show_time = false + @formatter.call("monkey", Time.new, "test", "mos def").should == "monkey: mos def\n" + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/log_spec.rb b/chef/spec/unit/log_spec.rb new file mode 100644 index 0000000000..14892b1af4 --- /dev/null +++ b/chef/spec/unit/log_spec.rb @@ -0,0 +1,62 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 'tempfile' +require 'logger' +require File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Log do + it "should accept regular options to Logger.new via init" do + tf = Tempfile.new("chef-test-log") + tf.open + lambda { Chef::Log.init(STDOUT) }.should_not raise_error + lambda { Chef::Log.init(tf) }.should_not raise_error + end + + it "should set the log level with :debug, :info, :warn, :error, or :fatal" do + levels = { + :debug => Logger::DEBUG, + :info => Logger::INFO, + :warn => Logger::WARN, + :error => Logger::ERROR, + :fatal => Logger::FATAL + } + levels.each do |symbol, constant| + Chef::Log.level(symbol) + Chef::Log.logger.level.should == constant + end + end + + it "should raise an ArgumentError if you try and set the level to something strange" do + lambda { Chef::Log.level(:the_roots) }.should raise_error(ArgumentError) + end + + it "should pass other method calls directly to logger" do + Chef::Log.level(:debug) + Chef::Log.should be_debug + lambda { Chef::Log.debug("Gimme some sugar!") }.should_not raise_error + end + + it "should default to STDOUT if init is called with no arguments" do + logger_mock = mock(Logger, :null_object => true) + Logger.stub!(:new).and_return(logger_mock) + Logger.should_receive(:new).with(STDOUT).and_return(logger_mock) + Chef::Log.init + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/mixin/params_validate_spec.rb b/chef/spec/unit/mixin/params_validate_spec.rb new file mode 100644 index 0000000000..937589b059 --- /dev/null +++ b/chef/spec/unit/mixin/params_validate_spec.rb @@ -0,0 +1,331 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +class TinyClass + include Chef::Mixin::ParamsValidate + + def music(is_good=true) + is_good + end +end + +describe Chef::Mixin::ParamsValidate do + before(:each) do + @vo = TinyClass.new() + end + + it "should allow a hash and a hash as arguments to validate" do + lambda { @vo.validate({:one => "two"}, {}) }.should_not raise_error(ArgumentError) + end + + it "should raise an argument error if validate is called incorrectly" do + lambda { @vo.validate("one", "two") }.should raise_error(ArgumentError) + end + + it "should require validation map keys to be symbols or strings" do + lambda { @vo.validate({:one => "two"}, { :one => true }) }.should_not raise_error(ArgumentError) + lambda { @vo.validate({:one => "two"}, { "one" => true }) }.should_not raise_error(ArgumentError) + lambda { @vo.validate({:one => "two"}, { Hash.new => true }) }.should raise_error(ArgumentError) + end + + it "should allow options to be required with true" do + lambda { @vo.validate({:one => "two"}, { :one => true }) }.should_not raise_error(ArgumentError) + end + + it "should allow options to be optional with false" do + lambda { @vo.validate({}, {:one => false})}.should_not raise_error(ArgumentError) + end + + it "should allow you to check what kind_of? thing an argument is with kind_of" do + lambda { + @vo.validate( + {:one => "string"}, + { + :one => { + :kind_of => String + } + } + ) + }.should_not raise_error(ArgumentError) + + lambda { + @vo.validate( + {:one => "string"}, + { + :one => { + :kind_of => Array + } + } + ) + }.should raise_error(ArgumentError) + end + + it "should allow you to specify an argument is required with required" do + lambda { + @vo.validate( + {:one => "string"}, + { + :one => { + :required => true + } + } + ) + }.should_not raise_error(ArgumentError) + + lambda { + @vo.validate( + {:two => "string"}, + { + :one => { + :required => true + } + } + ) + }.should raise_error(ArgumentError) + + lambda { + @vo.validate( + {:two => "string"}, + { + :one => { + :required => false + } + } + ) + }.should_not raise_error(ArgumentError) + end + + it "should allow you to specify whether an object has a method with respond_to" do + lambda { + @vo.validate( + {:one => @vo}, + { + :one => { + :respond_to => "validate" + } + } + ) + }.should_not raise_error(ArgumentError) + + lambda { + @vo.validate( + {:one => @vo}, + { + :one => { + :respond_to => "monkey" + } + } + ) + }.should raise_error(ArgumentError) + end + + it "should allow you to specify whether an object has all the given methods with respond_to and an array" do + lambda { + @vo.validate( + {:one => @vo}, + { + :one => { + :respond_to => ["validate", "music"] + } + } + ) + }.should_not raise_error(ArgumentError) + + lambda { + @vo.validate( + {:one => @vo}, + { + :one => { + :respond_to => ["monkey", "validate"] + } + } + ) + }.should raise_error(ArgumentError) + end + + it "should let you set a default value with default => value" do + arguments = Hash.new + @vo.validate(arguments, { + :one => { + :default => "is the loneliest number" + } + }) + arguments[:one].should == "is the loneliest number" + end + + it "should let you check regular expressions" do + lambda { + @vo.validate( + { :one => "is good" }, + { + :one => { + :regex => /^is good$/ + } + } + ) + }.should_not raise_error(ArgumentError) + + lambda { + @vo.validate( + { :one => "is good" }, + { + :one => { + :regex => /^is bad$/ + } + } + ) + }.should raise_error(ArgumentError) + end + + it "should let you specify your own callbacks" do + lambda { + @vo.validate( + { :one => "is good" }, + { + :one => { + :callbacks => { + "should be equal to is good" => lambda { |a| + a == "is good" + }, + } + } + } + ) + }.should_not raise_error(ArgumentError) + + lambda { + @vo.validate( + { :one => "is bad" }, + { + :one => { + :callbacks => { + "should be equal to 'is good'" => lambda { |a| + a == "is good" + }, + } + } + } + ) + }.should raise_error(ArgumentError) + end + + it "should let you combine checks" do + args = { :one => "is good", :two => "is bad" } + lambda { + @vo.validate( + args, + { + :one => { + :kind_of => String, + :respond_to => [ :to_s, :upcase ], + :regex => /^is good/, + :callbacks => { + "should be your friend" => lambda { |a| + a == "is good" + } + }, + :required => true + }, + :two => { + :kind_of => String, + :required => false + }, + :three => { :default => "neato mosquito" } + } + ) + }.should_not raise_error(ArgumentError) + args[:three].should == "neato mosquito" + lambda { + @vo.validate( + args, + { + :one => { + :kind_of => String, + :respond_to => [ :to_s, :upcase ], + :regex => /^is good/, + :callbacks => { + "should be your friend" => lambda { |a| + a == "is good" + } + }, + :required => true + }, + :two => { + :kind_of => Hash, + :required => false + }, + :three => { :default => "neato mosquito" } + } + ) + }.should raise_error(ArgumentError) + end + + it "should raise an ArgumentError if the validation map has an unknown check" do + lambda { @vo.validate( + { :one => "two" }, + { + :one => { + :busted => "check" + } + } + ) + }.should raise_error(ArgumentError) + end + + it "should accept keys that are strings in the options" do + lambda { + @vo.validate({ "one" => "two" }, { :one => { :regex => /^two$/ }}) + }.should_not raise_error(ArgumentError) + end + + it "should allow an array to kind_of" do + lambda { + @vo.validate( + {:one => "string"}, + { + :one => { + :kind_of => [ String, Array ] + } + } + ) + }.should_not raise_error(ArgumentError) + lambda { + @vo.validate( + {:one => ["string"]}, + { + :one => { + :kind_of => [ String, Array ] + } + } + ) + }.should_not raise_error(ArgumentError) + lambda { + @vo.validate( + {:one => Hash.new}, + { + :one => { + :kind_of => [ String, Array ] + } + } + ) + }.should raise_error(ArgumentError) + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/mixin/template_spec.rb b/chef/spec/unit/mixin/template_spec.rb new file mode 100644 index 0000000000..0ff929e8f6 --- /dev/null +++ b/chef/spec/unit/mixin/template_spec.rb @@ -0,0 +1,60 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +class TinyTemplateClass; include Chef::Mixin::Template; end + +describe Chef::Mixin::Template, "render_template" do + + before(:each) do + @template = "abcnews" + @context = { :fine => "dear" } + @eruby = mock(:erubis, { :evaluate => "elvis costello" }) + Erubis::Eruby.stub!(:new).and_return(@eruby) + @tempfile = mock(:tempfile, { :print => true, :close => true }) + Tempfile.stub!(:new).and_return(@tempfile) + @tiny_template = TinyTemplateClass.new + end + + it "should create a new Erubis object from the template" do + Erubis::Eruby.should_receive(:new).with("abcnews").and_return(@eruby) + @tiny_template.render_template(@template, @context) + end + + it "should evaluate the template with the provided context" do + @eruby.should_receive(:evaluate).with(@context).and_return(true) + @tiny_template.render_template(@template, @context) + end + + it "should create a tempfile for the resulting file" do + Tempfile.should_receive(:new).and_return(@tempfile) + @tiny_template.render_template(@template, @context) + end + + it "should print the contents of the resulting template to the tempfile" do + @tempfile.should_receive(:print).with("elvis costello").and_return(true) + @tiny_template.render_template(@template, @context) + end + + it "should close the tempfile" do + @tempfile.should_receive(:close).and_return(true) + @tiny_template.render_template(@template, @context) + end +end + diff --git a/chef/spec/unit/node_spec.rb b/chef/spec/unit/node_spec.rb new file mode 100644 index 0000000000..605bb8b628 --- /dev/null +++ b/chef/spec/unit/node_spec.rb @@ -0,0 +1,326 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Node, "new method" do + before(:each) do + Chef::Config.node_path(File.join(File.dirname(__FILE__), "..", "data", "nodes")) + @node = Chef::Node.new() + end + + it "should create a new Chef::Node" do + @node.should be_a_kind_of(Chef::Node) + end +end + +describe Chef::Node, "name" do + before(:each) do + Chef::Config.node_path(File.join(File.dirname(__FILE__), "..", "data", "nodes")) + @node = Chef::Node.new() + end + + it "should allow you to set a name with name(something)" do + lambda { @node.name("latte") }.should_not raise_error + end + + it "should return the name with name()" do + @node.name("latte") + @node.name.should eql("latte") + end + + it "should always have a string for name" do + lambda { @node.name(Hash.new) }.should raise_error(ArgumentError) + end + +end + +describe Chef::Node, "attributes" do + before(:each) do + Chef::Config.node_path(File.join(File.dirname(__FILE__), "..", "data", "nodes")) + @node = Chef::Node.new() + end + + it "should have attributes" do + @node.attribute.should be_a_kind_of(Hash) + end + + it "should allow attributes to be accessed by name or symbol directly on node[]" do + @node.attribute["locust"] = "something" + @node[:locust].should eql("something") + @node["locust"].should eql("something") + end + + it "should return nil if it cannot find an attribute with node[]" do + @node["secret"].should eql(nil) + end + + it "should allow you to set an attribute via node[]=" do + @node["secret"] = "shush" + @node["secret"].should eql("shush") + end + + it "should allow you to query whether an attribute exists with attribute?" do + @node.attribute["locust"] = "something" + @node.attribute?("locust").should eql(true) + @node.attribute?("no dice").should eql(false) + end + + it "should allow you to set an attribute via method_missing" do + @node.sunshine "is bright" + @node.attribute[:sunshine].should eql("is bright") + end + + it "should allow you get get an attribute via method_missing" do + @node.sunshine "is bright" + @node.sunshine.should eql("is bright") + end + + it "should raise an ArgumentError if you ask for an attribute that doesn't exist via method_missing" do + lambda { @node.sunshine }.should raise_error(ArgumentError) + end + + it "should allow you to iterate over attributes with each_attribute" do + @node.sunshine "is bright" + @node.canada "is a nice place" + seen_attributes = Hash.new + @node.each_attribute do |a,v| + seen_attributes[a] = v + end + seen_attributes.should have_key(:sunshine) + seen_attributes.should have_key(:canada) + seen_attributes[:sunshine].should == "is bright" + seen_attributes[:canada].should == "is a nice place" + end + +end + +describe Chef::Node, "recipes" do + before(:each) do + Chef::Config.node_path(File.join(File.dirname(__FILE__), "..", "data", "nodes")) + @node = Chef::Node.new() + end + + it "should have an array of recipes that should be applied" do + @node.recipes.should be_a_kind_of(Array) + end + + it "should allow you to query whether or not it has a recipe applied with recipe?" do + @node.recipes << "sunrise" + @node.recipe?("sunrise").should eql(true) + @node.recipe?("not at home").should eql(false) + end + + it "should allow you to set recipes with arguments" do + @node.recipes "one", "two" + @node.recipe?("one").should eql(true) + @node.recipe?("two").should eql(true) + end + +end + +describe Chef::Node, "from file" do + before(:each) do + Chef::Config.node_path(File.join(File.dirname(__FILE__), "..", "data", "nodes")) + @node = Chef::Node.new() + end + + it "should load a node from a ruby file" do + @node.from_file(File.join(File.dirname(__FILE__), "..", "data", "nodes", "test.rb")) + @node.name.should eql("test.example.com short") + @node.sunshine.should eql("in") + @node.something.should eql("else") + @node.recipes.should eql(["operations-master", "operations-monitoring"]) + end + + it "should raise an exception if the file cannot be found or read" do + lambda { @node.from_file("/tmp/monkeydiving") }.should raise_error(IOError) + end +end + +describe Chef::Node, "find_file" do + before(:each) do + Chef::Config.node_path(File.join(File.dirname(__FILE__), "..", "data", "nodes")) + @node = Chef::Node.new() + end + + it "should load a node from a file by fqdn" do + @node.find_file("test.example.com") + @node.name.should == "test.example.com" + end + + it "should load a node from a file by hostname" do + File.stub!(:exists?).and_return(true) + File.should_receive(:exists?).with(File.join(Chef::Config[:node_path], "test.example.com.rb")).and_return(false) + @node.find_file("test.example.com") + @node.name.should == "test.example.com short" + end + + it "should load a node from the default file" do + File.stub!(:exists?).and_return(true) + File.should_receive(:exists?).with(File.join(Chef::Config[:node_path], "test.example.com.rb")).and_return(false) + File.should_receive(:exists?).with(File.join(Chef::Config[:node_path], "test.rb")).and_return(false) + @node.find_file("test.example.com") + @node.name.should == "test.example.com default" + end + + it "should raise an ArgumentError if it cannot find any node file at all" do + File.stub!(:exists?).and_return(true) + File.should_receive(:exists?).with(File.join(Chef::Config[:node_path], "test.example.com.rb")).and_return(false) + File.should_receive(:exists?).with(File.join(Chef::Config[:node_path], "test.rb")).and_return(false) + File.should_receive(:exists?).with(File.join(Chef::Config[:node_path], "default.rb")).and_return(false) + lambda { @node.find_file("test.example.com") }.should raise_error(ArgumentError) + end +end + +describe Chef::Node, "json" do + before(:each) do + Chef::Config.node_path(File.join(File.dirname(__FILE__), "..", "data", "nodes")) + @node = Chef::Node.new() + end + + it "should serialize itself as json" do + @node.find_file("test.example.com") + json = @node.to_json() + json.should =~ /json_class/ + json.should =~ /name/ + json.should =~ /attributes/ + json.should =~ /recipes/ + end + + it "should deserialize itself from json" do + @node.find_file("test.example.com") + json = @node.to_json + serialized_node = JSON.parse(json) + serialized_node.should be_a_kind_of(Chef::Node) + serialized_node.name.should eql(@node.name) + @node.each_attribute do |k,v| + serialized_node[k].should eql(v) + end + serialized_node.recipes.should eql(@node.recipes) + end +end + +describe Chef::Node, "to_index" do + before(:each) do + Chef::Config.node_path(File.join(File.dirname(__FILE__), "..", "data", "nodes")) + @node = Chef::Node.new() + @node.foo("bar") + end + + it "should return a hash with :index attributes" do + @node.name("airplane") + @node.to_index.should == { :foo => "bar", :index_name => "node", :id => "node_airplane", :name => "airplane" } + end +end + + +describe Chef::Node, "to_s" do + before(:each) do + Chef::Config.node_path(File.join(File.dirname(__FILE__), "..", "data", "nodes")) + @node = Chef::Node.new() + end + + it "should turn into a string like node[name]" do + @node.name("airplane") + @node.to_s.should eql("node[airplane]") + end +end + +describe Chef::Node, "list" do + before(:each) do + mock_couch = mock("Chef::CouchDB") + mock_couch.stub!(:list).and_return( + { + "rows" => [ + { + "value" => "a", + "key" => "avenue" + } + ] + } + ) + Chef::CouchDB.stub!(:new).and_return(mock_couch) + end + + it "should retrieve a list of nodes from CouchDB" do + Chef::Node.list.should eql(["avenue"]) + end + + it "should return just the ids if inflate is false" do + Chef::Node.list(false).should eql(["avenue"]) + end + + it "should return the full objects if inflate is true" do + Chef::Node.list(true).should eql(["a"]) + end +end + +describe Chef::Node, "load" do + it "should load a node from couchdb by name" do + mock_couch = mock("Chef::CouchDB") + mock_couch.should_receive(:load).with("node", "coffee").and_return(true) + Chef::CouchDB.stub!(:new).and_return(mock_couch) + Chef::Node.load("coffee") + end +end + +describe Chef::Node, "destroy" do + it "should delete this node from couchdb" do + mock_couch = mock("Chef::CouchDB") + mock_couch.should_receive(:delete).with("node", "bob", 1).and_return(true) + Chef::CouchDB.stub!(:new).and_return(mock_couch) + node = Chef::Node.new + node.name "bob" + node.couchdb_rev = 1 + Chef::Queue.should_receive(:send_msg).with(:queue, :remove, node) + node.destroy + end +end + +describe Chef::Node, "save" do + before(:each) do + @mock_couch = mock("Chef::CouchDB") + @mock_couch.stub!(:store).and_return({ "rev" => 33 }) + Chef::CouchDB.stub!(:new).and_return(@mock_couch) + Chef::Queue.stub!(:send_msg).and_return(true) + @node = Chef::Node.new + @node.name "bob" + @node.couchdb_rev = 1 + end + + it "should save the node to couchdb" do + Chef::Queue.should_receive(:send_msg).with(:queue, :index, @node) + @mock_couch.should_receive(:store).with("node", "bob", @node).and_return({ "rev" => 33 }) + @node.save + end + + it "should store the new couchdb_rev" do + @node.save + @node.couchdb_rev.should eql(33) + end +end + +describe Chef::Node, "create_design_document" do + it "should create our design document" do + mock_couch = mock("Chef::CouchDB") + mock_couch.should_receive(:create_design_document).with("nodes", Chef::Node::DESIGN_DOCUMENT) + Chef::CouchDB.stub!(:new).and_return(mock_couch) + Chef::Node.create_design_document + end +end diff --git a/chef/spec/unit/openid_registration_spec.rb b/chef/spec/unit/openid_registration_spec.rb new file mode 100644 index 0000000000..d849369796 --- /dev/null +++ b/chef/spec/unit/openid_registration_spec.rb @@ -0,0 +1,153 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::OpenIDRegistration, "initialize" do + it "should return a new Chef::OpenIDRegistration object" do + Chef::OpenIDRegistration.new.should be_kind_of(Chef::OpenIDRegistration) + end +end + +describe Chef::OpenIDRegistration, "set_password" do + it "should generate a salt for this object" do + oreg = Chef::OpenIDRegistration.new + oreg.salt.should eql(nil) + oreg.set_password("foolio") + oreg.salt.should_not eql(nil) + end + + it "should encrypt the password with the salt and the plaintext password" do + oreg = Chef::OpenIDRegistration.new + oreg.set_password("foolio") + oreg.password.should_not eql(nil) + end +end + +describe Chef::OpenIDRegistration, "to_json" do + it "should serialize itself as json" do + oreg = Chef::OpenIDRegistration.new + oreg.set_password("monkey") + json = oreg.to_json + %w{json_class chef_type name salt password validated}.each do |verify| + json.should =~ /#{verify}/ + end + end +end + +describe Chef::OpenIDRegistration, "from_json" do + it "should serialize itself as json" do + oreg = Chef::OpenIDRegistration.new() + oreg.name = "foobar" + oreg.set_password("monkey") + oreg_json = oreg.to_json + nreg = JSON.parse(oreg_json) + nreg.should be_a_kind_of(Chef::OpenIDRegistration) + %w{name salt password validated}.each do |verify| + nreg.send(verify.to_sym).should eql(oreg.send(verify.to_sym)) + end + end +end + +describe Chef::OpenIDRegistration, "list" do + before(:each) do + @mock_couch = mock("Chef::CouchDB") + @mock_couch.stub!(:list).and_return({ + "rows" => [ + { + "value" => "a", + "key" => "avenue" + } + ] + }) + Chef::CouchDB.stub!(:new).and_return(@mock_couch) + end + + it "should retrieve a list of nodes from CouchDB" do + Chef::OpenIDRegistration.list.should eql(["avenue"]) + end + + it "should return just the ids if inflate is false" do + Chef::OpenIDRegistration.list(false).should eql(["avenue"]) + end + + it "should return the full objects if inflate is true" do + Chef::OpenIDRegistration.list(true).should eql(["a"]) + end +end + +describe Chef::OpenIDRegistration, "load" do + it "should load a registration from couchdb by name" do + @mock_couch = mock("Chef::CouchDB") + Chef::CouchDB.stub!(:new).and_return(@mock_couch) + @mock_couch.should_receive(:load).with("openid_registration", "coffee").and_return(true) + Chef::OpenIDRegistration.load("coffee") + end +end + +describe Chef::OpenIDRegistration, "destroy" do + it "should delete this registration from couchdb" do + @mock_couch = mock("Chef::CouchDB") + @mock_couch.should_receive(:delete).with("openid_registration", "bob", 1).and_return(true) + Chef::CouchDB.stub!(:new).and_return(@mock_couch) + reg = Chef::OpenIDRegistration.new + reg.name = "bob" + reg.couchdb_rev = 1 + reg.destroy + end +end + +describe Chef::OpenIDRegistration, "save" do + before(:each) do + @mock_couch = mock("Chef::CouchDB") + Chef::CouchDB.stub!(:new).and_return(@mock_couch) + @reg = Chef::OpenIDRegistration.new + @reg.name = "bob" + @reg.couchdb_rev = 1 + end + + it "should save the registration to couchdb" do + @mock_couch.should_receive(:store).with("openid_registration", "bob", @reg).and_return({ "rev" => 33 }) + @reg.save + end + + it "should store the new couchdb_rev" do + @mock_couch.stub!(:store).with("openid_registration", "bob", @reg).and_return({ "rev" => 33 }) + @reg.save + @reg.couchdb_rev.should eql(33) + end +end + +describe Chef::OpenIDRegistration, "create_design_document" do + it "should create our design document" do + mock_couch = mock("Chef::CouchDB") + mock_couch.should_receive(:create_design_document).with("registrations", Chef::OpenIDRegistration::DESIGN_DOCUMENT) + Chef::CouchDB.stub!(:new).and_return(mock_couch) + Chef::OpenIDRegistration.create_design_document + end +end + +describe Chef::OpenIDRegistration, "has_key?" do + it "should check with CouchDB for a registration with this key" do + @mock_couch = mock("Chef::CouchDB") + @mock_couch.should_receive(:has_key?).with("openid_registration", "bob").and_return(true) + Chef::CouchDB.stub!(:new).and_return(@mock_couch) + Chef::OpenIDRegistration.has_key?("bob") + end +end + diff --git a/chef/spec/unit/platform_spec.rb b/chef/spec/unit/platform_spec.rb new file mode 100644 index 0000000000..4eeeebed89 --- /dev/null +++ b/chef/spec/unit/platform_spec.rb @@ -0,0 +1,209 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Platform do + before(:each) do + Chef::Platform.platforms = { + :darwin => { + "9.2.2" => { + :file => "darwinian", + :else => "thing" + }, + :default => { + :file => "old school", + :snicker => "snack" + } + }, + :mars_volta => { + }, + :default => { + :file => Chef::Provider::File, + :pax => "brittania", + :cat => "nice" + } + } + end + + it "should allow you to look up a platform by name and version, returning the provider map for it" do + pmap = Chef::Platform.find("Darwin", "9.2.2") + pmap.should be_a_kind_of(Hash) + pmap[:file].should eql("darwinian") + end + + it "should use the default providers for an os if the specific version does not exist" do + pmap = Chef::Platform.find("Darwin", "1") + pmap.should be_a_kind_of(Hash) + pmap[:file].should eql("old school") + end + + it "should use the default providers if the os doesn't give me a default, but does exist" do + pmap = Chef::Platform.find("mars_volta", "1") + pmap.should be_a_kind_of(Hash) + pmap[:file].should eql(Chef::Provider::File) + end + + it "should use the default provider if the os does not exist" do + pmap = Chef::Platform.find("AIX", "1") + pmap.should be_a_kind_of(Hash) + pmap[:file].should eql(Chef::Provider::File) + end + + it "should merge the defaults for an os with the specific version" do + pmap = Chef::Platform.find("Darwin", "9.2.2") + pmap[:file].should eql("darwinian") + pmap[:snicker].should eql("snack") + end + + it "should merge the defaults for an os with the universal defaults" do + pmap = Chef::Platform.find("Darwin", "9.2.2") + pmap[:file].should eql("darwinian") + pmap[:pax].should eql("brittania") + end + + it "should allow you to look up a provider for a platform directly by symbol" do + Chef::Platform.find_provider("Darwin", "9.2.2", :file).should eql("darwinian") + end + + it "should raise an exception if a provider cannot be found for a resource type" do + lambda { Chef::Platform.find_provider("Darwin", "9.2.2", :coffee) }.should raise_error(ArgumentError) + end + + it "should look up a provider for a resource with a Chef::Resource object" do + kitty = Chef::Resource::Cat.new("loulou") + Chef::Platform.find_provider("Darwin", "9.2.2", kitty) + end + + it "should look up a provider with a node and a Chef::Resource object" do + kitty = Chef::Resource::Cat.new("loulou") + node = Chef::Node.new + node.name("Intel") + node.operatingsystem("Darwin") + node.operatingsystemversion("9.2.2") + Chef::Platform.find_provider_for_node(node, kitty).should eql("nice") + end + + it "should prefer lsbdistid over operatingsystem when looking up via node" do + kitty = Chef::Resource::Cat.new("loulou") + node = Chef::Node.new + node.name("Intel") + node.operatingsystem("Darwin") + node.operatingsystemversion("9.2.2") + node.lsbdistid("Not Linux") + Chef::Platform.set( + :platform => :not_linux, + :resource => :cat, + :provider => "bourbon" + ) + Chef::Platform.find_provider_for_node(node, kitty).should eql("bourbon") + end + + it "should prefer macosx_productnmae over operatingsystem when looking up via node" do + kitty = Chef::Resource::Cat.new("loulou") + node = Chef::Node.new + node.name("Intel") + node.operatingsystem("Darwin") + node.operatingsystemversion("9.2.2") + node.macosx_productname("Mac OS X") + Chef::Platform.set( + :platform => :mac_os_x, + :resource => :cat, + :provider => "bourbon" + ) + Chef::Platform.find_provider_for_node(node, kitty).should eql("bourbon") + end + + it "should prefer lsbdistrelease over operatingsystem when looking up via node" do + kitty = Chef::Resource::Cat.new("loulou") + node = Chef::Node.new + node.name("Intel") + node.operatingsystem("Darwin") + node.operatingsystemversion("9.2.2") + node.lsbdistrelease("10") + Chef::Platform.set( + :platform => :darwin, + :version => "10", + :resource => :cat, + :provider => "bourbon" + ) + Chef::Platform.find_provider_for_node(node, kitty).should eql("bourbon") + end + + it "should prefer macosx_productversion over operatingsystem when looking up via node" do + kitty = Chef::Resource::Cat.new("loulou") + node = Chef::Node.new + node.name("Intel") + node.operatingsystem("Darwin") + node.operatingsystemversion("9.2.2") + node.macosx_productversion("10") + Chef::Platform.set( + :platform => :darwin, + :version => "10", + :resource => :cat, + :provider => "bourbon" + ) + Chef::Platform.find_provider_for_node(node, kitty).should eql("bourbon") + end + + it "should update the provider map with map" do + Chef::Platform.set( + :platform => :darwin, + :version => "9.2.2", + :resource => :file, + :provider => "masterful" + ) + Chef::Platform.platforms[:darwin]["9.2.2"][:file].should eql("masterful") + Chef::Platform.set( + :platform => :darwin, + :resource => :file, + :provider => "masterful" + ) + Chef::Platform.platforms[:darwin][:default][:file].should eql("masterful") + Chef::Platform.set( + :resource => :file, + :provider => "masterful" + ) + Chef::Platform.platforms[:default][:file].should eql("masterful") + + Chef::Platform.set( + :platform => :hero, + :version => "9.2.2", + :resource => :file, + :provider => "masterful" + ) + Chef::Platform.platforms[:hero]["9.2.2"][:file].should eql("masterful") + + Chef::Platform.set( + :resource => :file, + :provider => "masterful" + ) + Chef::Platform.platforms[:default][:file].should eql("masterful") + + Chef::Platform.platforms = {} + + Chef::Platform.set( + :resource => :file, + :provider => "masterful" + ) + Chef::Platform.platforms[:default][:file].should eql("masterful") + + end + + +end
\ No newline at end of file diff --git a/chef/spec/unit/provider/directory_spec.rb b/chef/spec/unit/provider/directory_spec.rb new file mode 100644 index 0000000000..4978b27288 --- /dev/null +++ b/chef/spec/unit/provider/directory_spec.rb @@ -0,0 +1,98 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 'ostruct' + +require File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Provider::Directory do + before(:each) do + @new_resource = mock("New Resource", :null_object => true) + @new_resource.stub!(:name).and_return("directory") + @new_resource.stub!(:path).and_return("/tmp") + @new_resource.stub!(:owner).and_return(500) + @new_resource.stub!(:group).and_return(500) + @new_resource.stub!(:mode).and_return(0644) + @new_resource.stub!(:updated).and_return(false) + @node = Chef::Node.new + @node.name "latte" + @directory = Chef::Provider::Directory.new(@node, @new_resource) + end + + it "should load the current resource based on the new resource" do + File.should_receive(:exist?).once.and_return(true) + File.should_receive(:directory?).once.and_return(true) + cstats = mock("stats", :null_object => true) + cstats.stub!(:uid).and_return(500) + cstats.stub!(:gid).and_return(500) + cstats.stub!(:mode).and_return(0755) + File.should_receive(:stat).once.and_return(cstats) + @directory.load_current_resource + @directory.current_resource.path.should eql(@new_resource.path) + @directory.current_resource.owner.should eql(500) + @directory.current_resource.group.should eql(500) + @directory.current_resource.mode.should eql("755") + end + + it "should create a new directory on create, setting updated to true" do + load_mock_provider + File.should_receive(:exists?).once.and_return(false) + Dir.should_receive(:mkdir).with(@new_resource.path).once.and_return(true) + @directory.new_resource.should_receive(:updated=).with(true) + @directory.should_receive(:set_owner).once.and_return(true) + @directory.should_receive(:set_group).once.and_return(true) + @directory.should_receive(:set_mode).once.and_return(true) + @directory.action_create + end + + it "should not create the directory if it already exists" do + load_mock_provider + File.should_receive(:exists?).once.and_return(true) + Dir.should_not_receive(:mkdir).with(@new_resource.path) + @directory.stub!(:set_owner).and_return(true) + @directory.stub!(:set_group).and_return(true) + @directory.stub!(:set_mode).and_return(true) + @directory.action_create + end + + it "should delete the directory if it exists, and is writable with action_delete" do + load_mock_provider + File.should_receive(:exists?).once.and_return(true) + File.should_receive(:writable?).once.and_return(true) + Dir.should_receive(:delete).with(@new_resource.path).once.and_return(true) + @directory.action_delete + end + + it "should raise an exception if it cannot delete the file due to bad permissions" do + load_mock_provider + File.stub!(:exists?).and_return(true) + File.stub!(:writable?).and_return(false) + lambda { @directory.action_delete }.should raise_error(RuntimeError) + end + + def load_mock_provider + File.stub!(:exist?).and_return(true) + File.stub!(:directory?).and_return(true) + cstats = mock("stats", :null_object => true) + cstats.stub!(:uid).and_return(500) + cstats.stub!(:gid).and_return(500) + cstats.stub!(:mode).and_return(0755) + File.stub!(:stat).once.and_return(cstats) + @directory.load_current_resource + end +end
\ No newline at end of file diff --git a/chef/spec/unit/provider/file_spec.rb b/chef/spec/unit/provider/file_spec.rb new file mode 100644 index 0000000000..9ea628a357 --- /dev/null +++ b/chef/spec/unit/provider/file_spec.rb @@ -0,0 +1,224 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 'ostruct' + +require File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Provider::File do + before(:each) do + @resource = Chef::Resource::File.new("seattle") + @resource.path(File.join(File.dirname(__FILE__), "..", "..", "data", "seattle.txt")) + @node = Chef::Node.new + @node.name "latte" + @provider = Chef::Provider::File.new(@node, @resource) + end + + it "should return a Chef::Provider::File" do + @provider.should be_a_kind_of(Chef::Provider::File) + end + + it "should store the resource passed to new as new_resource" do + @provider.new_resource.should eql(@resource) + end + + it "should store the node passed to new as node" do + @provider.node.should eql(@node) + end + + it "should load a current resource based on the one specified at construction" do + @provider.load_current_resource + @provider.current_resource.should be_a_kind_of(Chef::Resource::File) + @provider.current_resource.name.should eql(@resource.name) + @provider.current_resource.path.should eql(@resource.path) + @provider.current_resource.owner.should_not eql(nil) + @provider.current_resource.group.should_not eql(nil) + @provider.current_resource.mode.should_not eql(nil) + end + + it "should load a mostly blank current resource if the file specified in new_resource doesn't exist/isn't readable" do + resource = Chef::Resource::File.new("seattle") + resource.path(File.join(File.dirname(__FILE__), "..", "..", "data", "woot.txt")) + node = Chef::Node.new + node.name "latte" + provider = Chef::Provider::File.new(node, resource) + provider.load_current_resource + provider.current_resource.should be_a_kind_of(Chef::Resource::File) + provider.current_resource.name.should eql(resource.name) + provider.current_resource.path.should eql(resource.path) + provider.current_resource.owner.should eql(nil) + provider.current_resource.group.should eql(nil) + provider.current_resource.mode.should eql(nil) + end + + it "should load the correct value for owner of the current resource" do + stats = File.stat(@resource.path) + @provider.load_current_resource + @provider.current_resource.owner.should eql(stats.uid) + end + + it "should load an md5 sum for an existing file" do + @provider.load_current_resource + @provider.current_resource.checksum("8d6152c7d62ea9188eda596c4d31e732") + end + + it "should compare the current owner with the requested owner" do + @provider.load_current_resource + @provider.new_resource.stub!(:owner).and_return("adam") + Etc.stub!(:getpwnam).and_return( + OpenStruct.new( + :name => "adam", + :passwd => "foo", + :uid => 501, + :gid => 501, + :gecos => "Adam Jacob", + :dir => "/Users/adam", + :shell => "/bin/zsh", + :change => "0", + :uclass => "", + :expire => 0 + ) + ) + @provider.current_resource.owner(501) + @provider.compare_owner.should eql(true) + + @provider.current_resource.owner(777) + @provider.compare_owner.should eql(false) + + @provider.new_resource.stub!(:owner).and_return(501) + @provider.current_resource.owner(501) + @provider.compare_owner.should eql(true) + + @provider.new_resource.stub!(:owner).and_return("501") + @provider.current_resource.owner(501) + @provider.compare_owner.should eql(true) + end + + it "should set the ownership on the file to the requested owner" do + @provider.load_current_resource + @provider.new_resource.stub!(:owner).and_return(9982398) + File.stub!(:chown).and_return(1) + File.should_receive(:chown).with(9982398, nil, @provider.current_resource.path) + lambda { @provider.set_owner }.should_not raise_error + end + + it "should raise an exception if you are not root and try to change ownership" do + @provider.load_current_resource + @provider.new_resource.stub!(:owner).and_return(0) + if Process.uid != 0 + lambda { @provider.set_owner }.should raise_error + end + end + + it "should compare the current group with the requested group" do + @provider.load_current_resource + @provider.new_resource.stub!(:group).and_return("adam") + Etc.stub!(:getgrnam).and_return( + OpenStruct.new( + :name => "adam", + :gid => 501 + ) + ) + @provider.current_resource.group(501) + @provider.compare_group.should eql(true) + + @provider.current_resource.group(777) + @provider.compare_group.should eql(false) + + @provider.new_resource.stub!(:group).and_return(501) + @provider.current_resource.group(501) + @provider.compare_group.should eql(true) + + @provider.new_resource.stub!(:group).and_return("501") + @provider.current_resource.group(501) + @provider.compare_group.should eql(true) + end + + it "should set the group on the file to the requested group" do + @provider.load_current_resource + @provider.new_resource.stub!(:group).and_return(9982398) + File.stub!(:chown).and_return(1) + File.should_receive(:chown).with(nil, 9982398, @provider.current_resource.path) + lambda { @provider.set_group }.should_not raise_error + end + + it "should raise an exception if you are not root and try to change the group" do + @provider.load_current_resource + @provider.new_resource.stub!(:group).and_return(0) + if Process.uid != 0 + lambda { @provider.set_group }.should raise_error + end + end + + it "should create the file if it is missing, then set the attributes on action_create" do + @provider.load_current_resource + @provider.new_resource.stub!(:owner).and_return(9982398) + @provider.new_resource.stub!(:group).and_return(9982398) + @provider.new_resource.stub!(:mode).and_return(0755) + @provider.new_resource.stub!(:path).and_return("/tmp/monkeyfoo") + File.stub!(:chown).and_return(1) + File.should_receive(:chown).with(nil, 9982398, @provider.new_resource.path) + File.stub!(:chown).and_return(1) + File.should_receive(:chown).with(9982398, nil, @provider.new_resource.path) + File.stub!(:open).and_return(1) + File.should_receive(:chmod).with(0755, @provider.new_resource.path).and_return(1) + File.should_receive(:open).with(@provider.new_resource.path, "w+") + @provider.action_create + end + + it "should delete the file if it exists and is writable on action_delete" do + @provider.load_current_resource + @provider.new_resource.stub!(:path).and_return("/tmp/monkeyfoo") + @provider.stub!(:backup).and_return(true) + File.should_receive("exists?").with(@provider.new_resource.path).and_return(true) + File.should_receive("writable?").with(@provider.new_resource.path).and_return(true) + File.should_receive(:delete).with(@provider.new_resource.path).and_return(true) + @provider.action_delete + end + + it "should raise an error if it cannot delete the file" do + @provider.load_current_resource + @provider.stub!(:backup).and_return(true) + @provider.new_resource.stub!(:path).and_return("/tmp/monkeyfoo") + File.should_receive("exists?").with(@provider.new_resource.path).and_return(false) + lambda { @provider.action_delete }.should raise_error() + end + + it "should update the atime/mtime on action_touch" do + @provider.load_current_resource + @provider.new_resource.stub!(:path).and_return("/tmp/monkeyfoo") + File.should_receive(:utime).once.and_return(1) + File.stub!(:open).and_return(1) + File.stub!(:chown).and_return(1) + File.stub!(:chmod).and_return(1) + @provider.action_touch + end + + it "should backup a file no more than :backup times" do + @provider.load_current_resource + @provider.new_resource.stub!(:path).and_return("/tmp/s-20080705111233") + @provider.new_resource.stub!(:backup).and_return(2) + Dir.stub!(:[]).and_return([ "/tmp/s-20080705111233", "/tmp/s-20080705111232", "/tmp/s-20080705111223"]) + FileUtils.should_receive(:rm).with("/tmp/s-20080705111232").once.and_return(true) + FileUtils.should_receive(:rm).with("/tmp/s-20080705111223").once.and_return(true) + FileUtils.stub!(:cp).and_return(true) + File.stub!(:exist?).and_return(true) + @provider.backup + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/provider/link_spec.rb b/chef/spec/unit/provider/link_spec.rb new file mode 100644 index 0000000000..462ab9409b --- /dev/null +++ b/chef/spec/unit/provider/link_spec.rb @@ -0,0 +1,147 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 'ostruct' + +require File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Provider::Link do + before(:each) do + @new_resource = mock("New Resource", :null_object => true) + @new_resource.stub!(:name).and_return("symlink") + @new_resource.stub!(:source_file).and_return("/tmp/fofile") + @new_resource.stub!(:target_file).and_return("/tmp/fofile-link") + @new_resource.stub!(:link_type).and_return(:symbolic) + @new_resource.stub!(:updated).and_return(false) + @node = Chef::Node.new + @node.name "latte" + @provider = Chef::Provider::Link.new(@node, @new_resource) + end + + it "should load the current resource based on the new resource" do + File.should_receive(:exists?).once.and_return(true) + File.should_receive(:symlink?).once.and_return(true) + File.should_receive(:readlink).once.and_return("/tmp/fofile") + @provider.load_current_resource + @provider.current_resource.name.should eql("symlink") + @provider.current_resource.source_file.should eql("/tmp/fofile") + @provider.current_resource.target_file.should eql("/tmp/fofile-link") + @provider.current_resource.link_type.should eql(:symbolic) + end + + it "should set the current resource's source_file to '' if the target_file doesn't exist" do + File.should_receive(:exists?).once.and_return(true) + File.should_receive(:symlink?).once.and_return(false) + @provider.load_current_resource + @provider.current_resource.source_file.should eql("") + end + + it "should load the current resource if it is a hard link" do + @new_resource.stub!(:link_type).and_return(:hard) + File.should_receive(:exists?).twice.and_return(true) + cstat = mock("stats", :null_object => true) + cstat.stub!(:ino).and_return(1) + File.should_receive(:stat).with("/tmp/fofile-link").and_return(cstat) + File.should_receive(:stat).with("/tmp/fofile").and_return(cstat) + @provider.load_current_resource + @provider.current_resource.name.should eql("symlink") + @provider.current_resource.source_file.should eql("/tmp/fofile") + @provider.current_resource.target_file.should eql("/tmp/fofile-link") + @provider.current_resource.link_type.should eql(:hard) + end + + it "should set the current resource's source_file to '' if the target_file doesn't exist" do + @new_resource.stub!(:link_type).and_return(:hard) + File.should_receive(:exists?).once.and_return(false) + @provider.load_current_resource + @provider.current_resource.source_file.should eql("") + end + + it "should set the current resource's source_file to '' if the two files arent hardlinked" do + @new_resource.stub!(:link_type).and_return(:hard) + File.stub!(:exists?).and_return(true) + cstat = mock("stats", :null_object => true) + cstat.stub!(:ino).and_return(0) + bstat = mock("stats", :null_object => true) + bstat.stub!(:ino).and_return(1) + File.should_receive(:stat).with("/tmp/fofile-link").and_return(cstat) + File.should_receive(:stat).with("/tmp/fofile").and_return(bstat) + @provider.load_current_resource + @provider.current_resource.source_file.should eql("") + end + + it "should create a new symlink on create, setting updated to true" do + load_mock_symlink_provider + @provider.current_resource.source_file("nil") + File.should_receive(:symlink).with(@new_resource.source_file, @new_resource.target_file).once.and_return(true) + @provider.new_resource.should_receive(:updated=).with(true) + @provider.action_create + end + + it "should not create a new symlink on create if it already exists" do + load_mock_symlink_provider + File.should_not_receive(:symlink).with(@new_resource.source_file, @new_resource.target_file) + @provider.action_create + end + + it "should create a new hard link on create, setting updated to true" do + load_mock_hardlink_provider + @provider.current_resource.source_file("nil") + File.should_receive(:link).with(@new_resource.source_file, @new_resource.target_file).once.and_return(true) + @provider.new_resource.should_receive(:updated=).with(true) + @provider.action_create + end + + it "should not create a new hard link on create if it already exists" do + load_mock_symlink_provider + File.should_not_receive(:link).with(@new_resource.source_file, @new_resource.target_file) + @provider.action_create + end + + it "should delete the link if it exists, and is writable with action_delete" do + load_mock_symlink_provider + File.should_receive(:exists?).once.and_return(true) + File.should_receive(:writable?).once.and_return(true) + File.should_receive(:delete).with(@new_resource.target_file).once.and_return(true) + @provider.action_delete + end + + it "should raise an exception if it cannot delete the link due to bad permissions" do + load_mock_symlink_provider + File.stub!(:exists?).and_return(true) + File.stub!(:writable?).and_return(false) + lambda { @provider.action_delete }.should raise_error(RuntimeError) + end + + def load_mock_symlink_provider + File.stub!(:exists?).and_return(true) + File.stub!(:symlink?).and_return(true) + File.stub!(:readlink).and_return("/tmp/fofile") + @provider.load_current_resource + end + + def load_mock_hardlink_provider + @new_resource.stub!(:link_type).and_return(:hard) + File.stub!(:exists?).twice.and_return(true) + cstat = mock("stats", :null_object => true) + cstat.stub!(:ino).and_return(1) + File.stub!(:stat).with("/tmp/fofile-link").and_return(cstat) + File.stub!(:stat).with("/tmp/fofile").and_return(cstat) + @provider.load_current_resource + end +end
\ No newline at end of file diff --git a/chef/spec/unit/provider/remote_file_spec.rb b/chef/spec/unit/provider/remote_file_spec.rb new file mode 100644 index 0000000000..51c48d0f9a --- /dev/null +++ b/chef/spec/unit/provider/remote_file_spec.rb @@ -0,0 +1,152 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Provider::RemoteFile, "action_create" do + before(:each) do + @resource = Chef::Resource::RemoteFile.new("seattle") + @resource.path(File.join(File.dirname(__FILE__), "..", "..", "data", "seattle.txt")) + @resource.source("http://foo") + @node = Chef::Node.new + @node.name "latte" + @provider = Chef::Provider::RemoteFile.new(@node, @resource) + @provider.current_resource = @resource.clone + end + + it "should call do_remote_file" do + @provider.should_receive(:do_remote_file).with(@resource.source, @resource.path) + @provider.action_create + end + +end + +describe Chef::Provider::RemoteFile, "do_remote_file" do + before(:each) do + @rest = mock(Chef::REST, { }) + @tempfile = mock(Tempfile, { :path => "/tmp/foo", }) + @rest.stub!(:get_rest).and_return(@tempfile) + @resource = Chef::Resource::RemoteFile.new("seattle") + @resource.path(File.join(File.dirname(__FILE__), "..", "..", "data", "seattle.txt")) + @resource.source("foo") + @resource.cookbook_name = "monkey" + @node = Chef::Node.new + @node.name "latte" + @node.fqdn "latte.local" + @provider = Chef::Provider::RemoteFile.new(@node, @resource) + @provider.stub!(:checksum).and_return("dad86c61eea237932f201009e5431609") + @provider.current_resource = @resource.clone + @provider.current_resource.checksum("dad86c61eea237932f201009e5431609") + File.stub!(:exists?).and_return(true) + FileUtils.stub!(:cp).and_return(true) + Chef::Platform.stub!(:find_platform_and_version).and_return([ :mac_os_x, "10.5.1" ]) + end + + def do_remote_file + Chef::REST.stub!(:new).and_return(@rest) + @provider.do_remote_file(@resource.source, @resource.path) + end + + it "should set the checksum if the file exists" do + @provider.should_receive(:checksum).with(@resource.path) + do_remote_file + end + + it "should not set the checksum if the file doesn't exist" do + File.stub!(:exists?).with(@resource.path).and_return(false) + @provider.should_not_receive(:checksum).with(@resource.path) + do_remote_file + end + + it "should call generate_url with the current checksum as an extra attribute" do + @provider.should_receive(:generate_url).with(@resource.source, "files", { :checksum => "dad86c61eea237932f201009e5431609"}) + do_remote_file + end + + it "should call get_rest with a correctly composed url" do + url = "cookbooks/#{@resource.cookbook_name}/files?id=#{@resource.source}" + url += "&platform=mac_os_x" + url += "&version=10.5.1" + url += "&fqdn=latte.local" + url += "&checksum=dad86c61eea237932f201009e5431609" + @rest.should_receive(:get_rest).with(url, true).and_return(@tempfile) + do_remote_file + end + + it "should not transfer the file if it has not been changed" do + r = Net::HTTPNotModified.new("one", "two", "three") + e = Net::HTTPRetriableError.new("304", r) + @rest.stub!(:get_rest).and_raise(e) + do_remote_file.should eql(false) + end + + it "should raise an exception if it's any other kind of retriable response than 304" do + r = Net::HTTPMovedPermanently.new("one", "two", "three") + e = Net::HTTPRetriableError.new("301", r) + @rest.stub!(:get_rest).and_raise(e) + lambda { do_remote_file }.should raise_error(Net::HTTPRetriableError) + end + + it "should raise an exception if anything else happens" do + r = Net::HTTPBadRequest.new("one", "two", "three") + e = Net::HTTPServerException.new("fake exception", r) + @rest.stub!(:get_rest).and_raise(e) + lambda { do_remote_file }.should raise_error(Net::HTTPServerException) + end + + it "should checksum the raw file" do + @provider.should_receive(:checksum).with(@tempfile.path).and_return("dad86c61eea237932f201009e5431608") + do_remote_file + end + + it "should backup the original file" do + @provider.should_receive(:backup).with(@resource.path).and_return(true) + do_remote_file + end + + it "should set the new resource to updated" do + @resource.should_receive(:updated=).with(true) + do_remote_file + end + + it "should copy the raw file to the new resource" do + FileUtils.should_receive(:cp).with(@tempfile.path, @resource.path).and_return(true) + do_remote_file + end + + it "should set the owner if provided" do + @resource.owner("adam") + @provider.should_receive(:set_owner).and_return(true) + do_remote_file + end + + it "should set the group if provided" do + @resource.group("adam") + @provider.should_receive(:set_group).and_return(true) + do_remote_file + end + + it "should set the mode if provided" do + @resource.mode(0676) + @provider.should_receive(:set_mode).and_return(true) + do_remote_file + end + +# TODO: Finish these tests + +end diff --git a/chef/spec/unit/provider/template_spec.rb b/chef/spec/unit/provider/template_spec.rb new file mode 100644 index 0000000000..abc98e2ea3 --- /dev/null +++ b/chef/spec/unit/provider/template_spec.rb @@ -0,0 +1,105 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Provider::Template, "action_create" do + before(:each) do + @rest = mock(Chef::REST, { :get_rest => "/tmp/foobar" }) + @tempfile = mock(Tempfile, { :path => "/tmp/foo", }) + @rest.stub!(:get_rest).and_return(@tempfile) + @resource = Chef::Resource::Template.new("seattle") + @resource.path(File.join(File.dirname(__FILE__), "..", "..", "data", "seattle.txt")) + @resource.source("http://foo") + @node = Chef::Node.new + @node.name "latte" + @provider = Chef::Provider::Template.new(@node, @resource) + @provider.stub!(:checksum).and_return("dad86c61eea237932f201009e5431609") + @provider.current_resource = @resource.clone + @provider.current_resource.checksum("dad86c61eea237932f201009e5431609") + FileUtils.stub!(:cp).and_return(true) + end + + def do_action_create + Chef::REST.stub!(:new).and_return(@rest) + @provider.action_create + end + + it "should get the template based on the resources source value" do + @rest.should_receive(:get_rest).with(@resource.source, true).and_return(@tempfile) + do_action_create + end + + it "should set the checksum of the new resource to the value of the returned template" do + @resource.should_receive(:checksum).with("dad86c61eea237932f201009e5431609").once + @resource.should_receive(:checksum).twice + do_action_create + end + + it "should not copy the tempfile to the real file if the checksums match" do + FileUtils.should_not_receive(:cp) + do_action_create + end + + it "should copy the tempfile to the real file if the checksums do not match" do + @provider.stub!(:checksum).and_return("dad86c61eea237932f201009e5431607") + FileUtils.should_receive(:cp).once + @provider.stub!(:backup).and_return(true) + do_action_create + end + + it "should set the owner if provided" do + @resource.owner("adam") + @provider.should_receive(:set_owner).and_return(true) + do_action_create + end + + it "should set the group if provided" do + @resource.group("adam") + @provider.should_receive(:set_group).and_return(true) + do_action_create + end + + it "should set the mode if provided" do + @resource.mode(0676) + @provider.should_receive(:set_mode).and_return(true) + do_action_create + end +end + +describe Chef::Provider::Template, "generate_url" do + + before(:each) do + @resource = Chef::Resource::Template.new("seattle") + @resource.cookbook_name = "daft" + @resource.path(File.join(File.dirname(__FILE__), "..", "..", "data", "seattle.txt")) + @node = Chef::Node.new + @node.name "latte" + @provider = Chef::Provider::Template.new(@node, @resource) + end + + it "should return a raw url if it starts with http" do + @provider.generate_url('http://foobar', "templates").should eql("http://foobar") + end + + it "should return a composed url if it does not start with http" do + Chef::Platform.stub!(:find_platform_and_version).and_return(["monkey", "1.0"]) + @node.fqdn("monkeynode") + @provider.generate_url('default/something', "templates").should eql("cookbooks/daft/templates?id=default/something&platform=monkey&version=1.0&fqdn=monkeynode") + end +end
\ No newline at end of file diff --git a/chef/spec/unit/provider_spec.rb b/chef/spec/unit/provider_spec.rb new file mode 100644 index 0000000000..0512012598 --- /dev/null +++ b/chef/spec/unit/provider_spec.rb @@ -0,0 +1,48 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Provider do + before(:each) do + @resource = Chef::Resource.new("funk") + @node = Chef::Node.new + @node.name "latte" + @provider = Chef::Provider.new(@node, @resource) + end + + it "should return a Chef::Provider" do + @provider.should be_a_kind_of(Chef::Provider) + end + + it "should store the resource passed to new as new_resource" do + @provider.new_resource.should eql(@resource) + end + + it "should store the node passed to new as node" do + @provider.node.should eql(@node) + end + + it "should have nil for current_resource by default" do + @provider.current_resource.should eql(nil) + end + + it "should return true for action_nothing" do + @provider.action_nothing.should eql(true) + end +end
\ No newline at end of file diff --git a/chef/spec/unit/queue_spec.rb b/chef/spec/unit/queue_spec.rb new file mode 100644 index 0000000000..e902f44631 --- /dev/null +++ b/chef/spec/unit/queue_spec.rb @@ -0,0 +1,105 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Queue do + + it "should connect to a stomp server on localhost and 61613" do + Stomp::Connection.should_receive(:open).with("", "", "localhost", 61613, false).once + Chef::Queue.connect + end + + it "should allow config options to override defaults on connect" do + Chef::Config[:queue_user] = "monkey" + Chef::Config[:queue_password] = "password" + Chef::Config[:queue_host] = "10.10.10.10" + Chef::Config[:queue_port] = 61614 + Stomp::Connection.should_receive(:open).with("monkey", "password", "10.10.10.10", 61614, false).once + Chef::Queue.connect + end + + it "should make a url based on type and name" do + Chef::Queue.make_url("topic", "goal").should eql("/topic/chef/goal") + Chef::Queue.make_url("queue", "pool").should eql("/queue/chef/pool") + end + + it "should allow you to subscribe to a queue" do + queue = mock("Queue", :null_object => true) + queue.should_receive(:subscribe).with(Chef::Queue.make_url(:topic, :node)).once + Stomp::Connection.stub!(:open).and_return(queue) + Chef::Queue.connect + Chef::Queue.subscribe(:topic, :node) + end + + it "should allow you to send a message" do + message = mock("Message", :null_object => true) + message.should_receive(:to_json).once.and_return("some json") + connection = mock("Connection", :null_object => true) + connection.should_receive(:send).with(Chef::Queue.make_url(:queue, :node), "some json").once.and_return(true) + Stomp::Connection.stub!(:open).and_return(connection) + Chef::Queue.connect + Chef::Queue.send_msg(:queue, :node, message) + end + + it "should receive a message with receive_msg" do + raw_msg = mock("Stomp Message", :null_object => true) + raw_msg.should_receive(:body).twice.and_return("the body") + connection = mock("Connection", :null_object => true) + connection.should_receive(:receive).once.and_return(raw_msg) + JSON.should_receive(:parse).with("the body").and_return("the body") + Stomp::Connection.stub!(:open).and_return(connection) + Chef::Queue.connect + Chef::Queue.receive_msg.should eql([ "the body", raw_msg ]) + end + + it "should poll for a message with poll_msg, returning a message if there is one" do + raw_msg = mock("Stomp Message", :null_object => true) + raw_msg.should_receive(:body).once.and_return("the body") + connection = mock("Connection", :null_object => true) + connection.should_receive(:poll).once.and_return(raw_msg) + JSON.should_receive(:parse).with("the body").and_return("the body") + Stomp::Connection.stub!(:open).and_return(connection) + Chef::Queue.connect + Chef::Queue.poll_msg.should eql("the body") + end + + it "should poll for a message with poll_msg, returning nil if there is not a message" do + connection = mock("Connection", :null_object => true) + connection.should_receive(:poll).once.and_return(nil) + JSON.should_not_receive(:parse).with(nil) + Stomp::Connection.stub!(:open).and_return(connection) + Chef::Queue.connect + Chef::Queue.poll_msg.should eql(nil) + end + + it "should raise an exception if you disconnect without a connection" do + Stomp::Connection.stub!(:open).and_return(nil) + Chef::Queue.connect + lambda { Chef::Queue.disconnect }.should raise_error(ArgumentError) + end + + it "should disconnect an active connection" do + connection = mock("Connection", :null_object => true) + connection.should_receive(:disconnect).once.and_return(true) + Stomp::Connection.stub!(:open).and_return(connection) + Chef::Queue.connect + Chef::Queue.disconnect + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/recipe_spec.rb b/chef/spec/unit/recipe_spec.rb new file mode 100644 index 0000000000..77d5354993 --- /dev/null +++ b/chef/spec/unit/recipe_spec.rb @@ -0,0 +1,143 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Recipe do + before(:each) do + @recipe = Chef::Recipe.new("hjk", "test", Chef::Node.new) + end + + it "should load a two word (zen_master) resource" do + lambda do + @recipe.zen_master "monkey" do + peace true + end + end.should_not raise_error(ArgumentError) + end + + it "should load a one word (cat) resource" do + lambda do + @recipe.cat "loulou" do + pretty_kitty true + end + end.should_not raise_error(ArgumentError) + end + + it "should throw an error if you access a resource that we can't find" do + lambda { @recipe.not_home { || } }.should raise_error(NameError) + end + + it "should allow regular errors (not NameErrors) to pass unchanged" do + lambda { + @recipe.cat { || raise ArgumentError, "You Suck" } + }.should raise_error(ArgumentError) + end + + it "should add our zen_master to the collection" do + @recipe.zen_master "monkey" do + peace true + end + @recipe.collection.lookup("zen_master[monkey]").name.should eql("monkey") + end + + it "should add our zen masters to the collection in the order they appear" do + %w{monkey dog cat}.each do |name| + @recipe.zen_master name do + peace true + end + end + @recipe.collection.each_index do |i| + case i + when 0 + @recipe.collection[i].name.should eql("monkey") + when 1 + @recipe.collection[i].name.should eql("dog") + when 2 + @recipe.collection[i].name.should eql("cat") + end + end + end + + it "should return the new resource after creating it" do + res = @recipe.zen_master "makoto" do + peace true + end + res.resource_name.should eql(:zen_master) + res.name.should eql("makoto") + end + + it "should handle an instance_eval properly" do + code = <<-CODE +zen_master "gnome" do + peace = true +end +CODE + lambda { @recipe.instance_eval(code) }.should_not raise_error + @recipe.resources(:zen_master => "gnome").name.should eql("gnome") + end + + it "should execute defined resources" do + crow_define = Chef::ResourceDefinition.new + crow_define.define :crow, :peace => false, :something => true do + zen_master "lao tzu" do + peace params[:peace] + something params[:something] + end + end + @recipe.definitions[:crow] = crow_define + @recipe.crow "mine" do + peace true + end + @recipe.resources(:zen_master => "lao tzu").name.should eql("lao tzu") + @recipe.resources(:zen_master => "lao tzu").something.should eql(true) + end + + it "should load a resource from a ruby file" do + @recipe.from_file(File.join(File.dirname(__FILE__), "..", "data", "recipes", "test.rb")) + res = @recipe.resources(:file => "/etc/nsswitch.conf") + res.name.should eql("/etc/nsswitch.conf") + res.action.should eql(:create) + res.owner.should eql("root") + res.group.should eql("root") + res.mode.should eql(0644) + end + + it "should raise an exception if the file cannot be found or read" do + lambda { @recipe.from_file("/tmp/monkeydiving") }.should raise_error(IOError) + end + + it "should evaluate another recipe with recipe_require" do + Chef::Config.cookbook_path File.join(File.dirname(__FILE__), "..", "data", "cookbooks") + @recipe.cookbook_loader.load_cookbooks + @recipe.require_recipe "openldap::gigantor" + res = @recipe.resources(:cat => "blanket") + res.name.should eql("blanket") + res.pretty_kitty.should eql(false) + end + + it "should load the default recipe for a cookbook if require_recipe is called without a ::" do + Chef::Config.cookbook_path File.join(File.dirname(__FILE__), "..", "data", "cookbooks") + @recipe.cookbook_loader.load_cookbooks + @recipe.require_recipe "openldap" + res = @recipe.resources(:cat => "blanket") + res.name.should eql("blanket") + res.pretty_kitty.should eql(true) + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/bash_spec.rb b/chef/spec/unit/resource/bash_spec.rb new file mode 100644 index 0000000000..f86e63d32e --- /dev/null +++ b/chef/spec/unit/resource/bash_spec.rb @@ -0,0 +1,40 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Bash do + + before(:each) do + @resource = Chef::Resource::Bash.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::Bash" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Bash) + end + + it "should have a resource name of :bash" do + @resource.resource_name.should eql(:bash) + end + + it "should have an interpreter of bash" do + @resource.interpreter.should eql("bash") + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/csh_spec.rb b/chef/spec/unit/resource/csh_spec.rb new file mode 100644 index 0000000000..67ba4499fa --- /dev/null +++ b/chef/spec/unit/resource/csh_spec.rb @@ -0,0 +1,40 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Csh do + + before(:each) do + @resource = Chef::Resource::Csh.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::Csh" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Csh) + end + + it "should have a resource name of :csh" do + @resource.resource_name.should eql(:csh) + end + + it "should have an interpreter of csh" do + @resource.interpreter.should eql("csh") + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/directory_spec.rb b/chef/spec/unit/resource/directory_spec.rb new file mode 100644 index 0000000000..98cc28fbc6 --- /dev/null +++ b/chef/spec/unit/resource/directory_spec.rb @@ -0,0 +1,79 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Directory do + + before(:each) do + @resource = Chef::Resource::Directory.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::Directory" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Directory) + end + + it "should have a name" do + @resource.name.should eql("fakey_fakerton") + end + + it "should have a default action of 'create'" do + @resource.action.should eql(:create) + end + + it "should accept create or delete for action" do + lambda { @resource.action "create" }.should_not raise_error(ArgumentError) + lambda { @resource.action "delete" }.should_not raise_error(ArgumentError) + lambda { @resource.action "blues" }.should raise_error(ArgumentError) + end + + it "should accept a group name or id for group" do + lambda { @resource.group "root" }.should_not raise_error(ArgumentError) + lambda { @resource.group 123 }.should_not raise_error(ArgumentError) + lambda { @resource.group "root*goo" }.should raise_error(ArgumentError) + end + + it "should accept a valid unix file mode" do + lambda { @resource.mode 0444 }.should_not raise_error(ArgumentError) + lambda { @resource.mode 444 }.should_not raise_error(ArgumentError) + lambda { @resource.mode 4 }.should raise_error(ArgumentError) + end + + it "should accept a user name or id for owner" do + lambda { @resource.owner "root" }.should_not raise_error(ArgumentError) + lambda { @resource.owner 123 }.should_not raise_error(ArgumentError) + lambda { @resource.owner "root*goo" }.should raise_error(ArgumentError) + end + + it "should use the object name as the path by default" do + @resource.path.should eql("fakey_fakerton") + end + + it "should accept a string as the path" do + lambda { @resource.path "/tmp" }.should_not raise_error(ArgumentError) + lambda { @resource.path Hash.new }.should raise_error(ArgumentError) + end + + it "should allow you to have specify whether the action is recursive with true/false" do + lambda { @resource.recursive true }.should_not raise_error(ArgumentError) + lambda { @resource.recursive false }.should_not raise_error(ArgumentError) + lambda { @resource.recursive "monkey" }.should raise_error(ArgumentError) + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/execute_spec.rb b/chef/spec/unit/resource/execute_spec.rb new file mode 100644 index 0000000000..94cb3d3d92 --- /dev/null +++ b/chef/spec/unit/resource/execute_spec.rb @@ -0,0 +1,102 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Execute do + + before(:each) do + @resource = Chef::Resource::Execute.new("some command") + end + + it "should create a new Chef::Resource::Execute" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Execute) + end + + it "should set the command to the first argument to new" do + @resource.command.should eql("some command") + end + + it "should accept a string for the command to run" do + @resource.command "something" + @resource.command.should eql("something") + end + + it "should accept a string for the cwd" do + @resource.cwd "something" + @resource.cwd.should eql("something") + end + + it "should accept a hash for the environment" do + test_hash = { :one => :two } + @resource.environment(test_hash) + @resource.environment.should eql(test_hash) + end + + it "should accept a string for the group" do + @resource.group "something" + @resource.group.should eql("something") + end + + it "should accept an integer for the group" do + @resource.group 1 + @resource.group.should eql(1) + end + + it "should accept a string for onlyif" do + @resource.onlyif "woot" + @resource.onlyif.should eql("woot") + end + + it "should accept an array for the execution path" do + @resource.path ["woot"] + @resource.path.should eql(["woot"]) + end + + it "should accept an integer for the return code" do + @resource.returns 1 + @resource.returns.should eql(1) + end + + it "should accept an integer for the timeout" do + @resource.timeout 1 + @resource.timeout.should eql(1) + end + + it "should accept a string for unless" do + @resource.unless "woot" + @resource.unless.should eql("woot") + end + + it "should accept a string for the user" do + @resource.user "something" + @resource.user.should eql("something") + end + + it "should accept an integer for the user" do + @resource.user 1 + @resource.user.should eql(1) + end + + it "should accept a string for creates" do + @resource.creates "something" + @resource.creates.should eql("something") + end + +end diff --git a/chef/spec/unit/resource/file_spec.rb b/chef/spec/unit/resource/file_spec.rb new file mode 100644 index 0000000000..b67598d738 --- /dev/null +++ b/chef/spec/unit/resource/file_spec.rb @@ -0,0 +1,92 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::File do + + before(:each) do + @resource = Chef::Resource::File.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::File" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::File) + end + + it "should have a name" do + @resource.name.should eql("fakey_fakerton") + end + + it "should have a default action of 'create'" do + @resource.action.should eql("create") + end + + it "should be set to back up 5 files by default" do + @resource.backup.should eql(5) + end + + it "should only accept false or a number for backup" do + lambda { @resource.backup true }.should raise_error(ArgumentError) + lambda { @resource.backup false }.should_not raise_error(ArgumentError) + lambda { @resource.backup 10 }.should_not raise_error(ArgumentError) + lambda { @resource.backup "blues" }.should raise_error(ArgumentError) + end + + it "should accept an md5sum for checksum" do + lambda { @resource.checksum "bfda9e7a13afb123433667c2c7801d11" }.should_not raise_error(ArgumentError) + lambda { @resource.checksum "monkey!" }.should raise_error(ArgumentError) + end + + it "should accept create, delete or touch for action" do + lambda { @resource.action "create" }.should_not raise_error(ArgumentError) + lambda { @resource.action "delete" }.should_not raise_error(ArgumentError) + lambda { @resource.action "touch" }.should_not raise_error(ArgumentError) + lambda { @resource.action "blues" }.should raise_error(ArgumentError) + end + + it "should accept a group name or id for group" do + lambda { @resource.group "root" }.should_not raise_error(ArgumentError) + lambda { @resource.group 123 }.should_not raise_error(ArgumentError) + lambda { @resource.group "root*goo" }.should raise_error(ArgumentError) + end + + it "should accept a valid unix file mode" do + lambda { @resource.mode 0444 }.should_not raise_error(ArgumentError) + @resource.mode.should eql(0444) + lambda { @resource.mode 444 }.should_not raise_error(ArgumentError) + lambda { @resource.mode 4 }.should raise_error(ArgumentError) + end + + it "should accept a user name or id for owner" do + lambda { @resource.owner "root" }.should_not raise_error(ArgumentError) + lambda { @resource.owner 123 }.should_not raise_error(ArgumentError) + lambda { @resource.owner "root*goo" }.should raise_error(ArgumentError) + end + + it "should use the object name as the path by default" do + @resource.path.should eql("fakey_fakerton") + end + + it "should accept a string as the path" do + lambda { @resource.path "/tmp" }.should_not raise_error(ArgumentError) + @resource.path.should eql("/tmp") + lambda { @resource.path Hash.new }.should raise_error(ArgumentError) + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/link_spec.rb b/chef/spec/unit/resource/link_spec.rb new file mode 100644 index 0000000000..fa0810cced --- /dev/null +++ b/chef/spec/unit/resource/link_spec.rb @@ -0,0 +1,78 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Link do + + before(:each) do + @resource = Chef::Resource::Link.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::Link" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Link) + end + + it "should have a name" do + @resource.name.should eql("fakey_fakerton") + end + + it "should have a default action of 'create'" do + @resource.action.should eql(:create) + end + + it "should accept create or delete for action" do + lambda { @resource.action "create" }.should_not raise_error(ArgumentError) + lambda { @resource.action "delete" }.should_not raise_error(ArgumentError) + lambda { @resource.action "blues" }.should raise_error(ArgumentError) + end + + it "should use the object name as the source_file by default" do + @resource.source_file.should eql("fakey_fakerton") + end + + it "should accept a string as the source_file" do + lambda { @resource.source_file "/tmp" }.should_not raise_error(ArgumentError) + lambda { @resource.source_file Hash.new }.should raise_error(ArgumentError) + end + + it "should allow you to set a target_file" do + @resource.target_file "/tmp/foo" + @resource.target_file.should eql("/tmp/foo") + end + + it "should allow you to specify the link type" do + @resource.link_type "symbolic" + @resource.link_type.should eql(:symbolic) + end + + it "should default to a symbolic link" do + @resource.link_type.should eql(:symbolic) + end + + it "should accept a hard link_type" do + @resource.link_type :hard + @resource.link_type.should eql(:hard) + end + + it "should reject any other link_type but :hard and :symbolic" do + lambda { @resource.link_type "x-men" }.should raise_error(ArgumentError) + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/package_spec.rb b/chef/spec/unit/resource/package_spec.rb new file mode 100644 index 0000000000..8756e4f603 --- /dev/null +++ b/chef/spec/unit/resource/package_spec.rb @@ -0,0 +1,56 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Package do + + before(:each) do + @resource = Chef::Resource::Package.new("emacs") + end + + it "should create a new Chef::Resource::Package" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Package) + end + + it "should set the package_name to the first argument to new" do + @resource.package_name.should eql("emacs") + end + + it "should accept a string for the package name" do + @resource.package_name "something" + @resource.package_name.should eql("something") + end + + it "should accept a string for the version" do + @resource.version "something" + @resource.version.should eql("something") + end + + it "should accept a string for the response file" do + @resource.response_file "something" + @resource.response_file.should eql("something") + end + + it "should accept a string for the source" do + @resource.source "something" + @resource.source.should eql("something") + end + +end diff --git a/chef/spec/unit/resource/perl_spec.rb b/chef/spec/unit/resource/perl_spec.rb new file mode 100644 index 0000000000..528bcef08c --- /dev/null +++ b/chef/spec/unit/resource/perl_spec.rb @@ -0,0 +1,40 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Perl do + + before(:each) do + @resource = Chef::Resource::Perl.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::Perl" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Perl) + end + + it "should have a resource name of :perl" do + @resource.resource_name.should eql(:perl) + end + + it "should have an interpreter of perl" do + @resource.interpreter.should eql("perl") + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/python_spec.rb b/chef/spec/unit/resource/python_spec.rb new file mode 100644 index 0000000000..11c5b84d04 --- /dev/null +++ b/chef/spec/unit/resource/python_spec.rb @@ -0,0 +1,40 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Python do + + before(:each) do + @resource = Chef::Resource::Python.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::Python" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Python) + end + + it "should have a resource name of :python" do + @resource.resource_name.should eql(:python) + end + + it "should have an interpreter of python" do + @resource.interpreter.should eql("python") + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/remote_directory_spec.rb b/chef/spec/unit/resource/remote_directory_spec.rb new file mode 100644 index 0000000000..2d9ce122af --- /dev/null +++ b/chef/spec/unit/resource/remote_directory_spec.rb @@ -0,0 +1,71 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::RemoteDirectory do + + before(:each) do + @resource = Chef::Resource::RemoteDirectory.new("/etc/dunk") + end + + it "should create a new Chef::Resource::RemoteDirectory" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::RemoteDirectory) + end + + it "should set the path to the first argument to new" do + @resource.path.should eql("/etc/dunk") + end + + it "should accept a string for the remote directory source" do + @resource.source "foo" + @resource.source.should eql("foo") + end + + it "should accept a number for the remote files backup" do + @resource.files_backup 1 + @resource.files_backup.should eql(1) + end + + it "should accept false for the remote files backup" do + @resource.files_backup false + @resource.files_backup.should eql(false) + end + + it "should accept 3 or 4 digets for the files_mode" do + @resource.files_mode 100 + @resource.files_mode.should eql(100) + @resource.files_mode 1000 + @resource.files_mode.should eql(1000) + end + + it "should accept a string or number for the files group" do + @resource.files_group "heart" + @resource.files_group.should eql("heart") + @resource.files_group 1000 + @resource.files_group.should eql(1000) + end + + it "should accept a string or number for the files owner" do + @resource.files_owner "heart" + @resource.files_owner.should eql("heart") + @resource.files_owner 1000 + @resource.files_owner.should eql(1000) + end +end diff --git a/chef/spec/unit/resource/remote_file_spec.rb b/chef/spec/unit/resource/remote_file_spec.rb new file mode 100644 index 0000000000..4c052e0260 --- /dev/null +++ b/chef/spec/unit/resource/remote_file_spec.rb @@ -0,0 +1,38 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::RemoteFile do + + before(:each) do + @resource = Chef::Resource::RemoteFile.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::RemoteFile" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::File) + @resource.should be_a_kind_of(Chef::Resource::RemoteFile) + end + + it "should accept a string for the remote file source" do + @resource.source "something" + @resource.source.should eql("something") + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/ruby_spec.rb b/chef/spec/unit/resource/ruby_spec.rb new file mode 100644 index 0000000000..5351462154 --- /dev/null +++ b/chef/spec/unit/resource/ruby_spec.rb @@ -0,0 +1,40 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Ruby do + + before(:each) do + @resource = Chef::Resource::Ruby.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::Ruby" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Ruby) + end + + it "should have a resource name of :ruby" do + @resource.resource_name.should eql(:ruby) + end + + it "should have an interpreter of ruby" do + @resource.interpreter.should eql("ruby") + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/script_spec.rb b/chef/spec/unit/resource/script_spec.rb new file mode 100644 index 0000000000..f003028fd6 --- /dev/null +++ b/chef/spec/unit/resource/script_spec.rb @@ -0,0 +1,50 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Script do + + before(:each) do + @resource = Chef::Resource::Script.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::Script" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Script) + end + + it "should have a resource name of :script" do + @resource.resource_name.should eql(:script) + end + + it "should set command to the argument provided to new" do + @resource.command.should eql("fakey_fakerton") + end + + it "should accept a string for the code" do + @resource.code "hey jude" + @resource.code.should eql("hey jude") + end + + it "should accept a string for the interpreter" do + @resource.interpreter "naaaaNaNaNaaNaaNaaNaa" + @resource.interpreter.should eql("naaaaNaNaNaaNaaNaaNaa") + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource/service_spec.rb b/chef/spec/unit/resource/service_spec.rb new file mode 100644 index 0000000000..f5941f89ae --- /dev/null +++ b/chef/spec/unit/resource/service_spec.rb @@ -0,0 +1,100 @@ +# +# Author:: AJ Christensen (<aj@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Service do + + before(:each) do + @resource = Chef::Resource::Service.new("chef") + end + + it "should create a new Chef::Resource::Service" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::Service) + end + + it "should set the service_name to the first argument to new" do + @resource.service_name.should eql("chef") + end + + it "should set the pattern to be the service name by default" do + @resource.pattern.should eql("chef") + end + + it "should accept a string for the service name" do + @resource.service_name "something" + @resource.service_name.should eql("something") + end + + it "should accept a string for the service pattern" do + @resource.pattern ".*" + @resource.pattern.should eql(".*") + end + + it "should not accept a regexp for the service pattern" do + lambda { + @resource.pattern /.*/ + }.should raise_error(ArgumentError) + end + + it "should accept a string for the service start command" do + @resource.start_command "/etc/init.d/chef start" + @resource.start_command.should eql("/etc/init.d/chef start") + end + + it "should not accept a regexp for the service start command" do + lambda { + @resource.start_command /.*/ + }.should raise_error(ArgumentError) + end + + it "should accept a string for the service stop command" do + @resource.stop_command "/etc/init.d/chef stop" + @resource.stop_command.should eql("/etc/init.d/chef stop") + end + + it "should not accept a regexp for the service stop command" do + lambda { + @resource.stop_command /.*/ + }.should raise_error(ArgumentError) + end + + it "should accept a string for the service status command" do + @resource.status_command "/etc/init.d/chef status" + @resource.status_command.should eql("/etc/init.d/chef status") + end + + it "should not accept a regexp for the service status command" do + lambda { + @resource.status_command /.*/ + }.should raise_error(ArgumentError) + end + + it "should accept a string for the service restart command" do + @resource.restart_command "/etc/init.d/chef restart" + @resource.restart_command.should eql("/etc/init.d/chef restart") + end + + it "should not accept a regexp for the service restart command" do + lambda { + @resource.restart_command /.*/ + }.should raise_error(ArgumentError) + end + +end diff --git a/chef/spec/unit/resource/template_spec.rb b/chef/spec/unit/resource/template_spec.rb new file mode 100644 index 0000000000..106fe77bda --- /dev/null +++ b/chef/spec/unit/resource/template_spec.rb @@ -0,0 +1,43 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper")) + +describe Chef::Resource::Template do + + before(:each) do + @resource = Chef::Resource::Template.new("fakey_fakerton") + end + + it "should create a new Chef::Resource::Template" do + @resource.should be_a_kind_of(Chef::Resource) + @resource.should be_a_kind_of(Chef::Resource::File) + @resource.should be_a_kind_of(Chef::Resource::Template) + end + + it "should accept a string for the template source" do + @resource.source "something" + @resource.source.should eql("something") + end + + it "should accept a hash for the variable list" do + @resource.variables({ :reluctance => :awkward }) + @resource.variables.should == { :reluctance => :awkward } + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource_collection_spec.rb b/chef/spec/unit/resource_collection_spec.rb new file mode 100644 index 0000000000..37eeac18db --- /dev/null +++ b/chef/spec/unit/resource_collection_spec.rb @@ -0,0 +1,201 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::ResourceCollection do + + before(:each) do + @rc = Chef::ResourceCollection.new() + @resource = Chef::Resource::ZenMaster.new("makoto") + end + + it "should return a Chef::ResourceCollection" do + @rc.should be_kind_of(Chef::ResourceCollection) + end + + it "should accept Chef::Resources through [index]" do + lambda { @rc[0] = @resource }.should_not raise_error + lambda { @rc[0] = "string" }.should raise_error + end + + it "should not accept duplicate resources [index]=" do + @rc[0] = @resource + lambda { @rc[1] = @resource }.should raise_error(ArgumentError) + end + + it "should accept Chef::Resources through pushing" do + lambda { @rc.push(@resource) }.should_not raise_error + lambda { @rc.push("string") }.should raise_error + end + + it "should not accept duplicate resources through pushing" do + lambda { @rc.push(@resource) }.should_not raise_error + lambda { @rc.push(@resource) }.should raise_error(ArgumentError) + end + + it "should allow you to fetch Chef::Resources by position" do + @rc[0] = @resource + @rc[0].should eql(@resource) + end + + it "should accept the << operator" do + lambda { @rc << @resource }.should_not raise_error + end + + it "should not accept duplicate resources through the << operator" do + lambda { @rc << @resource }.should_not raise_error + lambda { @rc << @resource }.should raise_error(ArgumentError) + end + + it "should allow you to iterate over every resource in the collection" do + load_up_resources + results = Array.new + lambda { + @rc.each do |r| + results << r.name + end + }.should_not raise_error + results.each_index do |i| + case i + when 0 + results[i].should eql("dog") + when 1 + results[i].should eql("cat") + when 2 + results[i].should eql("monkey") + end + end + end + + it "should allow you to iterate over every resource by index" do + load_up_resources + results = Array.new + lambda { + @rc.each_index do |i| + results << @rc[i].name + end + }.should_not raise_error() + results.each_index do |i| + case i + when 0 + results[i].should eql("dog") + when 1 + results[i].should eql("cat") + when 2 + results[i].should eql("monkey") + end + end + end + + it "should allow you to find resources by name via lookup" do + zmr = Chef::Resource::ZenMaster.new("dog") + @rc << zmr + @rc.lookup(zmr.to_s).should eql(zmr) + + zmr = Chef::Resource::ZenMaster.new("cat") + @rc[0] = zmr + @rc.lookup(zmr).should eql(zmr) + + zmr = Chef::Resource::ZenMaster.new("monkey") + @rc.push(zmr) + @rc.lookup(zmr).should eql(zmr) + end + + it "should raise an exception if you send something strange to lookup" do + lambda { @rc.lookup(:symbol) }.should raise_error(ArgumentError) + end + + it "should raise an exception if it cannot find a resource with lookup" do + lambda { @rc.lookup("zen_master[dog]") }.should raise_error(ArgumentError) + end + + it "should find a resource by symbol and name (:zen_master => monkey)" do + load_up_resources + @rc.resources(:zen_master => "monkey").name.should eql("monkey") + end + + it "should find a resource by symbol and array of names (:zen_master => [a,b])" do + load_up_resources + results = @rc.resources(:zen_master => [ "monkey", "dog" ]) + results.length.should eql(2) + check_by_names(results, "monkey", "dog") + end + + it "should find resources of multiple kinds (:zen_master => a, :file => b)" do + load_up_resources + results = @rc.resources(:zen_master => "monkey", :file => "something") + results.length.should eql(2) + check_by_names(results, "monkey", "something") + end + + it "should find a resource by string zen_master[a]" do + load_up_resources + @rc.resources("zen_master[monkey]").name.should eql("monkey") + end + + it "should find resources by strings of zen_master[a,b]" do + load_up_resources + results = @rc.resources("zen_master[monkey,dog]") + results.length.should eql(2) + check_by_names(results, "monkey", "dog") + end + + it "should find resources of multiple types by strings of zen_master[a]" do + load_up_resources + results = @rc.resources("zen_master[monkey]", "file[something]") + results.length.should eql(2) + check_by_names(results, "monkey", "something") + end + + it "should raise an exception if you pass a bad name to resources" do + lambda { @rc.resources("michael jackson") }.should raise_error(ArgumentError) + end + + it "should raise an exception if you pass something other than a string or hash to resource" do + lambda { @rc.resources([Array.new]) }.should raise_error(ArgumentError) + end + + it "should serialize to json" do + json = @rc.to_json + json.should =~ /json_class/ + json.should =~ /instance_vars/ + end + + it "should deserialize itself from json" do + @rc << @resource + json = @rc.to_json + s_rc = JSON.parse(json) + s_rc.should be_a_kind_of(Chef::ResourceCollection) + s_rc[0].name.should eql(@resource.name) + end + + def check_by_names(results, *names) + names.each do |res_name| + results.detect{ |res| res.name == res_name }.should_not eql(nil) + end + end + + def load_up_resources + %w{dog cat monkey}.each do |n| + @rc << Chef::Resource::ZenMaster.new(n) + end + @rc << Chef::Resource::File.new("something") + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/resource_definition_spec.rb b/chef/spec/unit/resource_definition_spec.rb new file mode 100644 index 0000000000..8ba7556b70 --- /dev/null +++ b/chef/spec/unit/resource_definition_spec.rb @@ -0,0 +1,83 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::ResourceDefinition do + before(:each) do + @def = Chef::ResourceDefinition.new() + end + + it "should accept a new definition with a symbol for a name" do + lambda { + @def.define :smoke do + end + }.should_not raise_error(ArgumentError) + lambda { + @def.define "george washington" do + end + }.should raise_error(ArgumentError) + @def.name.should eql(:smoke) + end + + it "should accept a new definition with a hash" do + lambda { + @def.define :smoke, :cigar => "cuban", :cigarette => "marlboro" do + end + }.should_not raise_error(ArgumentError) + end + + it "should expose the prototype hash params in the params hash" do + @def.define :smoke, :cigar => "cuban", :cigarette => "marlboro" do + end + @def.params[:cigar].should eql("cuban") + @def.params[:cigarette].should eql("marlboro") + end + + it "should store the block passed to define as a proc under recipe" do + @def.define :smoke do + "I am what I am" + end + @def.recipe.should be_a_kind_of(Proc) + @def.recipe.call.should eql("I am what I am") + end + + it "should set paramaters based on method_missing" do + @def.mind "to fly" + @def.params[:mind].should eql("to fly") + end + + it "should raise an exception if prototype_params is not a hash" do + lambda { + @def.define :monkey, Array.new do + end + }.should raise_error(ArgumentError) + end + + it "should raise an exception if define is called without a block" do + lambda { + @def.define :monkey + }.should raise_error(ArgumentError) + end + + it "should load a description from a file" do + @def.from_file(File.join(File.dirname(__FILE__), "..", "data", "definitions", "test.rb")) + @def.name.should eql(:rico_suave) + @def.params[:rich].should eql("smooth") + end +end
\ No newline at end of file diff --git a/chef/spec/unit/resource_spec.rb b/chef/spec/unit/resource_spec.rb new file mode 100644 index 0000000000..3ca48060fa --- /dev/null +++ b/chef/spec/unit/resource_spec.rb @@ -0,0 +1,144 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Resource do + before(:each) do + @resource = Chef::Resource.new("funk") + end + + it "should create a new Chef::Resource" do + @resource.should be_a_kind_of(Chef::Resource) + end + + it "should have a name" do + @resource.name.should eql("funk") + end + + it "should let you set a new name" do + @resource.name "monkey" + @resource.name.should eql("monkey") + end + + it "should not be valid without a name" do + lambda { @resource.name false }.should raise_error(ArgumentError) + end + + it "should always have a string for name" do + lambda { @resource.name Hash.new }.should raise_error(ArgumentError) + end + + it "should accept true or false for noop" do + lambda { @resource.noop true }.should_not raise_error(ArgumentError) + lambda { @resource.noop false }.should_not raise_error(ArgumentError) + lambda { @resource.noop "eat it" }.should raise_error(ArgumentError) + end + + it "should make notified resources appear in the actions hash" do + @resource.collection << Chef::Resource::ZenMaster.new("coffee") + @resource.notifies :reload, @resource.resources(:zen_master => "coffee") + @resource.actions[:reload][:delayed][0].name.should eql("coffee") + end + + it "should make notified resources be capable of acting immediately" do + @resource.collection << Chef::Resource::ZenMaster.new("coffee") + @resource.notifies :reload, @resource.resources(:zen_master => "coffee"), :immediate + @resource.actions[:reload][:immediate][0].name.should eql("coffee") + end + + it "should raise an exception if told to act in other than :delay or :immediate(ly)" do + @resource.collection << Chef::Resource::ZenMaster.new("coffee") + lambda { + @resource.notifies :reload, @resource.resources(:zen_master => "coffee"), :someday + }.should raise_error(ArgumentError) + end + + it "should allow multiple notified resources appear in the actions hash" do + @resource.collection << Chef::Resource::ZenMaster.new("coffee") + @resource.notifies :reload, @resource.resources(:zen_master => "coffee") + @resource.actions[:reload][:delayed][0].name.should eql("coffee") + @resource.collection << Chef::Resource::ZenMaster.new("beans") + @resource.notifies :reload, @resource.resources(:zen_master => "beans") + @resource.actions[:reload][:delayed][1].name.should eql("beans") + end + + it "should make resources appear in the actions hash of subscribed nodes" do + @resource.collection << Chef::Resource::ZenMaster.new("coffee") + zr = @resource.resources(:zen_master => "coffee") + @resource.subscribes :reload, zr + zr.actions[:reload][:delayed][0].name.should eql("funk") + end + + it "should make resources appear in the actions hash of subscribed nodes" do + @resource.collection << Chef::Resource::ZenMaster.new("coffee") + zr = @resource.resources(:zen_master => "coffee") + @resource.subscribes :reload, zr + zr.actions[:reload][:delayed][0].name.should eql("funk") + + @resource.collection << Chef::Resource::ZenMaster.new("bean") + zrb = @resource.resources(:zen_master => "bean") + zrb.subscribes :reload, zr + zr.actions[:reload][:delayed][1].name.should eql("bean") + end + + it "should make subscribed resources be capable of acting immediately" do + @resource.collection << Chef::Resource::ZenMaster.new("coffee") + zr = @resource.resources(:zen_master => "coffee") + @resource.subscribes :reload, zr, :immediately + zr.actions[:reload][:immediate][0].name.should eql("funk") + end + + it "should return a value if not defined" do + zm = Chef::Resource::ZenMaster.new("coffee") + zm.something(true).should eql(true) + zm.something.should eql(true) + zm.something(false).should eql(false) + zm.something.should eql(false) + end + + it "should become a string like resource_name[name]" do + zm = Chef::Resource::ZenMaster.new("coffee") + zm.to_s.should eql("zen_master[coffee]") + end + + it "should return the arguments passed with 'is'" do + zm = Chef::Resource::ZenMaster.new("coffee") + res = zm.is("one", "two", "three") + res.should eql([ "one", "two", "three" ]) + end + + it "should allow arguments preceeded by is to methods" do + @resource.noop(@resource.is(true)) + @resource.noop.should eql(true) + end + + it "should serialize to json" do + json = @resource.to_json + json.should =~ /json_class/ + json.should =~ /instance_vars/ + end + + it "should deserialize itself from json" do + json = @resource.to_json + serialized_node = JSON.parse(json) + serialized_node.should be_a_kind_of(Chef::Resource) + serialized_node.name.should eql(@resource.name) + end + +end
\ No newline at end of file diff --git a/chef/spec/unit/rest_spec.rb b/chef/spec/unit/rest_spec.rb new file mode 100644 index 0000000000..56c2954a1a --- /dev/null +++ b/chef/spec/unit/rest_spec.rb @@ -0,0 +1,228 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) +require 'uri' +require 'net/https' + +describe Chef::REST, "initialize method" do + it "should create a new Chef::REST" do + Chef::REST.new("url").should be_kind_of(Chef::REST) + end +end + +describe Chef::REST, "get_rest method" do + it "should create a url from the path and base url" do + URI.should_receive(:parse).with("url/monkey") + r = Chef::REST.new("url") + r.stub!(:run_request) + r.get_rest("monkey") + end + + it "should call run_request :GET with the composed url object" do + URI.stub!(:parse).and_return(true) + r = Chef::REST.new("url") + r.should_receive(:run_request).with(:GET, true, false, 10, false).and_return(true) + r.get_rest("monkey") + end +end + +describe Chef::REST, "delete_rest method" do + it "should create a url from the path and base url" do + URI.should_receive(:parse).with("url/monkey") + r = Chef::REST.new("url") + r.stub!(:run_request) + r.delete_rest("monkey") + end + + it "should call run_request :DELETE with the composed url object" do + URI.stub!(:parse).and_return(true) + r = Chef::REST.new("url") + r.should_receive(:run_request).with(:DELETE, true).and_return(true) + r.delete_rest("monkey") + end +end + +describe Chef::REST, "post_rest method" do + it "should create a url from the path and base url" do + URI.should_receive(:parse).with("url/monkey") + r = Chef::REST.new("url") + r.stub!(:run_request) + r.post_rest("monkey", "data") + end + + it "should call run_request :POST with the composed url object and data" do + URI.stub!(:parse).and_return(true) + r = Chef::REST.new("url") + r.should_receive(:run_request).with(:POST, true, "data").and_return(true) + r.post_rest("monkey", "data") + end +end + +describe Chef::REST, "put_rest method" do + it "should create a url from the path and base url" do + URI.should_receive(:parse).with("url/monkey") + r = Chef::REST.new("url") + r.stub!(:run_request) + r.put_rest("monkey", "data") + end + + it "should call run_request :PUT with the composed url object and data" do + URI.stub!(:parse).and_return(true) + r = Chef::REST.new("url") + r.should_receive(:run_request).with(:PUT, true, "data").and_return(true) + r.put_rest("monkey", "data") + end +end + +describe Chef::REST, "run_request method" do + before(:each) do + @r = Chef::REST.new("url") + @url_mock = mock("URI", :null_object => true) + @url_mock.stub!(:host).and_return("one") + @url_mock.stub!(:port).and_return("80") + @url_mock.stub!(:path).and_return("/") + @url_mock.stub!(:query).and_return("foo=bar") + @url_mock.stub!(:scheme).and_return("https") + @url_mock.stub!(:to_s).and_return("https://one:80/?foo=bar") + @http_response_mock = mock("Net::HTTPSuccess", :null_object => true) + @http_response_mock.stub!(:kind_of?).with(Net::HTTPSuccess).and_return(true) + @http_response_mock.stub!(:body).and_return("ninja") + @http_mock = mock("Net::HTTP", :null_object => true) + @http_mock.stub!(:verify_mode=).and_return(true) + @http_mock.stub!(:read_timeout=).and_return(true) + @http_mock.stub!(:use_ssl=).with(true).and_return(true) + @data_mock = mock("Data", :null_object => true) + @data_mock.stub!(:to_json).and_return('{ "one": "two" }') + @request_mock = mock("Request", :null_object => true) + @request_mock.stub!(:body=).and_return(true) + @request_mock.stub!(:method).and_return(true) + @request_mock.stub!(:path).and_return(true) + @http_mock.stub!(:request).and_return(@http_response_mock) + @tf_mock = mock(Tempfile, { :print => true, :close => true }) + Tempfile.stub!(:new).with("chef-rest").and_return(@tf_mock) + end + + def do_run_request(method=:GET, data=false, limit=10, raw=false) + Net::HTTP.stub!(:new).and_return(@http_mock) + @r.run_request(method, @url_mock, data, limit, raw) + end + + it "should raise an exception if the redirect limit is 0" do + lambda { @r.run_request(:GET, "/", false, 0)}.should raise_error(ArgumentError) + end + + it "should use SSL if the url starts with https" do + @url_mock.should_receive(:scheme).and_return("https") + @http_mock.should_receive(:use_ssl=).with(true).and_return(true) + do_run_request + end + + it "should set the OpenSSL Verify Mode to verify_none if requested" do + @http_mock.should_receive(:verify_mode=).and_return(true) + do_run_request + end + + it "should set a read timeout based on the rest_timeout config option" do + Chef::Config[:rest_timeout] = 10 + @http_mock.should_receive(:read_timeout=).with(10).and_return(true) + do_run_request + end + + it "should build a new HTTP GET request" do + Net::HTTP::Get.should_receive(:new).with("/?foo=bar", + { 'Accept' => 'application/json' } + ).and_return(@request_mock) + do_run_request + end + + it "should build a new HTTP POST request" do + Net::HTTP::Post.should_receive(:new).with("/", + { 'Accept' => 'application/json', "Content-Type" => 'application/json' } + ).and_return(@request_mock) + do_run_request(:POST, @data_mock) + end + + it "should build a new HTTP PUT request" do + Net::HTTP::Put.should_receive(:new).with("/", + { 'Accept' => 'application/json', "Content-Type" => 'application/json' } + ).and_return(@request_mock) + do_run_request(:PUT, @data_mock) + end + + it "should build a new HTTP DELETE request" do + Net::HTTP::Delete.should_receive(:new).with("/?foo=bar", + { 'Accept' => 'application/json' } + ).and_return(@request_mock) + do_run_request(:DELETE) + end + + it "should raise an error if the method is not GET/PUT/POST/DELETE" do + lambda { do_run_request(:MONKEY) }.should raise_error(ArgumentError) + end + + it "should run an http request" do + @http_mock.should_receive(:request).and_return(@http_response_mock) + do_run_request + end + + it "should return the body of the response on success" do + do_run_request.should eql("ninja") + end + + it "should inflate the body as to an object if JSON is returned" do + @http_response_mock.stub!(:[]).with('content-type').and_return("application/json") + JSON.should_receive(:parse).with("ninja").and_return(true) + do_run_request + end + + it "should call run_request again on a Redirect response" do + @http_response_mock.stub!(:kind_of?).with(Net::HTTPSuccess).and_return(false) + @http_response_mock.stub!(:kind_of?).with(Net::HTTPFound).and_return(true) + @http_response_mock.stub!(:[]).with('location').and_return(@url_mock.path) + lambda { do_run_request(method=:GET, data=false, limit=1) }.should raise_error(ArgumentError) + end + + it "should raise an exception on an unsuccessful request" do + @http_response_mock.stub!(:kind_of?).with(Net::HTTPSuccess).and_return(false) + @http_response_mock.stub!(:kind_of?).with(Net::HTTPFound).and_return(false) + @http_response_mock.should_receive(:error!) + do_run_request + end + + it "should build a new HTTP GET request without the application/json accept header for raw reqs" do + Net::HTTP::Get.should_receive(:new).with("/?foo=bar", {}).and_return(@request_mock) + do_run_request(:GET, false, 10, true) + end + + it "should create a tempfile for the output of a raw request" do + Tempfile.should_receive(:new).with("chef-rest").and_return(@tf_mock) + do_run_request(:GET, false, 10, true).should eql(@tf_mock) + end + + it "should populate the tempfile with the value of the raw request" do + @tf_mock.should_receive(:print, "ninja").once.and_return(true) + do_run_request(:GET, false, 10, true) + end + + it "should close the tempfile if we're doing a raw request" do + @tf_mock.should_receive(:close).once.and_return(true) + do_run_request(:GET, false, 10, true) + end + +end diff --git a/chef/spec/unit/runner_spec.rb b/chef/spec/unit/runner_spec.rb new file mode 100644 index 0000000000..f73d5efb22 --- /dev/null +++ b/chef/spec/unit/runner_spec.rb @@ -0,0 +1,103 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Runner do + before(:each) do + @mock_node = mock("Node", :null_object => true) + @mock_collection = mock("Resource Collection", :null_object => true) + @mock_provider = mock("Provider", :null_object => true) + @mock_resource = mock("Resource", :null_object => true) + new_runner + end + + it "should require a Node and a ResourceCollection" do + @mock_node.should_receive(:kind_of?).once.and_return(true) + @mock_collection.should_receive(:kind_of?).once.and_return(true) + runner = Chef::Runner.new(@mock_node, @mock_collection) + runner.should be_a_kind_of(Chef::Runner) + end + + it "should raise an exception if you pass the wrong kind of object to new" do + @mock_node.stub!(:kind_of?).and_return(false) + @mock_collecton.stub!(:kind_of?).and_return(false) + lambda { Chef::Runner.new(@mock_node, @mock_collection) }.should raise_error(ArgumentError) + end + + it "should pass each resource in the collection to a provider" do + @collection.should_receive(:each).once + @runner.converge + end + + it "should use the provider specified by the resource (if it has one)" do + provider = Chef::Provider::Easy.new(@node, @collection[0]) + @collection[0].should_receive(:provider).once.and_return(Chef::Provider::Easy) + Chef::Provider::Easy.should_receive(:new).once.and_return(provider) + @runner.converge + end + + it "should use the platform provider if it has one" do + Chef::Platform.should_receive(:find_provider_for_node).once.and_return(Chef::Provider::SnakeOil) + @runner.converge + end + + it "should run the action for each resource" do + Chef::Platform.should_receive(:find_provider_for_node).once.and_return(Chef::Provider::SnakeOil) + provider = Chef::Provider::SnakeOil.new(@node, @collection[0]) + provider.should_receive(:action_sell).once.and_return(true) + Chef::Provider::SnakeOil.should_receive(:new).once.and_return(provider) + @runner.converge + end + + it "should execute immediate actions on changed resources" do + Chef::Platform.should_receive(:find_provider_for_node).exactly(3).times.and_return(Chef::Provider::SnakeOil) + provider = Chef::Provider::SnakeOil.new(@node, @collection[0]) + Chef::Provider::SnakeOil.should_receive(:new).exactly(3).times.and_return(provider) + @collection << Chef::Resource::Cat.new("peanut", @collection) + @collection[1].notifies :buy, @collection[0], :immediately + @collection[1].updated = true + provider.should_receive(:action_buy).once.and_return(true) + @runner.converge + end + + it "should execute delayed actions on changed resources" do + Chef::Platform.should_receive(:find_provider_for_node).exactly(3).times.and_return(Chef::Provider::SnakeOil) + provider = Chef::Provider::SnakeOil.new(@node, @collection[0]) + Chef::Provider::SnakeOil.should_receive(:new).exactly(3).times.and_return(provider) + @collection << Chef::Resource::Cat.new("peanut", @collection) + @collection[1].notifies :buy, @collection[0], :delayed + @collection[1].updated = true + provider.should_receive(:action_buy).once.and_return(true) + @runner.converge + end + + def new_runner + @node = Chef::Node.new + @node.name "latte" + @node.operatingsystem "mac_os_x" + @node.operatingsystemversion "10.5.1" + @collection = Chef::ResourceCollection.new() + @collection << Chef::Resource::Cat.new("loulou", @collection) + Chef::Platform.set( + :resource => :cat, + :provider => Chef::Provider::SnakeOil + ) + @runner = Chef::Runner.new(@node, @collection) + end +end
\ No newline at end of file diff --git a/chef/spec/unit/search_index_spec.rb b/chef/spec/unit/search_index_spec.rb new file mode 100644 index 0000000000..ff234c6a41 --- /dev/null +++ b/chef/spec/unit/search_index_spec.rb @@ -0,0 +1,136 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::SearchIndex, "initialize method" do + it "should create a new Chef::SearchIndex object" do + mf = mock("Ferret::Index::Index", :null_object => true) + Ferret::Index::Index.stub!(:new).and_return(mf) + Chef::SearchIndex.new.should be_kind_of(Chef::SearchIndex) + end + + it "should create a Ferret Indexer" do + mf = mock("Ferret::Index::Index", :null_object => true) + Ferret::Index::Index.should_receive(:new).and_return(mf) + Chef::SearchIndex.new + end +end + +describe Chef::SearchIndex, "create_index_object method" do + before(:each) do + @mf = mock("Ferret::Index::Index", :null_object => true) + @fakeobj = mock("ToIndex", :null_object => true) + @the_pigeon = { :index_name => "bird", :id => "pigeon" } + @fakeobj.stub!(:respond_to?).with(:to_index).and_return(true) + @fakeobj.stub!(:to_index).and_return(@the_pigeon) + Ferret::Index::Index.stub!(:new).and_return(@mf) + end + + def do_create_index_object + index = Chef::SearchIndex.new + index.create_index_object(@fakeobj) + end + + it "should call to_index if the passed object responds to it" do + @fakeobj.should_receive(:respond_to?).with(:to_index).and_return(true) + @fakeobj.should_receive(:to_index).and_return(@the_pigeon) + do_create_index_object + end + + it "should use a hash if the passed argument does not have to_index (but is a hash)" do + @fakeobj.stub!(:respond_to?).with(:to_index).and_return(false) + @fakeobj.should_receive(:kind_of?).with(Hash).and_return(true) + do_create_index_object + end + + it "should raise SearchIndex exception if the hash does not contain an :id field" do + @the_pigeon.delete(:id) + lambda { do_create_index_object }.should raise_error(Chef::Exception::SearchIndex) + end + + it "should raise SearchIndex exception if the hash does not contain an :index_name field" do + @the_pigeon.delete(:index_name) + lambda { do_create_index_object }.should raise_error(Chef::Exception::SearchIndex) + end + + it "should raise SearchIndex exception if the hash new_object cannot be indexed" do + @fakeobj.stub!(:respond_to?).and_return(false) + @fakeobj.stub!(:kind_of?).and_return(false) + lambda { do_create_index_object }.should raise_error(Chef::Exception::SearchIndex) + end + + it "should turn index hash keys in to symbols if it has strings" do + @the_pigeon["john"] = "and_yoko" + do_create_index_object.should have_key(:john) + end +end + +describe Chef::SearchIndex, "add method" do + before(:each) do + @mf = mock("Ferret::Index::Index", :null_object => true) + @fakeobj = mock("ToIndex", :null_object => true) + @the_pigeon = { :index_name => "bird", :id => "pigeon" } + @fakeobj.stub!(:respond_to?).with(:to_index).and_return(true) + @fakeobj.stub!(:to_index).and_return(@the_pigeon) + Ferret::Index::Index.stub!(:new).and_return(@mf) + end + + def do_add + index = Chef::SearchIndex.new + index.add(@fakeobj) + end + + it "should send the resulting hash to the index" do + @mf.should_receive(:add_document).with(@the_pigeon) + do_add + end +end + +describe Chef::SearchIndex, "delete method" do + before(:each) do + @mf = mock("Ferret::Index::Index", :null_object => true) + @fakeobj = mock("ToIndex", :null_object => true) + @the_pigeon = { :index_name => "bird", :id => "pigeon" } + @fakeobj.stub!(:respond_to?).with(:to_index).and_return(true) + @fakeobj.stub!(:to_index).and_return(@the_pigeon) + Ferret::Index::Index.stub!(:new).and_return(@mf) + end + + def do_delete(object) + index = Chef::SearchIndex.new + index.delete(object) + end + + it "should delete the resulting hash to the index" do + @mf.should_receive(:delete).with(@the_pigeon[:id]) + do_delete(@fakeobj) + end +end + +describe Chef::SearchIndex, "commit method" do + before(:each) do + @mf = mock("Ferret::Index::Index", :null_object => true) + Ferret::Index::Index.stub!(:new).and_return(@mf) + end + + it "should commit index to disk" do + @mf.should_receive(:commit) + Chef::SearchIndex.new.commit + end +end diff --git a/chef/spec/unit/search_spec.rb b/chef/spec/unit/search_spec.rb new file mode 100644 index 0000000000..94e3560a23 --- /dev/null +++ b/chef/spec/unit/search_spec.rb @@ -0,0 +1,74 @@ +# +# Author:: Adam Jacob (<adam@hjksolutions.com>) +# Copyright:: Copyright (c) 2008 HJK Solutions, LLC +# 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 File.expand_path(File.join(File.dirname(__FILE__), "..", "spec_helper")) + +describe Chef::Search, "initialize method" do + before(:each) do + @mf = mock("Ferret::Index::Index", :null_object => true) + Ferret::Index::Index.stub!(:new).and_return(@mf) + end + + it "should build a Chef::Search object" do + Chef::Search.new.should be_a_kind_of(Chef::Search) + end + + it "should build a Ferret search backend" do + Ferret::Index::Index.should_receive(:new).and_return(@mf) + Chef::Search.new + end +end + +describe Chef::Search, "search method" do + before(:each) do + @mf = mock("Ferret::Index::Index", :null_object => true) + end + + def do_search(type, query, &block) + Ferret::Index::Index.stub!(:new).and_return(@mf) + cs = Chef::Search.new + if Kernel.block_given? + cs.search(type, query, &block) + else + cs.search(type, query) + end + end + + it "should build the search query from the type and query provided" do + Ferret::Index::Index.stub!(:new).and_return(@mf) + cs = Chef::Search.new + cs.should_receive(:build_search_query).with(:node, "tag:monkey") + cs.search(:node, "tag:monkey") + end + + it "should call search_each with the custom block if a block is given" do + cp = lambda { |n,u| "noting to do here" } + @mf.should_receive(:search_each).with("index_name:node AND (tag:monkey)", { :limit => :all }, &cp) + do_search(:node, "tag:monkey", &cp) + end + + it "should call search_each if a block is not given" do + @mf.should_receive(:search_each).with("index_name:node AND (tag:monkey)", {:limit => :all}) + do_search(:node, "tag:monkey") + end + + it "should return the search results" do + @mf.should_receive(:search_each).with("index_name:node AND (tag:monkey)", :limit => :all).and_return(true) + do_search(:node, "tag:monkey").should eql([]) + end +end
\ No newline at end of file |