diff --git a/src/utils.jl b/src/utils.jl index 626474eb..c0b65577 100644 --- a/src/utils.jl +++ b/src/utils.jl @@ -90,11 +90,20 @@ istrivial(a) = a === nothing || isa(a, LineNumberNode) isgoto(stmt) = isa(stmt, Core.GotoNode) | isexpr(stmt, :gotoifnot) +function unwrap_where(ex::Expr) + while isexpr(ex, :where) + ex = ex.args[1] + end + return ex +end + function pushex!(exsigs::ExprsSigs, ex::Expr) uex = unwrap(ex) if is_doc_expr(uex) body = uex.args[4] - if isa(body, Expr) && body.head !== :call # don't trigger for docexprs like `"docstr" f(x::Int)` + # Don't trigger for exprs where the documented expression is just a signature + # (e.g. `"docstr" f(x::Int)`, `"docstr" f(x::T) where T` etc.) + if isa(body, Expr) && unwrap_where(body).head !== :call exsigs[RelocatableExpr(body)] = nothing end if length(uex.args) < 5 diff --git a/test/runtests.jl b/test/runtests.jl index 23f33d06..27627fcd 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -1112,6 +1112,23 @@ const issue639report = [] pop!(LOAD_PATH) end + do_test("doc expr signature") && @testset "Docstring attached to signatures" begin + md = Revise.ModuleExprsSigs(Main) + Revise.parse_source!(md, """ + module DocstringSigsOnly + function f end + "basecase" f(x) + "basecase with type" f(x::Int) + "basecase no varname" f(::Float64) + "where" f(x::T) where T <: Int8 + "where no varname" f(::T) where T <: String + end + """, "test2", Main) + # Simply test that the "bodies" of the doc exprs are not included as + # standalone expressions. + @test length(md[Main.DocstringSigsOnly]) == 6 # 1 func + 5 doc exprs + end + do_test("Undef in docstrings") && @testset "Undef in docstrings" begin fn = Base.find_source_file("abstractset.jl") # has lots of examples of """str""" func1, func2 mexsold = Revise.parse_source(fn, Base)