test-kitchen-provisioners 0.1

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.
@@ -0,0 +1,3 @@
1
+ Gemfile.lock
2
+ tmp
3
+ *.gem
@@ -0,0 +1,6 @@
1
+ language: ruby
2
+ rvm:
3
+ - 2.0.0
4
+ - 1.9.3
5
+ install: bundle install
6
+ script: rspec
data/Gemfile ADDED
@@ -0,0 +1,9 @@
1
+ source 'https://rubygems.org'
2
+
3
+ gemspec
4
+
5
+ gem "rake"
6
+
7
+ group :test do
8
+ gem "rspec", "~> 2.14"
9
+ end
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2013 David Genord II
2
+
3
+ MIT License
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining
6
+ a copy of this software and associated documentation files (the
7
+ "Software"), to deal in the Software without restriction, including
8
+ without limitation the rights to use, copy, modify, merge, publish,
9
+ distribute, sublicense, and/or sell copies of the Software, and to
10
+ permit persons to whom the Software is furnished to do so, subject to
11
+ the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be
14
+ included in all copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
@@ -0,0 +1,28 @@
1
+ test-kitchen-provisioners
2
+ =========================
3
+
4
+ A collection of provisioners to be used in [test-kitchen](https://github.com/opscode/test-kitchen) runs to
5
+ add functionality to the Provisioner system.
6
+
7
+ Unfortunately as Provisioners were not built with chaining in mind, these may come across as a bit hackish.
8
+
9
+ ## Usage
10
+
11
+ gem install test-kitchen-provisioners
12
+
13
+ Set the `provisioner` value in `.kitchen.yml` to one of the available provisioners.
14
+
15
+ ## Available Providers
16
+
17
+ ### ChefAptRubyRolesChefZero
18
+
19
+ provisioner: chef_apt_ruby_roles_chef_zero
20
+
21
+ A chain provisioners that ensure:
22
+
23
+ 1. Aptitude update is called before chef-zero starts. This lets us not have to make every single cookbook
24
+ reliant on the apt cookbook.
25
+ 2. Ruby roles get converted to JSON. chef-zero only knows how to work with JSON files, but we like Ruby.
26
+ This provider runs against the cookbook sandbox on the host machine before the files are uploaded to the VM
27
+ 3. Calls the built-in chef-zero provisioner to continue as normal.
28
+
File without changes
@@ -0,0 +1,16 @@
1
+ require 'kitchen'
2
+ require 'kitchen/provisioner/base'
3
+
4
+ module Kitchen
5
+ module Provisioner
6
+
7
+ # Apt Provisioner
8
+ # This makes sure that aptitude is up-to-date at the beginning of the
9
+ # chef run before cookbooks are pulled in.
10
+ class Apt < Base
11
+ def run_command
12
+ "#{sudo("aptitude")} update"
13
+ end
14
+ end
15
+ end
16
+ end
@@ -0,0 +1,58 @@
1
+ require 'kitchen/provisioner/apt'
2
+ require 'kitchen/provisioner/ruby_roles'
3
+ require 'kitchen/provisioner/chef_zero'
4
+
5
+ module Kitchen
6
+ module Provisioner
7
+
8
+ # This Provisioner combines the roles of the Apt, Ruby Roles, and ChefZero
9
+ # provisioners, to ensure a server is ready to run before ChefZero starts its magic
10
+ # Must be named "Chef..." otherwise TestKitchen won't understand that we intend to use chef
11
+ class ChefAptRubyRolesChefZero < Base
12
+ def initialize(instance, options)
13
+ super
14
+ @apt = Apt.new(instance, options)
15
+ @chef_zero = ChefZero.new(instance, options)
16
+ @ruby_roles = RubyRoles.new(instance, options)
17
+ end
18
+
19
+ def install_command
20
+ @chef_zero.install_command
21
+ end
22
+
23
+ def init_command
24
+ @chef_zero.init_command
25
+ end
26
+
27
+ # Let chef-zero create the sandbox then convert the Ruby-based role files
28
+ # into JSON before uploads happen (which happen right after this returns)
29
+ def create_sandbox
30
+ sandbox_path = @chef_zero.create_sandbox
31
+ @ruby_roles.roles_path = File.join(sandbox_path, "roles")
32
+ @ruby_roles.create_sandbox
33
+ sandbox_path
34
+ end
35
+
36
+ def prepare_command
37
+ @chef_zero.prepare_command
38
+ end
39
+
40
+ # Run the apt provisioner before starting up chef-zero
41
+ def run_command
42
+ [
43
+ @apt.run_command,
44
+ @chef_zero.run_command
45
+ ].join(" && ")
46
+ end
47
+
48
+ def cleanup_sandbox
49
+ @chef_zero.cleanup_sandbox
50
+ end
51
+
52
+ def home_path
53
+ @chef_zero.home_path
54
+ end
55
+ end
56
+
57
+ end
58
+ end
@@ -0,0 +1,32 @@
1
+ require 'kitchen'
2
+ require 'kitchen/provisioner/base'
3
+ require 'chef'
4
+
5
+ module Kitchen
6
+ module Provisioner
7
+
8
+ # RubyRoles Provisioner
9
+ # This provisioner, when given a path to a cookbook's roles directory, will iterate through
10
+ # each Ruby-written role and generate the JSON version.
11
+ # This then allows the ChefZero tool to pick up these roles, as it only knows about JSON
12
+ class RubyRoles < Base
13
+
14
+ attr_accessor :roles_path
15
+
16
+ def create_sandbox
17
+ # From https://gist.github.com/red56/834890
18
+ Dir.glob(File.join(@roles_path, "*.rb")).each do |ruby_role_file_name|
19
+ role = Chef::Role.new
20
+ role.from_file(ruby_role_file_name)
21
+ json_file_name = ruby_role_file_name.gsub(/rb$/, "json")
22
+
23
+ File.open(json_file_name, "w+") do |json_file|
24
+ json_file.write(JSON.pretty_generate(role))
25
+ end
26
+ end
27
+ end
28
+
29
+ end
30
+
31
+ end
32
+ end
@@ -0,0 +1,10 @@
1
+ require 'spec_helper'
2
+ require 'kitchen/provisioner/apt'
3
+
4
+ describe Kitchen::Provisioner::Apt do
5
+ it "runs aptitude update" do
6
+ provisioner = Kitchen::Provisioner::Apt.new(BogusInstance.new, nil)
7
+ expect(provisioner).to receive(:sudo).with("aptitude").and_return("aptitude")
8
+ expect(provisioner.run_command).to eq("aptitude update")
9
+ end
10
+ end
@@ -0,0 +1,69 @@
1
+ require 'spec_helper'
2
+ require 'kitchen/provisioner/chef_apt_ruby_roles_chef_zero'
3
+
4
+ describe Kitchen::Provisioner::ChefAptRubyRolesChefZero do
5
+
6
+ before do
7
+ @provisioner = Kitchen::Provisioner::ChefAptRubyRolesChefZero.new(BogusInstance.new, nil)
8
+ @apt = @provisioner.instance_variable_get("@apt")
9
+ @ruby_roles = @provisioner.instance_variable_get("@ruby_roles")
10
+ @chef_zero = @provisioner.instance_variable_get("@chef_zero")
11
+ end
12
+
13
+ it "installs chef zero" do
14
+ expect(@chef_zero).to receive(:install_command)
15
+ expect(@ruby_roles).not_to receive(:install_command)
16
+ expect(@apt).not_to receive(:install_command)
17
+
18
+ @provisioner.install_command
19
+ end
20
+
21
+ it "inits chef zero" do
22
+ expect(@chef_zero).to receive(:init_command)
23
+ expect(@ruby_roles).not_to receive(:init_command)
24
+ expect(@apt).not_to receive(:init_command)
25
+
26
+ @provisioner.init_command
27
+ end
28
+
29
+ it "creates sandbox for chef zero and converts ruby roles" do
30
+ expect(@chef_zero).to receive(:create_sandbox).and_return("path").ordered
31
+ expect(@ruby_roles).to receive(:roles_path=).with("path/roles").ordered
32
+ expect(@ruby_roles).to receive(:create_sandbox).ordered
33
+ expect(@apt).not_to receive(:create_sandbox)
34
+
35
+ @provisioner.create_sandbox
36
+ end
37
+
38
+ it "prepares chef zero" do
39
+ expect(@chef_zero).to receive(:prepare_command)
40
+ expect(@ruby_roles).not_to receive(:prepare_command)
41
+ expect(@apt).not_to receive(:prepare_command)
42
+
43
+ @provisioner.prepare_command
44
+ end
45
+
46
+ it "runs apt, ruby roles, then chef-zero, in that order" do
47
+ expect(@apt).to receive(:run_command).and_return("apt me").ordered
48
+ expect(@chef_zero).to receive(:run_command).and_return("chef me").ordered
49
+ expect(@ruby_roles).not_to receive(:run_command)
50
+
51
+ expect(@provisioner.run_command).to eq("apt me && chef me")
52
+ end
53
+
54
+ it "cleans up the chef zero sandbox" do
55
+ expect(@chef_zero).to receive(:cleanup_sandbox)
56
+ expect(@ruby_roles).not_to receive(:cleanup_sandbox)
57
+ expect(@apt).not_to receive(:cleanup_sandbox)
58
+
59
+ @provisioner.cleanup_sandbox
60
+ end
61
+
62
+ it "returns the home path of chef zero" do
63
+ expect(@chef_zero).to receive(:home_path)
64
+ expect(@ruby_roles).not_to receive(:home_path)
65
+ expect(@apt).not_to receive(:home_path)
66
+
67
+ @provisioner.home_path
68
+ end
69
+ end
@@ -0,0 +1,46 @@
1
+ require 'spec_helper'
2
+ require 'kitchen/provisioner/ruby_roles'
3
+
4
+ describe Kitchen::Provisioner::RubyRoles do
5
+ before do
6
+ # Sorry, can't use FakeFS because chef completely bypasses the
7
+ # entire File api when reading the file.
8
+ FileUtils.mkdir_p("tmp/roles")
9
+ end
10
+
11
+ after do
12
+ FileUtils.rm_r("tmp/roles")
13
+ end
14
+
15
+ it "converts any existing Ruby chef roles into JSON" do
16
+ File.open("tmp/roles/role1.rb", "w+") do |role|
17
+ role.puts <<EOC
18
+ name "role1"
19
+ description "I define how things work"
20
+
21
+ run_list("recipe[test::role1]")
22
+ EOC
23
+ end
24
+
25
+ File.open("tmp/roles/role2.rb", "w+") do |role|
26
+ role.puts <<EOC
27
+ name "role2"
28
+ description "I define how other things work"
29
+
30
+ run_list("recipe[test::role2]")
31
+ EOC
32
+ end
33
+
34
+ File.open("tmp/roles/role3.js", "w+") do |role|
35
+ role.puts "This is role thingy 3"
36
+ end
37
+
38
+ provisioner = Kitchen::Provisioner::RubyRoles.new(BogusInstance.new, nil)
39
+ provisioner.roles_path = "tmp/roles"
40
+ provisioner.create_sandbox
41
+
42
+ expect(File.exists?("tmp/roles/role1.json")).to be_true
43
+ expect(File.exists?("tmp/roles/role2.json")).to be_true
44
+ expect(File.exists?("tmp/roles/role3.json")).to be_false
45
+ end
46
+ end
@@ -0,0 +1,9 @@
1
+ RSpec.configure do |config|
2
+ config.order = "random"
3
+ config.expect_with(:rspec) {|c| c.syntax = :expect }
4
+ end
5
+
6
+ class BogusInstance
7
+ def logger
8
+ end
9
+ end
@@ -0,0 +1,18 @@
1
+ Gem::Specification.new do |gem|
2
+ gem.name = "test-kitchen-provisioners"
3
+ gem.version = "0.1"
4
+ gem.license = "MIT"
5
+ gem.authors = ["Collective Idea"]
6
+ gem.email = ["info@collectiveidea.com"]
7
+
8
+ gem.summary = "A collection of helpful provisioniners for working with Test Kitchen"
9
+ gem.description = gem.summary
10
+ gem.homepage = "https://github.com/collectiveidea/test-kitchen-provisioners"
11
+
12
+ gem.files = `git ls-files`.split($/)
13
+ gem.test_files = gem.files.grep(%r{spec})
14
+ gem.require_paths = ["lib"]
15
+
16
+ gem.add_dependency "chef", "~> 11.6.0"
17
+ gem.add_dependency "test-kitchen", "1.0.0.beta.2"
18
+ end
metadata ADDED
@@ -0,0 +1,98 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: test-kitchen-provisioners
3
+ version: !ruby/object:Gem::Version
4
+ version: '0.1'
5
+ prerelease:
6
+ platform: ruby
7
+ authors:
8
+ - Collective Idea
9
+ autorequire:
10
+ bindir: bin
11
+ cert_chain: []
12
+ date: 2013-08-03 00:00:00.000000000 Z
13
+ dependencies:
14
+ - !ruby/object:Gem::Dependency
15
+ name: chef
16
+ requirement: !ruby/object:Gem::Requirement
17
+ none: false
18
+ requirements:
19
+ - - ~>
20
+ - !ruby/object:Gem::Version
21
+ version: 11.6.0
22
+ type: :runtime
23
+ prerelease: false
24
+ version_requirements: !ruby/object:Gem::Requirement
25
+ none: false
26
+ requirements:
27
+ - - ~>
28
+ - !ruby/object:Gem::Version
29
+ version: 11.6.0
30
+ - !ruby/object:Gem::Dependency
31
+ name: test-kitchen
32
+ requirement: !ruby/object:Gem::Requirement
33
+ none: false
34
+ requirements:
35
+ - - '='
36
+ - !ruby/object:Gem::Version
37
+ version: 1.0.0.beta.2
38
+ type: :runtime
39
+ prerelease: false
40
+ version_requirements: !ruby/object:Gem::Requirement
41
+ none: false
42
+ requirements:
43
+ - - '='
44
+ - !ruby/object:Gem::Version
45
+ version: 1.0.0.beta.2
46
+ description: A collection of helpful provisioniners for working with Test Kitchen
47
+ email:
48
+ - info@collectiveidea.com
49
+ executables: []
50
+ extensions: []
51
+ extra_rdoc_files: []
52
+ files:
53
+ - .gitignore
54
+ - .travis.yml
55
+ - Gemfile
56
+ - LICENSE.txt
57
+ - README.md
58
+ - Rakefile
59
+ - lib/kitchen/provisioner/apt.rb
60
+ - lib/kitchen/provisioner/chef_apt_ruby_roles_chef_zero.rb
61
+ - lib/kitchen/provisioner/ruby_roles.rb
62
+ - spec/kitchen/provisioner/apt_spec.rb
63
+ - spec/kitchen/provisioner/chef_apt_ruby_roles_chef_zero_spec.rb
64
+ - spec/kitchen/provisioner/ruby_roles_spec.rb
65
+ - spec/spec_helper.rb
66
+ - test-kitchen-provisioners.gemspec
67
+ homepage: https://github.com/collectiveidea/test-kitchen-provisioners
68
+ licenses:
69
+ - MIT
70
+ post_install_message:
71
+ rdoc_options: []
72
+ require_paths:
73
+ - lib
74
+ required_ruby_version: !ruby/object:Gem::Requirement
75
+ none: false
76
+ requirements:
77
+ - - ! '>='
78
+ - !ruby/object:Gem::Version
79
+ version: '0'
80
+ required_rubygems_version: !ruby/object:Gem::Requirement
81
+ none: false
82
+ requirements:
83
+ - - ! '>='
84
+ - !ruby/object:Gem::Version
85
+ version: '0'
86
+ requirements: []
87
+ rubyforge_project:
88
+ rubygems_version: 1.8.25
89
+ signing_key:
90
+ specification_version: 3
91
+ summary: A collection of helpful provisioniners for working with Test Kitchen
92
+ test_files:
93
+ - spec/kitchen/provisioner/apt_spec.rb
94
+ - spec/kitchen/provisioner/chef_apt_ruby_roles_chef_zero_spec.rb
95
+ - spec/kitchen/provisioner/ruby_roles_spec.rb
96
+ - spec/spec_helper.rb
97
+ - test-kitchen-provisioners.gemspec
98
+ has_rdoc: