test-kitchen 1.7.0 → 1.7.1.dev

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 (181) hide show
  1. checksums.yaml +4 -4
  2. data/.cane +8 -8
  3. data/.gitattributes +3 -0
  4. data/.github/ISSUE_TEMPLATE.md +55 -55
  5. data/.gitignore +28 -28
  6. data/.kitchen.ci.yml +23 -23
  7. data/.kitchen.proxy.yml +27 -27
  8. data/.rubocop.yml +3 -3
  9. data/.travis.yml +70 -70
  10. data/.yardopts +3 -3
  11. data/Berksfile +3 -3
  12. data/CHANGELOG.md +1090 -1083
  13. data/CONTRIBUTING.md +14 -14
  14. data/Gemfile +19 -19
  15. data/Gemfile.proxy_tests +4 -4
  16. data/Guardfile +42 -42
  17. data/LICENSE +15 -15
  18. data/MAINTAINERS.md +23 -23
  19. data/README.md +135 -135
  20. data/Rakefile +61 -61
  21. data/appveyor.yml +44 -44
  22. data/features/kitchen_action_commands.feature +164 -164
  23. data/features/kitchen_command.feature +16 -16
  24. data/features/kitchen_console_command.feature +34 -34
  25. data/features/kitchen_defaults.feature +38 -38
  26. data/features/kitchen_diagnose_command.feature +96 -96
  27. data/features/kitchen_driver_create_command.feature +64 -64
  28. data/features/kitchen_driver_discover_command.feature +25 -25
  29. data/features/kitchen_help_command.feature +16 -16
  30. data/features/kitchen_init_command.feature +274 -274
  31. data/features/kitchen_list_command.feature +104 -104
  32. data/features/kitchen_login_command.feature +62 -62
  33. data/features/kitchen_sink_command.feature +30 -30
  34. data/features/kitchen_test_command.feature +88 -88
  35. data/features/step_definitions/gem_steps.rb +36 -36
  36. data/features/step_definitions/git_steps.rb +5 -5
  37. data/features/step_definitions/output_steps.rb +5 -5
  38. data/features/support/env.rb +75 -75
  39. data/lib/kitchen.rb +150 -150
  40. data/lib/kitchen/base64_stream.rb +55 -55
  41. data/lib/kitchen/cli.rb +419 -419
  42. data/lib/kitchen/collection.rb +55 -55
  43. data/lib/kitchen/color.rb +65 -65
  44. data/lib/kitchen/command.rb +185 -185
  45. data/lib/kitchen/command/action.rb +45 -45
  46. data/lib/kitchen/command/console.rb +58 -58
  47. data/lib/kitchen/command/diagnose.rb +92 -92
  48. data/lib/kitchen/command/driver_discover.rb +105 -105
  49. data/lib/kitchen/command/exec.rb +41 -41
  50. data/lib/kitchen/command/list.rb +119 -119
  51. data/lib/kitchen/command/login.rb +43 -43
  52. data/lib/kitchen/command/sink.rb +54 -54
  53. data/lib/kitchen/command/test.rb +51 -51
  54. data/lib/kitchen/config.rb +322 -322
  55. data/lib/kitchen/configurable.rb +529 -529
  56. data/lib/kitchen/data_munger.rb +959 -959
  57. data/lib/kitchen/diagnostic.rb +141 -141
  58. data/lib/kitchen/driver.rb +56 -56
  59. data/lib/kitchen/driver/base.rb +134 -134
  60. data/lib/kitchen/driver/dummy.rb +108 -108
  61. data/lib/kitchen/driver/proxy.rb +72 -72
  62. data/lib/kitchen/driver/ssh_base.rb +357 -357
  63. data/lib/kitchen/errors.rb +229 -229
  64. data/lib/kitchen/generator/driver_create.rb +177 -177
  65. data/lib/kitchen/generator/init.rb +296 -296
  66. data/lib/kitchen/instance.rb +662 -662
  67. data/lib/kitchen/lazy_hash.rb +142 -142
  68. data/lib/kitchen/loader/yaml.rb +349 -349
  69. data/lib/kitchen/logger.rb +423 -423
  70. data/lib/kitchen/logging.rb +56 -56
  71. data/lib/kitchen/login_command.rb +52 -52
  72. data/lib/kitchen/metadata_chopper.rb +52 -52
  73. data/lib/kitchen/platform.rb +67 -67
  74. data/lib/kitchen/provisioner.rb +54 -54
  75. data/lib/kitchen/provisioner/base.rb +236 -236
  76. data/lib/kitchen/provisioner/chef/berkshelf.rb +114 -114
  77. data/lib/kitchen/provisioner/chef/common_sandbox.rb +322 -322
  78. data/lib/kitchen/provisioner/chef/librarian.rb +112 -112
  79. data/lib/kitchen/provisioner/chef_apply.rb +124 -124
  80. data/lib/kitchen/provisioner/chef_base.rb +341 -341
  81. data/lib/kitchen/provisioner/chef_solo.rb +88 -88
  82. data/lib/kitchen/provisioner/chef_zero.rb +245 -245
  83. data/lib/kitchen/provisioner/dummy.rb +79 -79
  84. data/lib/kitchen/provisioner/shell.rb +138 -138
  85. data/lib/kitchen/rake_tasks.rb +63 -63
  86. data/lib/kitchen/shell_out.rb +93 -93
  87. data/lib/kitchen/ssh.rb +276 -276
  88. data/lib/kitchen/state_file.rb +120 -120
  89. data/lib/kitchen/suite.rb +51 -51
  90. data/lib/kitchen/thor_tasks.rb +66 -66
  91. data/lib/kitchen/transport.rb +54 -54
  92. data/lib/kitchen/transport/base.rb +176 -176
  93. data/lib/kitchen/transport/dummy.rb +79 -79
  94. data/lib/kitchen/transport/ssh.rb +364 -364
  95. data/lib/kitchen/transport/winrm.rb +486 -486
  96. data/lib/kitchen/util.rb +147 -147
  97. data/lib/kitchen/verifier.rb +55 -55
  98. data/lib/kitchen/verifier/base.rb +235 -235
  99. data/lib/kitchen/verifier/busser.rb +277 -277
  100. data/lib/kitchen/verifier/dummy.rb +79 -79
  101. data/lib/kitchen/verifier/shell.rb +101 -101
  102. data/lib/kitchen/version.rb +21 -21
  103. data/lib/vendor/hash_recursive_merge.rb +82 -82
  104. data/spec/kitchen/base64_stream_spec.rb +77 -77
  105. data/spec/kitchen/cli_spec.rb +56 -56
  106. data/spec/kitchen/collection_spec.rb +80 -80
  107. data/spec/kitchen/color_spec.rb +54 -54
  108. data/spec/kitchen/config_spec.rb +408 -408
  109. data/spec/kitchen/configurable_spec.rb +1095 -1095
  110. data/spec/kitchen/data_munger_spec.rb +2694 -2694
  111. data/spec/kitchen/diagnostic_spec.rb +129 -129
  112. data/spec/kitchen/driver/base_spec.rb +121 -121
  113. data/spec/kitchen/driver/dummy_spec.rb +199 -199
  114. data/spec/kitchen/driver/proxy_spec.rb +138 -138
  115. data/spec/kitchen/driver/ssh_base_spec.rb +1115 -1115
  116. data/spec/kitchen/driver_spec.rb +112 -112
  117. data/spec/kitchen/errors_spec.rb +309 -309
  118. data/spec/kitchen/instance_spec.rb +1419 -1419
  119. data/spec/kitchen/lazy_hash_spec.rb +117 -117
  120. data/spec/kitchen/loader/yaml_spec.rb +774 -774
  121. data/spec/kitchen/logger_spec.rb +429 -429
  122. data/spec/kitchen/logging_spec.rb +59 -59
  123. data/spec/kitchen/login_command_spec.rb +68 -68
  124. data/spec/kitchen/metadata_chopper_spec.rb +82 -82
  125. data/spec/kitchen/platform_spec.rb +89 -89
  126. data/spec/kitchen/provisioner/base_spec.rb +386 -386
  127. data/spec/kitchen/provisioner/chef_apply_spec.rb +136 -136
  128. data/spec/kitchen/provisioner/chef_base_spec.rb +1161 -1161
  129. data/spec/kitchen/provisioner/chef_solo_spec.rb +557 -557
  130. data/spec/kitchen/provisioner/chef_zero_spec.rb +1001 -1001
  131. data/spec/kitchen/provisioner/dummy_spec.rb +99 -99
  132. data/spec/kitchen/provisioner/shell_spec.rb +566 -566
  133. data/spec/kitchen/provisioner_spec.rb +107 -107
  134. data/spec/kitchen/shell_out_spec.rb +150 -150
  135. data/spec/kitchen/ssh_spec.rb +693 -693
  136. data/spec/kitchen/state_file_spec.rb +129 -129
  137. data/spec/kitchen/suite_spec.rb +62 -62
  138. data/spec/kitchen/transport/base_spec.rb +89 -89
  139. data/spec/kitchen/transport/ssh_spec.rb +1255 -1255
  140. data/spec/kitchen/transport/winrm_spec.rb +1143 -1143
  141. data/spec/kitchen/transport_spec.rb +112 -112
  142. data/spec/kitchen/util_spec.rb +165 -165
  143. data/spec/kitchen/verifier/base_spec.rb +362 -362
  144. data/spec/kitchen/verifier/busser_spec.rb +610 -610
  145. data/spec/kitchen/verifier/dummy_spec.rb +99 -99
  146. data/spec/kitchen/verifier/shell_spec.rb +160 -160
  147. data/spec/kitchen/verifier_spec.rb +120 -120
  148. data/spec/kitchen_spec.rb +114 -114
  149. data/spec/spec_helper.rb +85 -85
  150. data/spec/support/powershell_max_size_spec.rb +40 -40
  151. data/support/busser_install_command.ps1 +14 -14
  152. data/support/busser_install_command.sh +14 -14
  153. data/support/chef-client-zero.rb +77 -77
  154. data/support/chef_base_init_command.ps1 +18 -18
  155. data/support/chef_base_init_command.sh +2 -2
  156. data/support/chef_base_install_command.ps1 +85 -85
  157. data/support/chef_base_install_command.sh +229 -229
  158. data/support/chef_zero_prepare_command_legacy.ps1 +9 -9
  159. data/support/chef_zero_prepare_command_legacy.sh +10 -10
  160. data/support/download_helpers.sh +109 -109
  161. data/support/dummy-validation.pem +27 -27
  162. data/templates/driver/CHANGELOG.md.erb +3 -3
  163. data/templates/driver/Gemfile.erb +3 -3
  164. data/templates/driver/README.md.erb +64 -64
  165. data/templates/driver/Rakefile.erb +21 -21
  166. data/templates/driver/driver.rb.erb +23 -23
  167. data/templates/driver/gemspec.erb +29 -29
  168. data/templates/driver/gitignore.erb +17 -17
  169. data/templates/driver/license_apachev2.erb +15 -15
  170. data/templates/driver/license_lgplv3.erb +16 -16
  171. data/templates/driver/license_mit.erb +22 -22
  172. data/templates/driver/license_reserved.erb +5 -5
  173. data/templates/driver/tailor.erb +4 -4
  174. data/templates/driver/travis.yml.erb +11 -11
  175. data/templates/driver/version.rb.erb +12 -12
  176. data/templates/init/chefignore.erb +1 -1
  177. data/templates/init/kitchen.yml.erb +18 -18
  178. data/test-kitchen.gemspec +62 -62
  179. data/test/integration/default/default_spec.rb +3 -3
  180. data/testing_windows.md +37 -37
  181. metadata +5 -4
@@ -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