require 'set'

module Hyperstack
  class Autoloader
    # All files ever loaded.
    def self.history=(a)
      @@history = a
    end
    def self.history
      @@history
    end
    self.history = Set.new

    def self.load_paths=(a)
      @@load_paths = a
    end
    def self.load_paths
      @@load_paths
    end
    self.load_paths = []

    def self.loaded=(a)
      @@loaded = a
    end
    def self.loaded
      @@loaded
    end
    self.loaded = Set.new

    def self.loading=(a)
      @@loading = a
    end
    def self.loading
      @@loading
    end
    self.loading = []

    def self.const_missing(const_name, mod)
      # name.nil? is testing for anonymous
      from_mod = mod.name.nil? ? guess_for_anonymous(const_name) : mod
      load_missing_constant(from_mod, const_name)
    rescue Exception => e
      puts "HyperStack autoloader failed attempting to load #{mod}::#{const_name}.  Could be a bug in autoloader"
    end

    def self.guess_for_anonymous(const_name)
      if Object.const_defined?(const_name)
        raise NameError.new "#{const_name} cannot be autoloaded from an anonymous class or module", const_name
      else
        Object
      end
    end

    def self.load_missing_constant(from_mod, const_name)
      # see active_support/dependencies.rb in case of reloading on how to handle
      qualified_name = qualified_name_for(from_mod, const_name)
      qualified_path = underscore(qualified_name)

      module_path = search_for_module(qualified_path)
      if module_path
        if loading.include?(module_path)
          raise "Circular dependency detected while autoloading constant #{qualified_name}"
        else
          require_or_load(from_mod, module_path)
          raise LoadError, "Unable to autoload constant #{qualified_name}, expected #{module_path} to define it" unless from_mod.const_defined?(const_name, false)
          return from_mod.const_get(const_name)
        end
      elsif from_mod.respond_to?(:parent) &&  (parent = from_mod.parent) && parent != from_mod &&
            ! from_mod.parents.any? { |p| p.const_defined?(const_name, false) }
        begin
          return parent.const_missing(const_name)
        rescue NameError => e
          raise unless missing_name?(e, qualified_name_for(parent, const_name))
        end
      end
    end

    def self.missing_name?(e, name)
      mn = if /undefined/ !~ e.message
             $1 if /((::)?([A-Z]\w*)(::[A-Z]\w*)*)$/ =~ e.message
           end
      mn == name
    end

    # Returns the constant path for the provided parent and constant name.
    def self.qualified_name_for(mod, name)
      mod_name = to_constant_name(mod)
      mod_name == 'Object' ? name.to_s : "#{mod_name}::#{name}"
    end

    def self.require_or_load(from_mod, module_path)
      return if loaded.include?(module_path)
      loaded << module_path
      loading << module_path

      begin
        result = require module_path
      rescue Exception
        loaded.delete module_path
        raise LoadError, "Unable to autoload: require_or_load #{module_path} failed"
      ensure
        loading.pop
      end

      # Record history *after* loading so first load gets warnings.
      history << module_path
      result
      # end
    end

    def self.search_for_module(path)
      # oh my! imagine Bart Simpson, writing on the board:
      # "javascript is not ruby, javascript is not ruby, javascript is not ruby, ..."
      # then running home, starting irb, on the fly developing a chat client and opening a session with Homer at his workplace: "Hi Dad ..."
      load_paths.each do |load_path|
        mod_path = load_path + '/' + path
        return mod_path if `Opal.modules.hasOwnProperty(#{mod_path})`
      end
      return path if `Opal.modules.hasOwnProperty(#{path})`
      nil # Gee, I sure wish we had first_match ;-)
    end

    # Convert the provided const desc to a qualified constant name (as a string).
    # A module, class, symbol, or string may be provided.
    def self.to_constant_name(desc) #:nodoc:
      case desc
      when String then desc.sub(/^::/, '')
      when Symbol then desc.to_s
      when Module
        desc.name ||
          raise(ArgumentError, 'Anonymous modules have no name to be referenced by')
      else raise TypeError, "Not a valid constant descriptor: #{desc.inspect}"
      end
    end

    def self.underscore(string)
      string.gsub(/::/, '/').gsub(/([A-Z]+)([A-Z][a-z])/,'\1_\2').gsub(/([a-z\d])([A-Z])/,'\1_\2').tr("-", "_").downcase
    end
  end
end