From 67699b1b7ce5f486ba90278809454d903981725c Mon Sep 17 00:00:00 2001 From: Fredrik Ekre Date: Thu, 9 Nov 2023 14:11:54 +0100 Subject: [PATCH] Filter out more doc signature expressions Revise extracts the documented expression and puts that before the full doc expression in the revision queue. E.g. in `"docs" f(x) = x` there will be a revision of `f(x) = x` followed by a revision of the full expresion (`"docs" f(x) = x`). However, in many cases a docstring is only attached to a signature (not a function body/struct definition, etc.) and in this case the revision of the signature is not needed. Revise already filters out the base case, `"docs" f(x)`. This patch extends this filtering to also apply to `where`-clauses such as e.g. `"docs" f(x::T) where T <: Integer`. Concretely, this patch fixes #735, i.e. revising doc expressions where the signature doesn't lower without the context of the doc macro. As a bonus, slightly less work has to be done for e.g. `"docs" f(x::T) where T` since this is also filtered out. --- src/utils.jl | 17 +++++++++++++++-- test/runtests.jl | 17 +++++++++++++++++ 2 files changed, 32 insertions(+), 2 deletions(-) diff --git a/src/utils.jl b/src/utils.jl index 626474eb..c436c1dc 100644 --- a/src/utils.jl +++ b/src/utils.jl @@ -90,11 +90,24 @@ istrivial(a) = a === nothing || isa(a, LineNumberNode) isgoto(stmt) = isa(stmt, Core.GotoNode) | isexpr(stmt, :gotoifnot) +# Check whether the documented expression is just a signature, catches e.g. +# - "docs" f(...) +# - "docs" f(::T) where T +# - "docs" f(::T, ::S) where T where S +function is_doc_expr_sig_only(ex::Expr) + @assert is_doc_expr(ex) + sig = ex.args[4] # sig or body + while isexpr(sig, :where) + sig = sig.args[1] + end + return isexpr(sig, :call) +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)` + if !is_doc_expr_sig_only(uex) + body = uex.args[4] 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)