module Thor::Util

def self.constant_from_thor_path(str)

def self.constant_from_thor_path(str)
  make_constant(to_constant(str))
rescue NameError => e
  raise e unless e.message =~ /^uninitialized constant (.*)$/
  raise Error, "There was no available namespace `#{str}'."
end

def self.constant_to_thor_path(str, remove_default = true)

def self.constant_to_thor_path(str, remove_default = true)
  str = str.to_s.gsub(/^Thor::Tasks::/, "")
  str = snake_case(str).squeeze(":")
  str.gsub!(/^default/, '') if remove_default
  str
end

def self.constants_in_contents(str, file = __FILE__)

def self.constants_in_contents(str, file = __FILE__)
  klasses = ObjectSpace.classes.dup
  Module.new.class_eval(str, file)
  klasses = ObjectSpace.classes - klasses
  klasses = klasses.select {|k| k < Thor }
  klasses.map! {|k| k.to_s.gsub(/#<Module:\w+>::/, '')}
end

def self.full_const_get(obj, name)

def self.full_const_get(obj, name)
  list = name.split("::")
  list.shift if list.first.empty?
  list.each do |x| 
    # This is required because const_get tries to look for constants in the 
    # ancestor chain, but we only want constants that are HERE
    obj = obj.const_defined?(x) ? obj.const_get(x) : obj.const_missing(x)
  end
  obj
end

def self.make_constant(str, base = [Thor::Tasks, Object])

def self.make_constant(str, base = [Thor::Tasks, Object])
  which = base.find do |obj|
    full_const_get(obj, str) rescue nil
  end
  return full_const_get(which, str) if which
  raise NameError, "uninitialized constant #{str}"
end

def self.snake_case(str)

def self.snake_case(str)
  return str.downcase if str =~ /^[A-Z_]+$/
  str.gsub(/\B[A-Z]/, '_\&').squeeze('_') =~ /_*(.*)/
  return $+.downcase
end  

def self.to_constant(str)

def self.to_constant(str)
  str = 'default' if str.empty?
  str.gsub(/:(.?)/) { "::#{$1.upcase}" }.gsub(/(?:^|_)(.)/) { $1.upcase }
end