class Bundler::Index

def self.build

def self.build
  i = new
  yield i
  i
end

def <<(spec)

def <<(spec)
  arr = specs_by_name(spec.name)
  arr.delete_if do |s|
    same_version?(s.version, spec.version) && s.platform == spec.platform
  end
  arr << spec
  spec
end

def ==(o)

def ==(o)
  all? do |spec|
    other_spec = o[spec].first
    (spec.dependencies & other_spec.dependencies).empty? && spec.source == other_spec.source
  end
end

def add_source(index)

def add_source(index)
  if index.is_a?(Index)
    @sources << index
    @sources.uniq! # need to use uniq! here instead of checking for the item before adding
  else
    raise ArgumentError, "Source must be an index, not #{index.class}"
  end
end

def each(&blk)

def each(&blk)
  specs.values.each do |specs|
    specs.each(&blk)
  end
end

def empty?

def empty?
  each { return false }
  true
end

def initialize

def initialize
  @sources = []
  @cache = {}
  @specs = Hash.new { |h,k| h[k] = [] }
end

def initialize_copy(o)

def initialize_copy(o)
  super
  @sources = @sources.dup
  @cache = {}
  @specs = Hash.new { |h,k| h[k] = [] }
  o.specs.each do |name, array|
    @specs[name] = array.dup
  end
end

def inspect

def inspect
  "#<#{self.class}:0x#{object_id} sources=#{sources.map{|s| s.inspect}} specs.size=#{specs.size}>"
end

def local_search(query, base = nil)

def local_search(query, base = nil)
  case query
  when Gem::Specification, RemoteSpecification, LazySpecification, EndpointSpecification then search_by_spec(query)
  when String then specs_by_name(query)
  when Gem::Dependency then search_by_dependency(query, base)
  else
    raise "You can't search for a #{query.inspect}."
  end
end

def same_version?(a, b)

def same_version?(a, b)
  regex = /^(.*?)(?:\.0)*$/
  a.to_s[regex, 1] == b.to_s[regex, 1]
end

def same_version?(a, b)

def same_version?(a, b)
  a == b
end

def search(query, base = nil)

about, returning all of the results.
Search this index's specs, and any source indexes that this index knows
def search(query, base = nil)
  results = local_search(query, base)
  seen = Set.new(results.map { |spec| [spec.name, spec.version, spec.platform] })
  @sources.each do |source|
    source.search(query, base).each do |spec|
      lookup = [spec.name, spec.version, spec.platform]
      unless seen.include?(lookup)
        results << spec
        seen << lookup
      end
    end
  end
  results
end

def search_by_dependency(dependency, base = nil)

def search_by_dependency(dependency, base = nil)
  @cache[base || false] ||= {}
  @cache[base || false][dependency] ||= begin
    specs = specs_by_name(dependency.name) + (base || [])
    found = specs.select do |spec|
      if base # allow all platforms when searching from a lockfile
        dependency.matches_spec?(spec)
      else
        dependency.matches_spec?(spec) && Gem::Platform.match(spec.platform)
      end
    end
    wants_prerelease = dependency.requirement.prerelease?
    only_prerelease  = specs.all? {|spec| spec.version.prerelease? }
    unless wants_prerelease || only_prerelease
      found.reject! { |spec| spec.version.prerelease? }
    end
    found.sort_by {|s| [s.version, s.platform.to_s == 'ruby' ? "\0" : s.platform.to_s] }
  end
end

def search_by_spec(spec)

def search_by_spec(spec)
  specs_by_name(spec.name).select do |s|
    same_version?(s.version, spec.version) && Gem::Platform.new(s.platform) == Gem::Platform.new(spec.platform)
  end
end

def size

def size
  @sources.inject(@specs.size) do |size, source|
    size += source.size
  end
end

def source_types

def source_types
  sources.map{|s| s.class }.uniq
end

def spec_satisfies_dependency?(spec, dep)

def spec_satisfies_dependency?(spec, dep)
  return false unless dep.name == spec.name
  dep.requirement.satisfied_by?(spec.version)
end

def specs_by_name(name)

def specs_by_name(name)
  @specs[name]
end

def unmet_dependency_names

returns a list of the dependencies
def unmet_dependency_names
  dependency_names = specs.values.map do |array_of_s|
    array_of_s.map do |s|
      s.dependencies.map{|d| d.name }
    end
  end.flatten.uniq
  dependency_names.select{|name| name != 'bundler' && specs_by_name(name).empty? }
end

def use(other, override_dupes = false)

def use(other, override_dupes = false)
  return unless other
  other.each do |s|
    if (dupes = search_by_spec(s)) && dupes.any?
      next unless override_dupes
      @specs[s.name] -= dupes
    end
    @specs[s.name] << s
  end
  self
end