# # Author:: Fletcher Nichol () # # Copyright (C) 2015, Fletcher Nichol # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. require_relative "../errors" require_relative "../configurable" require_relative "../logging" module Kitchen module Verifier # Base class for a verifier. # # @author Fletcher Nichol class Base include Configurable include Logging default_config :http_proxy, nil default_config :https_proxy, nil default_config :ftp_proxy, nil default_config :root_path do |verifier| verifier.windows_os? ? '$env:TEMP\\verifier' : "/tmp/verifier" end default_config :sudo do |verifier| verifier.windows_os? ? nil : true end default_config :chef_omnibus_root, "/opt/chef" default_config :sudo_command do |verifier| verifier.windows_os? ? nil : "sudo -E" end default_config :command_prefix, nil default_config(:suite_name) { |busser| busser.instance.suite.name } # Creates a new Verifier object using the provided configuration data # which will be merged with any default configuration. # # @param config [Hash] provided verifier configuration def initialize(config = {}) init_config(config) end # Runs the verifier on the instance. # # @param state [Hash] mutable instance state # @raise [ActionFailed] if the action could not be completed def call(state) create_sandbox sandbox_dirs = Util.list_directory(sandbox_path) instance.transport.connection(state) do |conn| conn.execute(install_command) conn.execute(init_command) info("Transferring files to #{instance.to_str}") conn.upload(sandbox_dirs, config[:root_path]) debug("Transfer complete") conn.execute(prepare_command) conn.execute(run_command) info("Downloading files from #{instance.to_str}") config[:downloads].to_h.each do |remotes, local| debug("Downloading #{Array(remotes).join(", ")} to #{local}") conn.download(remotes, local) end debug("Download complete") end rescue Kitchen::Transport::TransportFailed => ex raise ActionFailed, ex.message ensure cleanup_sandbox end # Check system and configuration for common errors. # # @param state [Hash] mutable instance state # @returns [Boolean] Return true if a problem is found. def doctor(state) false end # Deletes the sandbox path. Without calling this method, the sandbox path # will persist after the process terminates. In other words, cleanup is # explicit. This method is safe to call multiple times. def cleanup_sandbox return if sandbox_path.nil? debug("Cleaning up local sandbox in #{sandbox_path}") FileUtils.rmtree(sandbox_path) end # Creates a temporary directory on the local workstation into which # verifier related files and directories can be copied or created. The # contents of this directory will be copied over to the instance before # invoking the verifier's run command. After this method completes, it # is expected that the contents of the sandbox is complete and ready for # copy to the remote instance. # # **Note:** any subclasses would be well advised to call super first when # overriding this method, for example: # # @example overriding `#create_sandbox` # # class MyVerifier < Kitchen::Verifier::Base # def create_sandbox # super # # any further file copies, preparations, etc. # end # end def create_sandbox @sandbox_path = Dir.mktmpdir("#{instance.name}-sandbox-") File.chmod(0755, sandbox_path) info("Preparing files for transfer") debug("Creating local sandbox in #{sandbox_path}") end # Generates a command string which will install and configure the # verifier software on an instance. If no work is required, then `nil` # will be returned. # # @return [String] a command string def install_command; end # Generates a command string which will perform any data initialization # or configuration required after the verifier software is installed # but before the sandbox has been transferred to the instance. If no work # is required, then `nil` will be returned. # # @return [String] a command string def init_command; end # Generates a command string which will perform any commands or # configuration required just before the main verifier run command but # after the sandbox has been transferred to the instance. If no work is # required, then `nil` will be returned. # # @return [String] a command string def prepare_command; end # Generates a command string which will invoke the main verifier # command on the prepared instance. If no work is required, then `nil` # will be returned. # # @return [String] a command string def run_command; end # Returns the absolute path to the sandbox directory or raises an # exception if `#create_sandbox` has not yet been called. # # @return [String] the absolute path to the sandbox directory # @raise [ClientError] if the sandbox directory has no yet been created # by calling `#create_sandbox` def sandbox_path @sandbox_path ||= begin raise ClientError, "Sandbox directory has not yet " \ "been created. Please run #{self.class}#create_sandox before " \ "trying to access the path." end end # Sets the API version for this verifier. If the verifier does not set # this value, then `nil` will be used and reported. # # Sets the API version for this verifier # # @example setting an API version # # module Kitchen # module Verifier # class NewVerifier < Kitchen::Verifier::Base # # kitchen_verifier_api_version 2 # # end # end # end # # @param version [Integer,String] a version number # def self.kitchen_verifier_api_version(version) @api_version = version end private # Builds a complete command given a variables String preamble and a file # containing shell code. # # @param vars [String] shell variables, as a String # @param file [String] file basename (without extension) containing # shell code # @return [String] command # @api private def shell_code_from_file(vars, file) src_file = File.join( File.dirname(__FILE__), %w{.. .. .. support}, file + (powershell_shell? ? ".ps1" : ".sh") ) wrap_shell_code([vars, "", IO.read(src_file)].join("\n")) end # Conditionally prefixes a command with a sudo command. # # @param command [String] command to be prefixed # @return [String] the command, conditionaly prefixed with sudo # @api private def sudo(script) config[:sudo] ? "#{config[:sudo_command]} #{script}" : script end # Conditionally prefixes a command with a command prefix. # This should generally be done after a command has been # conditionally prefixed by #sudo as certain platforms, such as # Cisco Nexus, require all commands to be run with a prefix to # obtain outbound network access. # # @param command [String] command to be prefixed # @return [String] the command, conditionally prefixed with the configured prefix # @api private def prefix_command(script) config[:command_prefix] ? "#{config[:command_prefix]} #{script}" : script end end end end