Class: Bundler::RubygemsIntegration
- Inherits:
-
Object
- Object
- Bundler::RubygemsIntegration
- Defined in:
- lib/bundler/rubygems_integration.rb
Constant Summary collapse
- EXT_LOCK =
Monitor.new
Instance Method Summary collapse
-
#add_default_gems_to(specs) ⇒ Object
Add default gems not already present in specs, and return them as a hash.
- #all_specs ⇒ Object
- #bin_path(gem, bin, ver) ⇒ Object
- #build(spec, skip_validation = false) ⇒ Object
- #build_args ⇒ Object
- #build_args=(args) ⇒ Object
- #build_gem(gem_dir, spec) ⇒ Object
- #clear_paths ⇒ Object
- #default_specs ⇒ Object
- #default_stubs ⇒ Object
- #download_gem(spec, uri, cache_dir, fetcher) ⇒ Object
- #ext_lock ⇒ Object
- #fetch_all_remote_specs(remote, gem_remote_fetcher) ⇒ Object
- #fetch_specs(remote, name, fetcher) ⇒ Object
- #find_bundler(version) ⇒ Object
- #find_name(name) ⇒ Object
- #gem_bindir ⇒ Object
- #gem_cache ⇒ Object
- #gem_dir ⇒ Object
- #gem_path ⇒ Object
- #inflate(obj) ⇒ Object
-
#initialize ⇒ RubygemsIntegration
constructor
A new instance of RubygemsIntegration.
- #installed_specs ⇒ Object
- #load_env_plugins ⇒ Object
- #load_plugin_files(plugin_files) ⇒ Object
- #load_plugins ⇒ Object
- #loaded_gem_paths ⇒ Object
- #loaded_specs(name) ⇒ Object
- #mark_loaded(spec) ⇒ Object
- #marshal_spec_dir ⇒ Object
- #method_visibility(klass, method) ⇒ Object
- #path(obj) ⇒ Object
- #path_separator ⇒ Object
- #plain_specs ⇒ Object
- #plain_specs=(specs) ⇒ Object
- #post_reset_hooks ⇒ Object
- #provides?(req_str) ⇒ Boolean
- #read_binary(path) ⇒ Object
- #redefine_method(klass, method, unbound_method = nil, &block) ⇒ Object
-
#replace_bin_path(specs_by_name) ⇒ Object
Used to give better error messages when activating specs outside of the current bundle.
-
#replace_entrypoints(specs) ⇒ Object
Replace or hook into RubyGems to provide a bundlerized view of the world.
- #replace_gem(specs, specs_by_name) ⇒ Object
- #reset ⇒ Object
- #reverse_rubygems_kernel_mixin ⇒ Object
- #ruby_engine ⇒ Object
- #security_policies ⇒ Object
- #security_policy_keys ⇒ Object
- #set_target_rbconfig(path) ⇒ Object
- #spec_cache_dirs ⇒ Object
- #spec_from_gem(path) ⇒ Object
- #stub_rubygems(specs) ⇒ Object
- #stub_set_spec(stub, spec) ⇒ Object
- #suffix_pattern ⇒ Object
- #ui=(obj) ⇒ Object
- #undo_replacements ⇒ Object
- #user_home ⇒ Object
- #validate(spec) ⇒ Object
- #version ⇒ Object
Constructor Details
#initialize ⇒ RubygemsIntegration
Returns a new instance of RubygemsIntegration.
11 12 13 |
# File 'lib/bundler/rubygems_integration.rb', line 11 def initialize @replaced_methods = {} end |
Instance Method Details
#add_default_gems_to(specs) ⇒ Object
Add default gems not already present in specs, and return them as a hash.
297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 |
# File 'lib/bundler/rubygems_integration.rb', line 297 def add_default_gems_to(specs) specs_by_name = specs.reduce({}) do |h, s| h[s.name] = s h end Bundler.rubygems.default_stubs.each do |stub| default_spec = stub.to_spec default_spec_name = default_spec.name next if specs_by_name.key?(default_spec_name) specs << default_spec specs_by_name[default_spec_name] = default_spec end specs_by_name end |
#all_specs ⇒ Object
432 433 434 435 436 437 438 |
# File 'lib/bundler/rubygems_integration.rb', line 432 def all_specs SharedHelpers.major_deprecation 2, "Bundler.rubygems.all_specs has been removed in favor of Bundler.rubygems.installed_specs" Gem::Specification.stubs.map do |stub| StubSpecification.from_stub(stub) end end |
#bin_path(gem, bin, ver) ⇒ Object
128 129 130 |
# File 'lib/bundler/rubygems_integration.rb', line 128 def bin_path(gem, bin, ver) Gem.bin_path(gem, bin, ver) end |
#build(spec, skip_validation = false) ⇒ Object
423 424 425 426 |
# File 'lib/bundler/rubygems_integration.rb', line 423 def build(spec, skip_validation = false) require "rubygems/package" Gem::Package.build(spec, skip_validation) end |
#build_args ⇒ Object
23 24 25 26 |
# File 'lib/bundler/rubygems_integration.rb', line 23 def build_args require "rubygems/command" Gem::Command.build_args end |
#build_args=(args) ⇒ Object
28 29 30 31 |
# File 'lib/bundler/rubygems_integration.rb', line 28 def build_args=(args) require "rubygems/command" Gem::Command.build_args = args end |
#build_gem(gem_dir, spec) ⇒ Object
162 163 164 |
# File 'lib/bundler/rubygems_integration.rb', line 162 def build_gem(gem_dir, spec) build(spec) end |
#clear_paths ⇒ Object
124 125 126 |
# File 'lib/bundler/rubygems_integration.rb', line 124 def clear_paths Gem.clear_paths end |
#default_specs ⇒ Object
446 447 448 449 450 |
# File 'lib/bundler/rubygems_integration.rb', line 446 def default_specs Gem::Specification.default_stubs.map do |stub| StubSpecification.from_stub(stub) end end |
#default_stubs ⇒ Object
460 461 462 |
# File 'lib/bundler/rubygems_integration.rb', line 460 def default_stubs Gem::Specification.default_stubs("*.gemspec") end |
#download_gem(spec, uri, cache_dir, fetcher) ⇒ Object
393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 |
# File 'lib/bundler/rubygems_integration.rb', line 393 def download_gem(spec, uri, cache_dir, fetcher) require "rubygems/remote_fetcher" uri = Bundler.settings.mirror_for(uri) redacted_uri = Gem::Uri.redact(uri) Bundler::Retry.new("download gem from #{redacted_uri}").attempts do gem_file_name = spec.file_name local_gem_path = File.join cache_dir, gem_file_name return if File.exist? local_gem_path begin remote_gem_path = uri + "gems/#{gem_file_name}" SharedHelpers.filesystem_access(local_gem_path) do fetcher.cache_update_path remote_gem_path, local_gem_path end rescue Gem::RemoteFetcher::FetchError raise if spec.original_platform == spec.platform original_gem_file_name = "#{spec.original_name}.gem" raise if gem_file_name == original_gem_file_name gem_file_name = original_gem_file_name retry end end rescue Gem::RemoteFetcher::FetchError => e raise Bundler::HTTPError, "Could not download gem from #{redacted_uri} due to underlying error <#{e.}>" end |
#ext_lock ⇒ Object
153 154 155 |
# File 'lib/bundler/rubygems_integration.rb', line 153 def ext_lock EXT_LOCK end |
#fetch_all_remote_specs(remote, gem_remote_fetcher) ⇒ Object
386 387 388 389 390 391 |
# File 'lib/bundler/rubygems_integration.rb', line 386 def fetch_all_remote_specs(remote, gem_remote_fetcher) specs = fetch_specs(remote, "specs", gem_remote_fetcher) pres = fetch_specs(remote, "prerelease_specs", gem_remote_fetcher) || [] specs.concat(pres) end |
#fetch_specs(remote, name, fetcher) ⇒ Object
374 375 376 377 378 379 380 381 382 383 384 |
# File 'lib/bundler/rubygems_integration.rb', line 374 def fetch_specs(remote, name, fetcher) require "rubygems/remote_fetcher" path = remote.uri.to_s + "#{name}.#{Gem.marshal_version}.gz" string = fetcher.fetch_path(path) specs = Bundler.safe_load_marshal(string) raise MarshalError, "Specs #{name} from #{remote} is expected to be an Array but was unexpected class #{specs.class}" unless specs.is_a?(Array) specs rescue Gem::RemoteFetcher::FetchError # it's okay for prerelease to fail raise unless name == "prerelease_specs" end |
#find_bundler(version) ⇒ Object
452 453 454 |
# File 'lib/bundler/rubygems_integration.rb', line 452 def find_bundler(version) find_name("bundler").find {|s| s.version.to_s == version } end |
#find_name(name) ⇒ Object
456 457 458 |
# File 'lib/bundler/rubygems_integration.rb', line 456 def find_name(name) Gem::Specification.stubs_for(name).map(&:to_spec) end |
#gem_bindir ⇒ Object
84 85 86 |
# File 'lib/bundler/rubygems_integration.rb', line 84 def gem_bindir Gem.bindir end |
#gem_cache ⇒ Object
108 109 110 |
# File 'lib/bundler/rubygems_integration.rb', line 108 def gem_cache gem_path.map {|p| File.("cache", p) } end |
#gem_dir ⇒ Object
80 81 82 |
# File 'lib/bundler/rubygems_integration.rb', line 80 def gem_dir Gem.dir end |
#gem_path ⇒ Object
92 93 94 |
# File 'lib/bundler/rubygems_integration.rb', line 92 def gem_path Gem.path end |
#inflate(obj) ⇒ Object
76 77 78 |
# File 'lib/bundler/rubygems_integration.rb', line 76 def inflate(obj) Gem::Util.inflate(obj) end |
#installed_specs ⇒ Object
440 441 442 443 444 |
# File 'lib/bundler/rubygems_integration.rb', line 440 def installed_specs Gem::Specification.stubs.reject(&:default_gem?).map do |stub| StubSpecification.from_stub(stub) end end |
#load_env_plugins ⇒ Object
145 146 147 |
# File 'lib/bundler/rubygems_integration.rb', line 145 def load_env_plugins Gem.load_env_plugins end |
#load_plugin_files(plugin_files) ⇒ Object
141 142 143 |
# File 'lib/bundler/rubygems_integration.rb', line 141 def load_plugin_files(plugin_files) Gem.load_plugin_files(plugin_files) end |
#load_plugins ⇒ Object
137 138 139 |
# File 'lib/bundler/rubygems_integration.rb', line 137 def load_plugins Gem.load_plugins end |
#loaded_gem_paths ⇒ Object
132 133 134 135 |
# File 'lib/bundler/rubygems_integration.rb', line 132 def loaded_gem_paths loaded_gem_paths = Gem.loaded_specs.map {|_, s| s.full_require_paths } loaded_gem_paths.flatten end |
#loaded_specs(name) ⇒ Object
37 38 39 |
# File 'lib/bundler/rubygems_integration.rb', line 37 def loaded_specs(name) Gem.loaded_specs[name] end |
#mark_loaded(spec) ⇒ Object
41 42 43 44 45 46 47 48 |
# File 'lib/bundler/rubygems_integration.rb', line 41 def mark_loaded(spec) if spec.respond_to?(:activated=) current = Gem.loaded_specs[spec.name] current.activated = false if current spec.activated = true end Gem.loaded_specs[spec.name] = spec end |
#marshal_spec_dir ⇒ Object
120 121 122 |
# File 'lib/bundler/rubygems_integration.rb', line 120 def marshal_spec_dir Gem::MARSHAL_SPEC_DIR end |
#method_visibility(klass, method) ⇒ Object
344 345 346 347 348 349 350 351 352 |
# File 'lib/bundler/rubygems_integration.rb', line 344 def method_visibility(klass, method) if klass.private_method_defined?(method) :private elsif klass.protected_method_defined?(method) :protected else :public end end |
#path(obj) ⇒ Object
64 65 66 |
# File 'lib/bundler/rubygems_integration.rb', line 64 def path(obj) obj.to_s end |
#path_separator ⇒ Object
428 429 430 |
# File 'lib/bundler/rubygems_integration.rb', line 428 def path_separator Gem.path_separator end |
#plain_specs ⇒ Object
366 367 368 |
# File 'lib/bundler/rubygems_integration.rb', line 366 def plain_specs Gem::Specification._all end |
#plain_specs=(specs) ⇒ Object
370 371 372 |
# File 'lib/bundler/rubygems_integration.rb', line 370 def plain_specs=(specs) Gem::Specification.all = specs end |
#post_reset_hooks ⇒ Object
100 101 102 |
# File 'lib/bundler/rubygems_integration.rb', line 100 def post_reset_hooks Gem.post_reset_hooks end |
#provides?(req_str) ⇒ Boolean
19 20 21 |
# File 'lib/bundler/rubygems_integration.rb', line 19 def provides?(req_str) Gem::Requirement.new(req_str).satisfied_by?(version) end |
#read_binary(path) ⇒ Object
72 73 74 |
# File 'lib/bundler/rubygems_integration.rb', line 72 def read_binary(path) Gem.read_binary(path) end |
#redefine_method(klass, method, unbound_method = nil, &block) ⇒ Object
323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 |
# File 'lib/bundler/rubygems_integration.rb', line 323 def redefine_method(klass, method, unbound_method = nil, &block) visibility = method_visibility(klass, method) begin if (instance_method = klass.instance_method(method)) && method != :initialize # doing this to ensure we also get private methods klass.send(:remove_method, method) end rescue NameError # method isn't defined nil end @replaced_methods[[method, klass]] = instance_method if unbound_method klass.send(:define_method, method, unbound_method) klass.send(visibility, method) elsif block klass.send(:define_method, method, &block) klass.send(visibility, method) end end |
#replace_bin_path(specs_by_name) ⇒ Object
Used to give better error messages when activating specs outside of the current bundle
236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 |
# File 'lib/bundler/rubygems_integration.rb', line 236 def replace_bin_path(specs_by_name) gem_class = (class << Gem; self; end) redefine_method(gem_class, :find_spec_for_exe) do |gem_name, *args| exec_name = args.first raise ArgumentError, "you must supply exec_name" unless exec_name spec_with_name = specs_by_name[gem_name] matching_specs_by_exec_name = specs_by_name.values.select {|s| s.executables.include?(exec_name) } spec = matching_specs_by_exec_name.delete(spec_with_name) unless spec || !matching_specs_by_exec_name.empty? = "can't find executable #{exec_name} for gem #{gem_name}" if spec_with_name.nil? += ". #{gem_name} is not currently included in the bundle, " \ "perhaps you meant to add it to your #{Bundler.default_gemfile.basename}?" end raise Gem::Exception, end unless spec spec = matching_specs_by_exec_name.shift warn \ "Bundler is using a binstub that was created for a different gem (#{spec.name}).\n" \ "You should run `bundle binstub #{gem_name}` " \ "to work around a system/bundle conflict." end unless matching_specs_by_exec_name.empty? conflicting_names = matching_specs_by_exec_name.map(&:name).join(", ") warn \ "The `#{exec_name}` executable in the `#{spec.name}` gem is being loaded, but it's also present in other gems (#{conflicting_names}).\n" \ "If you meant to run the executable for another gem, make sure you use a project specific binstub (`bundle binstub <gem_name>`).\n" \ "If you plan to use multiple conflicting executables, generate binstubs for them and disambiguate their names." end spec end end |
#replace_entrypoints(specs) ⇒ Object
Replace or hook into RubyGems to provide a bundlerized view of the world.
278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 |
# File 'lib/bundler/rubygems_integration.rb', line 278 def replace_entrypoints(specs) specs_by_name = add_default_gems_to(specs) reverse_rubygems_kernel_mixin begin # bundled_gems only provide with Ruby 3.3 or later require "bundled_gems" rescue LoadError else Gem::BUNDLED_GEMS.replace_require(specs) if Gem::BUNDLED_GEMS.respond_to?(:replace_require) end replace_gem(specs, specs_by_name) stub_rubygems(specs_by_name.values) replace_bin_path(specs_by_name) Gem.clear_paths end |
#replace_gem(specs, specs_by_name) ⇒ Object
192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 |
# File 'lib/bundler/rubygems_integration.rb', line 192 def replace_gem(specs, specs_by_name) executables = nil [::Kernel.singleton_class, ::Kernel].each do |kernel_class| redefine_method(kernel_class, :gem) do |dep, *reqs| if executables&.include?(File.basename(caller_locations(1, 1).first.path)) break end reqs.pop if reqs.last.is_a?(Hash) unless dep.respond_to?(:name) && dep.respond_to?(:requirement) dep = Gem::Dependency.new(dep, reqs) end if spec = specs_by_name[dep.name] return true if dep.matches_spec?(spec) end = if spec.nil? target_file = begin Bundler.default_gemfile.basename rescue GemfileNotFound "inline Gemfile" end "#{dep.name} is not part of the bundle." \ " Add it to your #{target_file}." else "can't activate #{dep}, already activated #{spec.full_name}. " \ "Make sure all dependencies are added to Gemfile." end e = Gem::LoadError.new() e.name = dep.name e.requirement = dep.requirement raise e end # backwards compatibility shim, see https://github.com/rubygems/bundler/issues/5102 kernel_class.send(:public, :gem) if Bundler.feature_flag.setup_makes_kernel_gem_public? end end |
#reset ⇒ Object
96 97 98 |
# File 'lib/bundler/rubygems_integration.rb', line 96 def reset Gem::Specification.reset end |
#reverse_rubygems_kernel_mixin ⇒ Object
179 180 181 182 183 184 185 186 187 188 189 190 |
# File 'lib/bundler/rubygems_integration.rb', line 179 def reverse_rubygems_kernel_mixin # Disable rubygems' gem activation system if Gem.respond_to?(:discover_gems_on_require=) Gem.discover_gems_on_require = false else [::Kernel.singleton_class, ::Kernel].each do |k| if k.private_method_defined?(:gem_original_require) redefine_method(k, :require, k.instance_method(:gem_original_require)) end end end end |
#ruby_engine ⇒ Object
68 69 70 |
# File 'lib/bundler/rubygems_integration.rb', line 68 def ruby_engine Gem.ruby_engine end |
#security_policies ⇒ Object
170 171 172 173 174 175 176 177 |
# File 'lib/bundler/rubygems_integration.rb', line 170 def security_policies @security_policies ||= begin require "rubygems/security" Gem::Security::Policies rescue LoadError, NameError {} end end |
#security_policy_keys ⇒ Object
166 167 168 |
# File 'lib/bundler/rubygems_integration.rb', line 166 def security_policy_keys %w[High Medium Low AlmostNo No].map {|level| "#{level}Security" } end |
#set_target_rbconfig(path) ⇒ Object
33 34 35 |
# File 'lib/bundler/rubygems_integration.rb', line 33 def set_target_rbconfig(path) Gem.set_target_rbconfig(path) end |
#spec_cache_dirs ⇒ Object
112 113 114 115 116 117 118 |
# File 'lib/bundler/rubygems_integration.rb', line 112 def spec_cache_dirs @spec_cache_dirs ||= begin dirs = gem_path.map {|dir| File.join(dir, "specifications") } dirs << Gem.spec_cache_dir dirs.uniq.select {|dir| File.directory? dir } end end |
#spec_from_gem(path) ⇒ Object
157 158 159 160 |
# File 'lib/bundler/rubygems_integration.rb', line 157 def spec_from_gem(path) require "rubygems/package" Gem::Package.new(path).spec end |
#stub_rubygems(specs) ⇒ Object
354 355 356 357 358 359 360 361 362 363 364 |
# File 'lib/bundler/rubygems_integration.rb', line 354 def stub_rubygems(specs) Gem::Specification.all = specs Gem.post_reset do Gem::Specification.all = specs end redefine_method((class << Gem; self; end), :finish_resolve) do |*| [] end end |
#stub_set_spec(stub, spec) ⇒ Object
60 61 62 |
# File 'lib/bundler/rubygems_integration.rb', line 60 def stub_set_spec(stub, spec) stub.instance_variable_set(:@spec, spec) end |
#suffix_pattern ⇒ Object
104 105 106 |
# File 'lib/bundler/rubygems_integration.rb', line 104 def suffix_pattern Gem.suffix_pattern end |
#ui=(obj) ⇒ Object
149 150 151 |
# File 'lib/bundler/rubygems_integration.rb', line 149 def ui=(obj) Gem::DefaultUserInteraction.ui = obj end |
#undo_replacements ⇒ Object
315 316 317 318 319 320 321 |
# File 'lib/bundler/rubygems_integration.rb', line 315 def undo_replacements @replaced_methods.each do |(sym, klass), method| redefine_method(klass, sym, method) end post_reset_hooks.reject! {|proc| proc.binding.source_location[0] == __FILE__ } @replaced_methods.clear end |
#user_home ⇒ Object
88 89 90 |
# File 'lib/bundler/rubygems_integration.rb', line 88 def user_home Gem.user_home end |
#validate(spec) ⇒ Object
50 51 52 53 54 55 56 57 58 |
# File 'lib/bundler/rubygems_integration.rb', line 50 def validate(spec) Bundler.ui.silence { spec.validate_for_resolution } rescue Gem::InvalidSpecificationException => e = "The gemspec at #{spec.loaded_from} is not valid. Please fix this gemspec.\n" \ "The validation error was '#{e.}'\n" raise Gem::InvalidSpecificationException.new() rescue Errno::ENOENT nil end |
#version ⇒ Object
15 16 17 |
# File 'lib/bundler/rubygems_integration.rb', line 15 def version @version ||= Gem.rubygems_version end |