Class: Bundler::LazySpecification
| Relationships & Source Files | |
| Super Chains via Extension / Inclusion / Inheritance | |
|
Instance Chain:
|
|
| Inherits: | Object |
| Defined in: | lib/bundler/lazy_specification.rb |
Class Method Summary
Instance Attribute Summary
- #dependencies (also: #runtime_dependencies) rw
- #force_ruby_platform rw
- #incomplete? ⇒ Boolean readonly
- #materialization readonly
- #missing? ⇒ Boolean readonly
-
#most_specific_locked_platform
rw
For backwards compatibility with existing lockfiles, if the most specific locked platform is not a specific platform like x86_64-linux or universal-java-11, then we keep the previous behaviour of resolving the best platform variant at materiliazation time.
- #name readonly
- #overrides rw
- #platform readonly
- #remote rw
- #required_ruby_version rw
- #required_rubygems_version rw
-
#runtime_dependencies
readonly
Alias for #dependencies.
- #source rw
- #source_changed? ⇒ Boolean readonly
- #version readonly
- #ruby_platform_materializes_to_ruby_platform? ⇒ Boolean readonly private
- #use_exact_resolved_specifications? ⇒ Boolean readonly private
MatchMetadata - Included
Instance Method Summary
- #==(other)
- #eql?(other) ⇒ Boolean
- #force_ruby_platform!
- #full_name
- #git_version
- #hash
- #inspect
- #lock_name
- #materialize_for_cache
- #materialize_for_installation(locked_platforms)
- #materialized_for_installation(locked_platforms = nil)
- #name_tuple
- #replace_source_with!(gemfile_source)
-
#satisfies?(dependency) ⇒ Boolean
Does this locked specification satisfy
dependency? - #to_lock
- #to_s
-
#candidate_platforms(locked_platforms: nil)
private
Platforms to try in order of preference.
-
#choose_compatible(candidates, fallback_to_non_installable: Bundler.frozen_bundle?)
private
If in frozen mode, we fallback to a non-installable candidate because by doing this we avoid re-resolving and potentially end up changing the lockfile, which is not allowed.
- #find_compatible_platform_spec(specs, locked_platforms: nil) private
-
#frozen_bundle_fallback(specs)
private
In frozen mode, accept any candidate.
- #materialize(query) {|matching_specs| ... } private
-
#resolve_best_platform(specs, locked_platforms: nil)
private
Try platforms in order of preference until finding a compatible spec.
-
#validate_dependencies(spec)
private
Validate dependencies of this locked spec are consistent with dependencies of the actual spec that was materialized.
ForcePlatform - Included
| #default_force_ruby_platform | The |
MatchPlatform - Included
MatchMetadata - Included
Constructor Details
.new(name, version, platform, source = nil, **materialization_options) ⇒ LazySpecification
# File 'lib/bundler/lazy_specification.rb', line 38
def initialize(name, version, platform, source = nil, **) @name = name @version = version @dependencies = [] @required_ruby_version = Gem::Requirement.default @required_rubygems_version = Gem::Requirement.default @platform = platform || Gem::Platform::RUBY @original_source = source @source = source @materialization_options = @force_ruby_platform = default_force_ruby_platform @most_specific_locked_platform = nil @materialization = nil end
Class Method Details
.from_spec(s)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 29
def self.from_spec(s) lazy_spec = new(s.name, s.version, s.platform, s.source) lazy_spec.dependencies = s.runtime_dependencies lazy_spec.required_ruby_version = s.required_ruby_version lazy_spec.required_rubygems_version = s.required_rubygems_version lazy_spec.overrides = s.overrides if s.is_a?(LazySpecification) lazy_spec end
Instance Attribute Details
#dependencies (rw) Also known as: #runtime_dependencies
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 12
attr_accessor :source, :remote, :force_ruby_platform, :dependencies, :required_ruby_version, :required_rubygems_version
#force_ruby_platform (rw)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 12
attr_accessor :source, :remote, :force_ruby_platform, :dependencies, :required_ruby_version, :required_rubygems_version
#incomplete? ⇒ Boolean (readonly)
[ GitHub ]
# File 'lib/bundler/lazy_specification.rb', line 59
def incomplete? @materialization.nil? end
#materialization (readonly)
[ GitHub ]
#missing? ⇒ Boolean (readonly)
[ GitHub ]
# File 'lib/bundler/lazy_specification.rb', line 55
def missing? @materialization == self end
#most_specific_locked_platform (rw)
For backwards compatibility with existing lockfiles, if the most specific locked platform is not a specific platform like x86_64-linux or universal-java-11, then we keep the previous behaviour of resolving the best platform variant at materiliazation time. For previous bundler versions (before 2.2.0) this was always the case (except when the lockfile only included non-ruby platforms), but we're also keeping this behaviour on newer bundlers unless users generate the lockfile from scratch or explicitly add a more specific platform.
# File 'lib/bundler/lazy_specification.rb', line 25
attr_accessor :most_specific_locked_platform
#name (readonly)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 11
attr_reader :name, :version, :platform, :materialization
#overrides (rw)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 13
attr_accessor :overrides
#platform (readonly)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 11
attr_reader :name, :version, :platform, :materialization
#remote (rw)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 12
attr_accessor :source, :remote, :force_ruby_platform, :dependencies, :required_ruby_version, :required_rubygems_version
#required_ruby_version (rw)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 12
attr_accessor :source, :remote, :force_ruby_platform, :dependencies, :required_ruby_version, :required_rubygems_version
#required_rubygems_version (rw)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 12
attr_accessor :source, :remote, :force_ruby_platform, :dependencies, :required_ruby_version, :required_rubygems_version
#ruby_platform_materializes_to_ruby_platform? ⇒ Boolean (readonly, private)
[ GitHub ]
# File 'lib/bundler/lazy_specification.rb', line 225
def ruby_platform_materializes_to_ruby_platform? generic_platform = Bundler.generic_local_platform == Gem::Platform::JAVA ? Gem::Platform::JAVA : Gem::Platform::RUBY (most_specific_locked_platform != generic_platform) || force_ruby_platform || Bundler.settings[:force_ruby_platform] end
#runtime_dependencies (readonly)
Alias for #dependencies.
# File 'lib/bundler/lazy_specification.rb', line 27
alias_method :runtime_dependencies, :dependencies
#source (rw)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 12
attr_accessor :source, :remote, :force_ruby_platform, :dependencies, :required_ruby_version, :required_rubygems_version
#source_changed? ⇒ Boolean (readonly)
[ GitHub ]
# File 'lib/bundler/lazy_specification.rb', line 63
def source_changed? @original_source != source end
#use_exact_resolved_specifications? ⇒ Boolean (readonly, private)
[ GitHub ]
# File 'lib/bundler/lazy_specification.rb', line 183
def use_exact_resolved_specifications? !source.is_a?(Source::Path) && ruby_platform_materializes_to_ruby_platform? end
#version (readonly)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 11
attr_reader :name, :version, :platform, :materialization
Instance Method Details
#==(other)
[ GitHub ]#candidate_platforms(locked_platforms: nil) (private)
Platforms to try in order of preference. Ruby platform is last since it requires compilation, but works when precompiled gems are incompatible. When a set of locked platforms is given (frozen mode), restrict the candidates to those, so we never materialize a platform that wasn't locked.
# File 'lib/bundler/lazy_specification.rb', line 207
def candidate_platforms(locked_platforms: nil) target = source.is_a?(Source::Path) ? platform : Bundler.local_platform platforms = [target, platform, Gem::Platform::RUBY].uniq return platforms unless locked_platforms platforms & locked_platforms end
#choose_compatible(candidates, fallback_to_non_installable: Bundler.frozen_bundle?) (private)
If in frozen mode, we fallback to a non-installable candidate because by doing this we avoid re-resolving and potentially end up changing the lockfile, which is not allowed. In that case, we will give a proper error about the mismatch higher up the stack, right before trying to install the bad gem.
# File 'lib/bundler/lazy_specification.rb', line 243
def choose_compatible(candidates, fallback_to_non_installable: Bundler.frozen_bundle?) override_list = overrides || [] search = candidates.reverse.find do |spec| spec.is_a?(StubSpecification) || spec.(override_list) end if search.nil? && fallback_to_non_installable search = candidates.last end if search validate_dependencies(search) if search.platform == platform search.locked_platform = platform if search.instance_of?(RemoteSpecification) || search.instance_of?(EndpointSpecification) end search end
#eql?(other) ⇒ Boolean
#find_compatible_platform_spec(specs, locked_platforms: nil) (private)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 194
def find_compatible_platform_spec(specs, locked_platforms: nil) candidate_platforms(locked_platforms: locked_platforms).each do |plat| candidates = MatchPlatform.select_best_platform_match(specs, plat) spec = choose_compatible(candidates, fallback_to_non_installable: false) return spec if spec end nil end
#force_ruby_platform!
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 169
def force_ruby_platform! @force_ruby_platform = true end
#frozen_bundle_fallback(specs) (private)
In frozen mode, accept any candidate. Will error at install time. When target differs from locked platform, prefer locked platform's candidates to preserve lockfile integrity.
# File 'lib/bundler/lazy_specification.rb', line 218
def frozen_bundle_fallback(specs) target = source.is_a?(Source::Path) ? platform : Bundler.local_platform fallback_platform = target == platform ? target : platform candidates = MatchPlatform.select_best_platform_match(specs, fallback_platform) choose_compatible(candidates) end
#full_name
[ GitHub ]#git_version
[ GitHub ]#hash
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 91
def hash full_name.hash end
#inspect
[ GitHub ]#lock_name
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 75
def lock_name @lock_name ||= name_tuple.lock_name end
#materialize(query) {|matching_specs| ... } (private)
# File 'lib/bundler/lazy_specification.rb', line 231
def materialize(query) matching_specs = source.specs.search(query) return self if matching_specs.empty? yield matching_specs end
#materialize_for_cache
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 127
def materialize_for_cache source.remote! materialize(self, &:first) end
#materialize_for_installation(locked_platforms)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 139
def materialize_for_installation(locked_platforms) source.local! if use_exact_resolved_specifications? spec = materialize(self) {|specs| choose_compatible(specs, fallback_to_non_installable: false) } return spec if spec # Exact spec is incompatible; in frozen mode, try to find a compatible platform variant # In non-frozen mode, return nil to trigger re-resolution and lockfile update if Bundler.frozen_bundle? materialize([name, version]) {|specs| resolve_best_platform(specs, locked_platforms: locked_platforms) } end else materialize([name, version]) {|specs| resolve_best_platform(specs) } end end
#materialized_for_installation(locked_platforms = nil)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 133
def materialized_for_installation(locked_platforms = nil) @materialization = materialize_for_installation(locked_platforms) self unless incomplete? end
#name_tuple
[ GitHub ]#replace_source_with!(gemfile_source)
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 173
def replace_source_with!(gemfile_source) return unless gemfile_source.can_lock?(self) @source = gemfile_source true end
#resolve_best_platform(specs, locked_platforms: nil) (private)
Try platforms in order of preference until finding a compatible spec. Used for legacy lockfiles and as a fallback when the exact locked spec is incompatible. Falls back to frozen bundle behavior if none match.
# File 'lib/bundler/lazy_specification.rb', line 190
def resolve_best_platform(specs, locked_platforms: nil) find_compatible_platform_spec(specs, locked_platforms: locked_platforms) || frozen_bundle_fallback(specs) end
#satisfies?(dependency) ⇒ Boolean
Does this locked specification satisfy dependency?
NOTE: Rubygems default requirement is ">= 0", which doesn't match prereleases of 0 versions, like "0.0.0.dev" or "0.0.0.SNAPSHOT". However, bundler users expect those to work. We need to make sure that Gemfile dependencies without explicit requirements (which use ">= 0" under the hood by default) are still valid for locked specs using this kind of versions. The method implements an ad-hoc fix for that. A better solution might be to change default rubygems requirement of dependencies to be ">= 0.A" but that's a major refactoring likely to break things. Hopefully we can attempt it in the future.
# File 'lib/bundler/lazy_specification.rb', line 109
def satisfies?(dependency) effective_requirement = dependency.requirement == Gem::Requirement.default ? Gem::Requirement.new(">= 0.A") : dependency.requirement @name == dependency.name && effective_requirement.satisfied_by?(Gem::Version.new(@version)) end
#to_lock
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 115
def to_lock out = String.new out << " #{lock_name}\n" dependencies.sort_by(&:to_s).uniq.each do |dep| next if dep.type == :development out << " #{dep.to_lock}\n" end out end
#to_s
[ GitHub ]# File 'lib/bundler/lazy_specification.rb', line 160
def to_s lock_name end
#validate_dependencies(spec) (private)
Validate dependencies of this locked spec are consistent with dependencies of the actual spec that was materialized.
Note that unless we are in strict mode (which we set during installation)
we don't validate dependencies of locally installed gems but
accept what's in the lockfile instead for performance, since loading
dependencies of locally installed gems would mean evaluating all gemspecs,
which would affect bundler/setup performance.
# File 'lib/bundler/lazy_specification.rb', line 268
def validate_dependencies(spec) if !@materialization_options[:strict] && spec.is_a?(StubSpecification) spec.dependencies = dependencies else if !source.is_a?(Source::Path) && spec.runtime_dependencies.sort != dependencies.sort raise IncorrectLockfileDependencies.new(self, spec.runtime_dependencies, dependencies) end end end