Можно ли "выгрузить" ("не требовать") библиотеку Ruby?
Я смотрю, чтобы загрузить несколько библиотек, заставить их сделать некоторую работу, а затем сделать противоположное require
чтобы избежать ошибок совместимости позже. Я не хочу, чтобы сделать дамп в файл и перезапустить оболочку, как объекты, созданные (например, data
) могут быть хорошо обработаны другими моими библиотеками, но не в присутствии ранних библиотек, которые я ищу.
Кто-нибудь получил какие-либо предложения или знаете, возможно ли это? Разговор 2006 года не пришел к большому выводу, кроме того, что "похоже, Вебрику удается как-то это сделать".
Этими библиотеками являются Google_drive и Nokogiri (библиотека Roo для обработки электронных таблиц зависит от Google_drive для онлайн-чтения / записи электронных таблиц, как описано по этой ссылке).
3 ответа
Как сказал @Alex, вы можете использовать Kernel#fork
создать новый процесс ruby, где вы будете require
ваши библиотеки. Новый разветвленный процесс будет иметь доступ к данным, загруженным в родительский процесс:
def talk(msg)
# this will allow us to see which process is
# talking
puts "#{Process.pid}: #{msg}"
end
# this data was loaded on the parent process
# and will be use in the child (and in the parent)
this_is_data = ["a", "b", "c"]
talk "I'm the father process, and I see #{this_is_data}"
# this will create a new ruby process
fork{
talk "I'm another process, and I also see #{this_is_data}"
talk "But when I change `this_is_data`, a new copy of it is created"
this_is_data << "d"
talk "My own #{this_is_data}"
}
# let's wait and give a chance to the child process
# finishes before the parent
sleep 3
talk "Now, in the father again, data is: #{this_is_data}"
Результат этого выполнения зависит от вашей машины, Process.id
вернет разные значения, но это будет так:
23520: I'm the father process, and I see ["a", "b", "c"]
23551: I'm another process, and I also see ["a", "b", "c"]
23551: But when I change `this_is_data`, a new copy of it is created
23551: My own ["a", "b", "c", "d"]
23520: Now, in the father again, data is: ["a", "b", "c"]
И это хорошо! Каждый процесс создан fork
это процесс на уровне операционной системы, который выполняется в собственном пространстве памяти.
Еще одна вещь, которую вы можете сделать, чтобы как-то управлять глобальными переменными, созданными при загрузке файла, это заменить использование require
от load
, Этот подход не решает все проблемы, о которых уже говорилось, но действительно может помочь. Смотрите следующие характеристики:
require "minitest/autorun"
describe "Loading files inside a scope" do
def create_lib_file(version)
libfile = <<CODE
class MyLibrary#{version}
VERSION = "0.0.#{version}"
end
class String
def omg_danger!
end
end
puts "loaded \#{MyLibrary#{version}::VERSION}"
CODE
File.write("my_library.rb", libfile)
end
after do
File.delete("my_library.rb") if File.exists?("my_library.rb")
end
describe "loading with require" do
it "sees the MyLibrary definition" do
create_lib_file("1")
require_relative "my_library.rb"
MyLibrary1::VERSION.must_be :==, "0.0.1"
"".respond_to?(:omg_danger!).must_be :==, true
end
end
describe "loading with #load " do
describe "without wrapping" do
it "sees the MyLibrary definition" do
create_lib_file("2")
load "my_library.rb"
MyLibrary2::VERSION.must_be :==, "0.0.2"
"".respond_to?(:omg_danger!).must_be :==, true
end
end
describe "using anonymous module wraping" do
it "doesn't sees MyLibrary definition" do
create_lib_file("3")
load "my_library.rb", true
->{ MyLibrary3 }.must_raise NameError
"".respond_to?(:omg_danger!).must_be :==, false
end
end
end
end
И результат выполнения:
Run options: --seed 16453
# Running tests:
loaded 0.0.3
.loaded 0.0.2
.loaded 0.0.1
.
Finished tests in 0.004707s, 637.3486 tests/s, 1274.6973 assertions/s.
3 tests, 6 assertions, 0 failures, 0 errors, 0 skips
Несмотря на то, что обычно говорят, с помощью этого процесса можно отказаться от запроса / выгрузки пакетов.
- Предполагая, что требуемый файл хранится как
d:/foo.rb
с этим простым содержанием:
class Foo
end
- Поскольку любой класс, модуль или метод определяется в Ruby как константа, вы можете сначала отсоединить его:
irb(main):001:0> require 'd:/foo.rb'
=> true
irb(main):002:0> defined? Foo
=> "constant"
irb(main):003:0> Object.send(:remove_const, :Foo)
=> Foo
irb(main):004:0> defined? Foo
=> nil
- Уже требуемые / загруженные файлы записываются в глобальную переменную.
$"
, затем вам нужно очистить от него то, что вам уже требовалось:
irb(main):005:0> $".select{|r| r.include? 'foo.rb'}
=> ["d:/foo.rb"]
irb(main):006:0> $".delete('d:/foo.rb')
=> "d:/foo.rb"
irb(main):007:0> $".select{|r| r.include? 'foo.rb'}
=> []
- Теперь вы можете снова потребовать свой файл, и все будет обновлено и доступно.
irb(main):008:0> require 'd:/foo.rb'
=> true
irb(main):009:0> $".select{|r| r.include? 'foo.rb'}
=> ["d:/foo.rb"]
irb(main):010:0> defined? Foo
=> "constant"
irb(main):011:0> Foo.new
=> #<Foo:0x000000033ff8d8>
Я не знаю ни одного способа выгрузки файла, но вы можете сбросить выбранные вручную глобальные переменные в константы nil и undefine (что достаточно близко):
class Foo; end
Object.constants.include?(:Foo)
Object.send(:remove_const, :Foo)
Object.constants.include?(:Foo)
Foo # NameError: uninitialized constant Foo
В зависимости от ваших конфликтов вы также можете временно переименовать конфликтующие классы:
Bar = Foo
Object.send(:remove_const, :Foo)
do_stuff
Foo = Bar
К сожалению, пара характеристик Ruby закончились вашим желанием чисто "разгрузить" библиотеку. Во-первых, "загрузка" библиотеки Ruby может запускать произвольный код Ruby. Во-вторых, существующие константы и методы могут быть динамически переопределены в Ruby.
Если библиотека Ruby определяет только новые классы и модули, вы можете просто отменить их определение, как указано @Denis. Однако в этом случае "ошибки совместимости" очень маловероятны, даже если вы просто оставите их как есть. Если библиотека обезьяна вносит исправления в базовые классы Ruby, создает обработчики сигналов или устанавливает ловушки трассировки или at_exit
хуки, очень и очень трудно будет отследить все, что изменилось, и аккуратно отменить изменения.
Лучше всего сначала загрузить данные, а затем использовать что-то вроде Process#fork
раскошелиться на новую оболочку, а затем загрузить библиотеки. Когда вы закончите, убейте дочернюю оболочку и вернитесь к родителю. Ваши данные все еще будут там.
https://github.com/burke/zeus и https://github.com/jonleighton/spring используют аналогичные методы, чтобы избежать повторного ожидания загрузки Rails. Возможно, вы сможете адаптировать некоторые части их работы.