Hello all,
I’ve run into an issue regarding the use of const_missing. This issue
revolves around the fact that these two cases are supposed to behave
differently from each other:
irb(main):001:0> B = 10
irb(main):002:0> module A; B; end
=> 10
irb(main):003:0> A::B
NameError: uninitialized constant A::B
from (irb):3
However it seems impossible to detect which case we are in within the
const_missing method. A more detailed explanation follows:
Suppose we are lazy and avoid typing “require ‘my_module.rb’” by
defining a const_missing that performs that require when MyModule is
first referenced. Now, suppose we are writing my_class, and wish to
include MyModule. So, we put this code in my_class.rb:
class MyClass
include MyModule
…
end
Our const_missing handler is called, and it loads ‘my_module.rb’,
finds the now present MyModule constant, and returns it. Everything
is good.
However, let’s say we’re slightly misguided and do MyClass::MyModule.
Since this is not defined, our const_missing handler is once again
called. ‘my_module.rb’ is thus loaded again, (usually to no effect,)
and MyModule returned.
This is a pretty clear violation of the semantics of ::. Indeed, Ruby
will produce a warning regarding our misbehaved const_missing. That
said, there does not seem to be a way to behave correctly; from
inside the const_missing handler there is no way to tell which case
the constant is missing in.
One way to ‘fix’ this is to have our const_missing look in it’s
parent modules for the missing constant. If we find it there, we know
this is a case of A::B. However this only works if the missing
constant has been defined in one of the parents. Thus, we can still
load ::B using A::B, but afterwards A::B will fail.
So, sorry for rambling on. The real question is: Can const_missing
detect if we are in a “A::B” case rather than “module A; B; end” ?
Thanks for your time and any replies,
Regards,
Nicholas S.