module Bundler::GemHelpers

def generic(p)

def generic(p)
  GENERIC_CACHE[p] ||= begin
    _, found = GENERICS.find do |match, _generic|
      p.os == match.os && (!match.cpu || p.cpu == match.cpu)
    end
    found || Gem::Platform::RUBY
  end
end

def generic_local_platform

def generic_local_platform
  generic(local_platform)
end

def generic_local_platform_is_ruby?

def generic_local_platform_is_ruby?
  generic_local_platform == Gem::Platform::RUBY
end

def local_platform

def local_platform
  Bundler.local_platform
end

def platform_specificity_match(spec_platform, user_platform)

def platform_specificity_match(spec_platform, user_platform)
  spec_platform = Gem::Platform.new(spec_platform)
  PlatformMatch.specificity_score(spec_platform, user_platform)
end

def same_deps(spec, exemplary_spec)

def same_deps(spec, exemplary_spec)
  same_runtime_deps = spec.dependencies.sort == exemplary_spec.dependencies.sort
  same_metadata_deps = spec.required_ruby_version == exemplary_spec.required_ruby_version && spec.required_rubygems_version == exemplary_spec.required_rubygems_version
  same_runtime_deps && same_metadata_deps
end

def same_specificity(platform, spec, exemplary_spec)

def same_specificity(platform, spec, exemplary_spec)
  platform_specificity_match(spec.platform, platform) == platform_specificity_match(exemplary_spec.platform, platform)
end

def select_best_local_platform_match(specs, force_ruby: false)

def select_best_local_platform_match(specs, force_ruby: false)
  select_best_platform_match(specs, local_platform, force_ruby: force_ruby).map(&:materialize_for_installation).compact
end

def select_best_platform_match(specs, platform, force_ruby: false, prefer_locked: false)

def select_best_platform_match(specs, platform, force_ruby: false, prefer_locked: false)
  matching = if force_ruby
    specs.select {|spec| spec.match_platform(Gem::Platform::RUBY) && spec.force_ruby_platform! }
  else
    specs.select {|spec| spec.match_platform(platform) }
  end
  if prefer_locked
    locked_originally = matching.select {|spec| spec.is_a?(LazySpecification) }
    return locked_originally if locked_originally.any?
  end
  sort_best_platform_match(matching, platform)
end

def sort_best_platform_match(matching, platform)

def sort_best_platform_match(matching, platform)
  exact = matching.select {|spec| spec.platform == platform }
  return exact if exact.any?
  sorted_matching = matching.sort_by {|spec| platform_specificity_match(spec.platform, platform) }
  exemplary_spec = sorted_matching.first
  sorted_matching.take_while {|spec| same_specificity(platform, spec, exemplary_spec) && same_deps(spec, exemplary_spec) }
end