# -*- encoding: utf-8 -*- # # Author:: Fletcher Nichol () # # Copyright (C) 2012, 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 'thor/util' module Kitchen module Driver # Value object to track a shell command that will be passed to Kernel.exec # for execution. # # @author Fletcher Nichol class LoginCommand attr_reader :cmd_array, :options def initialize(cmd_array, options = {}) @cmd_array = cmd_array @options = options end end # Base class for a driver. A driver is responsible for carrying out the # lifecycle activities of an instance, such as creating, converging, and # destroying an instance. # # @author Fletcher Nichol class Base include ShellOut include Logging attr_writer :instance class << self attr_reader :serial_actions end def initialize(config = {}) @config = config self.class.defaults.each do |attr, value| @config[attr] = value unless @config[attr] end Array(self.class.validations).each do |tuple| tuple.last.call(tuple.first, config[tuple.first]) end end # Provides hash-like access to configuration keys. # # @param attr [Object] configuration key # @return [Object] value at configuration key def [](attr) config[attr] end # Creates an instance. # # @param state [Hash] mutable instance and driver state # @raise [ActionFailed] if the action could not be completed def create(state) ; end # Converges a running instance. # # @param state [Hash] mutable instance and driver state # @raise [ActionFailed] if the action could not be completed def converge(state) ; end # Sets up an instance. # # @param state [Hash] mutable instance and driver state # @raise [ActionFailed] if the action could not be completed def setup(state) ; end # Verifies a converged instance. # # @param state [Hash] mutable instance and driver state # @raise [ActionFailed] if the action could not be completed def verify(state) ; end # Destroys an instance. # # @param state [Hash] mutable instance and driver state # @raise [ActionFailed] if the action could not be completed def destroy(state) ; end # Returns the shell command that will log into an instance. # # @param state [Hash] mutable instance and driver state # @return [LoginCommand] an object containing the array of command line # tokens and exec options to be used in a fork/exec # @raise [ActionFailed] if the action could not be completed def login_command(state) raise ActionFailed, "Remote login is not supported in this driver." end # Performs whatever tests that may be required to ensure that this driver # will be able to function in the current environment. This may involve # checking for the presence of certain directories, software installed, # etc. # # @raise [UserError] if the driver will not be able to perform or if a # documented dependency is missing from the system def verify_dependencies ; end protected attr_reader :config, :instance ACTION_METHODS = %w{create converge setup verify destroy}. map(&:to_sym).freeze def logger instance ? instance.logger : Kitchen.logger end def puts(msg) info(msg) end def print(msg) info(msg) end def run_command(cmd, options = {}) base_options = { :use_sudo => config[:use_sudo], :log_subject => Thor::Util.snake_case(self.class.to_s) }.merge(options) super(cmd, base_options) end def busser_setup_cmd busser.setup_cmd end def busser_sync_cmd busser.sync_cmd end def busser_run_cmd busser.run_cmd end def busser @busser ||= begin raise ClientError, "Instance must be set for Driver" if instance.nil? Busser.new(instance.suite.name) end end def self.defaults @defaults ||= Hash.new.merge(super_defaults) end def self.super_defaults klass = self.superclass if klass.respond_to?(:defaults) klass.defaults else Hash.new end end def self.default_config(attr, value) defaults[attr] = value end def self.validations @validations end def self.required_config(attr, &block) @validations = [] if @validations.nil? if ! block_given? klass = self block = lambda do |attr, value| if value.nil? || value.to_s.empty? raise UserError, "#{klass}#config[:#{attr}] cannot be blank" end end end @validations << [attr, block] end def self.no_parallel_for(*methods) Array(methods).each do |meth| if ! ACTION_METHODS.include?(meth) raise ClientError, "##{meth} is not a valid no_parallel_for method" end end @serial_actions ||= [] @serial_actions += methods end end end end