Bug #16831
closedRunning `Pathname#glob` with `File::FNM_DOTMATCH` option loses `.` and `..`
Description
Running Dir.glob, Pathname.glob with File::FNM_DOTMATCH option keeps . and .. as their basename:
require 'pathname'
pathname = Pathname.pwd
#=> #<Pathname:/Users/jnito/dev>
# Dir.glob keeps "." and ".."
Dir.glob(pathname.join('*'), File::FNM_DOTMATCH).sort[0..1]
#=> ["/Users/jnito/dev/.", "/Users/jnito/dev/.."]
# Pathname.glob keeps "." and ".." too
Pathname.glob(pathname.join('*'), File::FNM_DOTMATCH).sort[0..1]
#=> [#<Pathname:/Users/jnito/dev/.>, #<Pathname:/Users/jnito/dev/..>]
I expect Pathname#glob with File::FNM_DOTMATCH option has same behavior, but it loses . and ..:
# Pathname#glob loses "." and ".."
pathname.glob('*', File::FNM_DOTMATCH).sort[0..1]
#=> [#<Pathname:/Users/jnito>, #<Pathname:/Users/jnito/dev>]
I wanted to replace my code from Pathname.glob(pathname.join('*'), File::FNM_DOTMATCH) to pathname.glob('*', File::FNM_DOTMATCH), but I couldn't do due to their incompatibility. So I want Pathname#glob to keep . and ...
Updated by nobu (Nobuyoshi Nakada) over 5 years ago
Actually . and .. entries are kept.
Just Pathname("/Users/jnito/dev/..") is cleaned up as Pathname("/Users/jnito").
Updated by jnchito (Junichi Ito) over 5 years ago
nobu (Nobuyoshi Nakada) wrote in #note-1:
Actually
.and..entries are kept.
JustPathname("/Users/jnito/dev/..")is cleaned up asPathname("/Users/jnito").
That's true, but I feel the following code should have consistent results:
require 'pathname'
pathname = Pathname.pwd
Pathname(Dir.glob(pathname.join('*'), File::FNM_DOTMATCH).sort[0]).basename.to_s
#=> .
Pathname.glob(pathname.join('*'), File::FNM_DOTMATCH).sort[0].basename.to_s
#=> .
# This line doesn't return "."
pathname.glob('*', File::FNM_DOTMATCH).sort[0].basename.to_s
#=> jnito
Updated by Dan0042 (Daniel DeLorme) over 5 years ago
I don't think those can have consistent results, because Pathname#glob is equivalent to using the base option of Dir.glob and then joining the paths, which is not the same as joining the paths into a glob and then invoking Dir.glob. In particular they have different behavior if the base contains globbing characters. And I don't think there's a method of Pathname that allows to join without cleaning the path.
b = Pathname.new("ext")
b.glob("*").min #=> #<Pathname:ext/-test->
Dir.glob("*", base: b).min #=> "-test-"
Dir.glob(b+"*").min #=> "ext/-test-"
b = Pathname.new("{ext,man}")
b.glob("*").min #=> nil because there's no literal "{ext,man}" dir
Dir.glob("*", base: b).min #=> nil
Dir.glob(b+"*").min #=> "ext/-test-"
Updated by jnchito (Junichi Ito) over 5 years ago
Thank you for your explanation. With your hint and reading the implementation code solved my question.
Dan0042 (Daniel DeLorme) wrote in #note-3:
I don't think those can have consistent results, because Pathname#glob is equivalent to using the
baseoption ofDir.globand then joining the paths, which is not the same as joining the paths into a glob and then invokingDir.glob. In particular they have different behavior if the base contains globbing characters. And I don't think there's a method of Pathname that allows to join without cleaning the path.b = Pathname.new("ext") b.glob("*").min #=> #<Pathname:ext/-test-> Dir.glob("*", base: b).min #=> "-test-" Dir.glob(b+"*").min #=> "ext/-test-" b = Pathname.new("{ext,man}") b.glob("*").min #=> nil because there's no literal "{ext,man}" dir Dir.glob("*", base: b).min #=> nil Dir.glob(b+"*").min #=> "ext/-test-"
Dan0042 (Daniel DeLorme) wrote in #note-3:
I don't think those can have consistent results, because Pathname#glob is equivalent to using the
baseoption ofDir.globand then joining the paths, which is not the same as joining the paths into a glob and then invokingDir.glob. In particular they have different behavior if the base contains globbing characters. And I don't think there's a method of Pathname that allows to join without cleaning the path.b = Pathname.new("ext") b.glob("*").min #=> #<Pathname:ext/-test-> Dir.glob("*", base: b).min #=> "-test-" Dir.glob(b+"*").min #=> "ext/-test-" b = Pathname.new("{ext,man}") b.glob("*").min #=> nil because there's no literal "{ext,man}" dir Dir.glob("*", base: b).min #=> nil Dir.glob(b+"*").min #=> "ext/-test-"
Updated by jeremyevans0 (Jeremy Evans) over 5 years ago
- Status changed from Open to Closed
Updated by Eregon (Benoit Daloze) about 5 years ago
- Related to Bug #17280: Dir.glob with FNM_DOTMATCH matches ".." and "." and results in duplicated entries added