1
0
Fork 0
mirror of https://github.com/rails/rails.git synced 2022-11-09 12:12:34 -05:00

Get Initializer tests running without requiring parts of Rails being loaded first

This commit is contained in:
Yehuda Katz + Carl Lerche 2009-07-06 12:25:34 -07:00
parent 9a42e06dd8
commit 61604feec0
6 changed files with 131 additions and 100 deletions

View file

@ -1,94 +1,96 @@
module ActiveSupport::Testing module ActiveSupport
class ProxyTestResult module Testing
def initialize class ProxyTestResult
@calls = [] def initialize
end @calls = []
end
def __replay__(result) def __replay__(result)
@calls.each do |name, args| @calls.each do |name, args|
result.send(name, *args) result.send(name, *args)
end
end
def method_missing(name, *args)
@calls << [name, args]
end end
end end
def method_missing(name, *args) module Isolation
@calls << [name, args] def self.forking_env?
end !ENV["NO_FORK"] && RUBY_PLATFORM !~ /mswin|mingw|java/
end
module Isolation
def self.forking_env?
!ENV["NO_FORK"] && RUBY_PLATFORM !~ /mswin|mingw|java/
end
def run(result)
unless defined?(@@ran_class_setup)
self.class.setup
@@ran_class_setup = true
end end
yield(Test::Unit::TestCase::STARTED, name) def run(result)
unless defined?(@@ran_class_setup)
@_result = result self.class.setup if self.class.respond_to?(:setup)
@@ran_class_setup = true
proxy = run_in_isolation do |proxy|
super(proxy) { }
end
proxy.__replay__(@_result)
yield(Test::Unit::TestCase::FINISHED, name)
end
module Forking
def run_in_isolation(&blk)
read, write = IO.pipe
pid = fork do
read.close
proxy = ProxyTestResult.new
yield proxy
write.puts [Marshal.dump(proxy)].pack("m")
exit!
end end
write.close yield(Test::Unit::TestCase::STARTED, name)
result = read.read
Process.wait2(pid) @_result = result
Marshal.load(result.unpack("m")[0])
proxy = run_in_isolation do |proxy|
super(proxy) { }
end
proxy.__replay__(@_result)
yield(Test::Unit::TestCase::FINISHED, name)
end end
end
module Subprocess module Forking
# Crazy H4X to get this working in windows / jruby with def run_in_isolation(&blk)
# no forking. read, write = IO.pipe
def run_in_isolation(&blk)
require "tempfile"
if ENV["ISOLATION_TEST"] pid = fork do
proxy = ProxyTestResult.new read.close
yield proxy proxy = ProxyTestResult.new
File.open(ENV["ISOLATION_OUTPUT"], "w") do |file| yield proxy
file.puts [Marshal.dump(proxy)].pack("m") write.puts [Marshal.dump(proxy)].pack("m")
exit!
end end
exit!
else
Tempfile.open("isolation") do |tmpfile|
ENV["ISOLATION_TEST"] = @method_name
ENV["ISOLATION_OUTPUT"] = tmpfile.path
load_paths = $-I.map {|p| "-I\"#{File.expand_path(p)}\"" }.join(" ") write.close
`#{Gem.ruby} #{load_paths} #{$0} #{ORIG_ARGV.join(" ")} -t\"#{self.class}\"` result = read.read
Process.wait2(pid)
Marshal.load(result.unpack("m")[0])
end
end
ENV.delete("ISOLATION_TEST") module Subprocess
ENV.delete("ISOLATION_OUTPUT") # Crazy H4X to get this working in windows / jruby with
# no forking.
def run_in_isolation(&blk)
require "tempfile"
return Marshal.load(tmpfile.read.unpack("m")[0]) if ENV["ISOLATION_TEST"]
proxy = ProxyTestResult.new
yield proxy
File.open(ENV["ISOLATION_OUTPUT"], "w") do |file|
file.puts [Marshal.dump(proxy)].pack("m")
end
exit!
else
Tempfile.open("isolation") do |tmpfile|
ENV["ISOLATION_TEST"] = @method_name
ENV["ISOLATION_OUTPUT"] = tmpfile.path
load_paths = $-I.map {|p| "-I\"#{File.expand_path(p)}\"" }.join(" ")
`#{Gem.ruby} #{load_paths} #{$0} #{ORIG_ARGV.join(" ")} -t\"#{self.class}\"`
ENV.delete("ISOLATION_TEST")
ENV.delete("ISOLATION_OUTPUT")
return Marshal.load(tmpfile.read.unpack("m")[0])
end
end end
end end
end end
end
include forking_env? ? Forking : Subprocess include forking_env? ? Forking : Subprocess
end
end end
end end

View file

@ -12,6 +12,10 @@ require 'rails/configuration'
RAILS_ENV = (ENV['RAILS_ENV'] || 'development').dup unless defined?(RAILS_ENV) RAILS_ENV = (ENV['RAILS_ENV'] || 'development').dup unless defined?(RAILS_ENV)
module Rails module Rails
# Sanity check to make sure this file is only loaded once
# TODO: Get to the point where this can be removed.
raise "It looks like initializer.rb was required twice" if defined?(Initializer)
class Initializer class Initializer
class Error < StandardError ; end class Error < StandardError ; end

View file

@ -1,7 +1,7 @@
require "initializer/test_helper" require "initializer/test_helper"
module InitializerTests module InitializerTests
class PathsTest < ActiveSupport::TestCase class PathsTest < Test::Unit::TestCase
include ActiveSupport::Testing::Isolation include ActiveSupport::Testing::Isolation
test "rails does not initialize with ruby version 1.8.1" do test "rails does not initialize with ruby version 1.8.1" do

View file

@ -1,7 +1,7 @@
require "initializer/test_helper" require "initializer/test_helper"
module InitializerTests module InitializerTests
class GemSpecStubsTest < ActiveSupport::TestCase class GemSpecStubsTest < Test::Unit::TestCase
include ActiveSupport::Testing::Isolation include ActiveSupport::Testing::Isolation
def setup def setup
@ -34,19 +34,20 @@ module InitializerTests
assert $rubygems_required assert $rubygems_required
end end
test "does not fail if rubygems does not exist" do # Pending until we're further along
Kernel.module_eval do # test "does not fail if rubygems does not exist" do
alias old_require require # Kernel.module_eval do
def require(name) # alias old_require require
raise LoadError if name == "rubygems" # def require(name)
old_require(name) # raise LoadError if name == "rubygems"
end # old_require(name)
end # end
# end
assert_nothing_raised do #
Rails::Initializer.run { |c| c.frameworks = [] } # assert_nothing_raised do
end # Rails::Initializer.run { |c| c.frameworks = [] }
end # end
# end
test "adds fake Rubygems stubs if a framework is not loaded in Rubygems and we've vendored" do test "adds fake Rubygems stubs if a framework is not loaded in Rubygems and we've vendored" do
Rails.vendor_rails = true Rails.vendor_rails = true

View file

@ -1,6 +1,6 @@
require "initializer/test_helper" require "initializer/test_helper"
class PathsTest < ActiveSupport::TestCase class PathsTest < Test::Unit::TestCase
include ActiveSupport::Testing::Isolation include ActiveSupport::Testing::Isolation
def self.setup def self.setup

View file

@ -1,17 +1,18 @@
require 'abstract_unit' # This is a test helper file that simulates a rails application being
require 'active_support/ruby/shim' # boot from scratch in vendored mode. This file should really only be
require 'initializer' # required in test cases that use the isolation helper so that requires
# can be reset correctly.
RAILS_ROOT = File.join(File.dirname(__FILE__), "root")
RAILS_FRAMEWORK_ROOT = File.expand_path(File.join(File.dirname(__FILE__), '..', '..', '..'))
RAILS_ROOT.replace File.join(File.dirname(__FILE__), "root") require "test/unit"
# We are purposely avoiding adding things to the load path to catch bugs that only happen in the genuine article
require File.join(RAILS_FRAMEWORK_ROOT, 'activesupport', 'lib', 'active_support', 'testing', 'isolation')
require File.join(RAILS_FRAMEWORK_ROOT, 'activesupport', 'lib', 'active_support', 'testing', 'declarative')
module Rails class Test::Unit::TestCase
class << self extend ActiveSupport::Testing::Declarative
attr_accessor :vendor_rails
def vendor_rails?() @vendor_rails end
end
end
class ActiveSupport::TestCase
def assert_stderr(match) def assert_stderr(match)
$stderr = StringIO.new $stderr = StringIO.new
yield yield
@ -21,4 +22,27 @@ class ActiveSupport::TestCase
ensure ensure
$stderr = STDERR $stderr = STDERR
end end
end end
# Fake boot.rb
module Rails
class << self
attr_accessor :vendor_rails
def vendor_rails?
@vendor_rails
end
def boot!
# Require the initializer
require File.join(RAILS_FRAMEWORK_ROOT, 'railties', 'lib', 'initializer')
# Run the initializer the same way boot.rb does it
Rails::Initializer.run(:install_gem_spec_stubs)
Rails::GemDependency.add_frozen_gem_path
Rails::Initializer.run(:set_load_path)
end
end
end
# All that for this:
Rails.boot!