Update .init! to attempt repair if instructed

This commit is contained in:
Chris Roberts 2016-11-02 10:36:44 -07:00
parent d5c5561302
commit 3fb5913af1
1 changed files with 35 additions and 17 deletions

View File

@ -31,7 +31,7 @@ module Vagrant
# Initializes Bundler and the various gem paths so that we can begin # Initializes Bundler and the various gem paths so that we can begin
# loading gems. This must only be called once. # loading gems. This must only be called once.
def init!(plugins) def init!(plugins, repair=false)
# Add HashiCorp RubyGems source # Add HashiCorp RubyGems source
Gem.sources << HASHICORP_GEMSTORE Gem.sources << HASHICORP_GEMSTORE
@ -48,25 +48,43 @@ module Vagrant
# Sets that we can resolve our dependencies from # Sets that we can resolve our dependencies from
current_set = Gem::Resolver::CurrentSet.new current_set = Gem::Resolver::CurrentSet.new
plugin_set = Gem::Resolver::VendorSet.new plugin_set = Gem::Resolver::VendorSet.new
repair_result = nil
# Register all known plugin specifications to the plugin set begin
Dir.glob(plugin_gem_path.join('specifications/*.gemspec').to_s).each do |spec_path| # Register all known plugin specifications to the plugin set
spec = Gem::Specification.load(spec_path) Dir.glob(plugin_gem_path.join('specifications/*.gemspec').to_s).each do |spec_path|
desired_spec_path = File.join(spec.gem_dir, "#{spec.name}.gemspec") spec = Gem::Specification.load(spec_path)
# Vendor set requires the spec to be within the gem directory. Some gems will package their desired_spec_path = File.join(spec.gem_dir, "#{spec.name}.gemspec")
# spec file, and that's not what we want to load. # Vendor set requires the spec to be within the gem directory. Some gems will package their
if !File.exist?(desired_spec_path) || !FileUtils.cmp(spec.spec_file, desired_spec_path) # spec file, and that's not what we want to load.
File.write(desired_spec_path, spec.to_ruby) if !File.exist?(desired_spec_path) || !FileUtils.cmp(spec.spec_file, desired_spec_path)
File.write(desired_spec_path, spec.to_ruby)
end
plugin_set.add_vendor_gem(spec.name, spec.gem_dir)
end
# Compose set for resolution
composed_set = Gem::Resolver.compose_sets(current_set, plugin_set)
# Resolve the request set to ensure proper activation order
solution = request_set.resolve(composed_set)
rescue Gem::UnsatisfiableDependencyError => failure
if repair
raise failure if @init_retried
install(plugins)
@init_retried = true
retry
else
$stderr.puts "Vagrant failed to properly initialize due to an error while"
$stderr.puts "while attempting to load configured plugins. This can be caused"
$stderr.puts "by manually tampering with the 'plugins.json' file, or by a"
$stderr.puts "recent Vagrant upgrade. To fix this problem, please run:\n\n"
$stderr.puts " vagrant plugin repair\n\n"
$stderr.puts "The error message is shown below:\n\n"
$stderr.puts failure.message
exit 1
end end
plugin_set.add_vendor_gem(spec.name, spec.gem_dir)
end end
# Compose set for resolution
composed_set = Gem::Resolver.compose_sets(current_set, plugin_set)
# Resolve the request set to ensure proper activation order
solution = request_set.resolve(composed_set)
# Activate the gems # Activate the gems
begin begin
retried = false retried = false