Module: Zeitwerk::Loader::Callbacks
Instance Method Summary collapse
-
#on_dir_autoloaded(dir) ⇒ Object
Invoked from our decorated Kernel#require when a managed directory is autoloaded.
-
#on_file_autoloaded(file) ⇒ Object
Invoked from our decorated Kernel#require when a managed file is autoloaded.
-
#on_namespace_loaded(namespace) ⇒ Object
Invoked when a class or module is created or reopened, either from the tracer or from module autovivification.
Methods included from RealModName
Instance Method Details
#on_dir_autoloaded(dir) ⇒ Object
Invoked from our decorated Kernel#require when a managed directory is autoloaded.
34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 |
# File 'lib/zeitwerk/loader/callbacks.rb', line 34 def on_dir_autoloaded(dir) # Module#autoload does not serialize concurrent requires, and we handle # directories ourselves, so the callback needs to account for concurrency. # # Multi-threading would introduce a race condition here in which thread t1 # autovivifies the module, and while autoloads for its children are being # set, thread t2 autoloads the same namespace. # # Without the mutex and subsequent delete call, t2 would reset the module. # That not only would reassign the constant (undesirable per se) but, worse, # the module object created by t2 wouldn't have any of the autoloads for its # children, since t1 would have correctly deleted its namespace_dirs entry. mutex2.synchronize do if cref = autoloads.delete(dir) autovivified_module = cref[0].const_set(cref[1], Module.new) cpath = autovivified_module.name log("module #{cpath} autovivified from directory #{dir}") if logger to_unload[cpath] = [dir, cref] if reloading_enabled? # We don't unregister `dir` in the registry because concurrent threads # wouldn't find a loader associated to it in Kernel#require and would # try to require the directory. Instead, we are going to keep track of # these to be able to unregister later if eager loading. autoloaded_dirs << dir on_namespace_loaded(autovivified_module) run_on_load_callbacks(cpath, autovivified_module, dir) unless on_load_callbacks.empty? end end end |
#on_file_autoloaded(file) ⇒ Object
Invoked from our decorated Kernel#require when a managed file is autoloaded.
10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 |
# File 'lib/zeitwerk/loader/callbacks.rb', line 10 def on_file_autoloaded(file) cref = autoloads.delete(file) cpath = cpath(*cref) Zeitwerk::Registry.unregister_autoload(file) if cdef?(*cref) log("constant #{cpath} loaded from file #{file}") if logger to_unload[cpath] = [file, cref] if reloading_enabled? run_on_load_callbacks(cpath, cget(*cref), file) unless on_load_callbacks.empty? else msg = "expected file #{file} to define constant #{cpath}, but didn't" log(msg) if logger crem(*cref) to_unload[cpath] = [file, cref] if reloading_enabled? raise Zeitwerk::NameError.new(msg, cref.last) end end |
#on_namespace_loaded(namespace) ⇒ Object
Invoked when a class or module is created or reopened, either from the tracer or from module autovivification. If the namespace has matching subdirectories, we descend into them now.
73 74 75 76 77 78 79 |
# File 'lib/zeitwerk/loader/callbacks.rb', line 73 def on_namespace_loaded(namespace) if dirs = namespace_dirs.delete(real_mod_name(namespace)) dirs.each do |dir| set_autoloads_in_dir(dir, namespace) end end end |