chef 11.12.0.alpha.1-x86-mingw32 → 11.12.0.rc.1-x86-mingw32
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/lib/chef/api_client/registration.rb +46 -9
- data/lib/chef/application.rb +1 -0
- data/lib/chef/application/client.rb +25 -24
- data/lib/chef/client.rb +34 -0
- data/lib/chef/config.rb +11 -0
- data/lib/chef/cookbook/chefignore.rb +10 -2
- data/lib/chef/cookbook/metadata.rb +31 -3
- data/lib/chef/cookbook/synchronizer.rb +2 -2
- data/lib/chef/cookbook/syntax_check.rb +4 -4
- data/lib/chef/encrypted_data_bag_item.rb +37 -1
- data/lib/chef/exceptions.rb +1 -0
- data/lib/chef/guard_interpreter/default_guard_interpreter.rb +42 -0
- data/lib/chef/guard_interpreter/resource_guard_interpreter.rb +122 -0
- data/lib/chef/http.rb +0 -1
- data/lib/chef/http/decompressor.rb +7 -4
- data/lib/chef/http/simple.rb +5 -0
- data/lib/chef/http/validate_content_length.rb +28 -12
- data/lib/chef/knife.rb +1 -0
- data/lib/chef/knife/client_bulk_delete.rb +48 -9
- data/lib/chef/knife/client_delete.rb +4 -4
- data/lib/chef/knife/cookbook_bulk_delete.rb +1 -1
- data/lib/chef/knife/cookbook_upload.rb +17 -7
- data/lib/chef/knife/core/bootstrap_context.rb +1 -1
- data/lib/chef/knife/core/ui.rb +42 -5
- data/lib/chef/knife/node_run_list_add.rb +31 -2
- data/lib/chef/knife/ssh.rb +44 -31
- data/lib/chef/knife/ssl_check.rb +213 -0
- data/lib/chef/knife/ssl_fetch.rb +145 -0
- data/lib/chef/mixin/deep_merge.rb +13 -5
- data/lib/chef/mixin/shell_out.rb +9 -3
- data/lib/chef/node.rb +23 -4
- data/lib/chef/node/immutable_collections.rb +32 -0
- data/lib/chef/platform/provider_mapping.rb +21 -18
- data/lib/chef/platform/query_helpers.rb +10 -2
- data/lib/chef/policy_builder/expand_node_object.rb +3 -6
- data/lib/chef/provider/cron.rb +25 -3
- data/lib/chef/provider/mount/mount.rb +1 -1
- data/lib/chef/provider/package/dpkg.rb +2 -1
- data/lib/chef/provider/package/windows.rb +80 -0
- data/lib/chef/provider/package/windows/msi.rb +69 -0
- data/lib/chef/provider/powershell_script.rb +19 -6
- data/lib/chef/provider/service/solaris.rb +11 -7
- data/lib/chef/resource.rb +18 -5
- data/lib/chef/resource/conditional.rb +20 -7
- data/lib/chef/resource/cron.rb +18 -2
- data/lib/chef/resource/execute.rb +0 -2
- data/lib/chef/resource/powershell_script.rb +23 -1
- data/lib/chef/resource/script.rb +25 -0
- data/lib/chef/resource/subversion.rb +4 -0
- data/lib/chef/resource/windows_package.rb +79 -0
- data/lib/chef/resource/windows_script.rb +0 -5
- data/lib/chef/resources.rb +1 -0
- data/lib/chef/rest.rb +6 -1
- data/lib/chef/run_context.rb +22 -2
- data/lib/chef/run_context/cookbook_compiler.rb +12 -0
- data/lib/chef/util/editor.rb +92 -0
- data/lib/chef/util/file_edit.rb +22 -54
- data/lib/chef/version.rb +2 -2
- data/lib/chef/win32/api/installer.rb +166 -0
- data/lib/chef/win32/version.rb +8 -0
- data/spec/data/standalone_cookbook/Gemfile +1 -0
- data/spec/data/standalone_cookbook/chefignore +9 -0
- data/spec/data/standalone_cookbook/recipes/default.rb +3 -0
- data/spec/data/standalone_cookbook/vendor/bundle/ruby/2.0.0/gems/multi_json-1.9.0/lib/multi_json.rb +1 -0
- data/spec/functional/resource/powershell_spec.rb +262 -1
- data/spec/functional/win32/versions_spec.rb +3 -3
- data/spec/integration/knife/chefignore_spec.rb +1 -2
- data/spec/integration/knife/raw_spec.rb +8 -13
- data/spec/integration/knife/redirection_spec.rb +6 -14
- data/spec/integration/solo/solo_spec.rb +19 -0
- data/spec/support/shared/functional/windows_script.rb +1 -1
- data/spec/support/shared/integration/app_server_support.rb +42 -0
- data/spec/support/shared/integration/integration_helper.rb +1 -0
- data/spec/support/shared/unit/script_resource.rb +38 -0
- data/spec/unit/api_client/registration_spec.rb +109 -38
- data/spec/unit/application/client_spec.rb +48 -1
- data/spec/unit/cookbook/chefignore_spec.rb +10 -0
- data/spec/unit/cookbook/metadata_spec.rb +45 -1
- data/spec/unit/cookbook/syntax_check_spec.rb +28 -0
- data/spec/unit/cookbook_spec.rb +0 -10
- data/spec/unit/guard_interpreter/resource_guard_interpreter_spec.rb +56 -0
- data/spec/unit/http/simple_spec.rb +32 -0
- data/spec/unit/http/validate_content_length_spec.rb +187 -0
- data/spec/unit/knife/bootstrap_spec.rb +13 -4
- data/spec/unit/knife/client_bulk_delete_spec.rb +123 -38
- data/spec/unit/knife/client_delete_spec.rb +4 -4
- data/spec/unit/knife/cookbook_upload_spec.rb +181 -88
- data/spec/unit/knife/core/bootstrap_context_spec.rb +11 -1
- data/spec/unit/knife/core/ui_spec.rb +109 -38
- data/spec/unit/knife/node_run_list_add_spec.rb +24 -1
- data/spec/unit/knife/ssh_spec.rb +17 -6
- data/spec/unit/knife/ssl_check_spec.rb +187 -0
- data/spec/unit/knife/ssl_fetch_spec.rb +151 -0
- data/spec/unit/mixin/deep_merge_spec.rb +17 -0
- data/spec/unit/node/immutable_collections_spec.rb +55 -0
- data/spec/unit/node_spec.rb +9 -0
- data/spec/unit/platform/query_helpers_spec.rb +32 -0
- data/spec/unit/platform_spec.rb +193 -175
- data/spec/unit/policy_builder/expand_node_object_spec.rb +1 -1
- data/spec/unit/provider/cron_spec.rb +175 -1
- data/spec/unit/provider/mount/mount_spec.rb +33 -3
- data/spec/unit/provider/package/dpkg_spec.rb +4 -0
- data/spec/unit/provider/package/windows/msi_spec.rb +60 -0
- data/spec/unit/provider/package/windows_spec.rb +80 -0
- data/spec/unit/provider/service/macosx_spec.rb +3 -3
- data/spec/unit/provider/service/solaris_smf_service_spec.rb +35 -10
- data/spec/unit/pure_application_spec.rb +32 -0
- data/spec/unit/recipe_spec.rb +4 -0
- data/spec/unit/resource/conditional_spec.rb +13 -12
- data/spec/unit/resource/cron_spec.rb +7 -2
- data/spec/unit/resource/powershell_spec.rb +85 -2
- data/spec/unit/resource/subversion_spec.rb +5 -0
- data/spec/unit/resource/windows_package_spec.rb +74 -0
- data/spec/unit/resource_spec.rb +23 -1
- data/spec/unit/rest_spec.rb +15 -0
- data/spec/unit/run_context/cookbook_compiler_spec.rb +12 -0
- data/spec/unit/run_context_spec.rb +7 -0
- data/spec/unit/util/editor_spec.rb +152 -0
- data/spec/unit/util/file_edit_spec.rb +37 -1
- metadata +41 -30
@@ -127,7 +127,7 @@ describe Chef::Application::Client, "configure_chef" do
|
|
127
127
|
end
|
128
128
|
|
129
129
|
describe Chef::Application::Client, "run_application", :unix_only do
|
130
|
-
before do
|
130
|
+
before(:each) do
|
131
131
|
@pipe = IO.pipe
|
132
132
|
@app = Chef::Application::Client.new
|
133
133
|
@app.stub(:run_chef_client) do
|
@@ -147,4 +147,51 @@ describe Chef::Application::Client, "run_application", :unix_only do
|
|
147
147
|
IO.select([@pipe[0]], nil, nil, 0).should_not be_nil
|
148
148
|
@pipe[0].gets.should == "finished\n"
|
149
149
|
end
|
150
|
+
|
151
|
+
describe "when splay is set" do
|
152
|
+
before do
|
153
|
+
Chef::Config[:splay] = 10
|
154
|
+
Chef::Config[:interval] = 10
|
155
|
+
|
156
|
+
run_count = 0
|
157
|
+
|
158
|
+
# uncomment to debug failures...
|
159
|
+
# Chef::Log.init($stderr)
|
160
|
+
# Chef::Log.level = :debug
|
161
|
+
|
162
|
+
@app.stub(:run_chef_client) do
|
163
|
+
|
164
|
+
run_count += 1
|
165
|
+
if run_count > 3
|
166
|
+
exit 0
|
167
|
+
end
|
168
|
+
|
169
|
+
# If everything is fine, sending USR1 to self should prevent
|
170
|
+
# app to go into splay sleep forever.
|
171
|
+
Process.kill("USR1", Process.pid)
|
172
|
+
end
|
173
|
+
|
174
|
+
number_of_sleep_calls = 0
|
175
|
+
|
176
|
+
# This is a very complicated way of writing
|
177
|
+
# @app.should_receive(:sleep).once.
|
178
|
+
# We have to do it this way because the main loop of
|
179
|
+
# Chef::Application::Client swallows most exceptions, and we need to be
|
180
|
+
# able to expose our expectation failures to the parent process in the test.
|
181
|
+
@app.stub(:sleep) do |arg|
|
182
|
+
number_of_sleep_calls += 1
|
183
|
+
if number_of_sleep_calls > 1
|
184
|
+
exit 127
|
185
|
+
end
|
186
|
+
end
|
187
|
+
end
|
188
|
+
|
189
|
+
it "shouldn't sleep when sent USR1" do
|
190
|
+
pid = fork do
|
191
|
+
@app.run_application
|
192
|
+
end
|
193
|
+
_pid, result = Process.waitpid2(pid)
|
194
|
+
result.exitstatus.should == 0
|
195
|
+
end
|
196
|
+
end
|
150
197
|
end
|
@@ -36,4 +36,14 @@ describe Chef::Cookbook::Chefignore do
|
|
36
36
|
@chefignore.ignored?('recipes/ignoreme.rb').should be_true
|
37
37
|
@chefignore.ignored?('recipes/dontignoreme.rb').should be_false
|
38
38
|
end
|
39
|
+
|
40
|
+
context "when using the single cookbook pattern" do
|
41
|
+
before do
|
42
|
+
@chefignore = Chef::Cookbook::Chefignore.new(File.join(CHEF_SPEC_DATA, 'standalone_cookbook'))
|
43
|
+
end
|
44
|
+
|
45
|
+
it "loads the globs in the chefignore file" do
|
46
|
+
@chefignore.ignores.should =~ %w[recipes/ignoreme.rb ignored vendor/bundle/*]
|
47
|
+
end
|
48
|
+
end
|
39
49
|
end
|
@@ -402,7 +402,7 @@ describe Chef::Cookbook::Metadata do
|
|
402
402
|
@meta.attributes["db/mysql/databases"][:recipes].should == []
|
403
403
|
end
|
404
404
|
|
405
|
-
it "should allow the default value to be a string, array, or
|
405
|
+
it "should allow the default value to be a string, array, hash, boolean or numeric" do
|
406
406
|
lambda {
|
407
407
|
@meta.attribute("db/mysql/databases", :default => [])
|
408
408
|
}.should_not raise_error
|
@@ -412,11 +412,55 @@ describe Chef::Cookbook::Metadata do
|
|
412
412
|
lambda {
|
413
413
|
@meta.attribute("db/mysql/databases", :default => "alice in chains")
|
414
414
|
}.should_not raise_error
|
415
|
+
lambda {
|
416
|
+
@meta.attribute("db/mysql/databases", :default => 1337)
|
417
|
+
}.should_not raise_error
|
418
|
+
lambda {
|
419
|
+
@meta.attribute("db/mysql/databases", :default => true)
|
420
|
+
}.should_not raise_error
|
415
421
|
lambda {
|
416
422
|
@meta.attribute("db/mysql/databases", :required => :not_gonna_do_it)
|
417
423
|
}.should raise_error(ArgumentError)
|
418
424
|
end
|
419
425
|
|
426
|
+
it "should limit the types allowed in the choice array" do
|
427
|
+
options = {
|
428
|
+
:type => "string",
|
429
|
+
:choice => [ "test1", "test2" ],
|
430
|
+
:default => "test1"
|
431
|
+
}
|
432
|
+
lambda {
|
433
|
+
@meta.attribute("test_cookbook/test", options)
|
434
|
+
}.should_not raise_error
|
435
|
+
|
436
|
+
options = {
|
437
|
+
:type => "boolean",
|
438
|
+
:choice => [ true, false ],
|
439
|
+
:default => true
|
440
|
+
}
|
441
|
+
lambda {
|
442
|
+
@meta.attribute("test_cookbook/test", options)
|
443
|
+
}.should_not raise_error
|
444
|
+
|
445
|
+
options = {
|
446
|
+
:type => "numeric",
|
447
|
+
:choice => [ 1337, 420 ],
|
448
|
+
:default => 1337
|
449
|
+
}
|
450
|
+
lambda {
|
451
|
+
@meta.attribute("test_cookbook/test", options)
|
452
|
+
}.should_not raise_error
|
453
|
+
|
454
|
+
options = {
|
455
|
+
:type => "numeric",
|
456
|
+
:choice => [ true, "false" ],
|
457
|
+
:default => false
|
458
|
+
}
|
459
|
+
lambda {
|
460
|
+
@meta.attribute("test_cookbook/test", options)
|
461
|
+
}.should raise_error
|
462
|
+
end
|
463
|
+
|
420
464
|
it "should error if default used with calculated" do
|
421
465
|
lambda {
|
422
466
|
attrs = {
|
@@ -24,6 +24,17 @@ describe Chef::Cookbook::SyntaxCheck do
|
|
24
24
|
let(:cookbook_path) { File.join(CHEF_SPEC_DATA, 'cookbooks', 'openldap') }
|
25
25
|
let(:syntax_check) { Chef::Cookbook::SyntaxCheck.new(cookbook_path) }
|
26
26
|
|
27
|
+
let(:open_ldap_cookbook_files) {
|
28
|
+
%w{ attributes/default.rb
|
29
|
+
attributes/smokey.rb
|
30
|
+
definitions/client.rb
|
31
|
+
definitions/server.rb
|
32
|
+
metadata.rb
|
33
|
+
recipes/default.rb
|
34
|
+
recipes/gigantor.rb
|
35
|
+
recipes/one.rb }.map{ |f| File.join(cookbook_path, f) }
|
36
|
+
}
|
37
|
+
|
27
38
|
before do
|
28
39
|
Chef::Log.logger = Logger.new(StringIO.new)
|
29
40
|
Chef::Log.level = :warn # suppress "Syntax OK" messages
|
@@ -47,6 +58,23 @@ describe Chef::Cookbook::SyntaxCheck do
|
|
47
58
|
Chef::Config[:cookbook_path] = File.dirname(cookbook_path)
|
48
59
|
syntax_check = Chef::Cookbook::SyntaxCheck.for_cookbook(:openldap)
|
49
60
|
syntax_check.cookbook_path.should == cookbook_path
|
61
|
+
syntax_check.ruby_files.sort.should == open_ldap_cookbook_files.sort
|
62
|
+
end
|
63
|
+
|
64
|
+
it "creates a syntax checker given the cookbook name and cookbook_path" do
|
65
|
+
syntax_check = Chef::Cookbook::SyntaxCheck.for_cookbook(:openldap, File.join(CHEF_SPEC_DATA, 'cookbooks'))
|
66
|
+
syntax_check.cookbook_path.should == cookbook_path
|
67
|
+
syntax_check.ruby_files.sort.should == open_ldap_cookbook_files.sort
|
68
|
+
end
|
69
|
+
|
70
|
+
context "when using a standalone cookbook" do
|
71
|
+
let(:cookbook_path) { File.join(CHEF_SPEC_DATA, 'standalone_cookbook') }
|
72
|
+
|
73
|
+
it "creates a syntax checker given the cookbook name and cookbook_path for a standalone cookbook" do
|
74
|
+
syntax_check = Chef::Cookbook::SyntaxCheck.for_cookbook(:standalone_cookbook, CHEF_SPEC_DATA)
|
75
|
+
syntax_check.cookbook_path.should == cookbook_path
|
76
|
+
syntax_check.ruby_files.should == [File.join(cookbook_path, 'recipes/default.rb')]
|
77
|
+
end
|
50
78
|
end
|
51
79
|
|
52
80
|
describe "when first created" do
|
data/spec/unit/cookbook_spec.rb
CHANGED
@@ -68,16 +68,6 @@ describe Chef::CookbookVersion do
|
|
68
68
|
@cookbook.preferred_filename(@node, :files, 'a-filename', 'the-checksum').should be_nil
|
69
69
|
end
|
70
70
|
|
71
|
-
it "should allow you to include a fully-qualified recipe using the DSL" do
|
72
|
-
# DSL method include_recipe allows multiple arguments, so extract the first
|
73
|
-
@node.should_receive(:loaded_recipe).with(:openldap, "gigantor")
|
74
|
-
recipe = @run_context.include_recipe("openldap::gigantor").first
|
75
|
-
|
76
|
-
recipe.recipe_name.should == "gigantor"
|
77
|
-
recipe.cookbook_name.should == :openldap
|
78
|
-
@run_context.resource_collection[0].name.should == "blanket"
|
79
|
-
end
|
80
|
-
|
81
71
|
it "should raise an ArgumentException if you try to load a bad recipe name" do
|
82
72
|
lambda { @cookbook.load_recipe("doesnt_exist", @node) }.should raise_error(ArgumentError)
|
83
73
|
end
|
@@ -0,0 +1,56 @@
|
|
1
|
+
#
|
2
|
+
# Author:: Adam Edwards (<adamed@getchef.com>)
|
3
|
+
# Copyright:: Copyright (c) 2014 Chef Software, Inc.
|
4
|
+
# License:: Apache License, Version 2.0
|
5
|
+
#
|
6
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
7
|
+
# you may not use this file except in compliance with the License.
|
8
|
+
# You may obtain a copy of the License at
|
9
|
+
#
|
10
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
11
|
+
#
|
12
|
+
# Unless required by applicable law or agreed to in writing, software
|
13
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
14
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
15
|
+
# See the License for the specific language governing permissions and
|
16
|
+
# limitations under the License.
|
17
|
+
#
|
18
|
+
|
19
|
+
require 'spec_helper'
|
20
|
+
|
21
|
+
describe Chef::GuardInterpreter::ResourceGuardInterpreter do
|
22
|
+
before(:each) do
|
23
|
+
node = Chef::Node.new
|
24
|
+
|
25
|
+
node.default["kernel"] = Hash.new
|
26
|
+
node.default["kernel"][:machine] = :x86_64.to_s
|
27
|
+
|
28
|
+
run_context = Chef::RunContext.new(node, nil, nil)
|
29
|
+
|
30
|
+
@resource = Chef::Resource.new("powershell_unit_test", run_context)
|
31
|
+
@resource.stub(:run_action)
|
32
|
+
@resource.stub(:updated).and_return(true)
|
33
|
+
end
|
34
|
+
|
35
|
+
describe "when evaluating a guard resource" do
|
36
|
+
let(:resource) { @resource }
|
37
|
+
|
38
|
+
it "should allow guard interpreter to be set to Chef::Resource::Script" do
|
39
|
+
resource.guard_interpreter(:script)
|
40
|
+
allow_any_instance_of(Chef::GuardInterpreter::ResourceGuardInterpreter).to receive(:evaluate_action).and_return(false)
|
41
|
+
resource.only_if("echo hi")
|
42
|
+
end
|
43
|
+
|
44
|
+
it "should allow guard interpreter to be set to Chef::Resource::PowershellScript derived indirectly from Chef::Resource::Script" do
|
45
|
+
resource.guard_interpreter(:powershell_script)
|
46
|
+
allow_any_instance_of(Chef::GuardInterpreter::ResourceGuardInterpreter).to receive(:evaluate_action).and_return(false)
|
47
|
+
resource.only_if("echo hi")
|
48
|
+
end
|
49
|
+
|
50
|
+
it "should raise an exception if guard_interpreter is set to a resource not derived from Chef::Resource::Script" do
|
51
|
+
resource.guard_interpreter(:file)
|
52
|
+
expect { resource.only_if("echo hi") }.to raise_error ArgumentError
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
@@ -0,0 +1,32 @@
|
|
1
|
+
#
|
2
|
+
# Author:: Serdar Sutay (<serdar@opscode.com>)
|
3
|
+
# Copyright:: Copyright (c) 2014 Chef Software, Inc.
|
4
|
+
# License:: Apache License, Version 2.0
|
5
|
+
#
|
6
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
7
|
+
# you may not use this file except in compliance with the License.
|
8
|
+
# You may obtain a copy of the License at
|
9
|
+
#
|
10
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
11
|
+
#
|
12
|
+
# Unless required by applicable law or agreed to in writing, software
|
13
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
14
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
15
|
+
# See the License for the specific language governing permissions and
|
16
|
+
# limitations under the License.
|
17
|
+
#
|
18
|
+
|
19
|
+
require 'spec_helper'
|
20
|
+
|
21
|
+
describe Chef::HTTP::Simple do
|
22
|
+
it "should have content length validation middleware after compressor middleware" do
|
23
|
+
client = Chef::HTTP::Simple.new("dummy.com")
|
24
|
+
middlewares = client.instance_variable_get(:@middlewares)
|
25
|
+
content_length = middlewares.find_index { |e| e.is_a? Chef::HTTP::ValidateContentLength }
|
26
|
+
decompressor = middlewares.find_index { |e| e.is_a? Chef::HTTP::Decompressor }
|
27
|
+
|
28
|
+
content_length.should_not be_nil
|
29
|
+
decompressor.should_not be_nil
|
30
|
+
(decompressor < content_length).should be_true
|
31
|
+
end
|
32
|
+
end
|
@@ -0,0 +1,187 @@
|
|
1
|
+
#
|
2
|
+
# Author:: Serdar Sutay (<serdar@opscode.com>)
|
3
|
+
# Copyright:: Copyright (c) 2014 Chef Software, Inc.
|
4
|
+
# License:: Apache License, Version 2.0
|
5
|
+
#
|
6
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
7
|
+
# you may not use this file except in compliance with the License.
|
8
|
+
# You may obtain a copy of the License at
|
9
|
+
#
|
10
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
11
|
+
#
|
12
|
+
# Unless required by applicable law or agreed to in writing, software
|
13
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
14
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
15
|
+
# See the License for the specific language governing permissions and
|
16
|
+
# limitations under the License.
|
17
|
+
#
|
18
|
+
|
19
|
+
require 'spec_helper'
|
20
|
+
require 'stringio'
|
21
|
+
|
22
|
+
describe Chef::HTTP::ValidateContentLength do
|
23
|
+
class TestClient < Chef::HTTP
|
24
|
+
use Chef::HTTP::ValidateContentLength
|
25
|
+
end
|
26
|
+
|
27
|
+
let(:method) { "GET" }
|
28
|
+
let(:url) { "http://dummy.com" }
|
29
|
+
let(:headers) { {} }
|
30
|
+
let(:data) { false }
|
31
|
+
|
32
|
+
let(:request) { }
|
33
|
+
let(:return_value) { "200" }
|
34
|
+
|
35
|
+
# Test Variables
|
36
|
+
let(:request_type) { :streaming }
|
37
|
+
let(:content_length_value) { 23 }
|
38
|
+
let(:streaming_length) { 23 }
|
39
|
+
let(:response_body) { "Thanks for checking in." }
|
40
|
+
let(:response_headers) {
|
41
|
+
{
|
42
|
+
"content-length" => content_length_value
|
43
|
+
}
|
44
|
+
}
|
45
|
+
|
46
|
+
let(:response) {
|
47
|
+
m = double('HttpResponse', :body => response_body)
|
48
|
+
m.stub(:[]) do |key|
|
49
|
+
response_headers[key]
|
50
|
+
end
|
51
|
+
|
52
|
+
m
|
53
|
+
}
|
54
|
+
|
55
|
+
let(:middleware) {
|
56
|
+
client = TestClient.new(url)
|
57
|
+
client.middlewares[0]
|
58
|
+
}
|
59
|
+
|
60
|
+
def run_content_length_validation
|
61
|
+
stream_handler = middleware.stream_response_handler(response)
|
62
|
+
middleware.handle_request(method, url, headers, data)
|
63
|
+
|
64
|
+
case request_type
|
65
|
+
when :streaming
|
66
|
+
# First stream the data
|
67
|
+
data_length = streaming_length
|
68
|
+
while data_length > 0
|
69
|
+
chunk_size = data_length > 10 ? 10 : data_length
|
70
|
+
stream_handler.handle_chunk(double("Chunk", :bytesize => chunk_size))
|
71
|
+
data_length -= chunk_size
|
72
|
+
end
|
73
|
+
|
74
|
+
# Finally call stream complete
|
75
|
+
middleware.handle_stream_complete(response, request, return_value)
|
76
|
+
when :direct
|
77
|
+
middleware.handle_response(response, request, return_value)
|
78
|
+
else
|
79
|
+
raise "Unknown request_type: #{request_type}"
|
80
|
+
end
|
81
|
+
end
|
82
|
+
|
83
|
+
let(:debug_stream) { StringIO.new }
|
84
|
+
let(:debug_output) { debug_stream.string }
|
85
|
+
|
86
|
+
before(:each) {
|
87
|
+
Chef::Log.level = :debug
|
88
|
+
Chef::Log.stub(:debug) do |message|
|
89
|
+
debug_stream.puts message
|
90
|
+
end
|
91
|
+
}
|
92
|
+
|
93
|
+
describe "without response body" do
|
94
|
+
let(:request_type) { :direct }
|
95
|
+
let(:response_body) { "Thanks for checking in." }
|
96
|
+
|
97
|
+
it "shouldn't raise error" do
|
98
|
+
lambda { run_content_length_validation }.should_not raise_error
|
99
|
+
end
|
100
|
+
end
|
101
|
+
|
102
|
+
describe "without Content-Length header" do
|
103
|
+
let(:response_headers) { { } }
|
104
|
+
|
105
|
+
[ "direct", "streaming" ].each do |req_type|
|
106
|
+
describe "when running #{req_type} request" do
|
107
|
+
let(:request_type) { req_type.to_sym }
|
108
|
+
|
109
|
+
it "should skip validation and log for debug" do
|
110
|
+
run_content_length_validation
|
111
|
+
debug_output.should include("HTTP server did not include a Content-Length header in response")
|
112
|
+
end
|
113
|
+
end
|
114
|
+
end
|
115
|
+
end
|
116
|
+
|
117
|
+
describe "with correct Content-Length header" do
|
118
|
+
[ "direct", "streaming" ].each do |req_type|
|
119
|
+
describe "when running #{req_type} request" do
|
120
|
+
let(:request_type) { req_type.to_sym }
|
121
|
+
|
122
|
+
it "should validate correctly" do
|
123
|
+
run_content_length_validation
|
124
|
+
debug_output.should include("Content-Length validated correctly.")
|
125
|
+
end
|
126
|
+
end
|
127
|
+
end
|
128
|
+
end
|
129
|
+
|
130
|
+
describe "with wrong Content-Length header" do
|
131
|
+
let(:content_length_value) { 25 }
|
132
|
+
[ "direct", "streaming" ].each do |req_type|
|
133
|
+
describe "when running #{req_type} request" do
|
134
|
+
let(:request_type) { req_type.to_sym }
|
135
|
+
|
136
|
+
it "should raise ContentLengthMismatch error" do
|
137
|
+
lambda { run_content_length_validation }.should raise_error(Chef::Exceptions::ContentLengthMismatch)
|
138
|
+
end
|
139
|
+
end
|
140
|
+
end
|
141
|
+
end
|
142
|
+
|
143
|
+
describe "when download is interrupted" do
|
144
|
+
let(:streaming_length) { 12 }
|
145
|
+
|
146
|
+
it "should raise ContentLengthMismatch error" do
|
147
|
+
lambda { run_content_length_validation }.should raise_error(Chef::Exceptions::ContentLengthMismatch)
|
148
|
+
end
|
149
|
+
end
|
150
|
+
|
151
|
+
describe "when Transfer-Encoding & Content-Length is set" do
|
152
|
+
let(:response_headers) {
|
153
|
+
{
|
154
|
+
"content-length" => content_length_value,
|
155
|
+
"transfer-encoding" => "chunked"
|
156
|
+
}
|
157
|
+
}
|
158
|
+
|
159
|
+
[ "direct", "streaming" ].each do |req_type|
|
160
|
+
describe "when running #{req_type} request" do
|
161
|
+
let(:request_type) { req_type.to_sym }
|
162
|
+
|
163
|
+
it "should skip validation and log for debug" do
|
164
|
+
run_content_length_validation
|
165
|
+
debug_output.should include("Transfer-Encoding header is set, skipping Content-Length check.")
|
166
|
+
end
|
167
|
+
end
|
168
|
+
end
|
169
|
+
end
|
170
|
+
|
171
|
+
describe "when client is being reused" do
|
172
|
+
before do
|
173
|
+
run_content_length_validation
|
174
|
+
debug_output.should include("Content-Length validated correctly.")
|
175
|
+
end
|
176
|
+
|
177
|
+
it "should reset internal counter" do
|
178
|
+
middleware.instance_variable_get(:@content_length_counter).should be_nil
|
179
|
+
end
|
180
|
+
|
181
|
+
it "should validate correctly second time" do
|
182
|
+
run_content_length_validation
|
183
|
+
debug_output.should include("Content-Length validated correctly.")
|
184
|
+
end
|
185
|
+
end
|
186
|
+
|
187
|
+
end
|