From 2efb2d85e2ba710516a9726dd6bf6b349f5d6369 Mon Sep 17 00:00:00 2001 From: Shuhei Kadowaki Date: Sat, 10 Oct 2020 15:55:56 +0900 Subject: [PATCH 1/3] define at-invokelatest macro: - offers easier syntax to call `Baes.invokelatest` - `@invokelatest f(args...; kwargs...)` will simply be expanded into `Base.invokelatst(f, args...; kwargs...)` --- base/util.jl | 26 ++++++++++++++++++++++++++ doc/src/base/base.md | 1 + test/misc.jl | 25 +++++++++++++++++++++++++ 3 files changed, 52 insertions(+) diff --git a/base/util.jl b/base/util.jl index 0e7087890f81b..637dd175575a8 100644 --- a/base/util.jl +++ b/base/util.jl @@ -509,6 +509,32 @@ function _kwdef!(blk, params_args, call_args) blk end +""" + @invokelatest f(args...; kwargs...) + +Provides a convenient way to call [`Base.invokelatest`](@ref). +`@invokelatest f(args...; kwargs...)` will simply be expanded into +`Base.invokelatst(f, args...; kwargs...)`. +""" +macro invokelatest(ex) + @assert is_expr(ex, :call) "call expression f(args...; kwargs...) should be given" + + f = first(ex.args) + args = [] + kwargs = [] + for x in ex.args[2:end] + if is_expr(x, :parameters) + append!(kwargs, x.args) + elseif is_expr(x, :kw) + push!(kwargs, x) + else + push!(args, x) + end + end + + esc(:($(GlobalRef(Base, :invokelatest))($(f), $(args...); $(kwargs...)))) +end + # testing """ diff --git a/doc/src/base/base.md b/doc/src/base/base.md index 3b2f82763bafa..47d581ba76ce6 100644 --- a/doc/src/base/base.md +++ b/doc/src/base/base.md @@ -235,6 +235,7 @@ Base.hasmethod Core.applicable Core.invoke Base.invokelatest +Base.@invokelatest new Base.:(|>) Base.:(∘) diff --git a/test/misc.jl b/test/misc.jl index 62caf29aa7571..70af5595b1074 100644 --- a/test/misc.jl +++ b/test/misc.jl @@ -647,6 +647,31 @@ let foo() = begin @test foo() == 1 end +module atinvokelatest +f(x) = 1 +g(x, y; z=0) = x * y + z +end + +let foo() = begin + @eval atinvokelatest.f(x::Int) = 3 + return Base.@invokelatest atinvokelatest.f(0) + end + @test foo() == 3 +end + +let foo() = begin + @eval atinvokelatest.f(x::Int) = 3 + return Base.@invokelatest atinvokelatest.f(0) + end + @test foo() == 3 + + bar() = begin + @eval atinvokelatest.g(x::Int, y::Int; z=3) = z + return Base.@invokelatest atinvokelatest.g(2, 3; z=1) + end + @test bar() == 1 +end + # Endian tests # For now, we only support little endian. # Add an `Sys.ARCH` test for big endian when/if we add support for that. From 5e467742b8050797a54bb8f5cf8d8004f84c5dd1 Mon Sep 17 00:00:00 2001 From: Shuhei Kadowaki <40514306+aviatesk@users.noreply.github.com> Date: Thu, 12 Nov 2020 00:06:31 +0900 Subject: [PATCH 2/3] Update base/util.jl Co-authored-by: Jeff Bezanson --- base/util.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/base/util.jl b/base/util.jl index 637dd175575a8..a14c08997e856 100644 --- a/base/util.jl +++ b/base/util.jl @@ -514,7 +514,7 @@ end Provides a convenient way to call [`Base.invokelatest`](@ref). `@invokelatest f(args...; kwargs...)` will simply be expanded into -`Base.invokelatst(f, args...; kwargs...)`. +`Base.invokelatest(f, args...; kwargs...)`. """ macro invokelatest(ex) @assert is_expr(ex, :call) "call expression f(args...; kwargs...) should be given" From 7aa460719bee7c538689d1b43c27a75e867a1b67 Mon Sep 17 00:00:00 2001 From: Shuhei Kadowaki Date: Sun, 22 Nov 2020 02:53:20 +0900 Subject: [PATCH 3/3] throw `ArgumentError` instead --- base/util.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/base/util.jl b/base/util.jl index a14c08997e856..348b09f3f4c1c 100644 --- a/base/util.jl +++ b/base/util.jl @@ -517,7 +517,7 @@ Provides a convenient way to call [`Base.invokelatest`](@ref). `Base.invokelatest(f, args...; kwargs...)`. """ macro invokelatest(ex) - @assert is_expr(ex, :call) "call expression f(args...; kwargs...) should be given" + is_expr(ex, :call) || throw(ArgumentError("a call expression f(args...; kwargs...) should be given")) f = first(ex.args) args = []