ExprTools provides tooling for working with Julia expressions during metaprogramming. This package aims to provide light-weight performant tooling without requiring additional package dependencies.
Currently, this package provides the splitdef
, signature
and combinedef
functions which are useful for inspecting and manipulating function definition expressions.
splitdef
works on a function definition expression and returns aDict
of its parts.combinedef
takes aDict
fromsplitdef
and builds it into an expression.signature
works on aMethod
, or the type-tuplesig
field of a method, returning a similarDict
that holds the parts of the expressions that would form its signature.
As well as several helpers that are useful in combination with them.
args_tuple_expr
applies to aDict
fromsplitdef
orsignature
to generate an expression for a tuple of its arguments.parameters
which return the type-parameters of a type, and so is useful for working with the type-tuple that comes out of thesig
field of aMethod
e.g.
julia> using ExprTools
julia> ex = :(
function Base.f(x::T, y::T) where T
x + y
end
)
:(function Base.f(x::T, y::T) where T
#= none:3 =#
x + y
end)
julia> def = splitdef(ex)
Dict{Symbol,Any} with 5 entries:
:args => Any[:(x::T), :(y::T)]
:body => quote…
:name => :(Base.f)
:head => :function
:whereparams => Any[:T]
julia> def[:name] = :g;
julia> def[:head] = :(=);
julia> args_tuple_expr(def)
:((x, y))
julia> def[:body] = :(*($(args_tuple_expr(def))...));
julia> g_expr = combinedef(def)
:((g(x::T, y::T) where T) = (*)((x, y)...))
julia> eval(g_expr)
g (generic function with 1 method)
julia> g_method = first(methods(g))
g(x::T, y::T) where T in Main
julia> parameters(g_method.sig)
svec(typeof(g), T, T)
julia> signature(g_method)
Dict{Symbol, Any} with 3 entries:
:name => :g
:args => Expr[:(x::T), :(y::T)]
:whereparams => Any[:T]
"ExprTools: Metaprogramming from reflection" by Frames White