Skip to content

Commit 4bd0a8f

Browse files
committed
Fix #31368: joinpath works on collections of paths
1 parent ce1cf9f commit 4bd0a8f

File tree

2 files changed

+53
-10
lines changed

2 files changed

+53
-10
lines changed

base/path.jl

+44-10
Original file line numberDiff line numberDiff line change
@@ -248,16 +248,19 @@ function splitpath(p::String)
248248
return out
249249
end
250250

251-
joinpath(path::AbstractString)::String = path
252-
253251
if Sys.iswindows()
254252

255-
function joinpath(path::AbstractString, paths::AbstractString...)::String
256-
result_drive, result_path = splitdrive(path)
253+
function joinpath(paths::Union{Tuple, AbstractVector})::String
254+
assertstring(x) = x isa AbstractString || throw(ArgumentError("path component is not a string: $(repr(x))"))
255+
256+
isempty(paths) && return path_separator
257+
assertstring(paths[1])
258+
result_drive, result_path = splitdrive(paths[1])
257259

258-
local p_drive, p_path
259-
for p in paths
260-
p_drive, p_path = splitdrive(p)
260+
p_path = ""
261+
for i in firstindex(paths)+1:lastindex(paths)
262+
assertstring(paths[i])
263+
p_drive, p_path = splitdrive(paths[i])
261264

262265
if startswith(p_path, ('\\', '/'))
263266
# second path is absolute
@@ -293,8 +296,15 @@ end
293296

294297
else
295298

296-
function joinpath(path::AbstractString, paths::AbstractString...)::String
297-
for p in paths
299+
function joinpath(paths::Union{Tuple, AbstractVector})::String
300+
assertstring(x) = x isa AbstractString || throw(ArgumentError("path component is not a string: $(repr(x))"))
301+
302+
isempty(paths) && return path_separator
303+
assertstring(paths[1])
304+
path = paths[1]
305+
for i in firstindex(paths)+1:lastindex(paths)
306+
p = paths[i]
307+
assertstring(p)
298308
if isabspath(p)
299309
path = p
300310
elseif isempty(path) || path[end] == '/'
@@ -308,6 +318,29 @@ end
308318

309319
end # os-test
310320

321+
"""
322+
joinpath(parts) -> String
323+
324+
Join collection of path components into a full path. If some argument is an absolute path or
325+
(on Windows) has a drive specification that doesn't match the drive computed for
326+
the join of the preceding paths, then prior components are dropped.
327+
328+
Note on Windows since there is a current directory for each drive, `joinpath("c:", "foo")`
329+
represents a path relative to the current directory on drive "c:" so this is equal to "c:foo",
330+
not "c:\\foo". Furthermore, `joinpath` treats this as a non-absolute path and ignores the drive
331+
letter casing, hence `joinpath("C:\\A","c:b") = "C:\\A\\b"`.
332+
333+
# Examples
334+
```jldoctest
335+
julia> joinpath(["/home/myuser", "example.jl"])
336+
"/home/myuser/example.jl"
337+
338+
julia> joinpath(())
339+
"$(path_separator)"
340+
```
341+
"""
342+
joinpath
343+
311344
"""
312345
joinpath(parts::AbstractString...) -> String
313346
@@ -326,7 +359,8 @@ julia> joinpath("/home/myuser", "example.jl")
326359
"/home/myuser/example.jl"
327360
```
328361
"""
329-
joinpath
362+
joinpath(path::AbstractString)::String = path
363+
joinpath(path::AbstractString, paths::AbstractString...)::String = joinpath((path, paths...))
330364

331365
"""
332366
normpath(path::AbstractString) -> String

test/path.jl

+9
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,11 @@
4242
@test joinpath(S("foo"), S(homedir())) == homedir()
4343
@test joinpath(S(abspath("foo")), S(homedir())) == homedir()
4444

45+
@test joinpath(()) == sep
46+
for str in map(S, [sep, "a$(sep)b", "a$(sep)b$(sep)c", "a$(sep)b$(sep)c$(sep)d"])
47+
@test str == joinpath(splitpath(str))
48+
end
49+
4550
if Sys.iswindows()
4651
@test joinpath(S("foo"),S("bar:baz")) == "bar:baz"
4752
@test joinpath(S("C:"),S("foo"),S("D:"),S("bar")) == "D:bar"
@@ -58,6 +63,10 @@
5863
@test joinpath(S("\\\\server\\share"),S("a")) == "\\\\server\\share\\a"
5964
@test joinpath(S("\\\\server\\share\\"), S("a")) == "\\\\server\\share\\a"
6065

66+
for str in map(S, ["c:\\", "c:\\a", "c:\\a\\b", "c:\\a\\b\\c", "c:\\a\\b\\c\\d"])
67+
@test str == joinpath(splitpath(str))
68+
end
69+
6170
elseif Sys.isunix()
6271
@test joinpath(S("foo"),S("bar:baz")) == "foo$(sep)bar:baz"
6372
@test joinpath(S("C:"),S("foo"),S("D:"),S("bar")) == "C:$(sep)foo$(sep)D:$(sep)bar"

0 commit comments

Comments
 (0)