require'set'# This is the latest iteration of the gem dependency resolving algorithm. As of now,# it can resolve (as a success of failure) any set of gem dependencies we throw at it# in a reasonable amount of time. The most iterations I've seen it take is about 150.# The actual implementation of the algorithm is not as good as it could be yet, but that# can come later.# Extending Gem classes to add necessary tracking informationmoduleGemclassDependencydefrequired_by@required_by||=[]endendclassSpecificationdefrequired_by@required_by||=[]endendendmoduleBundlerclassResolverattr_reader:errors# Figures out the best possible configuration of gems that satisfies# the list of passed dependencies and any child dependencies without# causing any gem activation errors.## ==== Parameters# *dependencies<Gem::Dependency>:: The list of dependencies to resolve## ==== Returns# <GemBundle>,nil:: If the list of dependencies can be resolved, a# collection of gemspecs is returned. Otherwise, nil is returned.defself.resolve(requirements,index,source_requirements={})resolver=new(index,source_requirements)result=catch(:success)doresolver.resolve(requirements,{})output=resolver.errors.inject("")do|o,(conflict,(origin,requirement))|iforigino<<" Conflict on: #{conflict.inspect}:\n"o<<" * #{conflict} (#{origin.version}) activated by #{origin.required_by.first}\n"o<<" * #{requirement} required by #{requirement.required_by.first}\n"elseo<<" #{requirement} not found in any of the sources\n"o<<" required by #{requirement.required_by.first}\n"endo<<" All possible versions of origin requirements conflict."endraiseVersionConflict,"No compatible versions could be found for required dependencies:\n#{output}"nilendifresult# Order gems in order of dependencies. Every gem's dependency is at# a smaller index in the array.ordered=[]result.values.eachdo|spec1|index=nilplace=ordered.detectdo|spec2|spec1.dependencies.any?{|d|d.name==spec2.name}endplace?ordered.insert(ordered.index(place),spec1):ordered<<spec1endordered.reverseendenddefinitialize(index,source_requirements)@errors={}@stack=[]@index=index@source_requirements=source_requirementsenddefdebugputsyieldifdefined?($debug)&&$debugenddefresolve(reqs,activated)# If the requirements are empty, then we are in a success state. Aka, all# gem dependencies have been resolved.throw:success,activatedifreqs.empty?debug{STDIN.gets;print"\e[2J\e[f";"==== Iterating ====\n\n"}# Sort dependencies so that the ones that are easiest to resolve are first.# Easiest to resolve is defined by:# 1) Is this gem already activated?# 2) Do the version requirements include prereleased gems?# 3) Sort by number of gems available in the source.reqs=reqs.sort_bydo|a|[activated[a.name]?0:1,a.version_requirements.prerelease??0:1,@errors[a.name]?0:1,activated[a.name]?0:search(a).size]enddebug{"Activated:\n"+activated.values.map{|a|" #{a.name} (#{a.version})"}.join("\n")}debug{"Requirements:\n"+reqs.map{|r|" #{r.name} (#{r.version_requirements})"}.join("\n")}activated=activated.dup# Pull off the first requirement so that we can resolve itcurrent=reqs.shiftdebug{"Attempting:\n#{current.name} (#{current.version_requirements})"}# Check if the gem has already been activated, if it has, we will make sure# that the currently activated gem satisfies the requirement.ifexisting=activated[current.name]ifcurrent.version_requirements.satisfied_by?(existing.version)debug{" * [SUCCESS] Already activated"}@errors.delete(existing.name)# Since the current requirement is satisfied, we can continue resolving# the remaining requirements.resolve(reqs,activated)elsedebug{" * [FAIL] Already activated"}@errors[existing.name]=[existing,current]debug{current.required_by.map{|d|" * #{d.name} (#{d.version_requirements})"}.join("\n")}# debug { " * All current conflicts:\n" + @errors.keys.map { |c| " - #{c}" }.join("\n") }# Since the current requirement conflicts with an activated gem, we need# to backtrack to the current requirement's parent and try another version# of it (maybe the current requirement won't be present anymore). If the# current requirement is a root level requirement, we need to jump back to# where the conflicting gem was activated.parent=current.required_by.last||existing.required_by.last# We track the spot where the current gem was activated because we need# to keep a list of every spot a failure happened.debug{" -> Jumping to: #{parent.name}"}throwparent.name,existing.required_by.last.nameendelse# There are no activated gems for the current requirement, so we are going# to find all gems that match the current requirement and try them in decending# order. We also need to keep a set of all conflicts that happen while trying# this gem. This is so that if no versions work, we can figure out the best# place to backtrack to.conflicts=Set.new# Fetch all gem versions matching the requirement## TODO: Warn / error when no matching versions are found.matching_versions=search(current)ifmatching_versions.empty?ifcurrent.required_by.empty?location=current.source?current.source.to_s:"any of the sources"raiseGemNotFound,"Could not find gem '#{current}' in #{location}"else@errors[current.name]=[nil,current]endendmatching_versions.reverse_eachdo|spec|conflict=resolve_requirement(spec,current,reqs.dup,activated.dup)conflicts<<conflictifconflictend# If the current requirement is a root level gem and we have conflicts, we# can figure out the best spot to backtrack to.ifcurrent.required_by.empty?&&!conflicts.empty?# Check the current "catch" stack for the first one that is included in the# conflicts set. That is where the parent of the conflicting gem was required.# By jumping back to this spot, we can try other version of the parent of# the conflicting gem, hopefully finding a combination that activates correctly.@stack.reverse_eachdo|savepoint|ifconflicts.include?(savepoint)debug{" -> Jumping to: #{savepoint}"}throwsavepointendendendendenddefresolve_requirement(spec,requirement,reqs,activated)# We are going to try activating the spec. We need to keep track of stack of# requirements that got us to the point of activating this gem.spec.required_by.replacerequirement.required_byspec.required_by<<requirementactivated[spec.name]=specdebug{" Activating: #{spec.name} (#{spec.version})"}debug{spec.required_by.map{|d|" * #{d.name} (#{d.version_requirements})"}.join("\n")}# Now, we have to loop through all child dependencies and add them to our# array of requirements.debug{" Dependencies"}spec.dependencies.eachdo|dep|nextifdep.type==:developmentdebug{" * #{dep.name} (#{dep.version_requirements})"}dep.required_by.replace(requirement.required_by)dep.required_by<<requirementreqs<<depend# We create a savepoint and mark it by the name of the requirement that caused# the gem to be activated. If the activated gem ever conflicts, we are able to# jump back to this point and try another version of the gem.length=@stack.length@stack<<requirement.nameretval=catch(requirement.name)doresolve(reqs,activated)end# Since we're doing a lot of throw / catches. A push does not necessarily match# up to a pop. So, we simply slice the stack back to what it was before the catch# block.@stack.slice!(length..-1)retvalenddefsearch(dep)index=@source_requirements[dep.name]||@indexindex.search(dep)endendend