chdir now can be nested inside itself, errors if the directory doesn't

exist and we can still use absolute paths inside it.
This commit is contained in:
Jeff Hodges 2009-05-30 20:21:44 -07:00
parent 35c88244de
commit b3bc7f6615
2 changed files with 61 additions and 5 deletions

View File

@ -152,11 +152,16 @@ module FakeFS
module FileSystem module FileSystem
extend self extend self
def dir_levels
@dir_levels
end
def fs def fs
@fs ||= MockDir.new('.') @fs ||= MockDir.new('.')
end end
def clear def clear
@dir_levels = []
@fs = nil @fs = nil
end end
@ -165,7 +170,7 @@ module FakeFS
end end
def find(path) def find(path)
parts = path_parts(path) parts = path_parts(normalize_path(path))
target = parts[0...-1].inject(fs) do |dir, part| target = parts[0...-1].inject(fs) do |dir, part|
dir[part] || {} dir[part] || {}
@ -180,7 +185,7 @@ module FakeFS
end end
def add(path, object=MockDir.new) def add(path, object=MockDir.new)
parts = path_parts(path) parts = path_parts(normalize_path(path))
d = parts[0...-1].inject(fs) do |dir, part| d = parts[0...-1].inject(fs) do |dir, part|
dir[part] ||= MockDir.new(part, dir) dir[part] ||= MockDir.new(part, dir)
@ -220,16 +225,27 @@ module FakeFS
def chdir(dir, &blk) def chdir(dir, &blk)
raise "you must pass in a block" unless blk raise "you must pass in a block" unless blk
@old_fs = @fs
@fs = find(dir) new_dir = find(dir)
raise Errno::ENOENT unless new_dir
dir_levels.push dir
blk.call blk.call
ensure ensure
@fs = @old_fs dir_levels.pop
end end
def path_parts(path) def path_parts(path)
path.split(File::PATH_SEPARATOR).reject { |part| part.empty? } path.split(File::PATH_SEPARATOR).reject { |part| part.empty? }
end end
def normalize_path(path)
if path[0,1] == '/'
File.expand_path(path)
else
parts = dir_levels + [path]
File.expand_path(File.join(*parts))
end
end
end end
class MockFile class MockFile

View File

@ -171,6 +171,46 @@ class FakeFSTest < Test::Unit::TestCase
end end
end end
def test_chdir_shouldnt_keep_us_from_absolute_paths
FileUtils.mkdir_p '/path'
Dir.chdir '/path' do
File.open('foo', 'w'){|f| f.write 'foo'}
File.open('/foobar', 'w'){|f| f.write 'foo'}
end
assert_equal ['foo'], FileSystem.fs['path'].keys.sort
assert_equal ['foobar', 'path'], FileSystem.fs.keys.sort
Dir.chdir '/path' do
FileUtils.rm('foo')
FileUtils.rm('/foobar')
end
assert_equal [], FileSystem.fs['path'].keys.sort
assert_equal ['path'], FileSystem.fs.keys.sort
end
def test_chdir_should_be_nestable
FileUtils.mkdir_p '/path/me'
Dir.chdir '/path' do
File.open('foo', 'w'){|f| f.write 'foo'}
Dir.chdir 'me' do
File.open('foobar', 'w'){|f| f.write 'foo'}
end
end
assert_equal ['foo','me'], FileSystem.fs['path'].keys.sort
assert_equal ['foobar'], FileSystem.fs['path']['me'].keys.sort
end
def test_chdir_should_flop_over_and_die_if_the_dir_doesnt_exist
assert_raise(Errno::ENOENT) do
Dir.chdir('/nope') do
1
end
end
end
def test_chdir_shouldnt_lose_state_because_of_errors def test_chdir_shouldnt_lose_state_because_of_errors
FileUtils.mkdir_p '/path' FileUtils.mkdir_p '/path'