Skip to content

Commit da23ed5

Browse files
committed
Make reinterpretting Unions of nothings/missing and a bitstype work
remove mistakenly included files remove mistakenly included files
1 parent c1c6a17 commit da23ed5

File tree

4 files changed

+72
-25
lines changed

4 files changed

+72
-25
lines changed

base/missing.jl

+27
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,33 @@ convert(::Type{T}, x::T) where {T>:Union{Missing, Nothing}} = x
6969
convert(::Type{T}, x) where {T>:Missing} = convert(nonmissingtype_checked(T), x)
7070
convert(::Type{T}, x) where {T>:Union{Missing, Nothing}} = convert(nonmissingtype_checked(nonnothingtype_checked(T)), x)
7171

72+
for Cs in ((:Missing,), (:Nothing,), (:Missing, :Nothing))
73+
@eval function Base.reinterpret(
74+
::Type{<:T}, # Note this is effectively same as Type{T} since will error if T is abstract
75+
xs::Array{Union{T, $(Cs...)},N}
76+
) where {T,N}
77+
isbitstype(T) || throw(ArgumentError("cannot reinterpret `$(eltype(xs))`, type `$(T)` is not a bits type"))
78+
return unsafe_wrap(Array{T,N}, Ptr{T}(pointer(xs)), length(xs))
79+
end
80+
81+
#==
82+
@eval function Base.convert(
83+
::Type{Array{Q,N}},
84+
xs::Array{Union{T, $(Cs...)},N}
85+
) where {N} where {T<:Q} where Q
86+
87+
all(x->x isa T, xs) || throw(ArgumentError("cannot convert $(eltype(xs)) to $T"))
88+
if isbitstype(T) &&
89+
reinterpret(T, xs)
90+
else
91+
T[x for x in xs]
92+
end
93+
end
94+
==#
95+
end
96+
# Resolve ambiguity
97+
#Base.convert(::Type{Array{Any,1}}, xs::Array{Any,1}) = xs
98+
7299

73100
# Comparison operators
74101
==(::Missing, ::Missing) = missing

base/regex.jl

+3-7
Original file line numberDiff line numberDiff line change
@@ -145,13 +145,13 @@ function show(io::IO, m::RegexMatch)
145145
idx_to_capture_name = PCRE.capture_names(m.regex.regex)
146146
if !isempty(m.captures)
147147
print(io, ", ")
148-
for (i, capture) in enumerate(m)
148+
for i = 1:length(m.captures)
149149
# If the capture group is named, show the name.
150150
# Otherwise show its index.
151151
capture_name = get(idx_to_capture_name, i, i)
152152
print(io, capture_name, "=")
153-
show(io, capture)
154-
if i < length(m)
153+
show(io, m.captures[i])
154+
if i < length(m.captures)
155155
print(io, ", ")
156156
end
157157
end
@@ -168,10 +168,6 @@ function getindex(m::RegexMatch, name::Symbol)
168168
end
169169
getindex(m::RegexMatch, name::AbstractString) = m[Symbol(name)]
170170

171-
iterate(m::RegexMatch, args...) = iterate(m.captures, args...)
172-
length(m::RegexMatch) = length(m.captures)
173-
eltype(m::RegexMatch) = eltype(m.captures)
174-
175171
function occursin(r::Regex, s::AbstractString; offset::Integer=0)
176172
compile(r)
177173
return PCRE.exec_r(r.regex, String(s), offset, r.match_options)

test/missing.jl

+42
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,48 @@ end
2323
@test_throws MethodError convert(Union{Int, Missing}, "a")
2424
end
2525

26+
mutable struct NotABitsType a end
27+
==(n1::NotABitsType,n2::NotABitsType) = n1.a == n2.a
28+
29+
@testset "convert/reinterpret Arrays of Unions" begin
30+
unions(T) = (Union{T, Missing}, Union{T, Nothing}, Union{T, Missing, Nothing})
31+
corrupters(::Type{T}) where T = (x for x in (missing, nothing) if x isa T)
32+
33+
@testset "bits type ($U)" for U in unions(Int)
34+
xo = U[1, 2, 3]
35+
xr = reinterpret(Int, xo)
36+
#xc = convert(Vector{Int}, xo)
37+
#@test xc isa Vector{Int}
38+
@test xr isa Vector{Int}
39+
@test xr == xo
40+
#@test xc == xo
41+
# should not have allocated new memory
42+
#@test pointer(xc) == pointer(xo)
43+
@test pointer(xr) == pointer(xo)
44+
45+
@test_throws ArgumentError reinterpret(Integer, xo)
46+
47+
@testset "With non-target type values" begin
48+
yo = U[1, 2, 3, corrupters(U)...]
49+
#@test_throws ArgumentError convert(Vector{Int}, yo)
50+
yr = reinterpret(Int, yo)
51+
@test yr isa Vector{Int}
52+
# No comment on what happens for the nonInt values, unspecified behavour
53+
@test yr[1:3] == yo[1:3]
54+
@test length(yo) == length(yr)
55+
end
56+
end
57+
@testset "non-bits type ($U)" for U in unions(NotABitsType)
58+
xo = U[NotABitsType(1), NotABitsType(2), NotABitsType(3)]
59+
@test_throws ArgumentError reinterpret(Int, xo)
60+
#xc = convert(Vector{NotABitsType}, x)
61+
#@test xc isa Vector{Int}
62+
#@test xc == xo
63+
# should have allocated new memory
64+
#@test pointer(xc) != pointer(xo)
65+
end
66+
end
67+
2668
@testset "promote rules" begin
2769
@test promote_type(Missing, Missing) == Missing
2870
@test promote_type(Missing, Int) == Union{Missing, Int}

test/regex.jl

-18
Original file line numberDiff line numberDiff line change
@@ -136,24 +136,6 @@
136136
@test r"this|that"^2 == r"(?:this|that){2}"
137137
end
138138

139-
@testset "iterate" begin
140-
m = match(r"(.) test (.+)", "a test 123")
141-
@test first(m) == "a"
142-
@test collect(m) == ["a", "123"]
143-
@test for (i, capture) in enumerate(m)
144-
i == 1 && @test capture == "a"
145-
i == 2 && @test capture == "123"
146-
end
147-
end
148-
149-
@testset "Destructuring dispatch" begin
150-
handle(::Nothing) = "not found"
151-
handle((capture,)::RegexMatch) = "found $capture"
152-
153-
@test handle(match(r"a (\d)", "xyz")) == "not found"
154-
@test handle(match(r"a (\d)", "a 1")) == "found 1"
155-
end
156-
157139
# Test that PCRE throws the correct kind of error
158140
# TODO: Uncomment this once the corresponding change has propagated to CI
159141
#@test_throws ErrorException Base.PCRE.info(C_NULL, Base.PCRE.INFO_NAMECOUNT, UInt32)

0 commit comments

Comments
 (0)