Skip to content

Commit

Permalink
Fix joinpath on Windows and improve joinpath on linux
Browse files Browse the repository at this point in the history
  • Loading branch information
musm committed Oct 5, 2019
1 parent bf8c324 commit 39ff677
Show file tree
Hide file tree
Showing 2 changed files with 79 additions and 15 deletions.
75 changes: 62 additions & 13 deletions base/path.jl
Original file line number Diff line number Diff line change
Expand Up @@ -259,20 +259,69 @@ julia> joinpath("/home/myuser", "example.jl")
"/home/myuser/example.jl"
```
"""
joinpath(a::AbstractString, b::AbstractString, c::AbstractString...) = joinpath(joinpath(a,b), c...)
joinpath(a::AbstractString) = a

function joinpath(a::String, b::String)
isabspath(b) && return b
A, a = splitdrive(a)
B, b = splitdrive(b)
!isempty(B) && A != B && return string(B,b)
C = isempty(B) ? A : B
isempty(a) ? string(C,b) :
occursin(path_separator_re, a[end:end]) ? string(C,a,b) :
string(C,a,pathsep(a,b),b)
joinpath

joinpath(path::AbstractString) = path

if Sys.iswindows()

function joinpath(path::AbstractString, paths::AbstractString...)
result_drive, result_path = splitdrive(path)

local p_drive, p_path
for p in paths
p_drive, p_path = splitdrive(p)

if !isempty(p_path) && p_path[1] ('\\', '/')
# second path is absolute
if !isempty(p_drive) || !isempty(result_drive)
result_drive = p_drive
end
result_path = p_path
continue
elseif !isempty(p_drive) && p_drive != result_drive
if lowercase(p_drive) != lowercase(result_drive)
# different drives, ignore the first path entirely
result_drive = p_drive
result_path = p_path
continue
end
# same drive in different case
result_drive = p_drive
end

# second path is relative to the first
if !isempty(result_path) && result_path[end] ('\\', '/')
result_path *= path_separator
end

result_path = result_path * p_path
end

# add separator between UNC and non-absolute path
if !isempty(p_path) && result_path[1] ('\\', '/') && !isempty(result_drive) && result_drive[end] != ':'
return result_drive * path_separator * result_path
end

return result_drive * result_path
end

else

function joinpath(path::AbstractString, paths::AbstractString...)
for p in paths
if !isempty(p) && p[1] == path_separator
path = p
elseif isempty(path) || path[end] == path_separator
path *= p
else
path *= path_separator * p
end
end
return path
end
joinpath(a::AbstractString, b::AbstractString) = joinpath(String(a), String(b))

end # os-test

"""
normpath(path::AbstractString) -> AbstractString
Expand Down
19 changes: 17 additions & 2 deletions test/path.jl
Original file line number Diff line number Diff line change
Expand Up @@ -31,15 +31,30 @@
@test isdirpath(S(".."))
end
@testset "joinpath" begin
@test joinpath(S("")) == ""
@test joinpath(S("foo")) == "foo"
@test joinpath(S("foo"), S("bar")) == "foo$(sep)bar"
@test joinpath(S("foo"), S(homedir())) == homedir()
@test joinpath(S(abspath("foo")), S(homedir())) == homedir()
@test joinpath(S("foo"), S("bar"), S("baz")) == "foo$(sep)bar$(sep)baz"
@test joinpath(S("foo"), S(""), S("baz")) == "foo$(sep)baz"
@test joinpath(S("foo"), S(""), S("")) == "foo$(sep)"
@test joinpath(S("foo"), S(""), S(""), S("bar")) == "foo$(sep)bar"

if Sys.iswindows()
@test joinpath(S("foo"),S("bar:baz")) == "bar:baz"
@test joinpath(S("C:"),S("foo"),S("D:"),S("bar")) == "D:bar"
@test joinpath(S("C:"),S("foo"),S("D:bar"),S("baz")) == "D:bar$(sep)baz"

# relative folders and case-insensitive drive letters
@test joinpath(S("C:\\a\\b"), S("c:c\\e")) == "c:\\a\\b\\c\\e"

# UNC paths
@test _joinpath(S("\\\\server"), S("share")) == "\\\\server\\share"
@test _joinpath(S("\\\\server"), S("share"), S("a")) == "\\\\server\\share\\a"
@test _joinpath(S("\\\\server\\"), S("share"), S("a")) == "\\\\server\\share\\a"
@test _joinpath(S("\\\\server"), S("share"), S("a"), S("b")) == "\\\\server\\share\\a\\b"
@test _joinpath(S("\\\\server\\share"),S("a")) == "\\\\server\\share\\a"
@test _joinpath(S("\\\\server\\share\\"), S("a")) == "\\\\server\\share\\a"

elseif Sys.isunix()
@test joinpath(S("foo"),S("bar:baz")) == "foo$(sep)bar:baz"
@test joinpath(S("C:"),S("foo"),S("D:"),S("bar")) == "C:$(sep)foo$(sep)D:$(sep)bar"
Expand Down

0 comments on commit 39ff677

Please sign in to comment.