summaryrefslogtreecommitdiff
path: root/lib/bundler/gem_version_promoter.rb
diff options
context:
space:
mode:
authorchrismo <chrismo@clabs.org>2016-06-20 01:25:07 -0500
committerchrismo <chrismo@clabs.org>2016-07-08 19:35:57 -0500
commitb26b54cc6f993309512f1a10733f17af49122d67 (patch)
tree4b9b7e0e809ee7f2745eb1ec958aaefba29d231d /lib/bundler/gem_version_promoter.rb
parent5f63cad355e7736acaefab3c40266d033f67a4b8 (diff)
downloadbundler-b26b54cc6f993309512f1a10733f17af49122d67.tar.gz
GemVersionPromoter refactor
UpdateOptions which was then renamed to DependencySearch is now called GemVersionPromoter, cuz I can't name this damn class. It's in its own file now, so there's that. I took a shot at moving Resolver#search_for into it, but had naively overlooked a few instance variables and such and it just didn't make as much sense as I'd first envisioned. Probably some other smaller classes in between perhaps. GemVersionPromoter class now caching its results, too, and I moved out the return from it back into Resolver as it made more sense there. As a standalone class, it may make sense to have this actually implement :major sorting, but maybe later.
Diffstat (limited to 'lib/bundler/gem_version_promoter.rb')
-rw-r--r--lib/bundler/gem_version_promoter.rb126
1 files changed, 126 insertions, 0 deletions
diff --git a/lib/bundler/gem_version_promoter.rb b/lib/bundler/gem_version_promoter.rb
new file mode 100644
index 0000000000..19de170a45
--- /dev/null
+++ b/lib/bundler/gem_version_promoter.rb
@@ -0,0 +1,126 @@
+# frozen_string_literal: true
+module Bundler
+ class GemVersionPromoter
+ attr_accessor :level, :strict, :minimal
+
+ def initialize(locked_specs = SpecSet.new([]), unlock_gems = [])
+ @level_default = :major
+ @level = @level_default
+ @strict = false
+ @minimal = false
+ @locked_specs = locked_specs
+ @unlock_gems = unlock_gems
+ @sort_versions = {}
+ end
+
+ def level=(value)
+ v = begin
+ case value
+ when String, Symbol
+ value.to_sym
+ end
+ end
+
+ @level = [:major, :minor, :patch].include?(v) ? v : @level_default
+ end
+
+ def sort_versions(dep, dep_specs)
+ before_result = "before sort_versions: #{debug_format_result(dep, dep_specs).inspect}"
+
+ result = @sort_versions[dep] ||= begin
+ gem_name = dep.name
+
+ # An Array per version returned, different entries for different platforms.
+ # We just need the version here so it's ok to hard code this to the first instance.
+ locked_spec = @locked_specs[gem_name].first
+
+ if @strict
+ filter_dep_specs(dep_specs, locked_spec)
+ else
+ sort_dep_specs(dep_specs, locked_spec)
+ end.tap do |specs|
+ if ENV["DEBUG_PATCH_RESOLVER"] # MODO: proper debug flag name, check and proper debug output
+ STDERR.puts before_result
+ STDERR.puts " after sort_versions: #{debug_format_result(dep, specs).inspect}"
+ end
+ end
+ end
+ result.dup # not ideal, but elsewhere in bundler the resulting array is occasionally emptied, corrupting the cache.
+ end
+
+ private
+
+ def filter_dep_specs(specs, locked_spec)
+ res = specs.select do |sg|
+ # SpecGroup is grouped by name/version, multiple entries for multiple platforms.
+ # We only need the name, which will be the same, so hard coding to first is ok.
+ gem_spec = sg.first
+
+ if locked_spec
+ gsv = gem_spec.version
+ lsv = locked_spec.version
+
+ must_match = @level == :minor ? [0] : [0, 1]
+
+ matches = must_match.map {|idx| gsv.segments[idx] == lsv.segments[idx] }
+ (matches.uniq == [true]) ? (gsv >= lsv) : false
+ else
+ true
+ end
+ end
+
+ sort_dep_specs(res, locked_spec)
+ end
+
+ # reminder: sort still filters anything older than locked version
+ # :major bundle update behavior can move a gem to an older version
+ # in order to satisfy the dependency tree.
+ def sort_dep_specs(specs, locked_spec)
+ return specs unless locked_spec
+ gem_name = locked_spec.name
+ locked_version = locked_spec.version
+
+ filtered = specs.select {|s| s.first.version >= locked_version }
+
+ filtered.sort do |a, b|
+ a_ver = a.first.version
+ b_ver = b.first.version
+ case
+ when a_ver.segments[0] != b_ver.segments[0]
+ b_ver <=> a_ver
+ when !(@level == :minor) && (a_ver.segments[1] != b_ver.segments[1])
+ b_ver <=> a_ver
+ when @minimal && !unlocking_gem?(gem_name)
+ b_ver <=> a_ver
+ when @minimal && unlocking_gem?(gem_name) &&
+ ![a_ver, b_ver].include?(locked_version) # MODO: revisit this case
+ b_ver <=> a_ver
+ else
+ a_ver <=> b_ver
+ end
+ end.tap do |result|
+ unless unlocking_gem?(gem_name)
+ move_version_to_end(specs, locked_version, result)
+ end
+ end
+ end
+
+ def unlocking_gem?(gem_name)
+ @unlock_gems.empty? || @unlock_gems.include?(gem_name)
+ end
+
+ def move_version_to_end(specs, version, result)
+ spec_group = specs.detect {|s| s.first.version.to_s == version.to_s }
+ return unless spec_group
+ result.reject! {|s| s.first.version.to_s == version.to_s }
+ result << spec_group
+ end
+
+ def debug_format_result(dep, res)
+ a = [dep.to_s,
+ res.map {|sg| [sg.version, sg.dependencies_for_activated_platforms.map {|dp| [dp.name, dp.requirement.to_s] }] }]
+ [a.first, a.last.map {|sg_data| [sg_data.first.version, sg_data.last.map {|aa| aa.join(" ") }] },
+ @level, @strict ? :strict : :not_strict, @minimal ? :minimal : :not_minimal]
+ end
+ end
+end