# # Author:: Fletcher Nichol () # # Copyright (C) 2012, 2013, 2014, 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 "thor/util" module Kitchen # Stateless utility methods used in different contexts. Essentially a mini # PassiveSupport library. # # @author Fletcher Nichol module Util # Returns the standard library Logger level constants for a given symbol # representation. # # @param symbol [Symbol] symbol representation of a logger level (:debug, # :info, :warn, :error, :fatal) # @return [Integer] Logger::Severity constant value or nil if input is not # valid def self.to_logger_level(symbol) return nil unless %i{debug info warn error fatal}.include?(symbol) Logger.const_get(symbol.to_s.upcase) end # Returns the symbol represenation of a logging levels for a given # standard library Logger::Severity constant. # # @param const [Integer] Logger::Severity constant value for a logging # level (Logger::DEBUG, Logger::INFO, Logger::WARN, Logger::ERROR, # Logger::FATAL) # @return [Symbol] symbol representation of the logging level def self.from_logger_level(const) case const when Logger::DEBUG then :debug when Logger::INFO then :info when Logger::WARN then :warn when Logger::ERROR then :error else :fatal end end # Returns a new Hash with all key values coerced to symbols. All keys # within a Hash are coerced by calling #to_sym and hashes within arrays # and other hashes are traversed. # # @param obj [Object] the hash to be processed. While intended for # hashes, this method safely processes arbitrary objects # @return [Object] a converted hash with all keys as symbols def self.symbolized_hash(obj) if obj.is_a?(Hash) obj.inject({}) { |h, (k, v)| h[k.to_sym] = symbolized_hash(v); h } elsif obj.is_a?(Array) obj.inject([]) { |a, e| a << symbolized_hash(e); a } else obj end end # Returns a new Hash with all key values coerced to strings. All keys # within a Hash are coerced by calling #to_s and hashes with arrays # and other hashes are traversed. # # @param obj [Object] the hash to be processed. While intended for # hashes, this method safely processes arbitrary objects # @return [Object] a converted hash with all keys as strings def self.stringified_hash(obj) if obj.is_a?(Hash) obj.inject({}) { |h, (k, v)| h[k.to_s] = stringified_hash(v); h } elsif obj.is_a?(Array) obj.inject([]) { |a, e| a << stringified_hash(e); a } else obj end end # Returns a formatted string representing a duration in seconds. # # @param total [Integer] the total number of seconds # @return [String] a formatted string of the form (XmYY.00s) def self.duration(total) total = 0 if total.nil? minutes = (total / 60).to_i seconds = (total - (minutes * 60)) format("(%dm%.2fs)", minutes, seconds) end # Generates a command (or series of commands) wrapped so that it can be # invoked on a remote instance or locally. # # This method uses the Bourne shell (/bin/sh) to maximize the chance of # cross platform portability on Unixlike systems. # # @param [String] the command # @return [String] a wrapped command string def self.wrap_command(cmd) cmd = "false" if cmd.nil? cmd = "true" if cmd.to_s.empty? cmd = cmd.sub(/\n\Z/, "") if cmd =~ /\n\Z/ "sh -c '\n#{cmd}\n'" end # Modifes the given string to strip leading whitespace on each line, the # amount which is calculated by using the first line of text. # # @example # # string = <<-STRING # a # b # c # STRING # Util.outdent!(string) # => "a\n b\nc\n" # # @param string [String] the string that will be modified # @return [String] the modified string def self.outdent!(string) string.gsub!(/^ {#{string.index(/[^ ]/)}}/, "") end # Returns a set of Bourne Shell (AKA /bin/sh) compatible helper # functions. This function is usually called inline in a string that # will be executed remotely on a test instance. # # @return [String] a string representation of useful helper functions def self.shell_helpers IO.read(File.join( File.dirname(__FILE__), %w{.. .. support download_helpers.sh} )) end # Lists the contents of the given directory. path will be prepended # to the list returned. '.' and '..' are never returned. # # @param path [String] the directory to list # @param include_dot [Boolean] if true, dot files will be included # @param recurse [Boolean] if true, listing will be recursive # @return A listing of the specified path # # @note You should prefer this method to using Dir.glob directly. The reason is # because Dir.glob behaves strangely on Windows. It wont accept '\' # and doesn't like fake directories (C:\Documents and Settings) # It also does not do any sort of error checking, so things one would # expect to fail just return an empty list # # @note Dir.chdir is applied to the process, thus it is not thread-safe # and must be synchronized. def self.list_directory(path, include_dot: false, recurse: false) # Things (such as tests) are relying on this to not blow up if # the directory does not exist return [] unless Dir.exist?(path) Kitchen.mutex_chdir.synchronize do Dir.chdir(path) do glob_pattern = if recurse "**/*" else "*" end flags = if include_dot [File::FNM_DOTMATCH] else [] end Dir.glob(glob_pattern, *flags) .reject { |f| [".", ".."].include?(f) } .map { |f| File.join(path, f) } end end end # Similar to Dir.glob. # # The difference is this function forces you to specify where to glob from. # You should glob from the path closest to what you want. The reason for this # is because if you have symlinks on windows of any kind, Dir.glob will not # traverse them. # # @param path [String] the directory to glob from # @param pattern [String] The pattern to match # @param flags [Integer] You can specify flags you would have passed to Dir.glob # @return Files matching the specified pattern in the given path # # @note Dir.chdir is applied to the process, thus it is not thread-safe # and must be synchronized. def self.safe_glob(path, pattern, *flags) return [] unless Dir.exist?(path) Kitchen.mutex_chdir.synchronize do Dir.chdir(path) do Dir.glob(pattern, *flags).map { |f| File.join(path, f) } end end end def self.camel_case(a_string) Thor::Util.camel_case(a_string) end def self.snake_case(a_string) Thor::Util.snake_case(a_string) end end end