Skip to content

Commit 2d5e963

Browse files
committed
Add @unpack_fields macro
1 parent 535605a commit 2d5e963

File tree

3 files changed

+168
-56
lines changed

3 files changed

+168
-56
lines changed

README.md

+28-7
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,8 @@
44
[![Coverage](https://codecov.io/gh/devmotion/SimpleUnPack.jl/branch/main/graph/badge.svg)](https://codecov.io/gh/devmotion/SimpleUnPack.jl)
55
[![Code Style: Blue](https://img.shields.io/badge/code%20style-blue-4495d1.svg)](https://github.com/invenia/BlueStyle)
66

7-
This package provides the `@unpack` macro for destructuring properties.
8-
Its behaviour is equivalent to the destructuring that was introduced in [Julia#39285](https://github.com/JuliaLang/julia/pull/39285) and is available in Julia >= 1.7.0-DEV.364.
7+
This package provides the `@unpack` and `@unpack_fields` macros for destructuring properties and fields, respectively.
8+
The behaviour of `@unpack` is equivalent to the destructuring that was introduced in [Julia#39285](https://github.com/JuliaLang/julia/pull/39285) and is available in Julia >= 1.7.0-DEV.364.
99

1010
## Examples
1111

@@ -24,18 +24,28 @@ julia> a
2424

2525
julia> b
2626
42
27+
28+
julia> @unpack_fields a, b = f(10)
29+
(b = 10, a = 5.0)
30+
31+
julia> a
32+
5.0
33+
34+
julia> b
35+
10
2736
```
2837

2938
An example with a custom struct in a function:
3039

3140
```julia
3241
julia> using SimpleUnPack
3342

34-
julia> struct MyStruct{T}
35-
x::T
43+
julia> struct MyStruct{X,Y}
44+
x::X
45+
y::Y
3646
end
3747

38-
julia> Base.getpropertynames(::MyStruct) = (:x, :y)
48+
julia> Base.propertynames(::MyStruct) = (:x, :y)
3949

4050
julia> function Base.getproperty(m::MyStruct, p::Symbol)
4151
if p === :y
@@ -50,8 +60,16 @@ julia> function g(m::MyStruct)
5060
return (; x, y)
5161
end;
5262

53-
julia> g(MyStruct(1.0))
63+
julia> g(MyStruct(1.0, -5))
5464
(x = 1.0, y = 42)
65+
66+
julia> function h(m::MyStruct)
67+
@unpack_fields x, y = m
68+
return (; x, y)
69+
end
70+
71+
julia> h(MyStruct(1.0, -5))
72+
(x = 1.0, y = -5)
5573
```
5674

5775
## Comparison with UnPack.jl
@@ -60,4 +78,7 @@ The syntax of `@unpack` is based on [`UnPack.@unpack`](https://github.com/mauro3
6078
However, `UnPack.@unpack` is more flexible and based on `UnPack.unpack` instad of `getproperty`.
6179
While `UnPack.unpack` falls back to `getproperty`, it also supports `AbstractDict`s with keys of type `Symbol` and `AbstractString`, and can be specialized for other types.
6280
Since `UnPack.unpack` dispatches on `Val(property)` instances, this increased flexibility comes at the cost of increased compilation times.
63-
Moreover, UnPack also provides an `UnPack.@pack!` macro for setting properties.
81+
82+
In contrast to SimpleUnPack, UnPack provides an `UnPack.@pack!` macro for setting properties.
83+
84+
However, currently UnPack does not support destructuring based on `getfield` only ([UnPack#23](https://github.com/mauro3/UnPack.jl/issues/23)).

src/SimpleUnPack.jl

+70-32
Original file line numberDiff line numberDiff line change
@@ -1,29 +1,68 @@
11
module SimpleUnPack
22

3-
export @unpack
3+
export @unpack, @unpack_fields
44

55
"""
66
@unpack a, b, ... = x
77
88
Destructure properties `a`, `b`, ... of `x` into variables of the same name.
99
1010
The behaviour of the macro is equivalent to `(; a, b, ...) = x` which was introduced in [Julia#39285](https://github.com/JuliaLang/julia/pull/39285) and is available in Julia >= 1.7.0-DEV.364.
11+
12+
See also [`@unpack_fields`](@ref)
1113
"""
1214
macro unpack(args)
13-
return unpack(args)
15+
# Extract names of properties and RHS
16+
names, rhs = split_names_rhs(:unpack, args)
17+
18+
# Construct destructuring expression
19+
expr = if VERSION >= v"1.7.0-DEV.364"
20+
# Fall back to destructuring in Base when available:
21+
# https://github.com/JuliaLang/julia/pull/39285
22+
Expr(:(=), Expr(:tuple, Expr(:parameters, (esc(p) for p in names)...)), esc(rhs))
23+
else
24+
destructuring_expr(:getproperty, names, rhs)
25+
end
26+
27+
return expr
28+
end
29+
30+
"""
31+
@unpack_fields a, b, ... = x
32+
33+
Destructure fields `a`, `b`, ... of `x` into variables of the same name.
34+
35+
See also [`@unpack`](@ref)
36+
"""
37+
macro unpack_fields(args)
38+
# Extract names of fields and RHS
39+
names, rhs = split_names_rhs(:unpack_fields, args)
40+
41+
# Construct destructuring expression
42+
expr = destructuring_expr(:getfield, names, rhs)
43+
44+
return expr
1445
end
1546

16-
function unpack(args)
17-
# Extract properties and RHS
18-
if !Meta.isexpr(args, :(=), 2)
47+
"""
48+
split_names_rhs(macrosym::Symbol, expr)
49+
50+
Split an expression `expr` of the form `a, b, ... = x` into a tuple consisting of a vector of symbols `a`, `b`, ..., and the right-hand side `x`.
51+
52+
The symbol `macro_name` specifies the macro from which this function is called.
53+
54+
This function is used internally with `macrosym = :unpack` and `macrosym = :unpack_fields`.
55+
"""
56+
function split_names_rhs(macrosym::Symbol, expr)
57+
if !Meta.isexpr(expr, :(=), 2)
1958
throw(
2059
ArgumentError(
21-
"`@unpack` can only be applied to expressions of the form `a, b, ... = x`"
60+
"`@$macrosym` can only be applied to expressions of the form `a, b, ... = x`",
2261
),
2362
)
2463
end
25-
lhs, rhs = args.args
26-
properties = if lhs isa Symbol
64+
lhs, rhs = expr.args
65+
names = if lhs isa Symbol
2766
[lhs]
2867
elseif Meta.isexpr(lhs, :tuple) &&
2968
!isempty(lhs.args) &&
@@ -32,34 +71,33 @@ function unpack(args)
3271
else
3372
throw(
3473
ArgumentError(
35-
"`@unpack` can only be applied to expressions of the form `a, b, ... = x`",
74+
"`@$macrosym` can only be applied to expressions of the form `a, b, ... = x`",
3675
),
3776
)
3877
end
78+
return names, rhs
79+
end
3980

40-
if VERSION >= v"1.7.0-DEV.364"
41-
# Fall back to destructuring in Base when available:
42-
# https://github.com/JuliaLang/julia/pull/39285
43-
return Expr(
44-
:(=), Expr(:tuple, Expr(:parameters, (esc(p) for p in properties)...)), esc(rhs)
45-
)
46-
else
47-
@gensym object
48-
block = Expr(:block)
49-
for p in properties
50-
push!(
51-
block.args,
52-
Expr(:(=), esc(p), Expr(:call, :getproperty, esc(object), QuoteNode(p))),
53-
)
54-
end
55-
return Base.remove_linenums!(
56-
quote
57-
$(esc(object)) = $(esc(rhs)) # In case the RHS is an expression
58-
$block
59-
$(esc(object)) # Return evaluation of rhs to ensure the behaviour is the same as (; ...) = rhs
60-
end,
61-
)
81+
"""
82+
destructuring_expr(fsym::Symbol, names, rhs)
83+
84+
Return an expression that destructures `rhs` based on a function of name `fsym` and keys `names` into variables of the same `names`.
85+
86+
This function is used internally with `fsym = :getproperty` and `fsym = :getfield`.
87+
"""
88+
function destructuring_expr(fsym::Symbol, names, rhs)
89+
@gensym object
90+
block = Expr(:block)
91+
for p in names
92+
push!(block.args, Expr(:(=), esc(p), Expr(:call, fsym, esc(object), QuoteNode(p))))
6293
end
94+
return Base.remove_linenums!(
95+
quote
96+
$(esc(object)) = $(esc(rhs)) # In case the RHS is an expression
97+
$block
98+
$(esc(object)) # Return evaluation of the RHS
99+
end,
100+
)
63101
end
64102

65-
end
103+
end # module

test/runtests.jl

+70-17
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,10 @@
11
using SimpleUnPack
22
using Test
33

4-
struct Property{X,Y}
4+
struct Property{X,Y,Z}
55
x::X
66
y::Y
7+
z::Z
78
end
89

910
Base.propertynames(::Property) = (:x, :y, :z)
@@ -56,19 +57,40 @@ end
5657
@unpack y = d
5758
@test y == 1.0
5859

59-
d = Struct(43, 2.0, "z2")
60-
@unpack x, z = d
60+
d = (x=43, y=2.0, z="z2")
61+
@unpack_fields x, z = d
6162
@test x == 43
6263
@test z == "z2"
63-
@unpack y = d
64+
@unpack_fields y = d
6465
@test y == 2.0
6566

66-
d = Property(44, 3.0)
67+
d = Struct(44, 3.0, "z3")
6768
@unpack x, z = d
6869
@test x == 44
69-
@test z == "z"
70+
@test z == "z3"
7071
@unpack y = d
7172
@test y == 3.0
73+
74+
d = Struct(45, 4.0, "z4")
75+
@unpack_fields x, z = d
76+
@test x == 45
77+
@test z == "z4"
78+
@unpack_fields y = d
79+
@test y == 4.0
80+
81+
d = Property(46, 5.0, "z5")
82+
@unpack x, z = d
83+
@test x == 46
84+
@test z == "z"
85+
@unpack y = d
86+
@test y == 5.0
87+
88+
d = Property(47, 6.0, "z6")
89+
@unpack_fields x, z = d
90+
@test x == 47
91+
@test z == "z6"
92+
@unpack_fields y = d
93+
@test y == 6.0
7294
end
7395

7496
@testset "Expression as RHS" begin
@@ -78,27 +100,53 @@ end
78100
@unpack y = (x=42, y=1.0, z="z1")
79101
@test y == 1.0
80102

81-
@unpack x, z = Struct(43, 2.0, "z2")
103+
@unpack_fields x, z = (x=43, y=2.0, z="z2")
82104
@test x == 43
83105
@test z == "z2"
84-
@unpack y = Struct(43, 2.0, "z2")
106+
@unpack_fields y = (x=43, y=2.0, z="z2")
85107
@test y == 2.0
86108

87-
@unpack x, z = Property(44, 3.0)
109+
@unpack x, z = Struct(44, 3.0, "z3")
88110
@test x == 44
89-
@test z == "z"
90-
@unpack y = Property(44, 3.0)
111+
@test z == "z3"
112+
@unpack y = Struct(44, 3.0, "z3")
91113
@test y == 3.0
114+
115+
@unpack_fields x, z = Struct(45, 4.0, "z4")
116+
@test x == 45
117+
@test z == "z4"
118+
@unpack_fields y = Struct(45, 4.0, "z4")
119+
@test y == 4.0
120+
121+
@unpack x, z = Property(46, 5.0, "z5")
122+
@test x == 46
123+
@test z == "z"
124+
@unpack y = Property(46, 5.0, "z5")
125+
@test y == 5.0
126+
127+
@unpack_fields x, z = Property(47, 6.0, "z6")
128+
@test x == 47
129+
@test z == "z6"
130+
@unpack_fields y = Property(47, 6.0, "z6")
131+
@test y == 6.0
92132
end
93133

94134
@testset "Type inference" begin
95-
function f(z)
96-
@unpack y, x = z
97-
return x, y
135+
function f(a)
136+
@unpack y, z, x = a
137+
return x, y, z
138+
end
139+
@test @inferred(f((; y=1.0, z="a", x=42))) == (42, 1.0, "a")
140+
@test @inferred(f(Struct(42, 1.0, "a"))) == (42, 1.0, "a")
141+
@test @inferred(f(Property(42, 1.0, "a"))) == (42, 1.0, "z")
142+
143+
function g(a)
144+
@unpack_fields y, z, x = a
145+
return x, y, z
98146
end
99-
@test @inferred(f((; y=1.0, z="z", x=42))) == (42, 1.0)
100-
@test @inferred(f(Struct(42, 1.0, "z"))) == (42, 1.0)
101-
@test @inferred(f(Property(42, 1.0))) == (42, 1.0)
147+
@test @inferred(g((; y=1.0, z="a", x=42))) == (42, 1.0, "a")
148+
@test @inferred(g(Struct(42, 1.0, "a"))) == (42, 1.0, "a")
149+
@test @inferred(g(Property(42, 1.0, "a"))) == (42, 1.0, "a")
102150
end
103151

104152
@testset "Errors" begin
@@ -107,5 +155,10 @@ end
107155
@test_macro_throws ArgumentError @unpack (; x=42, y=1.0)
108156
@test_macro_throws ArgumentError @unpack x, y, (; x=42, y=1.0)
109157
@test_macro_throws ArgumentError @unpack x, 1 = (; x=42, y=1.0)
158+
159+
@test_macro_throws ArgumentError @unpack_fields d
160+
@test_macro_throws ArgumentError @unpack_fields (; x=42, y=1.0)
161+
@test_macro_throws ArgumentError @unpack_fields x, y, (; x=42, y=1.0)
162+
@test_macro_throws ArgumentError @unpack_fields x, 1 = (; x=42, y=1.0)
110163
end
111164
end

0 commit comments

Comments
 (0)