test-kitchen 1.6.0 → 1.7.0

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.
Files changed (180) hide show
  1. checksums.yaml +4 -4
  2. data/.cane +8 -7
  3. data/.github/ISSUE_TEMPLATE.md +56 -0
  4. data/.gitignore +28 -27
  5. data/.kitchen.ci.yml +23 -0
  6. data/.kitchen.proxy.yml +27 -0
  7. data/.rubocop.yml +3 -3
  8. data/.travis.yml +70 -53
  9. data/.yardopts +3 -3
  10. data/Berksfile +3 -0
  11. data/CHANGELOG.md +1083 -1051
  12. data/CONTRIBUTING.md +14 -14
  13. data/Gemfile +19 -14
  14. data/Gemfile.proxy_tests +4 -5
  15. data/Guardfile +42 -42
  16. data/LICENSE +15 -15
  17. data/MAINTAINERS.md +23 -24
  18. data/README.md +135 -135
  19. data/Rakefile +61 -76
  20. data/appveyor.yml +44 -34
  21. data/features/kitchen_action_commands.feature +164 -164
  22. data/features/kitchen_command.feature +16 -16
  23. data/features/kitchen_console_command.feature +34 -34
  24. data/features/kitchen_defaults.feature +38 -38
  25. data/features/kitchen_diagnose_command.feature +96 -96
  26. data/features/kitchen_driver_create_command.feature +64 -64
  27. data/features/kitchen_driver_discover_command.feature +25 -25
  28. data/features/kitchen_help_command.feature +16 -16
  29. data/features/kitchen_init_command.feature +274 -274
  30. data/features/kitchen_list_command.feature +104 -104
  31. data/features/kitchen_login_command.feature +62 -62
  32. data/features/kitchen_sink_command.feature +30 -30
  33. data/features/kitchen_test_command.feature +88 -88
  34. data/features/step_definitions/gem_steps.rb +36 -36
  35. data/features/step_definitions/git_steps.rb +5 -5
  36. data/features/step_definitions/output_steps.rb +5 -5
  37. data/features/support/env.rb +75 -75
  38. data/lib/kitchen.rb +150 -150
  39. data/lib/kitchen/base64_stream.rb +55 -55
  40. data/lib/kitchen/cli.rb +419 -419
  41. data/lib/kitchen/collection.rb +55 -55
  42. data/lib/kitchen/color.rb +65 -65
  43. data/lib/kitchen/command.rb +185 -185
  44. data/lib/kitchen/command/action.rb +45 -45
  45. data/lib/kitchen/command/console.rb +58 -58
  46. data/lib/kitchen/command/diagnose.rb +92 -92
  47. data/lib/kitchen/command/driver_discover.rb +105 -105
  48. data/lib/kitchen/command/exec.rb +41 -41
  49. data/lib/kitchen/command/list.rb +119 -119
  50. data/lib/kitchen/command/login.rb +43 -43
  51. data/lib/kitchen/command/sink.rb +54 -54
  52. data/lib/kitchen/command/test.rb +51 -51
  53. data/lib/kitchen/config.rb +322 -322
  54. data/lib/kitchen/configurable.rb +529 -529
  55. data/lib/kitchen/data_munger.rb +959 -960
  56. data/lib/kitchen/diagnostic.rb +141 -141
  57. data/lib/kitchen/driver.rb +56 -56
  58. data/lib/kitchen/driver/base.rb +134 -134
  59. data/lib/kitchen/driver/dummy.rb +108 -108
  60. data/lib/kitchen/driver/proxy.rb +72 -72
  61. data/lib/kitchen/driver/ssh_base.rb +357 -357
  62. data/lib/kitchen/errors.rb +229 -229
  63. data/lib/kitchen/generator/driver_create.rb +177 -177
  64. data/lib/kitchen/generator/init.rb +296 -296
  65. data/lib/kitchen/instance.rb +662 -662
  66. data/lib/kitchen/lazy_hash.rb +142 -142
  67. data/lib/kitchen/loader/yaml.rb +349 -349
  68. data/lib/kitchen/logger.rb +423 -423
  69. data/lib/kitchen/logging.rb +56 -56
  70. data/lib/kitchen/login_command.rb +52 -52
  71. data/lib/kitchen/metadata_chopper.rb +52 -52
  72. data/lib/kitchen/platform.rb +67 -67
  73. data/lib/kitchen/provisioner.rb +54 -54
  74. data/lib/kitchen/provisioner/base.rb +236 -236
  75. data/lib/kitchen/provisioner/chef/berkshelf.rb +114 -114
  76. data/lib/kitchen/provisioner/chef/common_sandbox.rb +322 -322
  77. data/lib/kitchen/provisioner/chef/librarian.rb +112 -112
  78. data/lib/kitchen/provisioner/chef_apply.rb +124 -125
  79. data/lib/kitchen/provisioner/chef_base.rb +341 -294
  80. data/lib/kitchen/provisioner/chef_solo.rb +88 -89
  81. data/lib/kitchen/provisioner/chef_zero.rb +245 -245
  82. data/lib/kitchen/provisioner/dummy.rb +79 -79
  83. data/lib/kitchen/provisioner/shell.rb +138 -138
  84. data/lib/kitchen/rake_tasks.rb +63 -63
  85. data/lib/kitchen/shell_out.rb +93 -93
  86. data/lib/kitchen/ssh.rb +276 -276
  87. data/lib/kitchen/state_file.rb +120 -120
  88. data/lib/kitchen/suite.rb +51 -51
  89. data/lib/kitchen/thor_tasks.rb +66 -66
  90. data/lib/kitchen/transport.rb +54 -54
  91. data/lib/kitchen/transport/base.rb +176 -176
  92. data/lib/kitchen/transport/dummy.rb +79 -79
  93. data/lib/kitchen/transport/ssh.rb +364 -364
  94. data/lib/kitchen/transport/winrm.rb +486 -486
  95. data/lib/kitchen/util.rb +147 -147
  96. data/lib/kitchen/verifier.rb +55 -55
  97. data/lib/kitchen/verifier/base.rb +235 -235
  98. data/lib/kitchen/verifier/busser.rb +277 -277
  99. data/lib/kitchen/verifier/dummy.rb +79 -79
  100. data/lib/kitchen/verifier/shell.rb +101 -101
  101. data/lib/kitchen/version.rb +21 -21
  102. data/lib/vendor/hash_recursive_merge.rb +82 -82
  103. data/spec/kitchen/base64_stream_spec.rb +77 -77
  104. data/spec/kitchen/cli_spec.rb +56 -56
  105. data/spec/kitchen/collection_spec.rb +80 -80
  106. data/spec/kitchen/color_spec.rb +54 -54
  107. data/spec/kitchen/config_spec.rb +408 -408
  108. data/spec/kitchen/configurable_spec.rb +1095 -1062
  109. data/spec/kitchen/data_munger_spec.rb +2694 -2383
  110. data/spec/kitchen/diagnostic_spec.rb +129 -129
  111. data/spec/kitchen/driver/base_spec.rb +121 -121
  112. data/spec/kitchen/driver/dummy_spec.rb +199 -199
  113. data/spec/kitchen/driver/proxy_spec.rb +138 -138
  114. data/spec/kitchen/driver/ssh_base_spec.rb +1115 -1115
  115. data/spec/kitchen/driver_spec.rb +112 -112
  116. data/spec/kitchen/errors_spec.rb +309 -309
  117. data/spec/kitchen/instance_spec.rb +1419 -1419
  118. data/spec/kitchen/lazy_hash_spec.rb +117 -117
  119. data/spec/kitchen/loader/yaml_spec.rb +774 -774
  120. data/spec/kitchen/logger_spec.rb +429 -429
  121. data/spec/kitchen/logging_spec.rb +59 -59
  122. data/spec/kitchen/login_command_spec.rb +68 -68
  123. data/spec/kitchen/metadata_chopper_spec.rb +82 -82
  124. data/spec/kitchen/platform_spec.rb +89 -89
  125. data/spec/kitchen/provisioner/base_spec.rb +386 -386
  126. data/spec/kitchen/provisioner/chef_apply_spec.rb +136 -136
  127. data/spec/kitchen/provisioner/chef_base_spec.rb +1161 -1067
  128. data/spec/kitchen/provisioner/chef_solo_spec.rb +557 -557
  129. data/spec/kitchen/provisioner/chef_zero_spec.rb +1001 -1001
  130. data/spec/kitchen/provisioner/dummy_spec.rb +99 -99
  131. data/spec/kitchen/provisioner/shell_spec.rb +566 -566
  132. data/spec/kitchen/provisioner_spec.rb +107 -107
  133. data/spec/kitchen/shell_out_spec.rb +150 -150
  134. data/spec/kitchen/ssh_spec.rb +693 -693
  135. data/spec/kitchen/state_file_spec.rb +129 -129
  136. data/spec/kitchen/suite_spec.rb +62 -62
  137. data/spec/kitchen/transport/base_spec.rb +89 -89
  138. data/spec/kitchen/transport/ssh_spec.rb +1255 -1255
  139. data/spec/kitchen/transport/winrm_spec.rb +1143 -1143
  140. data/spec/kitchen/transport_spec.rb +112 -112
  141. data/spec/kitchen/util_spec.rb +165 -165
  142. data/spec/kitchen/verifier/base_spec.rb +362 -362
  143. data/spec/kitchen/verifier/busser_spec.rb +610 -610
  144. data/spec/kitchen/verifier/dummy_spec.rb +99 -99
  145. data/spec/kitchen/verifier/shell_spec.rb +160 -158
  146. data/spec/kitchen/verifier_spec.rb +120 -120
  147. data/spec/kitchen_spec.rb +114 -114
  148. data/spec/spec_helper.rb +85 -85
  149. data/spec/support/powershell_max_size_spec.rb +40 -40
  150. data/support/busser_install_command.ps1 +14 -14
  151. data/support/busser_install_command.sh +14 -14
  152. data/support/chef-client-zero.rb +77 -77
  153. data/support/chef_base_init_command.ps1 +18 -18
  154. data/support/chef_base_init_command.sh +2 -2
  155. data/support/chef_base_install_command.ps1 +85 -85
  156. data/support/chef_base_install_command.sh +229 -229
  157. data/support/chef_zero_prepare_command_legacy.ps1 +9 -9
  158. data/support/chef_zero_prepare_command_legacy.sh +10 -10
  159. data/support/download_helpers.sh +109 -109
  160. data/support/dummy-validation.pem +27 -27
  161. data/templates/driver/CHANGELOG.md.erb +3 -3
  162. data/templates/driver/Gemfile.erb +3 -3
  163. data/templates/driver/README.md.erb +64 -64
  164. data/templates/driver/Rakefile.erb +21 -21
  165. data/templates/driver/driver.rb.erb +23 -23
  166. data/templates/driver/gemspec.erb +29 -29
  167. data/templates/driver/gitignore.erb +17 -17
  168. data/templates/driver/license_apachev2.erb +15 -15
  169. data/templates/driver/license_lgplv3.erb +16 -16
  170. data/templates/driver/license_mit.erb +22 -22
  171. data/templates/driver/license_reserved.erb +5 -5
  172. data/templates/driver/tailor.erb +4 -4
  173. data/templates/driver/travis.yml.erb +11 -11
  174. data/templates/driver/version.rb.erb +12 -12
  175. data/templates/init/chefignore.erb +1 -1
  176. data/templates/init/kitchen.yml.erb +18 -18
  177. data/test-kitchen.gemspec +62 -62
  178. data/test/integration/default/default_spec.rb +3 -0
  179. data/testing_windows.md +37 -37
  180. metadata +23 -11
@@ -1,54 +1,54 @@
1
- # -*- encoding: utf-8 -*-
2
- #
3
- # Author:: Fletcher Nichol (<fnichol@nichol.ca>)
4
- #
5
- # Copyright (C) 2013, Fletcher Nichol
6
- #
7
- # Licensed under the Apache License, Version 2.0 (the "License");
8
- # you may not use this file except in compliance with the License.
9
- # You may obtain a copy of the License at
10
- #
11
- # http://www.apache.org/licenses/LICENSE-2.0
12
- #
13
- # Unless required by applicable law or agreed to in writing, software
14
- # distributed under the License is distributed on an "AS IS" BASIS,
15
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
- # See the License for the specific language governing permissions and
17
- # limitations under the License.
18
-
19
- require "thor/util"
20
-
21
- module Kitchen
22
-
23
- # A provisioner is responsible for generating the commands necessary to
24
- # install set up and use a configuration management tool such as Chef and
25
- # Puppet.
26
- #
27
- # @author Fletcher Nichol <fnichol@nichol.ca>
28
- module Provisioner
29
-
30
- # Default provisioner to use
31
- DEFAULT_PLUGIN = "chef_solo".freeze
32
-
33
- # Returns an instance of a provisioner given a plugin type string.
34
- #
35
- # @param plugin [String] a provisioner plugin type, to be constantized
36
- # @param config [Hash] a configuration hash to initialize the provisioner
37
- # @return [Provisioner::Base] a provisioner instance
38
- # @raise [ClientError] if a provisioner instance could not be created
39
- def self.for_plugin(plugin, config)
40
- first_load = require("kitchen/provisioner/#{plugin}")
41
-
42
- str_const = Thor::Util.camel_case(plugin)
43
- klass = const_get(str_const)
44
- object = klass.new(config)
45
- object.verify_dependencies if first_load
46
- object
47
- rescue LoadError, NameError
48
- raise ClientError,
49
- "Could not load the '#{plugin}' provisioner from the load path." \
50
- " Please ensure that your provisioner is installed as a gem or" \
51
- " included in your Gemfile if using Bundler."
52
- end
53
- end
54
- end
1
+ # -*- encoding: utf-8 -*-
2
+ #
3
+ # Author:: Fletcher Nichol (<fnichol@nichol.ca>)
4
+ #
5
+ # Copyright (C) 2013, Fletcher Nichol
6
+ #
7
+ # Licensed under the Apache License, Version 2.0 (the "License");
8
+ # you may not use this file except in compliance with the License.
9
+ # You may obtain a copy of the License at
10
+ #
11
+ # http://www.apache.org/licenses/LICENSE-2.0
12
+ #
13
+ # Unless required by applicable law or agreed to in writing, software
14
+ # distributed under the License is distributed on an "AS IS" BASIS,
15
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ # See the License for the specific language governing permissions and
17
+ # limitations under the License.
18
+
19
+ require "thor/util"
20
+
21
+ module Kitchen
22
+
23
+ # A provisioner is responsible for generating the commands necessary to
24
+ # install set up and use a configuration management tool such as Chef and
25
+ # Puppet.
26
+ #
27
+ # @author Fletcher Nichol <fnichol@nichol.ca>
28
+ module Provisioner
29
+
30
+ # Default provisioner to use
31
+ DEFAULT_PLUGIN = "chef_solo".freeze
32
+
33
+ # Returns an instance of a provisioner given a plugin type string.
34
+ #
35
+ # @param plugin [String] a provisioner plugin type, to be constantized
36
+ # @param config [Hash] a configuration hash to initialize the provisioner
37
+ # @return [Provisioner::Base] a provisioner instance
38
+ # @raise [ClientError] if a provisioner instance could not be created
39
+ def self.for_plugin(plugin, config)
40
+ first_load = require("kitchen/provisioner/#{plugin}")
41
+
42
+ str_const = Thor::Util.camel_case(plugin)
43
+ klass = const_get(str_const)
44
+ object = klass.new(config)
45
+ object.verify_dependencies if first_load
46
+ object
47
+ rescue LoadError, NameError
48
+ raise ClientError,
49
+ "Could not load the '#{plugin}' provisioner from the load path." \
50
+ " Please ensure that your provisioner is installed as a gem or" \
51
+ " included in your Gemfile if using Bundler."
52
+ end
53
+ end
54
+ end
@@ -1,236 +1,236 @@
1
- # -*- encoding: utf-8 -*-
2
- #
3
- # Author:: Fletcher Nichol (<fnichol@nichol.ca>)
4
- #
5
- # Copyright (C) 2013, Fletcher Nichol
6
- #
7
- # Licensed under the Apache License, Version 2.0 (the "License");
8
- # you may not use this file except in compliance with the License.
9
- # You may obtain a copy of the License at
10
- #
11
- # http://www.apache.org/licenses/LICENSE-2.0
12
- #
13
- # Unless required by applicable law or agreed to in writing, software
14
- # distributed under the License is distributed on an "AS IS" BASIS,
15
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
- # See the License for the specific language governing permissions and
17
- # limitations under the License.
18
-
19
- module Kitchen
20
-
21
- module Provisioner
22
-
23
- # Base class for a provisioner.
24
- #
25
- # @author Fletcher Nichol <fnichol@nichol.ca>
26
- class Base
27
-
28
- include Configurable
29
- include Logging
30
-
31
- default_config :http_proxy, nil
32
- default_config :https_proxy, nil
33
- default_config :ftp_proxy, nil
34
-
35
- default_config :root_path do |provisioner|
36
- provisioner.windows_os? ? "$env:TEMP\\kitchen" : "/tmp/kitchen"
37
- end
38
-
39
- default_config :sudo do |provisioner|
40
- provisioner.windows_os? ? nil : true
41
- end
42
-
43
- default_config :sudo_command do |provisioner|
44
- provisioner.windows_os? ? nil : "sudo -E"
45
- end
46
-
47
- default_config :command_prefix, nil
48
-
49
- expand_path_for :test_base_path
50
-
51
- # Constructs a new provisioner by providing a configuration hash.
52
- #
53
- # @param config [Hash] initial provided configuration
54
- def initialize(config = {})
55
- init_config(config)
56
- end
57
-
58
- # Runs the provisioner on the instance.
59
- #
60
- # @param state [Hash] mutable instance state
61
- # @raise [ActionFailed] if the action could not be completed
62
- def call(state)
63
- create_sandbox
64
- sandbox_dirs = Dir.glob(File.join(sandbox_path, "*"))
65
-
66
- instance.transport.connection(state) do |conn|
67
- conn.execute(install_command)
68
- conn.execute(init_command)
69
- info("Transferring files to #{instance.to_str}")
70
- conn.upload(sandbox_dirs, config[:root_path])
71
- debug("Transfer complete")
72
- conn.execute(prepare_command)
73
- conn.execute(run_command)
74
- end
75
- rescue Kitchen::Transport::TransportFailed => ex
76
- raise ActionFailed, ex.message
77
- ensure
78
- cleanup_sandbox
79
- end
80
-
81
- # Generates a command string which will install and configure the
82
- # provisioner software on an instance. If no work is required, then `nil`
83
- # will be returned.
84
- #
85
- # @return [String] a command string
86
- def install_command
87
- end
88
-
89
- # Generates a command string which will perform any data initialization
90
- # or configuration required after the provisioner software is installed
91
- # but before the sandbox has been transferred to the instance. If no work
92
- # is required, then `nil` will be returned.
93
- #
94
- # @return [String] a command string
95
- def init_command
96
- end
97
-
98
- # Generates a command string which will perform any commands or
99
- # configuration required just before the main provisioner run command but
100
- # after the sandbox has been transferred to the instance. If no work is
101
- # required, then `nil` will be returned.
102
- #
103
- # @return [String] a command string
104
- def prepare_command
105
- end
106
-
107
- # Generates a command string which will invoke the main provisioner
108
- # command on the prepared instance. If no work is required, then `nil`
109
- # will be returned.
110
- #
111
- # @return [String] a command string
112
- def run_command
113
- end
114
-
115
- # Creates a temporary directory on the local workstation into which
116
- # provisioner related files and directories can be copied or created. The
117
- # contents of this directory will be copied over to the instance before
118
- # invoking the provisioner's run command. After this method completes, it
119
- # is expected that the contents of the sandbox is complete and ready for
120
- # copy to the remote instance.
121
- #
122
- # **Note:** any subclasses would be well advised to call super first when
123
- # overriding this method, for example:
124
- #
125
- # @example overriding `#create_sandbox`
126
- #
127
- # class MyProvisioner < Kitchen::Provisioner::Base
128
- # def create_sandbox
129
- # super
130
- # # any further file copies, preparations, etc.
131
- # end
132
- # end
133
- def create_sandbox
134
- @sandbox_path = Dir.mktmpdir("#{instance.name}-sandbox-")
135
- File.chmod(0755, sandbox_path)
136
- info("Preparing files for transfer")
137
- debug("Creating local sandbox in #{sandbox_path}")
138
- end
139
-
140
- # Returns the absolute path to the sandbox directory or raises an
141
- # exception if `#create_sandbox` has not yet been called.
142
- #
143
- # @return [String] the absolute path to the sandbox directory
144
- # @raise [ClientError] if the sandbox directory has no yet been created
145
- # by calling `#create_sandbox`
146
- def sandbox_path
147
- @sandbox_path || (raise ClientError, "Sandbox directory has not yet " \
148
- "been created. Please run #{self.class}#create_sandox before " \
149
- "trying to access the path.")
150
- end
151
-
152
- # Deletes the sandbox path. Without calling this method, the sandbox path
153
- # will persist after the process terminates. In other words, cleanup is
154
- # explicit. This method is safe to call multiple times.
155
- def cleanup_sandbox
156
- return if sandbox_path.nil?
157
-
158
- debug("Cleaning up local sandbox in #{sandbox_path}")
159
- FileUtils.rmtree(sandbox_path)
160
- end
161
-
162
- # Sets the API version for this provisioner. If the provisioner does not
163
- # set this value, then `nil` will be used and reported.
164
- #
165
- # Sets the API version for this provisioner
166
- #
167
- # @example setting an API version
168
- #
169
- # module Kitchen
170
- # module Provisioner
171
- # class NewProvisioner < Kitchen::Provisioner::Base
172
- #
173
- # kitchen_provisioner_api_version 2
174
- #
175
- # end
176
- # end
177
- # end
178
- #
179
- # @param version [Integer,String] a version number
180
- #
181
- def self.kitchen_provisioner_api_version(version)
182
- @api_version = version
183
- end
184
-
185
- private
186
-
187
- # Builds a complete command given a variables String preamble and a file
188
- # containing shell code.
189
- #
190
- # @param vars [String] shell variables, as a String
191
- # @param file [String] file basename (without extension) containing
192
- # shell code
193
- # @return [String] command
194
- # @api private
195
- def shell_code_from_file(vars, file)
196
- src_file = File.join(
197
- File.dirname(__FILE__),
198
- %w[.. .. .. support],
199
- file + (powershell_shell? ? ".ps1" : ".sh")
200
- )
201
-
202
- wrap_shell_code([vars, "", IO.read(src_file)].join("\n"))
203
- end
204
-
205
- # Conditionally prefixes a command with a sudo command.
206
- #
207
- # @param command [String] command to be prefixed
208
- # @return [String] the command, conditionally prefixed with sudo
209
- # @api private
210
- def sudo(script)
211
- "#{sudo_command} #{script}".lstrip
212
- end
213
-
214
- # Returns the sudo command to use or empty string if sudo is not configured
215
- #
216
- # @return [String] the sudo command if sudo config is true
217
- # @api private
218
- def sudo_command
219
- config[:sudo] ? config[:sudo_command].to_s : ""
220
- end
221
-
222
- # Conditionally prefixes a command with a command prefix.
223
- # This should generally be done after a command has been
224
- # conditionally prefixed by #sudo as certain platforms, such as
225
- # Cisco Nexus, require all commands to be run with a prefix to
226
- # obtain outbound network access.
227
- #
228
- # @param command [String] command to be prefixed
229
- # @return [String] the command, conditionally prefixed with the configured prefix
230
- # @api private
231
- def prefix_command(script)
232
- config[:command_prefix] ? "#{config[:command_prefix]} #{script}" : script
233
- end
234
- end
235
- end
236
- end
1
+ # -*- encoding: utf-8 -*-
2
+ #
3
+ # Author:: Fletcher Nichol (<fnichol@nichol.ca>)
4
+ #
5
+ # Copyright (C) 2013, Fletcher Nichol
6
+ #
7
+ # Licensed under the Apache License, Version 2.0 (the "License");
8
+ # you may not use this file except in compliance with the License.
9
+ # You may obtain a copy of the License at
10
+ #
11
+ # http://www.apache.org/licenses/LICENSE-2.0
12
+ #
13
+ # Unless required by applicable law or agreed to in writing, software
14
+ # distributed under the License is distributed on an "AS IS" BASIS,
15
+ # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ # See the License for the specific language governing permissions and
17
+ # limitations under the License.
18
+
19
+ module Kitchen
20
+
21
+ module Provisioner
22
+
23
+ # Base class for a provisioner.
24
+ #
25
+ # @author Fletcher Nichol <fnichol@nichol.ca>
26
+ class Base
27
+
28
+ include Configurable
29
+ include Logging
30
+
31
+ default_config :http_proxy, nil
32
+ default_config :https_proxy, nil
33
+ default_config :ftp_proxy, nil
34
+
35
+ default_config :root_path do |provisioner|
36
+ provisioner.windows_os? ? "$env:TEMP\\kitchen" : "/tmp/kitchen"
37
+ end
38
+
39
+ default_config :sudo do |provisioner|
40
+ provisioner.windows_os? ? nil : true
41
+ end
42
+
43
+ default_config :sudo_command do |provisioner|
44
+ provisioner.windows_os? ? nil : "sudo -E"
45
+ end
46
+
47
+ default_config :command_prefix, nil
48
+
49
+ expand_path_for :test_base_path
50
+
51
+ # Constructs a new provisioner by providing a configuration hash.
52
+ #
53
+ # @param config [Hash] initial provided configuration
54
+ def initialize(config = {})
55
+ init_config(config)
56
+ end
57
+
58
+ # Runs the provisioner on the instance.
59
+ #
60
+ # @param state [Hash] mutable instance state
61
+ # @raise [ActionFailed] if the action could not be completed
62
+ def call(state)
63
+ create_sandbox
64
+ sandbox_dirs = Dir.glob(File.join(sandbox_path, "*"))
65
+
66
+ instance.transport.connection(state) do |conn|
67
+ conn.execute(install_command)
68
+ conn.execute(init_command)
69
+ info("Transferring files to #{instance.to_str}")
70
+ conn.upload(sandbox_dirs, config[:root_path])
71
+ debug("Transfer complete")
72
+ conn.execute(prepare_command)
73
+ conn.execute(run_command)
74
+ end
75
+ rescue Kitchen::Transport::TransportFailed => ex
76
+ raise ActionFailed, ex.message
77
+ ensure
78
+ cleanup_sandbox
79
+ end
80
+
81
+ # Generates a command string which will install and configure the
82
+ # provisioner software on an instance. If no work is required, then `nil`
83
+ # will be returned.
84
+ #
85
+ # @return [String] a command string
86
+ def install_command
87
+ end
88
+
89
+ # Generates a command string which will perform any data initialization
90
+ # or configuration required after the provisioner software is installed
91
+ # but before the sandbox has been transferred to the instance. If no work
92
+ # is required, then `nil` will be returned.
93
+ #
94
+ # @return [String] a command string
95
+ def init_command
96
+ end
97
+
98
+ # Generates a command string which will perform any commands or
99
+ # configuration required just before the main provisioner run command but
100
+ # after the sandbox has been transferred to the instance. If no work is
101
+ # required, then `nil` will be returned.
102
+ #
103
+ # @return [String] a command string
104
+ def prepare_command
105
+ end
106
+
107
+ # Generates a command string which will invoke the main provisioner
108
+ # command on the prepared instance. If no work is required, then `nil`
109
+ # will be returned.
110
+ #
111
+ # @return [String] a command string
112
+ def run_command
113
+ end
114
+
115
+ # Creates a temporary directory on the local workstation into which
116
+ # provisioner related files and directories can be copied or created. The
117
+ # contents of this directory will be copied over to the instance before
118
+ # invoking the provisioner's run command. After this method completes, it
119
+ # is expected that the contents of the sandbox is complete and ready for
120
+ # copy to the remote instance.
121
+ #
122
+ # **Note:** any subclasses would be well advised to call super first when
123
+ # overriding this method, for example:
124
+ #
125
+ # @example overriding `#create_sandbox`
126
+ #
127
+ # class MyProvisioner < Kitchen::Provisioner::Base
128
+ # def create_sandbox
129
+ # super
130
+ # # any further file copies, preparations, etc.
131
+ # end
132
+ # end
133
+ def create_sandbox
134
+ @sandbox_path = Dir.mktmpdir("#{instance.name}-sandbox-")
135
+ File.chmod(0755, sandbox_path)
136
+ info("Preparing files for transfer")
137
+ debug("Creating local sandbox in #{sandbox_path}")
138
+ end
139
+
140
+ # Returns the absolute path to the sandbox directory or raises an
141
+ # exception if `#create_sandbox` has not yet been called.
142
+ #
143
+ # @return [String] the absolute path to the sandbox directory
144
+ # @raise [ClientError] if the sandbox directory has no yet been created
145
+ # by calling `#create_sandbox`
146
+ def sandbox_path
147
+ @sandbox_path || (raise ClientError, "Sandbox directory has not yet " \
148
+ "been created. Please run #{self.class}#create_sandox before " \
149
+ "trying to access the path.")
150
+ end
151
+
152
+ # Deletes the sandbox path. Without calling this method, the sandbox path
153
+ # will persist after the process terminates. In other words, cleanup is
154
+ # explicit. This method is safe to call multiple times.
155
+ def cleanup_sandbox
156
+ return if sandbox_path.nil?
157
+
158
+ debug("Cleaning up local sandbox in #{sandbox_path}")
159
+ FileUtils.rmtree(sandbox_path)
160
+ end
161
+
162
+ # Sets the API version for this provisioner. If the provisioner does not
163
+ # set this value, then `nil` will be used and reported.
164
+ #
165
+ # Sets the API version for this provisioner
166
+ #
167
+ # @example setting an API version
168
+ #
169
+ # module Kitchen
170
+ # module Provisioner
171
+ # class NewProvisioner < Kitchen::Provisioner::Base
172
+ #
173
+ # kitchen_provisioner_api_version 2
174
+ #
175
+ # end
176
+ # end
177
+ # end
178
+ #
179
+ # @param version [Integer,String] a version number
180
+ #
181
+ def self.kitchen_provisioner_api_version(version)
182
+ @api_version = version
183
+ end
184
+
185
+ private
186
+
187
+ # Builds a complete command given a variables String preamble and a file
188
+ # containing shell code.
189
+ #
190
+ # @param vars [String] shell variables, as a String
191
+ # @param file [String] file basename (without extension) containing
192
+ # shell code
193
+ # @return [String] command
194
+ # @api private
195
+ def shell_code_from_file(vars, file)
196
+ src_file = File.join(
197
+ File.dirname(__FILE__),
198
+ %w[.. .. .. support],
199
+ file + (powershell_shell? ? ".ps1" : ".sh")
200
+ )
201
+
202
+ wrap_shell_code([vars, "", IO.read(src_file)].join("\n"))
203
+ end
204
+
205
+ # Conditionally prefixes a command with a sudo command.
206
+ #
207
+ # @param command [String] command to be prefixed
208
+ # @return [String] the command, conditionally prefixed with sudo
209
+ # @api private
210
+ def sudo(script)
211
+ "#{sudo_command} #{script}".lstrip
212
+ end
213
+
214
+ # Returns the sudo command to use or empty string if sudo is not configured
215
+ #
216
+ # @return [String] the sudo command if sudo config is true
217
+ # @api private
218
+ def sudo_command
219
+ config[:sudo] ? config[:sudo_command].to_s : ""
220
+ end
221
+
222
+ # Conditionally prefixes a command with a command prefix.
223
+ # This should generally be done after a command has been
224
+ # conditionally prefixed by #sudo as certain platforms, such as
225
+ # Cisco Nexus, require all commands to be run with a prefix to
226
+ # obtain outbound network access.
227
+ #
228
+ # @param command [String] command to be prefixed
229
+ # @return [String] the command, conditionally prefixed with the configured prefix
230
+ # @api private
231
+ def prefix_command(script)
232
+ config[:command_prefix] ? "#{config[:command_prefix]} #{script}" : script
233
+ end
234
+ end
235
+ end
236
+ end