зеркало из https://github.com/github/ruby.git
487 строки
16 KiB
Ruby
487 строки
16 KiB
Ruby
#!/usr/bin/env ruby
|
|
|
|
require 'test/unit'
|
|
require 'pathname'
|
|
|
|
require 'fileutils'
|
|
require 'tmpdir'
|
|
require 'enumerator'
|
|
|
|
class TestPathname < Test::Unit::TestCase
|
|
|
|
if RUBY_VERSION < "1.9"
|
|
FUNCALL = :__send__
|
|
else
|
|
FUNCALL = :funcall
|
|
end
|
|
|
|
def self.define_assertion(name, &block)
|
|
@defassert_num ||= {}
|
|
@defassert_num[name] ||= 0
|
|
@defassert_num[name] += 1
|
|
define_method("test_#{name}_#{@defassert_num[name]}", &block)
|
|
end
|
|
|
|
def self.defassert(name, result, *args)
|
|
define_assertion(name) {
|
|
assert_equal(result, self.send(name, *args), "#{name}(#{args.map {|a| a.inspect }.join(', ')})")
|
|
}
|
|
end
|
|
|
|
DOSISH = File::ALT_SEPARATOR != nil
|
|
DOSISH_DRIVE_LETTER = File.dirname("A:") == "A:."
|
|
DOSISH_UNC = File.dirname("//") == "//"
|
|
|
|
def cleanpath_aggressive(path)
|
|
Pathname.new(path).cleanpath.to_s
|
|
end
|
|
|
|
defassert(:cleanpath_aggressive, '/', '/')
|
|
defassert(:cleanpath_aggressive, '.', '')
|
|
defassert(:cleanpath_aggressive, '.', '.')
|
|
defassert(:cleanpath_aggressive, '..', '..')
|
|
defassert(:cleanpath_aggressive, 'a', 'a')
|
|
defassert(:cleanpath_aggressive, '/', '/.')
|
|
defassert(:cleanpath_aggressive, '/', '/..')
|
|
defassert(:cleanpath_aggressive, '/a', '/a')
|
|
defassert(:cleanpath_aggressive, '.', './')
|
|
defassert(:cleanpath_aggressive, '..', '../')
|
|
defassert(:cleanpath_aggressive, 'a', 'a/')
|
|
defassert(:cleanpath_aggressive, 'a/b', 'a//b')
|
|
defassert(:cleanpath_aggressive, 'a', 'a/.')
|
|
defassert(:cleanpath_aggressive, 'a', 'a/./')
|
|
defassert(:cleanpath_aggressive, '.', 'a/..')
|
|
defassert(:cleanpath_aggressive, '.', 'a/../')
|
|
defassert(:cleanpath_aggressive, '/a', '/a/.')
|
|
defassert(:cleanpath_aggressive, '..', './..')
|
|
defassert(:cleanpath_aggressive, '..', '../.')
|
|
defassert(:cleanpath_aggressive, '..', './../')
|
|
defassert(:cleanpath_aggressive, '..', '.././')
|
|
defassert(:cleanpath_aggressive, '/', '/./..')
|
|
defassert(:cleanpath_aggressive, '/', '/../.')
|
|
defassert(:cleanpath_aggressive, '/', '/./../')
|
|
defassert(:cleanpath_aggressive, '/', '/.././')
|
|
defassert(:cleanpath_aggressive, 'a/b/c', 'a/b/c')
|
|
defassert(:cleanpath_aggressive, 'b/c', './b/c')
|
|
defassert(:cleanpath_aggressive, 'a/c', 'a/./c')
|
|
defassert(:cleanpath_aggressive, 'a/b', 'a/b/.')
|
|
defassert(:cleanpath_aggressive, '.', 'a/../.')
|
|
defassert(:cleanpath_aggressive, '/a', '/../.././../a')
|
|
defassert(:cleanpath_aggressive, '../../d', 'a/b/../../../../c/../d')
|
|
|
|
if DOSISH_UNC
|
|
defassert(:cleanpath_aggressive, '//a/b/c', '//a/b/c/')
|
|
else
|
|
defassert(:cleanpath_aggressive, '/', '///')
|
|
defassert(:cleanpath_aggressive, '/a', '///a')
|
|
defassert(:cleanpath_aggressive, '/', '///..')
|
|
defassert(:cleanpath_aggressive, '/', '///.')
|
|
defassert(:cleanpath_aggressive, '/', '///a/../..')
|
|
end
|
|
|
|
def cleanpath_conservative(path)
|
|
Pathname.new(path).cleanpath(true).to_s
|
|
end
|
|
|
|
defassert(:cleanpath_conservative, '/', '/')
|
|
defassert(:cleanpath_conservative, '.', '')
|
|
defassert(:cleanpath_conservative, '.', '.')
|
|
defassert(:cleanpath_conservative, '..', '..')
|
|
defassert(:cleanpath_conservative, 'a', 'a')
|
|
defassert(:cleanpath_conservative, '/', '/.')
|
|
defassert(:cleanpath_conservative, '/', '/..')
|
|
defassert(:cleanpath_conservative, '/a', '/a')
|
|
defassert(:cleanpath_conservative, '.', './')
|
|
defassert(:cleanpath_conservative, '..', '../')
|
|
defassert(:cleanpath_conservative, 'a/', 'a/')
|
|
defassert(:cleanpath_conservative, 'a/b', 'a//b')
|
|
defassert(:cleanpath_conservative, 'a/.', 'a/.')
|
|
defassert(:cleanpath_conservative, 'a/.', 'a/./')
|
|
defassert(:cleanpath_conservative, 'a/..', 'a/../')
|
|
defassert(:cleanpath_conservative, '/a/.', '/a/.')
|
|
defassert(:cleanpath_conservative, '..', './..')
|
|
defassert(:cleanpath_conservative, '..', '../.')
|
|
defassert(:cleanpath_conservative, '..', './../')
|
|
defassert(:cleanpath_conservative, '..', '.././')
|
|
defassert(:cleanpath_conservative, '/', '/./..')
|
|
defassert(:cleanpath_conservative, '/', '/../.')
|
|
defassert(:cleanpath_conservative, '/', '/./../')
|
|
defassert(:cleanpath_conservative, '/', '/.././')
|
|
defassert(:cleanpath_conservative, 'a/b/c', 'a/b/c')
|
|
defassert(:cleanpath_conservative, 'b/c', './b/c')
|
|
defassert(:cleanpath_conservative, 'a/c', 'a/./c')
|
|
defassert(:cleanpath_conservative, 'a/b/.', 'a/b/.')
|
|
defassert(:cleanpath_conservative, 'a/..', 'a/../.')
|
|
defassert(:cleanpath_conservative, '/a', '/../.././../a')
|
|
defassert(:cleanpath_conservative, 'a/b/../../../../c/../d', 'a/b/../../../../c/../d')
|
|
|
|
if DOSISH_UNC
|
|
defassert(:cleanpath_conservative, '//', '//')
|
|
else
|
|
defassert(:cleanpath_conservative, '/', '//')
|
|
end
|
|
|
|
# has_trailing_separator?(path) -> bool
|
|
def has_trailing_separator?(path)
|
|
Pathname.allocate.send(FUNCALL, :has_trailing_separator?, path)
|
|
end
|
|
|
|
defassert(:has_trailing_separator?, false, "/")
|
|
defassert(:has_trailing_separator?, false, "///")
|
|
defassert(:has_trailing_separator?, false, "a")
|
|
defassert(:has_trailing_separator?, true, "a/")
|
|
|
|
def add_trailing_separator(path)
|
|
Pathname.allocate.send(FUNCALL, :add_trailing_separator, path)
|
|
end
|
|
|
|
def del_trailing_separator(path)
|
|
Pathname.allocate.send(FUNCALL, :del_trailing_separator, path)
|
|
end
|
|
|
|
defassert(:del_trailing_separator, "/", "/")
|
|
defassert(:del_trailing_separator, "/a", "/a")
|
|
defassert(:del_trailing_separator, "/a", "/a/")
|
|
defassert(:del_trailing_separator, "/a", "/a//")
|
|
defassert(:del_trailing_separator, ".", ".")
|
|
defassert(:del_trailing_separator, ".", "./")
|
|
defassert(:del_trailing_separator, ".", ".//")
|
|
|
|
if DOSISH_DRIVE_LETTER
|
|
defassert(:del_trailing_separator, "A:", "A:")
|
|
defassert(:del_trailing_separator, "A:/", "A:/")
|
|
defassert(:del_trailing_separator, "A:/", "A://")
|
|
defassert(:del_trailing_separator, "A:.", "A:.")
|
|
defassert(:del_trailing_separator, "A:.", "A:./")
|
|
defassert(:del_trailing_separator, "A:.", "A:.//")
|
|
end
|
|
|
|
if DOSISH_UNC
|
|
defassert(:del_trailing_separator, "//", "//")
|
|
defassert(:del_trailing_separator, "//a", "//a")
|
|
defassert(:del_trailing_separator, "//a", "//a/")
|
|
defassert(:del_trailing_separator, "//a", "//a//")
|
|
defassert(:del_trailing_separator, "//a/b", "//a/b")
|
|
defassert(:del_trailing_separator, "//a/b", "//a/b/")
|
|
defassert(:del_trailing_separator, "//a/b", "//a/b//")
|
|
defassert(:del_trailing_separator, "//a/b/c", "//a/b/c")
|
|
defassert(:del_trailing_separator, "//a/b/c", "//a/b/c/")
|
|
defassert(:del_trailing_separator, "//a/b/c", "//a/b/c//")
|
|
else
|
|
defassert(:del_trailing_separator, "/", "///")
|
|
defassert(:del_trailing_separator, "///a", "///a/")
|
|
end
|
|
|
|
if DOSISH
|
|
defassert(:del_trailing_separator, "a", "a\\")
|
|
defassert(:del_trailing_separator, "\225\\", "\225\\\\") # SJIS
|
|
end
|
|
|
|
def plus(path1, path2) # -> path
|
|
(Pathname.new(path1) + Pathname.new(path2)).to_s
|
|
end
|
|
|
|
defassert(:plus, '/', '/', '/')
|
|
defassert(:plus, 'a/b', 'a', 'b')
|
|
defassert(:plus, 'a', 'a', '.')
|
|
defassert(:plus, 'b', '.', 'b')
|
|
defassert(:plus, '.', '.', '.')
|
|
defassert(:plus, '/b', 'a', '/b')
|
|
|
|
defassert(:plus, '/', '/', '..')
|
|
defassert(:plus, '.', 'a', '..')
|
|
defassert(:plus, 'a', 'a/b', '..')
|
|
defassert(:plus, '../..', '..', '..')
|
|
defassert(:plus, '/c', '/', '../c')
|
|
defassert(:plus, 'c', 'a', '../c')
|
|
defassert(:plus, 'a/c', 'a/b', '../c')
|
|
defassert(:plus, '../../c', '..', '../c')
|
|
|
|
defassert(:plus, 'a//b/d//e', 'a//b/c', '../d//e')
|
|
|
|
def relative?(path)
|
|
Pathname.new(path).relative?
|
|
end
|
|
|
|
defassert(:relative?, false, '/')
|
|
defassert(:relative?, false, '/a')
|
|
defassert(:relative?, false, '/..')
|
|
defassert(:relative?, true, 'a')
|
|
defassert(:relative?, true, 'a/b')
|
|
|
|
if DOSISH_DRIVE_LETTER
|
|
defassert(:relative?, false, 'A:')
|
|
defassert(:relative?, false, 'A:/')
|
|
defassert(:relative?, false, 'A:/a')
|
|
end
|
|
|
|
if File.dirname('//') == '//'
|
|
defassert(:relative?, false, '//')
|
|
defassert(:relative?, false, '//a')
|
|
defassert(:relative?, false, '//a/')
|
|
defassert(:relative?, false, '//a/b')
|
|
defassert(:relative?, false, '//a/b/')
|
|
defassert(:relative?, false, '//a/b/c')
|
|
end
|
|
|
|
def relative_path_from(dest_directory, base_directory)
|
|
Pathname.new(dest_directory).relative_path_from(Pathname.new(base_directory)).to_s
|
|
end
|
|
|
|
defassert(:relative_path_from, "../a", "a", "b")
|
|
defassert(:relative_path_from, "../a", "a", "b/")
|
|
defassert(:relative_path_from, "../a", "a/", "b")
|
|
defassert(:relative_path_from, "../a", "a/", "b/")
|
|
defassert(:relative_path_from, "../a", "/a", "/b")
|
|
defassert(:relative_path_from, "../a", "/a", "/b/")
|
|
defassert(:relative_path_from, "../a", "/a/", "/b")
|
|
defassert(:relative_path_from, "../a", "/a/", "/b/")
|
|
|
|
defassert(:relative_path_from, "../b", "a/b", "a/c")
|
|
defassert(:relative_path_from, "../a", "../a", "../b")
|
|
|
|
defassert(:relative_path_from, "a", "a", ".")
|
|
defassert(:relative_path_from, "..", ".", "a")
|
|
|
|
defassert(:relative_path_from, ".", ".", ".")
|
|
defassert(:relative_path_from, ".", "..", "..")
|
|
defassert(:relative_path_from, "..", "..", ".")
|
|
|
|
defassert(:relative_path_from, "c/d", "/a/b/c/d", "/a/b")
|
|
defassert(:relative_path_from, "../..", "/a/b", "/a/b/c/d")
|
|
defassert(:relative_path_from, "../../../../e", "/e", "/a/b/c/d")
|
|
defassert(:relative_path_from, "../b/c", "a/b/c", "a/d")
|
|
|
|
defassert(:relative_path_from, "../a", "/../a", "/b")
|
|
defassert(:relative_path_from, "../../a", "../a", "b")
|
|
defassert(:relative_path_from, ".", "/a/../../b", "/b")
|
|
defassert(:relative_path_from, "..", "a/..", "a")
|
|
defassert(:relative_path_from, ".", "a/../b", "b")
|
|
|
|
defassert(:relative_path_from, "a", "a", "b/..")
|
|
defassert(:relative_path_from, "b/c", "b/c", "b/..")
|
|
|
|
def self.defassert_raise(name, exc, *args)
|
|
define_assertion(name) {
|
|
message = "#{name}(#{args.map {|a| a.inspect }.join(', ')})"
|
|
assert_raise(exc, message) { self.send(name, *args) }
|
|
}
|
|
end
|
|
|
|
defassert_raise(:relative_path_from, ArgumentError, "/", ".")
|
|
defassert_raise(:relative_path_from, ArgumentError, ".", "/")
|
|
defassert_raise(:relative_path_from, ArgumentError, "a", "..")
|
|
defassert_raise(:relative_path_from, ArgumentError, ".", "..")
|
|
|
|
def realpath(path)
|
|
Pathname.new(path).realpath.to_s
|
|
end
|
|
|
|
def test_realpath
|
|
begin
|
|
File.symlink(nil, nil)
|
|
rescue NotImplementedError
|
|
return
|
|
rescue TypeError
|
|
end
|
|
dir = "#{Dir.tmpdir}/tst-pathname-#$$"
|
|
Dir.mkdir(dir)
|
|
begin
|
|
File.symlink("not-exist-target", "#{dir}/not-exist")
|
|
assert_raise(Errno::ENOENT) { realpath("#{dir}/not-exist") }
|
|
File.symlink("loop", "#{dir}/loop")
|
|
assert_raise(Errno::ELOOP) { realpath("#{dir}/loop") }
|
|
ensure
|
|
FileUtils.rmtree(dir)
|
|
end
|
|
end
|
|
|
|
def descend(path)
|
|
Pathname.new(path).enum_for(:descend).map {|v| v.to_s }
|
|
end
|
|
|
|
defassert(:descend, %w[/ /a /a/b /a/b/c], "/a/b/c")
|
|
defassert(:descend, %w[a a/b a/b/c], "a/b/c")
|
|
defassert(:descend, %w[. ./a ./a/b ./a/b/c], "./a/b/c")
|
|
defassert(:descend, %w[a/], "a/")
|
|
|
|
def ascend(path)
|
|
Pathname.new(path).enum_for(:ascend).map {|v| v.to_s }
|
|
end
|
|
|
|
defassert(:ascend, %w[/a/b/c /a/b /a /], "/a/b/c")
|
|
defassert(:ascend, %w[a/b/c a/b a], "a/b/c")
|
|
defassert(:ascend, %w[./a/b/c ./a/b ./a .], "./a/b/c")
|
|
defassert(:ascend, %w[a/], "a/")
|
|
|
|
def test_initialize
|
|
p1 = Pathname.new('a')
|
|
assert_equal('a', p1.to_s)
|
|
p2 = Pathname.new(p1)
|
|
assert_equal(p1, p2)
|
|
end
|
|
|
|
def test_initialize_nul
|
|
assert_raise(ArgumentError) { Pathname.new("a\0") }
|
|
end
|
|
|
|
class AnotherStringLike # :nodoc:
|
|
def initialize(s) @s = s end
|
|
def to_str() @s end
|
|
def ==(other) @s == other end
|
|
end
|
|
|
|
def test_equality
|
|
obj = Pathname.new("a")
|
|
str = "a"
|
|
sym = :a
|
|
ano = AnotherStringLike.new("a")
|
|
assert_equal(false, obj == str)
|
|
assert_equal(false, str == obj)
|
|
assert_equal(false, obj == ano)
|
|
assert_equal(false, ano == obj)
|
|
assert_equal(false, obj == sym)
|
|
assert_equal(false, sym == obj)
|
|
|
|
obj2 = Pathname.new("a")
|
|
assert_equal(true, obj == obj2)
|
|
assert_equal(true, obj === obj2)
|
|
assert_equal(true, obj.eql?(obj2))
|
|
end
|
|
|
|
def test_hashkey
|
|
h = {}
|
|
h[Pathname.new("a")] = 1
|
|
h[Pathname.new("a")] = 2
|
|
assert_equal(1, h.size)
|
|
end
|
|
|
|
def assert_pathname_cmp(e, s1, s2)
|
|
p1 = Pathname.new(s1)
|
|
p2 = Pathname.new(s2)
|
|
r = p1 <=> p2
|
|
assert(e == r,
|
|
"#{p1.inspect} <=> #{p2.inspect}: <#{e}> expected but was <#{r}>")
|
|
end
|
|
def test_comparison
|
|
assert_pathname_cmp( 0, "a", "a")
|
|
assert_pathname_cmp( 1, "b", "a")
|
|
assert_pathname_cmp(-1, "a", "b")
|
|
ss = %w(
|
|
a
|
|
a/
|
|
a/b
|
|
a.
|
|
a0
|
|
)
|
|
s1 = ss.shift
|
|
ss.each {|s2|
|
|
assert_pathname_cmp(-1, s1, s2)
|
|
s1 = s2
|
|
}
|
|
end
|
|
|
|
def test_comparison_string
|
|
assert_equal(nil, Pathname.new("a") <=> "a")
|
|
assert_equal(nil, "a" <=> Pathname.new("a"))
|
|
end
|
|
|
|
def pathsub(path, pat, repl) Pathname.new(path).sub(pat, repl).to_s end
|
|
defassert(:pathsub, "a.o", "a.c", /\.c\z/, ".o")
|
|
|
|
def root?(path)
|
|
Pathname.new(path).root?
|
|
end
|
|
|
|
defassert(:root?, true, "/")
|
|
defassert(:root?, true, "//")
|
|
defassert(:root?, true, "///")
|
|
defassert(:root?, false, "")
|
|
defassert(:root?, false, "a")
|
|
|
|
def test_destructive_update
|
|
path = Pathname.new("a")
|
|
path.to_s.replace "b"
|
|
assert_equal(Pathname.new("a"), path)
|
|
end
|
|
|
|
def test_null_character
|
|
assert_raise(ArgumentError) { Pathname.new("\0") }
|
|
end
|
|
|
|
def test_taint
|
|
obj = Pathname.new("a"); assert_same(obj, obj.taint)
|
|
obj = Pathname.new("a"); assert_same(obj, obj.untaint)
|
|
|
|
assert_equal(false, Pathname.new("a" ) .tainted?)
|
|
assert_equal(false, Pathname.new("a" ) .to_s.tainted?)
|
|
assert_equal(true, Pathname.new("a" ).taint .tainted?)
|
|
assert_equal(true, Pathname.new("a" ).taint.to_s.tainted?)
|
|
assert_equal(true, Pathname.new("a".taint) .tainted?)
|
|
assert_equal(true, Pathname.new("a".taint) .to_s.tainted?)
|
|
assert_equal(true, Pathname.new("a".taint).taint .tainted?)
|
|
assert_equal(true, Pathname.new("a".taint).taint.to_s.tainted?)
|
|
|
|
str = "a"
|
|
path = Pathname.new(str)
|
|
str.taint
|
|
assert_equal(false, path .tainted?)
|
|
assert_equal(false, path.to_s.tainted?)
|
|
end
|
|
|
|
def test_untaint
|
|
obj = Pathname.new("a"); assert_same(obj, obj.untaint)
|
|
|
|
assert_equal(false, Pathname.new("a").taint.untaint .tainted?)
|
|
assert_equal(false, Pathname.new("a").taint.untaint.to_s.tainted?)
|
|
|
|
str = "a".taint
|
|
path = Pathname.new(str)
|
|
str.untaint
|
|
assert_equal(true, path .tainted?)
|
|
assert_equal(true, path.to_s.tainted?)
|
|
end
|
|
|
|
def test_freeze
|
|
obj = Pathname.new("a"); assert_same(obj, obj.freeze)
|
|
|
|
assert_equal(false, Pathname.new("a" ) .frozen?)
|
|
assert_equal(false, Pathname.new("a".freeze) .frozen?)
|
|
assert_equal(true, Pathname.new("a" ).freeze .frozen?)
|
|
assert_equal(true, Pathname.new("a".freeze).freeze .frozen?)
|
|
assert_equal(false, Pathname.new("a" ) .to_s.frozen?)
|
|
assert_equal(false, Pathname.new("a".freeze) .to_s.frozen?)
|
|
assert_equal(false, Pathname.new("a" ).freeze.to_s.frozen?)
|
|
assert_equal(false, Pathname.new("a".freeze).freeze.to_s.frozen?)
|
|
end
|
|
|
|
def test_to_s
|
|
str = "a"
|
|
obj = Pathname.new(str)
|
|
assert_equal(str, obj.to_s)
|
|
assert_not_same(str, obj.to_s)
|
|
assert_not_same(obj.to_s, obj.to_s)
|
|
end
|
|
|
|
def test_kernel_open
|
|
count = 0
|
|
result = Kernel.open(Pathname.new(__FILE__)) {|f|
|
|
assert(File.identical?(__FILE__, f))
|
|
count += 1
|
|
2
|
|
}
|
|
assert_equal(1, count)
|
|
assert_equal(2, result)
|
|
end
|
|
|
|
def test_each_filename
|
|
result = []
|
|
Pathname.new("/usr/bin/ruby").each_filename {|f| result << f }
|
|
assert_equal(%w[usr bin ruby], result)
|
|
end
|
|
|
|
def test_kernel_pathname
|
|
assert_equal(Pathname.new("a"), Pathname("a"))
|
|
end
|
|
end
|