aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--activesupport/CHANGELOG2
-rw-r--r--activesupport/lib/active_support/dependencies.rb12
-rw-r--r--activesupport/test/dependencies_test.rb16
3 files changed, 29 insertions, 1 deletions
diff --git a/activesupport/CHANGELOG b/activesupport/CHANGELOG
index 90b476f667..485e8edc83 100644
--- a/activesupport/CHANGELOG
+++ b/activesupport/CHANGELOG
@@ -1,5 +1,7 @@
*SVN*
+* Stop using defined? in Dependencies.qualified_const_defined? since defined? may invoke const_missing. [Nicholas Seckar]
+
* Dependencies can autoload directories of nested classes. [Jeremy Kemper]
Example:
invoice.rb class Invoice
diff --git a/activesupport/lib/active_support/dependencies.rb b/activesupport/lib/active_support/dependencies.rb
index 6241a822a4..b8a8a9b630 100644
--- a/activesupport/lib/active_support/dependencies.rb
+++ b/activesupport/lib/active_support/dependencies.rb
@@ -96,7 +96,17 @@ module Dependencies #:nodoc:
def qualified_const_defined?(path)
raise NameError, "#{path.inspect} is not a valid constant name!" unless
/^(::)?([A-Z]\w*)(::[A-Z]\w*)*$/ =~ path
- Object.module_eval("defined?(#{path})", __FILE__, __LINE__)
+
+ names = path.split('::')
+ names.shift if names.first.empty?
+
+ # We can't use defined? because it will invoke const_missing for the parent
+ # of the name we are checking.
+ names.inject(Object) do |mod, name|
+ return false unless mod.const_defined? name
+ mod.const_get name
+ end
+ return true
end
# Given +path+ return an array of constant paths which would cause Dependencies
diff --git a/activesupport/test/dependencies_test.rb b/activesupport/test/dependencies_test.rb
index 92f3e63e1a..3d0ada47f1 100644
--- a/activesupport/test/dependencies_test.rb
+++ b/activesupport/test/dependencies_test.rb
@@ -1,5 +1,13 @@
require File.dirname(__FILE__) + '/abstract_unit'
+module ModuleWithMissing
+ mattr_accessor :missing_count
+ def self.const_missing(name)
+ self.missing_count += 1
+ name
+ end
+end
+
class DependenciesTest < Test::Unit::TestCase
def teardown
@@ -249,6 +257,14 @@ class DependenciesTest < Test::Unit::TestCase
assert Dependencies.qualified_const_defined?("::Test::Unit::TestCase")
end
+ def test_qualified_const_defined_should_not_call_method_missing
+ ModuleWithMissing.missing_count = 0
+ assert ! Dependencies.qualified_const_defined?("ModuleWithMissing::A")
+ assert_equal 0, ModuleWithMissing.missing_count
+ assert ! Dependencies.qualified_const_defined?("ModuleWithMissing::A::B")
+ assert_equal 0, ModuleWithMissing.missing_count
+ end
+
def test_autoloaded?
with_loading 'autoloading_fixtures' do
assert ! Dependencies.autoloaded?("ModuleFolder")