https://github.com/JuliaLang/julia
Tip revision: b487ae69279fcfa077f1f9927dd01f0c1f42ed33 authored by Elliot Saba on 07 November 2022, 17:36:05 UTC
Change timeout kill signals to SIGQUIT from SIGTERM
Change timeout kill signals to SIGQUIT from SIGTERM
Tip revision: b487ae6
show.jl
# This file is a part of Julia. License is MIT: https://julialang.org/license
using LinearAlgebra
# For curmod_*
include("testenv.jl")
replstr(x, kv::Pair...) = sprint((io,x) -> show(IOContext(io, :limit => true, :displaysize => (24, 80), kv...), MIME("text/plain"), x), x)
showstr(x, kv::Pair...) = sprint((io,x) -> show(IOContext(io, :limit => true, :displaysize => (24, 80), kv...), x), x)
@testset "IOContext" begin
io = IOBuffer()
ioc = IOContext(io)
@test ioc.io == io
@test ioc.dict == Base.ImmutableDict{Symbol, Any}()
ioc = IOContext(io, :x => 1)
@test ioc.io == io
@test ioc.dict == Base.ImmutableDict{Symbol, Any}(:x, 1)
ioc = IOContext(io, :x => 1, :y => 2)
@test ioc.io == io
@test ioc.dict == Base.ImmutableDict(Base.ImmutableDict{Symbol, Any}(:x, 1),
:y => 2)
@test Base.ImmutableDict((key => ioc[key] for key in keys(ioc))...) == ioc.dict
@test keys(IOBuffer()) isa Base.KeySet
@test length(keys(IOBuffer())) == 0
end
@test replstr(Array{Any}(undef, 2)) == "2-element Vector{Any}:\n #undef\n #undef"
@test replstr(Array{Any}(undef, 2,2)) == "2×2 Matrix{Any}:\n #undef #undef\n #undef #undef"
@test replstr(Array{Any}(undef, 2,2,2)) == "2×2×2 Array{Any, 3}:\n[:, :, 1] =\n #undef #undef\n #undef #undef\n\n[:, :, 2] =\n #undef #undef\n #undef #undef"
@test replstr([1f10]) == "1-element Vector{Float32}:\n 1.0f10"
struct T5589
names::Vector{String}
end
@test replstr(T5589(Vector{String}(undef, 100))) == "$(curmod_prefix)T5589([#undef, #undef, #undef, #undef, #undef, #undef, #undef, #undef, #undef, #undef … #undef, #undef, #undef, #undef, #undef, #undef, #undef, #undef, #undef, #undef])"
@test replstr(Meta.parse("mutable struct X end")) == ":(mutable struct X\n #= none:1 =#\n end)"
@test replstr(Meta.parse("struct X end")) == ":(struct X\n #= none:1 =#\n end)"
let s = "ccall(:f, Int, (Ptr{Cvoid},), &x)"
@test replstr(Meta.parse(s)) == ":($s)"
end
# recursive array printing
# issue #10353
let a = Any[]
push!(a,a)
show(IOBuffer(), a)
push!(a,a)
show(IOBuffer(), a)
end
# expression printing
macro test_repr(x)
# this is a macro instead of function so we can avoid getting useful backtraces :)
return :(test_repr($(esc(x))))
end
macro weak_test_repr(x)
# this is a macro instead of function so we can avoid getting useful backtraces :)
return :(test_repr($(esc(x)), true))
end
function test_repr(x::String, remove_linenums::Bool = false)
# Note: We can't just compare x1 and x2 because interpolated
# strings get converted to string Exprs by the first show().
# This could produce a few false positives, but until string
# interpolation works we don't really have a choice.
#
# Rectification: comparing x1 and x2 seems to be working
x1 = Meta.parse(x)
x2 = eval(Meta.parse(repr(x1)))
x3 = eval(Meta.parse(repr(x2)))
if !remove_linenums
if ! (x1 == x2 == x3)
error(string(
"\nrepr test (Rule 2) failed:",
"\noriginal: ", x,
"\n\npreparsed: ", x1, "\n", sprint(dump, x1),
"\n\nparsed: ", x2, "\n", sprint(dump, x2),
"\n\nreparsed: ", x3, "\n", sprint(dump, x3),
"\n\n"))
end
@test x1 == x2 == x3
end
x4 = Base.remove_linenums!(Meta.parse(x))
x5 = eval(Base.remove_linenums!(Meta.parse(repr(x4))))
x6 = eval(Base.remove_linenums!(Meta.parse(repr(x5))))
if ! (x4 == x5 == x6)
error(string(
"\nrepr test (Rule 2) without line numbers failed:",
"\noriginal: ", x,
"\n\npreparsed: ", x4, "\n", sprint(dump, x4),
"\n\nparsed: ", x5, "\n", sprint(dump, x5),
"\n\nreparsed: ", x6, "\n", sprint(dump, x6),
"\n\n"))
end
@test x4 == x5 == x6
@test Base.remove_linenums!(x1) ==
Base.remove_linenums!(x2) ==
Base.remove_linenums!(x3) ==
x4 == x5 == x6
if isa(x1, Expr) && remove_linenums
if Base.remove_linenums!(Meta.parse(string(x1))) != x1
error(string(
"\nstring test (Rule 1) failed:",
"\noriginal: ", x,
"\n\npreparsed: ", x1, "\n", sprint(dump, x4),
"\n\nstring(preparsed): ", string(x1),
"\n\nBase.remove_linenums!(Meta.parse(string(preparsed))): ",
Base.remove_linenums!(Meta.parse(string(x1))), "\n",
sprint(dump, Base.remove_linenums!(Meta.parse(string(x1)))),
"\n\n"))
end
@test Base.remove_linenums!(Meta.parse(string(x1))) == x1
elseif isa(x1, Expr)
if Meta.parse(string(x1)) != x1
error(string(
"\nstring test (Rule 1) failed:",
"\noriginal: ", x,
"\n\npreparsed: ", x1, "\n", sprint(dump, x4),
"\n\nstring(preparsed): ", string(x1),
"\n\nMeta.parse(string(preparsed)): ",
Meta.parse(string(x1)), "\n",
sprint(dump, Meta.parse(string(x1))),
"\n\n"))
end
@test Meta.parse(string(x1)) == x1
end
end
# primitive types
@test_repr "x"
@test_repr "123"
@test_repr "\"123\""
@test_repr ":()"
@test_repr ":(x, y)"
# basic expressions
@test_repr "x + y"
@test_repr "2e"
@test_repr "2*e1"
@test_repr "2*E1"
@test_repr "2*f1"
@test_repr "0x00*a"
@test_repr "!x"
@test_repr "f(1, 2, 3)"
@test_repr "x = ~y"
@test_repr ":(:x, :y)"
@test_repr ":(:(:(x)))"
@test_repr "-\"\""
@test_repr "-(<=)"
@test_repr "\$x"
@test_repr "\$(\"x\")"
# order of operations
@test_repr "x + y * z"
@test_repr "x * y + z"
@test_repr "x * (y + z)"
@test_repr "!x^y"
@test_repr "!x^(y+z)"
@test_repr "!(x^y+z)"
@test_repr "x^-y"
@test_repr "x^-(y+z)"
@test_repr "x^-f(y+z)"
@test_repr "+(w-x)^-f(y+z)"
@test_repr "w = ((x = y) = z)" # parens aren't necessary, but not wrong
@test_repr "w = ((x, y) = z)" # parens aren't necessary, but not wrong
@test_repr "a & b && c"
@test_repr "a & (b && c)"
@test_repr "(a => b) in c"
@test_repr "a => b in c"
@test_repr "*(a..., b)"
@test_repr "+(a, b, c...)"
@test_repr "f((x...)...)"
# precedence tie resolution
@test_repr "(a * b) * (c * d)"
@test_repr "(a / b) / (c / d / e)"
@test_repr "(a == b == c) != (c == d < e)"
# Exponentiation (>= operator_precedence(:^)) and unary operators
@test_repr "(-1)^a"
@test_repr "(-2.1)^-1"
@test_repr "(-x)^a"
@test_repr "(-a)^-1"
@test_repr "(!x)↑!a"
@test_repr "(!x).a"
@test_repr "(!x)::a"
# invalid UTF-8 strings
@test_repr "\"\\ud800\""
@test_repr "\"\\udfff\""
@test_repr "\"\\xc0\\xb0\""
@test_repr "\"\\xe0\\xb0\\xb0\""
@test_repr "\"\\xf0\\xb0\\xb0\\xb0\""
# import statements
@test_repr "using A"
@test_repr "using A, B.C, D"
@test_repr "using A: b"
@test_repr "using A: a, x, y.z"
@test_repr "using A.B.C: a, x, y.z"
@test_repr "using ..A: a, x, y.z"
@test_repr "import A"
@test_repr "import A, B.C, D"
@test_repr "import A: b"
@test_repr "import A: a, x, y.z"
@test_repr "import A.B.C: a, x, y.z"
@test_repr "import ..A: a, x, y.z"
@test_repr "import A.B, C.D"
@test_repr "import A as B"
@test_repr "import A.x as y"
@test_repr "import A: x as y"
@test_repr "import A.B: x, y as z"
@test_repr "import A.B: x, y as z, a.b as c, xx"
# keyword args (issue #34023 and #32775)
@test_repr "f(a, b=c)"
@test_repr "f(a, b! = c)"
@test_repr "T{x=1}"
@test_repr "[a=1]"
@test_repr "a[x=1]"
@test_repr "f(; a=1)"
@test_repr "f(b=2; a=1)"
@test_repr "@f(1, y=3)"
@test_repr "n + (x=1)"
@test_repr "(;x=1)"
@test_repr "(x,;x=1)"
@test_repr "(a=1,;x=1)"
@test_repr "(a=1,b=2;x=1,y,:z=>2)"
@test repr(:((a,;b))) == ":((a,; b))"
@test repr(:((a=1,;x=2))) == ":((a = 1,; x = 2))"
@test repr(:((a=1,3;x=2))) == ":((a = 1, 3; x = 2))"
@test repr(:(g(a,; b))) == ":(g(a; b))"
@test repr(:(;)) == ":((;))"
@test repr(:(-(;x))) == ":(-(; x))"
@test repr(:(+(1, 2;x))) == ":(+(1, 2; x))"
@test repr(:(1:2...)) == ":(1:2...)"
@test repr(:(1 := 2)) == ":(1 := 2)"
@test repr(:(1 ≔ 2)) == ":(1 ≔ 2)"
@test repr(:(1 ⩴ 2)) == ":(1 ⩴ 2)"
@test repr(:(1 ≕ 2)) == ":(1 ≕ 2)"
@test repr(:(∓ 1)) == ":(∓1)"
@test repr(:(± 1)) == ":(±1)"
for ex in [Expr(:call, :f, Expr(:(=), :x, 1)),
Expr(:ref, :f, Expr(:(=), :x, 1)),
Expr(:vect, 1, 2, Expr(:kw, :x, 1)),
Expr(:kw, :a, :b),
Expr(:curly, :T, Expr(:kw, :x, 1)),
Expr(:call, :+, :n, Expr(:kw, :x, 1)),
:((a=1,; $(Expr(:(=), :x, 2)))),
:(($(Expr(:(=), :a, 1)),; x = 2)),
Expr(:tuple, Expr(:parameters)),
Expr(:call, :*, 0, :x01),
Expr(:call, :*, 0, :b01),
Expr(:call, :*, 0, :o01)]
@test eval(Meta.parse(repr(ex))) == ex
end
@test repr(Expr(:using, :Foo)) == ":(\$(Expr(:using, :Foo)))"
@test repr(Expr(:using, Expr(:(.), ))) == ":(\$(Expr(:using, :(\$(Expr(:.))))))"
@test repr(Expr(:import, :Foo)) == ":(\$(Expr(:import, :Foo)))"
@test repr(Expr(:import, Expr(:(.), ))) == ":(\$(Expr(:import, :(\$(Expr(:.))))))"
@test repr(Expr(:using, Expr(:(.), :A))) == ":(using A)"
@test repr(Expr(:using, Expr(:(.), :A),
Expr(:(.), :B))) == ":(using A, B)"
@test repr(Expr(:using, Expr(:(.), :A),
Expr(:(.), :B, :C),
Expr(:(.), :D))) == ":(using A, B.C, D)"
@test repr(Expr(:using, Expr(:(.), :A, :B),
Expr(:(.), :C, :D))) == ":(using A.B, C.D)"
@test repr(Expr(:import, Expr(:(.), :A))) == ":(import A)"
@test repr(Expr(:import, Expr(:(.), :A),
Expr(:(.), :B))) == ":(import A, B)"
@test repr(Expr(:import, Expr(:(.), :A),
Expr(:(.), :B, :(C)),
Expr(:(.), :D))) == ":(import A, B.C, D)"
@test repr(Expr(:import, Expr(:(.), :A, :B),
Expr(:(.), :C, :D))) == ":(import A.B, C.D)"
# https://github.com/JuliaLang/julia/issues/49168
@test repr(:(using A: (..))) == ":(using A: (..))"
@test repr(:(using A: (..) as twodots)) == ":(using A: (..) as twodots)"
# range syntax
@test_repr "1:2"
@test_repr "3:4:5"
let ex4 = Expr(:call, :(:), 1, 2, 3, 4),
ex1 = Expr(:call, :(:), 1)
@test eval(Meta.parse(repr(ex4))) == ex4
@test eval(Meta.parse(repr(ex1))) == ex1
end
# Complex
# Meta.parse(repr(:(...))) returns a double-quoted block, so we need to eval twice to unquote it
@test iszero(eval(eval(Meta.parse(repr(:($(1 + 2im) - $(1 + 2im)))))))
# control structures (shamelessly stolen from base/bitarray.jl)
@weak_test_repr """mutable struct BitArray{N} <: AbstractArray{Bool, N}
# line meta
chunks::Vector{UInt64}
# line meta
len::Int
# line meta
dims::NTuple{N,Int}
# line meta
function BitArray(undef, dims::Int...)
# line meta
if length(dims) != N
# line meta
error(\"number of dimensions must be \$N (got \$(length(dims)))\")
end
# line meta
n = 1
# line meta
for d in dims
# line meta
if d < 0
# line meta
error(\"dimension size must be non-negative (got \$d)\")
end
# line meta
n *= d
end
# line meta
nc = num_bit_chunks(n)
# line meta
chunks = Vector{UInt64}(undef, nc)
# line meta
if nc > 0
# line meta
chunks[end] = UInt64(0)
end
# line meta
b = new(chunks, n)
# line meta
if N != 1
# line meta
b.dims = dims
end
# line meta
return b
end
end"""
@weak_test_repr """function copy_chunks(dest::Vector{UInt64}, pos_d::Integer, src::Vector{UInt64}, pos_s::Integer, numbits::Integer)
# line meta
if numbits == 0
# line meta
return
end
# line meta
if dest === src && pos_d > pos_s
# line meta
return copy_chunks_rtol(dest, pos_d, pos_s, numbits)
end
# line meta
kd0, ld0 = get_chunks_id(pos_d)
# line meta
kd1, ld1 = get_chunks_id(pos_d + numbits - 1)
# line meta
ks0, ls0 = get_chunks_id(pos_s)
# line meta
ks1, ls1 = get_chunks_id(pos_s + numbits - 1)
# line meta
delta_kd = kd1 - kd0
# line meta
delta_ks = ks1 - ks0
# line meta
u = _msk64
# line meta
if delta_kd == 0
# line meta
msk_d0 = ~(u << ld0) | (u << ld1 << 1)
else
# line meta
msk_d0 = ~(u << ld0)
# line meta
msk_d1 = (u << ld1 << 1)
end
# line meta
if delta_ks == 0
# line meta
msk_s0 = (u << ls0) & ~(u << ls1 << 1)
else
# line meta
msk_s0 = (u << ls0)
end
# line meta
chunk_s0 = glue_src_bitchunks(src, ks0, ks1, msk_s0, ls0)
# line meta
dest[kd0] = (dest[kd0] & msk_d0) | ((chunk_s0 << ld0) & ~msk_d0)
# line meta
if delta_kd == 0
# line meta
return
end
# line meta
for i = 1 : kd1 - kd0 - 1
# line meta
chunk_s1 = glue_src_bitchunks(src, ks0 + i, ks1, msk_s0, ls0)
# line meta
chunk_s = (chunk_s0 >>> (63 - ld0) >>> 1) | (chunk_s1 << ld0)
# line meta
dest[kd0 + i] = chunk_s
# line meta
chunk_s0 = chunk_s1
end
# line meta
if ks1 >= ks0 + delta_kd
# line meta
chunk_s1 = glue_src_bitchunks(src, ks0 + delta_kd, ks1, msk_s0, ls0)
else
# line meta
chunk_s1 = UInt64(0)
end
# line meta
chunk_s = (chunk_s0 >>> (63 - ld0) >>> 1) | (chunk_s1 << ld0)
# line meta
dest[kd1] = (dest[kd1] & msk_d1) | (chunk_s & ~msk_d1)
# line meta
return
end"""
@weak_test_repr """if a
# line meta
b
end
"""
@weak_test_repr """if a
# line meta
b
elseif c
# line meta
d
end
"""
@weak_test_repr """if a
# line meta
b
elseif c
# line meta
d
else
# line meta
e
end
"""
@weak_test_repr """if a
# line meta
b
elseif c
# line meta
d
elseif e
# line meta
f
end
"""
@weak_test_repr """f(x, y) do z, w
# line meta
a
# line meta
b
end
"""
@weak_test_repr """f(x, y) do z
# line meta
a
# line meta
b
end
"""
# issue #7188
@test sprint(show, :foo) == ":foo"
@test sprint(show, Symbol("foo bar")) == "Symbol(\"foo bar\")"
@test sprint(show, Symbol("foo \"bar")) == "Symbol(\"foo \\\"bar\")"
@test sprint(show, :+) == ":+"
@test sprint(show, :end) == ":end"
# make sure :var"'" prints correctly
@test sprint(show, Symbol("'")) == "Symbol(\"'\")"
@test_repr "var\"'\" = 5"
# isidentifier
@test Meta.isidentifier("x")
@test Meta.isidentifier("x1")
@test !Meta.isidentifier("x.1")
@test !Meta.isidentifier("1x")
@test Meta.isidentifier(Symbol("x"))
@test Meta.isidentifier(Symbol("x1"))
@test !Meta.isidentifier(Symbol("x.1"))
@test !Meta.isidentifier(Symbol("1x"))
# issue #32408: Printing of names which are invalid identifiers
# Invalid identifiers which need `var` quoting:
@test sprint(show, Expr(:call, :foo, Symbol("##"))) == ":(foo(var\"##\"))"
@test sprint(show, Expr(:call, :foo, Symbol("a-b"))) == ":(foo(var\"a-b\"))"
@test sprint(show, :(export var"#")) == ":(export var\"#\")"
@test sprint(show, :(import A: var"#")) == ":(import A: var\"#\")"
@test sprint(show, :(macro var"#" end)) == ":(macro var\"#\" end)"
@test sprint(show, :"x$(var"#")y") == ":(\"x\$(var\"#\")y\")"
# Macro-like names outside macro calls
@test sprint(show, Expr(:call, :foo, Symbol("@bar"))) == ":(foo(var\"@bar\"))"
@test sprint(show, :(export @foo)) == ":(export @foo)"
@test sprint(show, :(import A.B: c.@d)) == ":(import A.B: c.@d)"
@test sprint(show, :(using A.@foo)) == ":(using A.@foo)"
# Hidden macro names
@test sprint(show, Expr(:macrocall, Symbol("@#"), nothing, :a)) == ":(@var\"#\" a)"
# Test that public expressions are rendered nicely
# though they are hard to create with quotes because public is not a context dependant keyword
@test sprint(show, Expr(:public, Symbol("@foo"))) == ":(public @foo)"
@test sprint(show, Expr(:public, :f,:o,:o)) == ":(public f, o, o)"
s = sprint(show, :(module A; public x; end))
@test match(r"^:\(module A\n #= .* =#\n #= .* =#\n public x\n end\)$", s) !== nothing
# PR #38418
module M1 var"#foo#"() = 2 end
@test occursin("M1.var\"#foo#\"", sprint(show, M1.var"#foo#", context = :module=>@__MODULE__))
# PR #43932
module var"#43932#" end
@test endswith(sprint(show, var"#43932#"), ".var\"#43932#\"")
# issue #12477
@test sprint(show, Union{Int64, Int32, Int16, Int8, Float64}) == "Union{Float64, Int16, Int32, Int64, Int8}"
# Function and array reference precedence
@test_repr "([2] + 3)[1]"
@test_repr "foo.bar[1]"
@test_repr "foo.bar()"
@test_repr "(foo + bar)()"
# issue #7921
@test replace(sprint(show, Expr(:function, :(==(a, b)), Expr(:block,:(return a == b)))),
r"\s+" => " ") == ":(function ==(a, b) return a == b end)"
# unicode operator printing
@test sprint(show, :(1 ⊕ (2 ⊗ 3))) == ":(1 ⊕ 2 ⊗ 3)"
@test sprint(show, :((1 ⊕ 2) ⊗ 3)) == ":((1 ⊕ 2) ⊗ 3)"
# issue #8155
@test_repr "foo(x,y; z=bar)"
@test_repr "foo(x,y,z=bar)"
@test_repr "Int[i for i=1:10]"
@test_repr "Int[(i, j) for (i, j) in zip(1:10,1:0)]"
@test_repr "[1 2 3; 4 5 6; 7 8 9]'"
@weak_test_repr "baremodule X
# line meta
# line meta
import ...B.c
# line meta
import D
# line meta
import B.C.D.E.F.g
end"
@weak_test_repr "baremodule Y
# line meta
# line meta
export A, B, C
# line meta
export D, E, F
end"
# issue #19840
@test_repr "Array{Int}(undef, 0)"
@test_repr "Array{Int}(undef, 0,0)"
@test_repr "Array{Int}(undef, 0,0,0)"
@test_repr "Array{Int}(undef, 0,1)"
@test_repr "Array{Int}(undef, 0,0,1)"
# issue #8994
@test_repr "get! => 2"
@test_repr "(<) : 2"
@test_repr "(<) :: T"
@test_repr "S{(<) <: T}"
@test_repr "+ + +"
# issue #9474
for s in ("(1::Int64 == 1::Int64)::Bool", "(1:2:3) + 4", "x = 1:2:3")
local s
@test sprint(show, Meta.parse(s)) == ":("*s*")"
end
# parametric type instantiation printing
struct TParametricPrint{a}; end
@test sprint(show, :(TParametricPrint{false}())) == ":(TParametricPrint{false}())"
# issue #9797
let q1 = Meta.parse(repr(:("$(a)b"))),
q2 = Meta.parse(repr(:("$ab")))
@test isa(q1, Expr)
@test q1.args[1].head === :string
@test q1.args[1].args == [:a, "b"]
@test isa(q2, Expr)
@test q2.args[1].head === :string
@test q2.args[1].args == [:ab,]
end
x8d003 = 2
let a = Expr(:quote,Expr(:$,:x8d003))
@test eval(Meta.parse(repr(a))) == a
@test eval(eval(Meta.parse(repr(a)))) == 2
end
# issue #11413
@test string(:(*{1, 2})) == "*{1, 2}"
@test string(:(*{1, x})) == "*{1, x}"
@test string(:(-{x})) == "-{x}"
# issue #11393
@test_repr "@m(x, y) + z"
@test_repr "(@m(x, y), z)"
@test_repr "[@m(x, y), z]"
@test_repr "A[@m(x, y), z]"
@test_repr "T{@m(x, y), z}"
@test_repr "@m x @n(y) z"
@test_repr "f(@m(x, y); z=@n(a))"
@test_repr "@m(x, y).z"
@test_repr "::@m(x, y) + z"
@test_repr "[@m(x) y z]"
@test_repr "[@m(x) y; z]"
test_repr("let @m(x), y=z; end", true)
@test repr(:(@m x y)) == ":(#= $(@__FILE__):$(@__LINE__) =# @m x y)"
@test string(:(@m x y)) == "#= $(@__FILE__):$(@__LINE__) =# @m x y"
@test string(:(@m x y;)) == "begin\n #= $(@__FILE__):$(@__LINE__) =# @m x y\nend"
# issue #11436
@test_repr "1 => 2 => 3"
@test_repr "1 => (2 => 3)"
@test_repr "(1 => 2) => 3"
# pr 12008
@test_repr "primitive type A B end"
@test_repr "primitive type B 100 end"
@test repr(:(primitive type A B end)) == ":(primitive type A B end)"
@test repr(:(primitive type B 100 end)) == ":(primitive type B 100 end)"
# `where` syntax
@test_repr "A where T<:B"
@test_repr "A where T<:(Array{T} where T<:Real)"
@test_repr "Array{T} where {S<:Real, T<:Array{S}}"
@test_repr "x::Array{T} where T"
@test_repr "(a::b) where T"
@test_repr "a::b where T"
@test_repr "X where (T=1)"
@test_repr "X where T = 1"
@test_repr "Array{<:Real}"
@test_repr "Array{>:Real}"
@test repr(Base.typename(Array)) == "typename(Array)"
let oldout = stdout, olderr = stderr
local rdout, wrout, rderr, wrerr, out, err, rd, wr, io
try
# pr 16917
rdout, wrout = redirect_stdout()
@test wrout === stdout
out = @async read(rdout, String)
rderr, wrerr = redirect_stderr()
@test wrerr === stderr
err = @async read(rderr, String)
@test dump(Int64) === nothing
if !Sys.iswindows()
close(wrout)
close(wrerr)
end
for io in (Core.stdout, Core.stderr)
Core.println(io, "TESTA")
println(io, "TESTB")
print(io, 'Α', 1)
Core.print(io, 'Β', 2)
Core.show(io, "A")
println(io)
end
Core.println("A")
Core.print("1", 2, 3.0)
Core.show("C")
Core.println()
redirect_stdout(oldout)
redirect_stderr(olderr)
close(wrout)
close(wrerr)
@test fetch(out) == "primitive type Int64 <: Signed\nTESTA\nTESTB\nΑ1Β2\"A\"\nA\n123\"C\"\n"
@test fetch(err) == "TESTA\nTESTB\nΑ1Β2\"A\"\n"
finally
redirect_stdout(oldout)
redirect_stderr(olderr)
end
end
let filename = tempname()
ret = open(filename, "w") do f
redirect_stdout(f) do
println("hello")
[1,3]
end
end
@test ret == [1,3]
@test chomp(read(filename, String)) == "hello"
ret = open(filename, "w") do f
redirect_stderr(f) do
println(stderr, "WARNING: hello")
[2]
end
end
@test ret == [2]
# stdin is unavailable on the workers. Run test on master.
@test occursin("WARNING: hello", read(filename, String))
ret = Core.eval(Main, quote
remotecall_fetch(1, $filename) do fname
open(fname) do f
redirect_stdin(f) do
readline()
end
end
end
end)
@test occursin("WARNING: hello", ret)
rm(filename)
end
# issue #13127
function f13127()
buf = IOBuffer()
f() = 1
show(buf, f)
String(take!(buf))
end
@test startswith(f13127(), "$(@__MODULE__).var\"#f")
@test startswith(sprint(show, typeof(x->x), context = :module=>@__MODULE__), "var\"")
#test methodshow.jl functions
@test Base.inbase(Base)
@test !Base.inbase(LinearAlgebra)
@test !Base.inbase(Core)
let repr = sprint(show, "text/plain", methods(Base.inbase))
@test occursin("inbase(m::Module)", repr)
end
let repr = sprint(show, "text/html", methods(Base.inbase))
@test occursin("inbase(m::<b>Module</b>)", repr)
end
f5971(x, y...; z=1, w...) = nothing
let repr = sprint(show, "text/plain", methods(f5971))
@test occursin("f5971(x, y...; z, w...)", repr)
end
let repr = sprint(show, "text/html", methods(f5971))
@test occursin("f5971(x, y...; <i>z, w...</i>)", repr)
end
f16580(x, y...; z=1, w=y+x, q...) = nothing
let repr = sprint(show, "text/html", methods(f16580))
@test occursin("f16580(x, y...; <i>z, w, q...</i>)", repr)
end
# Just check it doesn't error
f46594(::Vararg{T, 2}) where T = 1
let repr = sprint(show, "text/html", first(methods(f46594)))
@test occursin("f46594(::Vararg{T, 2}) where T", replace(repr, r"</?[A-Za-z]>"=>""))
end
function triangular_methodshow(x::T1, y::T2) where {T2<:Integer, T1<:T2}
end
let repr = sprint(show, "text/plain", methods(triangular_methodshow))
@test occursin("where {T2<:Integer, T1<:T2}", repr)
end
struct S45879{P} end
let ms = methods(S45879)
@test ms isa Base.MethodList
@test length(ms) == 0
@test sprint(show, Base.MethodList(Method[], typeof(S45879).name.mt)) isa String
end
function f49475(a=12.0; b) end
let ms = methods(f49475)
@test length(ms) == 2
repr1 = sprint(show, "text/plain", ms[1])
repr2 = sprint(show, "text/plain", ms[2])
@test occursin("f49475(; ...)", repr1) || occursin("f49475(; ...)", repr2)
end
if isempty(Base.GIT_VERSION_INFO.commit)
@test occursin("https://github.com/JuliaLang/julia/tree/v$VERSION/base/special/trig.jl#L", Base.url(which(sin, (Float64,))))
else
@test occursin("https://github.com/JuliaLang/julia/tree/$(Base.GIT_VERSION_INFO.commit)/base/special/trig.jl#L", Base.url(which(sin, (Float64,))))
end
# Method location correction (Revise integration)
dummyloc(m::Method) = :nofile, Int32(123456789)
Base.methodloc_callback[] = dummyloc
let repr = sprint(show, "text/plain", methods(Base.inbase))
@test occursin("nofile:123456789", repr)
end
let repr = sprint(show, "text/html", methods(Base.inbase))
@test occursin("nofile:123456789", repr)
end
Base.methodloc_callback[] = nothing
@testset "matrix printing" begin
# print_matrix should be able to handle small and large objects easily, test by
# calling show. This also indirectly tests print_matrix_row, which
# is used repeatedly by print_matrix.
# This fits on screen:
@test replstr(Matrix(1.0I, 10, 10)) == "10×10 Matrix{Float64}:\n 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0\n 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0\n 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0\n 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0 0.0\n 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0 0.0\n 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0 0.0\n 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0 0.0\n 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0 0.0\n 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0 0.0\n 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 1.0"
# an array too long vertically to fit on screen, and too long horizontally:
@test replstr(Vector(1.:100.)) == "100-element Vector{Float64}:\n 1.0\n 2.0\n 3.0\n 4.0\n 5.0\n 6.0\n 7.0\n 8.0\n 9.0\n 10.0\n ⋮\n 92.0\n 93.0\n 94.0\n 95.0\n 96.0\n 97.0\n 98.0\n 99.0\n 100.0"
@test occursin(r"1×100 adjoint\(::Vector{Float64}\) with eltype Float64:\n 1.0 2.0 3.0 4.0 5.0 6.0 7.0 … 95.0 96.0 97.0 98.0 99.0 100.0", replstr(Vector(1.:100.)'))
# too big in both directions to fit on screen:
@test replstr((1.:100.)*(1:100)') == "100×100 Matrix{Float64}:\n 1.0 2.0 3.0 4.0 5.0 6.0 … 97.0 98.0 99.0 100.0\n 2.0 4.0 6.0 8.0 10.0 12.0 194.0 196.0 198.0 200.0\n 3.0 6.0 9.0 12.0 15.0 18.0 291.0 294.0 297.0 300.0\n 4.0 8.0 12.0 16.0 20.0 24.0 388.0 392.0 396.0 400.0\n 5.0 10.0 15.0 20.0 25.0 30.0 485.0 490.0 495.0 500.0\n 6.0 12.0 18.0 24.0 30.0 36.0 … 582.0 588.0 594.0 600.0\n 7.0 14.0 21.0 28.0 35.0 42.0 679.0 686.0 693.0 700.0\n 8.0 16.0 24.0 32.0 40.0 48.0 776.0 784.0 792.0 800.0\n 9.0 18.0 27.0 36.0 45.0 54.0 873.0 882.0 891.0 900.0\n 10.0 20.0 30.0 40.0 50.0 60.0 970.0 980.0 990.0 1000.0\n ⋮ ⋮ ⋱ \n 92.0 184.0 276.0 368.0 460.0 552.0 8924.0 9016.0 9108.0 9200.0\n 93.0 186.0 279.0 372.0 465.0 558.0 9021.0 9114.0 9207.0 9300.0\n 94.0 188.0 282.0 376.0 470.0 564.0 9118.0 9212.0 9306.0 9400.0\n 95.0 190.0 285.0 380.0 475.0 570.0 9215.0 9310.0 9405.0 9500.0\n 96.0 192.0 288.0 384.0 480.0 576.0 … 9312.0 9408.0 9504.0 9600.0\n 97.0 194.0 291.0 388.0 485.0 582.0 9409.0 9506.0 9603.0 9700.0\n 98.0 196.0 294.0 392.0 490.0 588.0 9506.0 9604.0 9702.0 9800.0\n 99.0 198.0 297.0 396.0 495.0 594.0 9603.0 9702.0 9801.0 9900.0\n 100.0 200.0 300.0 400.0 500.0 600.0 9700.0 9800.0 9900.0 10000.0"
# test that no spurious visual lines are added when one element spans multiple lines
v = fill!(Array{Any}(undef, 9), 0)
v[1] = "look I'm wide! --- " ^ 9
r = replstr(v)
@test startswith(r, "9-element Vector{Any}:\n \"look I'm wide! ---")
@test endswith(r, "look I'm wide! --- \"\n 0\n 0\n 0\n 0\n 0\n 0\n 0\n 0")
# test vertical/diagonal ellipsis
v = fill!(Array{Any}(undef, 50), 0)
v[1] = "look I'm wide! --- " ^ 9
r = replstr(v)
@test startswith(r, "50-element Vector{Any}:\n \"look I'm wide! ---")
@test endswith(r, "look I'm wide! --- \"\n 0\n 0\n 0\n 0\n 0\n 0\n 0\n 0\n 0\n ⋮\n 0\n 0\n 0\n 0\n 0\n 0\n 0\n 0\n 0")
r = replstr([fill(0, 50) v])
@test startswith(r, "50×2 Matrix{Any}:\n 0 … \"look I'm wide! ---")
@test endswith(r, "look I'm wide! --- \"\n 0 0\n 0 0\n 0 0\n 0 0\n 0 … 0\n 0 0\n 0 0\n 0 0\n 0 0\n ⋮ ⋱ \n 0 0\n 0 0\n 0 0\n 0 0\n 0 … 0\n 0 0\n 0 0\n 0 0\n 0 0")
# issue #34659
@test replstr(Int32[]) == "Int32[]"
@test replstr([Int32[]]) == "1-element Vector{Vector{Int32}}:\n []"
@test replstr(permutedims([Int32[],Int32[]])) == "1×2 Matrix{Vector{Int32}}:\n [] []"
@test replstr(permutedims([Dict(),Dict()])) == "1×2 Matrix{Dict{Any, Any}}:\n Dict() Dict()"
@test replstr(permutedims([undef,undef])) == "1×2 Matrix{UndefInitializer}:\n UndefInitializer() UndefInitializer()"
@test replstr([zeros(3,0),zeros(2,0)]) == "2-element Vector{Matrix{Float64}}:\n 3×0 Matrix{Float64}\n 2×0 Matrix{Float64}"
end
# string show with elision
@testset "string show with elision" begin
@testset "elision logic" begin
strs = ["A", "∀", "∀A", "A∀", "😃"]
for limit = 0:100, len = 0:100, str in strs
str = str^len
str = str[1:nextind(str, 0, len)]
out = sprint() do io
show(io, MIME"text/plain"(), str; limit)
end
lower = length("\"\" ⋯ $(ncodeunits(str)) bytes ⋯ \"\"")
limit = max(limit, lower)
if length(str) + 2 ≤ limit
@test eval(Meta.parse(out)) == str
else
@test limit-!isascii(str) <= length(out) <= limit
re = r"(\"[^\"]*\") ⋯ (\d+) bytes ⋯ (\"[^\"]*\")"
m = match(re, out)
head = eval(Meta.parse(m.captures[1]))
tail = eval(Meta.parse(m.captures[3]))
skip = parse(Int, m.captures[2])
@test startswith(str, head)
@test endswith(str, tail)
@test ncodeunits(str) ==
ncodeunits(head) + skip + ncodeunits(tail)
end
end
end
@testset "default elision limit" begin
r = replstr("x"^1000)
@test length(r) == 7*80
@test r == repr("x"^271) * " ⋯ 459 bytes ⋯ " * repr("x"^270)
r = replstr(["x"^1000])
@test length(r) < 120
@test r == "1-element Vector{String}:\n " * repr("x"^31) * " ⋯ 939 bytes ⋯ " * repr("x"^30)
end
end
# Issue 14121
@test_repr "(A'x)'"
# issue #14481
@test_repr "in(1,2,3)"
@test_repr "<(1,2,3)"
@test_repr "+(1,2,3)"
@test_repr "-(1,2,3)"
@test_repr "*(1,2,3)"
# issue #15309
let ex,
l1 = Expr(:line, 42),
l2 = Expr(:line, 42, :myfile),
l2n = LineNumberNode(42)
@test string(l2n) == "#= line 42 =#"
@test string(l2) == "#= myfile:42 =#"
@test string(l1) == string(l2n)
ex = Expr(:block, l1, :x, l2, :y, l2n, :z)
@test replace(string(ex)," " => "") == replace("""
begin
#= line 42 =#
x
#= myfile:42 =#
y
#= line 42 =#
z
end""", " " => "")
end
# Test the printing of whatever form of line number representation
# that is used in the arguments to a macro looks the same as for
# regular quoting
macro strquote(ex)
return QuoteNode(string(ex))
end
let str_ex2a = @strquote(begin x end), str_ex2b = string(quote x end)
@test str_ex2a == str_ex2b
end
# test structured zero matrix printing for select structured types
let A = reshape(1:16, 4, 4)
@test occursin(r"4×4 (LinearAlgebra\.)?Diagonal{Int(32|64), Vector{Int(32|64)}}:\n 1 ⋅ ⋅ ⋅\n ⋅ 6 ⋅ ⋅\n ⋅ ⋅ 11 ⋅\n ⋅ ⋅ ⋅ 16", replstr(Diagonal(A)))
@test occursin(r"4×4 (LinearAlgebra\.)?Bidiagonal{Int(32|64), Vector{Int(32|64)}}:\n 1 5 ⋅ ⋅\n ⋅ 6 10 ⋅\n ⋅ ⋅ 11 15\n ⋅ ⋅ ⋅ 16", replstr(Bidiagonal(A, :U)))
@test occursin(r"4×4 (LinearAlgebra\.)?Bidiagonal{Int(32|64), Vector{Int(32|64)}}:\n 1 ⋅ ⋅ ⋅\n 2 6 ⋅ ⋅\n ⋅ 7 11 ⋅\n ⋅ ⋅ 12 16", replstr(Bidiagonal(A, :L)))
@test occursin(r"4×4 (LinearAlgebra\.)?SymTridiagonal{Int(32|64), Vector{Int(32|64)}}:\n 2 7 ⋅ ⋅\n 7 12 17 ⋅\n ⋅ 17 22 27\n ⋅ ⋅ 27 32", replstr(SymTridiagonal(A + A')))
@test occursin(r"4×4 (LinearAlgebra\.)?Tridiagonal{Int(32|64), Vector{Int(32|64)}}:\n 1 5 ⋅ ⋅\n 2 6 10 ⋅\n ⋅ 7 11 15\n ⋅ ⋅ 12 16", replstr(Tridiagonal(diag(A, -1), diag(A), diag(A, +1))))
@test occursin(r"4×4 (LinearAlgebra\.)?UpperTriangular{Int(32|64), Matrix{Int(32|64)}}:\n 1 5 9 13\n ⋅ 6 10 14\n ⋅ ⋅ 11 15\n ⋅ ⋅ ⋅ 16", replstr(UpperTriangular(copy(A))))
@test occursin(r"4×4 (LinearAlgebra\.)?LowerTriangular{Int(32|64), Matrix{Int(32|64)}}:\n 1 ⋅ ⋅ ⋅\n 2 6 ⋅ ⋅\n 3 7 11 ⋅\n 4 8 12 16", replstr(LowerTriangular(copy(A))))
end
# Vararg methods in method tables
function test_mt(f, str)
mt = methods(f)
@test length(mt) == 1
defs = first(mt)
io = IOBuffer()
show(io, defs)
strio = String(take!(io))
strio = split(strio, " at")[1]
@test strio[1:length(str)] == str
end
show_f1(x...) = [x...]
show_f2(x::Vararg{Any}) = [x...]
show_f3(x::Vararg) = [x...]
show_f4(x::Vararg{Any,3}) = [x...]
show_f5(A::AbstractArray{T, N}, indices::Vararg{Int,N}) where {T, N} = [indices...]
test_mt(show_f1, "show_f1(x...)")
test_mt(show_f2, "show_f2(x...)")
test_mt(show_f3, "show_f3(x...)")
test_mt(show_f4, "show_f4(x::Vararg{Any, 3})")
test_mt(show_f5, "show_f5(A::AbstractArray{T, N}, indices::Vararg{$Int, N})")
# Issue #15525, printing of vcat
@test sprint(show, :([a;])) == ":([a;])"
@test sprint(show, :([a; b])) == ":([a; b])"
@test_repr "[a;]"
@test_repr "[a; b]"
# other brackets and braces
@test_repr "[a]"
@test_repr "[a,b]"
@test_repr "[a;b;c]"
@test_repr "[a b]"
@test_repr "[a b;]"
@test_repr "[a b c]"
@test_repr "[a b; c d]"
@test_repr "{a}"
@test_repr "{a,b}"
@test_repr "{a;b;c}"
@test_repr "{a b}"
@test_repr "{a b;}"
@test_repr "{a b c}"
@test_repr "{a b; c d}"
# typed vcat and hcat
@test_repr "T[a]"
@test_repr "T[a,b]"
@test_repr "T[a;b;c]"
@test_repr "T[a b]"
@test_repr "T[a b;]"
@test_repr "T[a b c]"
@test_repr "T[a b; c d]"
@test_repr repr(Expr(:quote, Expr(:typed_vcat, Expr(:$, :a), 1)))
@test_repr repr(Expr(:quote, Expr(:typed_hcat, Expr(:$, :a), 1)))
@test_repr "Expr(:quote, Expr(:typed_vcat, Expr(:\$, :a), 1))"
@test_repr "Expr(:quote, Expr(:typed_hcat, Expr(:\$, :a), 1))"
@test_repr repr(Expr(:quote, Expr(:typed_vcat, Expr(:$, :a), 1)))
@test_repr repr(Expr(:quote, Expr(:typed_hcat, Expr(:$, :a), 1)))
# Printing of :(function f end)
@test sprint(show, :(function f end)) == ":(function f end)"
@test_repr "function g end"
# Printing of :(function (x...) end)
@test startswith(replstr(Meta.parse("function (x...) end")), ":(function (x...,)")
# Printing of macro definitions
@test sprint(show, :(macro m end)) == ":(macro m end)"
@test_repr "macro m end"
@test sprint(show, Expr(:macro, Expr(:call, :m, :ex), Expr(:block, :m))) ==
":(macro m(ex)\n m\n end)"
@weak_test_repr """macro identity(ex)
# line meta
esc(ex)
end"""
@weak_test_repr """macro m(a,b)
# line meta
quote
# line meta
\$a + \$b
end
end"""
# fallback printing + nested quotes and unquotes
@weak_test_repr repr(Expr(:block, LineNumberNode(0, :none),
Expr(:exotic_head, Expr(:$, :x))))
@test_repr repr(Expr(:exotic_head, Expr(:call, :+, 1, Expr(:quote, Expr(:$, Expr(:$, :y))))))
@test_repr repr(Expr(:quote, Expr(:$, Expr(:exotic_head, Expr(:call, :+, 1, Expr(:$, :y))))))
@test_repr repr(Expr(:$, Expr(:exotic_head, Expr(:call, :+, 1, Expr(:$, :y)))))
@test_repr "Expr(:block, LineNumberNode(0, :none), Expr(:exotic_head, Expr(:\$, :x)))"
@test_repr "Expr(:exotic_head, Expr(:call, :+, 1, \$y))"
@test_repr "Expr(:exotic_head, Expr(:call, :+, 1, \$\$y))"
@test_repr ":(Expr(:exotic_head, Expr(:call, :+, 1, \$y)))"
@test_repr ":(:(Expr(:exotic_head, Expr(:call, :+, 1, \$\$y))))"
@test repr(Expr(:exotic_head, Expr(:call, :+, 1, :(Expr(:$, :y))))) ==
":(\$(Expr(:exotic_head, :(1 + Expr(:\$, :y)))))"
@test repr(Expr(:block, Expr(:(=), :y, 2),
Expr(:quote, Expr(:exotic_head,
Expr(:call, :+, 1, Expr(:$, :y)))))) ==
"""
quote
y = 2
\$(Expr(:quote, :(\$(Expr(:exotic_head, :(1 + \$(Expr(:\$, :y))))))))
end"""
@test repr(eval(Expr(:block, Expr(:(=), :y, 2),
Expr(:quote, Expr(:exotic_head,
Expr(:call, :+, 1, Expr(:$, :y))))))) ==
":(\$(Expr(:exotic_head, :(1 + 2))))"
# nested quotes and blocks
@test_repr "Expr(:quote, Expr(:block, :a, :b))"
@weak_test_repr repr(Expr(:quote, Expr(:block, LineNumberNode(0, :none), :a, LineNumberNode(0, :none), :b)))
@test_broken repr(Expr(:quote, Expr(:block, :a, :b))) ==
":(quote
a
b
end)"
@test_repr "Expr(:quote, Expr(:block, :a))"
@weak_test_repr repr(Expr(:quote, Expr(:block, LineNumberNode(0, :none), :a)))
@test_broken repr(Expr(:quote, Expr(:block, :a))) ==
":(quote
a
end)"
@test_repr "Expr(:quote, Expr(:block, :(a + b)))"
@weak_test_repr repr(Expr(:quote, Expr(:block, LineNumberNode(0, :none), :(a + b))))
@test_broken repr(Expr(:quote, Expr(:block, :(a + b)))) ==
":(quote
a + b
end)"
# QuoteNode + quotes and unquotes
@test_repr "QuoteNode(\$x)"
@test_repr "QuoteNode(\$\$x)"
@test_repr ":(QuoteNode(\$x))"
@test_repr ":(:(QuoteNode(\$\$x)))"
@test repr(QuoteNode(Expr(:$, :x))) == ":(\$(QuoteNode(:(\$(Expr(:\$, :x))))))"
@test repr(QuoteNode(Expr(:quote, Expr(:$, :x)))) == ":(\$(QuoteNode(:(\$(Expr(:quote, :(\$(Expr(:\$, :x)))))))))"
@test repr(Expr(:quote, QuoteNode(Expr(:$, :x)))) == ":(\$(Expr(:quote, :(\$(QuoteNode(:(\$(Expr(:\$, :x)))))))))"
@test repr(Expr(:quote, Expr(:quote, Expr(:foo)))) == ":(\$(Expr(:quote, :(\$(Expr(:quote, :(\$(Expr(:foo)))))))))"
# unquoting
@test_repr "\$y"
@test_repr "\$\$y"
@weak_test_repr """
begin
# line meta
\$y
end"""
@weak_test_repr """
begin
# line meta
\$\$y
end"""
@test_repr ":(\$\$y)"
@test_repr repr(Expr(:$, :y))
# with reference to https://github.com/JuliaLang/julia/commit/9ef17207d5f99c7a0019cbbe0e58f77e7c4c1d21
y856739 = 2
x856739 = :y856739
z856739 = [:a, :b]
@test_repr repr(:(:(f($$x856739))))
@test_broken repr(:(:(f($$x856739)))) == ":(:(f(\$y856739)))"
@test repr(eval(:(:(f($$x856739))))) == ":(f(2))"
@test_repr repr(:(:(f($x856739))))
@test_broken repr(:(:(f($x856739)))) == ":(:(f(\$x856739)))"
@test repr(eval(:(:(f($x856739))))) == ":(f(y856739))"
@test_repr repr(:(:(f($(($z856739)...)))))
@test_broken repr(:(:(f($(($z856739)...))))) == ":(:(f(\$([:a, :b]...))))"
@test repr(eval(:(:(f($(($z856739)...)))))) == ":(f(a, b))"
# string interpolation, if this is what the comment in test_rep function
# definition talk about
@test repr(Expr(:string, "foo", :x, "bar")) == ":(\"foo\$(x)bar\")"
@test Meta.parse(string(Expr(:string, "foo", :x, "bar"))) == Expr(:string, "foo", :x, "bar")
@test repr(Meta.parse("\"foo\$(x)bar\"")) == ":(\"foo\$(x)bar\")"
@test_repr "\"foo\$(x)bar\""
# Printing of macrocall expressions with qualified macroname argument
@test sprint(show, Expr(:macrocall,
GlobalRef(Base, Symbol("@m")),
LineNumberNode(0, :none), :a, :b)) ==
":(#= none:0 =# Base.@m a b)"
@test sprint(show, Expr(:macrocall,
Expr(:(.), :Base, Expr(:quote, Symbol("@m"))),
LineNumberNode(0, :none), :a, :b)) ==
":(#= none:0 =# Base.@m a b)"
@test sprint(show, Expr(:macrocall,
Expr(:(.), Base, Expr(:quote, Symbol("@m"))),
LineNumberNode(0, :none), :a, :b)) ==
":(#= none:0 =# (Base).@m a b)"
@test sprint(show, Expr(:macrocall,
Expr(:(.), :Base, QuoteNode(Symbol("@m"))),
LineNumberNode(0, :none), :a, :b)) ==
":(#= none:0 =# Base.@m a b)"
@test sprint(show, Expr(:macrocall,
Expr(:(.), Base, QuoteNode(Symbol("@m"))),
LineNumberNode(0, :none), :a, :b)) ==
":(#= none:0 =# (Base).@m a b)"
# issue #34080
@test endswith(repr(:(a.b.@c x y)), "a.b.@c x y)")
@test endswith(repr(:((1+2).@x a)), "(1 + 2).@x a)")
@test repr(Expr(:(.),
Expr(:(.), :Base, QuoteNode(Symbol("Enums"))),
QuoteNode(Symbol("@enum")))) == ":(Base.Enums.var\"@enum\")"
# Printing of special macro syntaxes
# `a b c`
@test sprint(show, Expr(:macrocall,
GlobalRef(Core, Symbol("@cmd")),
LineNumberNode(0, :none), "a b c")) == ":(`a b c`)"
@test sprint(show, Expr(:macrocall,
Expr(:(.), :Core, Expr(:quote, Symbol("@cmd"))),
LineNumberNode(0, :none), "a b c")) == ":(#= none:0 =# Core.@cmd \"a b c\")"
@test sprint(show, Expr(:macrocall,
Expr(:(.), Core, Expr(:quote, Symbol("@cmd"))),
LineNumberNode(0, :none), "a b c")) == ":(#= none:0 =# (Core).@cmd \"a b c\")"
@test sprint(show, Expr(:macrocall,
Expr(:(.), :Core, QuoteNode(Symbol("@cmd"))),
LineNumberNode(0, :none), "a b c")) == ":(#= none:0 =# Core.@cmd \"a b c\")"
@test sprint(show, Expr(:macrocall,
Expr(:(.), Core, QuoteNode(Symbol("@cmd"))),
LineNumberNode(0, :none), "a b c")) == ":(#= none:0 =# (Core).@cmd \"a b c\")"
@test_repr "`a b c`"
@test sprint(show, Meta.parse("`a b c`")) == ":(`a b c`)"
# a"b" and a"b"c
@test_repr "a\"b\""
@test_repr "a\"b\"c"
@test_repr "aa\"b\""
@test_repr "a\"b\"cc"
@test sprint(show, Meta.parse("a\"b\"")) == ":(a\"b\")"
@test sprint(show, Meta.parse("a\"b\"c")) == ":(a\"b\"c)"
@test sprint(show, Meta.parse("aa\"b\"")) == ":(aa\"b\")"
@test sprint(show, Meta.parse("a\"b\"cc")) == ":(a\"b\"cc)"
@test sprint(show, Meta.parse("a\"\"\"issue \"35305\" \"\"\"")) == ":(a\"issue \\\"35305\\\" \")"
@test sprint(show, Meta.parse("a\"\$\"")) == ":(a\"\$\")"
@test sprint(show, Meta.parse("a\"\\b\"")) == ":(a\"\\b\")"
# 11111111111111111111, 0xfffffffffffffffff, 1111...many digits...
@test sprint(show, Meta.parse("11111111111111111111")) == ":(11111111111111111111)"
# @test_repr "Base.@int128_str \"11111111111111111111\""
@test sprint(show, Meta.parse("Base.@int128_str \"11111111111111111111\"")) ==
":(#= none:1 =# Base.@int128_str \"11111111111111111111\")"
@test sprint(show, Meta.parse("11111111111111111111")) == ":(11111111111111111111)"
@test sprint(show, Meta.parse("0xfffffffffffffffff")) == ":(0xfffffffffffffffff)"
@test sprint(show, Meta.parse("11111111111111111111111111111111111111111111111111111111111111")) ==
":(11111111111111111111111111111111111111111111111111111111111111)"
# Issue #15765 printing of continue and break
@test sprint(show, :(continue)) == ":(continue)"
@test sprint(show, :(break)) == ":(break)"
@test_repr "continue"
@test_repr "break"
let x = [], y = [], z = Base.ImmutableDict(x => y)
push!(x, y)
push!(y, x)
push!(y, z)
@test replstr(x) == "1-element Vector{Any}:\n Any[Any[#= circular reference @-2 =#], Base.ImmutableDict{Vector{Any}, Vector{Any}}([#= circular reference @-3 =#] => [#= circular reference @-2 =#])]"
@test repr(z) == "Base.ImmutableDict{Vector{Any}, Vector{Any}}([Any[Any[#= circular reference @-2 =#], Base.ImmutableDict{Vector{Any}, Vector{Any}}(#= circular reference @-3 =#)]] => [Any[Any[#= circular reference @-2 =#]], Base.ImmutableDict{Vector{Any}, Vector{Any}}(#= circular reference @-2 =#)])"
@test sprint(dump, x) == """
Array{Any}((1,))
1: Array{Any}((2,))
1: Array{Any}((1,))#= circular reference @-2 =#
2: Base.ImmutableDict{Vector{Any}, Vector{Any}}
parent: Base.ImmutableDict{Vector{Any}, Vector{Any}}
parent: #undef
key: #undef
value: #undef
key: Array{Any}((1,))#= circular reference @-3 =#
value: Array{Any}((2,))#= circular reference @-2 =#
"""
dz = sprint(dump, z)
@test 10 < countlines(IOBuffer(dz)) < 40
@test sum(Returns(1), eachmatch(r"circular reference", dz)) == 4
end
# PR 16221
# Printing of upper and lower bound of a TypeVar
@test string(TypeVar(:V, Signed, Real)) == "Signed<:V<:Real"
# Printing of primary type in type parameter place should not show the type
# parameter names.
@test string(Array) == "Array"
@test string(Tuple{Array}) == "Tuple{Array}"
# PR #16651
@test !occursin("\u2026", repr(fill(1.,10,10)))
@test occursin("\u2026", sprint((io, x) -> show(IOContext(io, :limit => true), x), fill(1.,30,30)))
let io = IOBuffer()
ioc = IOContext(io, :limit => true)
@test sprint(show, ioc) == "IOContext($(sprint(show, ioc.io)))"
end
@testset "PR 17117: print_array" begin
s = IOBuffer(Vector{UInt8}(), read=true, write=true)
Base.print_array(s, [1, 2, 3])
@test String(take!(s)) == " 1\n 2\n 3"
close(s)
s2 = IOBuffer(Vector{UInt8}(), read=true, write=true)
z = zeros(0,0,0,0,0,0,0,0)
Base.print_array(s2, z)
@test String(take!(s2)) == ""
close(s2)
end
module M30442
struct T end
end
@testset "Dump types" begin
let repr = sprint(dump, :(x = 1))
@test repr == "Expr\n head: Symbol =\n args: Array{Any}((2,))\n 1: Symbol x\n 2: $Int 1\n"
end
let repr = sprint(dump, Pair{String,Int64})
@test repr == "struct Pair{String, Int64} <: Any\n first::String\n second::Int64\n"
end
let repr = sprint(dump, Tuple)
@test repr == "Tuple <: Any\n"
end
let repr = sprint(dump, Int64)
@test repr == "primitive type Int64 <: Signed\n"
end
let repr = sprint(dump, Any)
@test repr == "abstract type Any\n"
end
let repr = sprint(dump, Integer)
@test occursin("abstract type Integer <: Real", repr)
@test !occursin("Any", repr)
end
let repr = sprint(dump, Union{Integer, Float32})
@test repr == "Union{Integer, Float32}\n" || repr == "Union{Float32, Integer}\n"
end
let repr = sprint(show, Union{String, M30442.T})
@test repr == "Union{$(curmod_prefix)M30442.T, String}" ||
repr == "Union{String, $(curmod_prefix)M30442.T}"
end
let repr = sprint(dump, Ptr{UInt8}(UInt(1)))
@test repr == "Ptr{UInt8}($(Base.repr(UInt(1))))\n"
end
let repr = sprint(dump, Core.svec())
@test repr == "empty SimpleVector\n"
end
let repr = sprint(dump, sin)
@test repr == "sin (function of type typeof(sin))\n"
end
let repr = sprint(dump, Test)
@test repr == "Module Test\n"
end
let repr = sprint(dump, nothing)
@test repr == "Nothing nothing\n"
end
let a = Vector{Any}(undef, 10000)
a[2] = "elemA"
a[4] = "elemB"
a[11] = "elemC"
repr = sprint(dump, a; context=(:limit => true), sizehint=0)
@test repr == "Array{Any}((10000,))\n 1: #undef\n 2: String \"elemA\"\n 3: #undef\n 4: String \"elemB\"\n 5: #undef\n ...\n 9996: #undef\n 9997: #undef\n 9998: #undef\n 9999: #undef\n 10000: #undef\n"
end
@test occursin("NamedTuple", sprint(dump, NamedTuple))
# issue 36495, dumping a partial NamedTupled shouldn't error
@test occursin("NamedTuple", sprint(dump, NamedTuple{(:foo,:bar)}))
end
# issue #17338
@test repr(Core.svec(1, 2)) == "svec(1, 2)"
# showing generator and comprehension expressions
@test repr(:(x for x in y for z in w)) == ":((x for x = y for z = w))"
@test repr(:(x for x in y if aa for z in w if bb)) == ":((x for x = y if aa for z = w if bb))"
@test repr(:([x for x = y])) == ":([x for x = y])"
@test repr(:([x for x = y if z])) == ":([x for x = y if z])"
@test repr(:(z for z = 1:5, y = 1:5)) == ":((z for z = 1:5, y = 1:5))"
@test_repr "(x for i in a, b in c)"
@test_repr "(x for a in b, c in d for e in f)"
for op in (:(.=), :(.+=), :(.&=))
@test repr(Meta.parse("x $op y")) == ":(x $op y)"
end
# pretty-printing of compact broadcast expressions (#17289)
@test repr(:(f.(X, Y))) == ":(f.(X, Y))"
@test repr(:(f.(X))) == ":(f.(X))"
@test repr(:(f.())) == ":(f.())"
# broadcasted operators (#26517)
@test_repr ":(y .= (+).(x, (*).(3, sin.(x))))"
@test repr(:(y .= (+).(x, (*).(3, (sin).(x))))) == ":(y .= (+).(x, (*).(3, sin.(x))))"
# pretty-printing of other `.` exprs
test_repr("a.b")
test_repr("a.in")
test_repr(":a.b")
test_repr("a.:+")
test_repr("(+).a")
test_repr("(+).:-")
test_repr("(!).:~")
test_repr("a.:(begin
#= none:3 =#
end)", true)
test_repr("a.:(=)")
test_repr("a.:(:)")
test_repr("(:).a")
@test eval(eval(Meta.parse(repr(:`ls x y`)))) == `ls x y`
@test repr(Expr(:., :a, :b, :c)) == ":(\$(Expr(:., :a, :b, :c)))"
@test repr(Expr(:., :a, :b)) == ":(\$(Expr(:., :a, :b)))"
@test repr(Expr(:., :a)) == ":(\$(Expr(:., :a)))"
@test repr(Expr(:.)) == ":(\$(Expr(:.)))"
@test repr(GlobalRef(Main, :a)) == ":(Main.a)"
@test repr(GlobalRef(Main, :in)) == ":(Main.in)"
@test repr(GlobalRef(Main, :+)) == ":(Main.:+)"
@test repr(GlobalRef(Main, :(:))) == ":(Main.:(:))"
# Test compact printing of homogeneous tuples
@test repr(NTuple{7,Int64}) == "NTuple{7, Int64}"
@test repr(Tuple{Float64, Float64, Float64, Float64}) == "NTuple{4, Float64}"
@test repr(Tuple{Float32, Float32, Float32}) == "Tuple{Float32, Float32, Float32}"
@test repr(Tuple{String, Int64, Int64, Int64}) == "Tuple{String, Int64, Int64, Int64}"
@test repr(Tuple{String, Int64, Int64, Int64, Int64}) == "Tuple{String, Vararg{Int64, 4}}"
@test repr(NTuple) == "NTuple{N, T} where {N, T}"
@test repr(Tuple{NTuple{N}, Vararg{NTuple{N}, 4}} where N) == "NTuple{5, NTuple{N, T} where T} where N"
@test repr(Tuple{Float64, NTuple{N}, Vararg{NTuple{N}, 4}} where N) == "Tuple{Float64, Vararg{NTuple{N, T} where T, 5}} where N"
# Test printing of NamedTuples using the macro syntax
@test repr(@NamedTuple{kw::Int64}) == "@NamedTuple{kw::Int64}"
@test repr(@NamedTuple{kw::Union{Float64, Int64}, kw2::Int64}) == "@NamedTuple{kw::Union{Float64, Int64}, kw2::Int64}"
@test repr(@NamedTuple{kw::@NamedTuple{kw2::Int64}}) == "@NamedTuple{kw::@NamedTuple{kw2::Int64}}"
@test repr(@NamedTuple{kw::NTuple{7, Int64}}) == "@NamedTuple{kw::NTuple{7, Int64}}"
@test repr(@NamedTuple{a::Float64, b}) == "@NamedTuple{a::Float64, b}"
@test repr(@NamedTuple{var"#"::Int64}) == "@NamedTuple{var\"#\"::Int64}"
# Test general printing of `Base.Pairs` (it should not use the `@Kwargs` macro syntax)
@test repr(@Kwargs{init::Int}) == "Base.Pairs{Symbol, $Int, Tuple{Symbol}, @NamedTuple{init::$Int}}"
@testset "issue #42931" begin
@test repr(NTuple{4, :A}) == "Tuple{:A, :A, :A, :A}"
@test repr(NTuple{3, :A}) == "Tuple{:A, :A, :A}"
@test repr(NTuple{2, :A}) == "Tuple{:A, :A}"
@test repr(NTuple{1, :A}) == "Tuple{:A}"
@test repr(NTuple{0, :A}) == "Tuple{}"
@test repr(Tuple{:A, :A, :A, :B}) == "Tuple{:A, :A, :A, :B}"
@test repr(Tuple{:A, :A, :A, :A}) == "Tuple{:A, :A, :A, :A}"
@test repr(Tuple{:A, :A, :A}) == "Tuple{:A, :A, :A}"
@test repr(Tuple{:A}) == "Tuple{:A}"
@test repr(Tuple{}) == "Tuple{}"
@test repr(Tuple{Vararg{N, 10}} where N) == "NTuple{10, N} where N"
@test repr(Tuple{Vararg{10, N}} where N) == "Tuple{Vararg{10, N}} where N"
end
# Test that REPL/mime display of invalid UTF-8 data doesn't throw an exception:
@test isa(repr("text/plain", String(UInt8[0x00:0xff;])), String)
# don't use julia-specific `f` in Float32 printing (PR #18053)
@test sprint(print, 1f-7) == "1.0e-7"
@test string(1f-7) == "1.0e-7"
let d = TextDisplay(IOBuffer())
@test_throws MethodError display(d, "text/foobar", [3 1 4])
try
display(d, "text/foobar", [3 1 4])
catch e
@test e.f == show
end
end
struct TypeWith4Params{a,b,c,d}
end
@test endswith(string(TypeWith4Params{Int8,Int8,Int8,Int8}), "TypeWith4Params{Int8, Int8, Int8, Int8}")
# issues #20332 and #20781
struct T20332{T}
end
(::T20332{T})(x) where T = 0
let m = which(T20332{Int}(), (Int,)),
mi = Core.Compiler.specialize_method(m, Tuple{T20332{T}, Int} where T, Core.svec())
# test that this doesn't throw an error
@test occursin("MethodInstance for", repr(mi))
# issue #41928
str = sprint(mi; context=:color=>true) do io, mi
printstyled(io, mi; color=:light_cyan)
end
@test !occursin("\U1b[0m", str)
end
@test sprint(show, Main) == "Main"
@test sprint(Base.show_supertypes, Int64) == "Int64 <: Signed <: Integer <: Real <: Number <: Any"
@test sprint(Base.show_supertypes, Vector{String}) == "Vector{String} <: DenseVector{String} <: AbstractVector{String} <: Any"
# static_show
function static_shown(x)
p = Pipe()
Base.link_pipe!(p, reader_supports_async=true, writer_supports_async=true)
ccall(:jl_static_show, Cvoid, (Ptr{Cvoid}, Any), p.in, x)
@async close(p.in)
return read(p.out, String)
end
# Test for PR 17803
@test static_shown(Int128(-1)) == "Int128(0xffffffffffffffffffffffffffffffff)"
# PR #22160
@test static_shown(:aa) == ":aa"
@test static_shown(:+) == ":+"
@test static_shown(://) == "://"
@test static_shown(://=) == "://="
@test static_shown(Symbol("")) == ":var\"\""
@test static_shown(Symbol("a/b")) == ":var\"a/b\""
@test static_shown(Symbol("a-b")) == ":var\"a-b\""
@test static_shown(UnionAll) == "UnionAll"
@test static_shown(QuoteNode(:x)) == ":(:x)"
@test static_shown(:!) == ":!"
@test static_shown("\"") == "\"\\\"\""
@test static_shown("\$") == "\"\\\$\""
@test static_shown("\\") == "\"\\\\\""
@test static_shown("a\x80b") == "\"a\\x80b\""
@test static_shown("a\x80\$\\b") == "\"a\\x80\\\$\\\\b\""
@test static_shown(GlobalRef(Main, :var"a#b")) == "Main.var\"a#b\""
@test static_shown(GlobalRef(Main, :+)) == "Main.:(+)"
@test static_shown((a = 3, ! = 4, var"a b" = 5)) == "(a=3, (!)=4, var\"a b\"=5)"
# PR #38049
@test static_shown(sum) == "Base.sum"
@test static_shown(+) == "Base.:(+)"
@test static_shown(typeof(+)) == "typeof(Base.:(+))"
struct var"#X#" end
var"#f#"() = 2
struct var"%X%" end # Invalid name without '#'
# (Just to make this test more sustainable,) we don't necessarily need to test the exact
# output format, just ensure that it prints at least the parts we expect:
@test occursin(".var\"#X#\"", static_shown(var"#X#")) # Leading `.` tests it printed a module name.
@test occursin(r"Set{var\"[^\"]+\"} where var\"[^\"]+\"", static_shown(Set{<:Any}))
# Test that static_shown is returning valid, correct julia expressions
@testset "static_show() prints valid julia" begin
@testset for v in (
var"#X#",
var"#X#"(),
var"%X%",
var"%X%"(),
Vector,
Vector{<:Any},
Vector{var"#X#"},
+,
typeof(+),
var"#f#",
typeof(var"#f#"),
)
@test v == eval(Meta.parse(static_shown(v)))
end
end
# Test that static show prints something reasonable for `<:Function` types
@test static_shown(:) == "Base.Colon()"
# Test @show
let fname = tempname()
try
open(fname, "w") do fout
redirect_stdout(fout) do
@show zeros(2, 2)
end
end
@test read(fname, String) == "zeros(2, 2) = [0.0 0.0; 0.0 0.0]\n"
finally
rm(fname, force=true)
end
end
module ModFWithParams
struct f_with_params{t} <: Function end
(::f_with_params)(x) = 2x
end
let io = IOBuffer()
show(io, MIME"text/html"(), ModFWithParams.f_with_params.body.name.mt)
@test occursin("ModFWithParams.f_with_params", String(take!(io)))
end
@testset "printing of Val's" begin
@test sprint(show, Val(Float64)) == "Val{Float64}()" # Val of a type
@test sprint(show, Val(:Float64)) == "Val{:Float64}()" # Val of a symbol
@test sprint(show, Val(true)) == "Val{true}()" # Val of a value
end
@testset "printing of Pair's" begin
for (p, s) in (Pair(1.0,2.0) => "1.0 => 2.0",
Pair(Pair(1,2), Pair(3,4)) => "(1 => 2) => (3 => 4)",
Pair{Integer,Int64}(1, 2) => "Pair{Integer, Int64}(1, 2)",
(Pair{Integer,Int64}(1, 2) => 3) => "Pair{Integer, Int64}(1, 2) => 3",
((1+2im) => (3+4im)) => "1 + 2im => 3 + 4im",
(1 => 2 => Pair{Real,Int64}(3, 4)) => "1 => (2 => Pair{Real, Int64}(3, 4))")
local s
@test sprint(show, p) == s
end
# - when the context has :compact=>false, print pair's member non-compactly
# - if one member is printed as "Pair{...}(...)", no need to put parens around
s = IOBuffer()
show(IOContext(s, :compact => false), (1=>2) => Pair{Any,Any}(3,4))
@test String(take!(s)) == "(1 => 2) => Pair{Any, Any}(3, 4)"
# issue #28327
d = Dict(Pair{Integer,Integer}(1,2)=>Pair{Integer,Integer}(1,2))
@test showstr(d) == "Dict{Pair{Integer, Integer}, Pair{Integer, Integer}}((1 => 2) => (1 => 2))" # correct parenthesis
# issue #29536
d = Dict((+)=>1)
@test showstr(d) == "Dict((+) => 1)"
d = Dict("+"=>1)
@test showstr(d) == "Dict(\"+\" => 1)"
struct Foo
a::Int
end
struct Bar
a::Int
end
d = Dict([Bar(1), Bar(2)] => [Foo(1), Foo(2)])
@test showstr(d) == "Dict{Vector{$(curmod_prefix)Bar}, Vector{$(curmod_prefix)Foo}}([$(curmod_prefix)Bar(1), $(curmod_prefix)Bar(2)] => [$(curmod_prefix)Foo(1), $(curmod_prefix)Foo(2)])"
@test sprint(show, MIME("text/plain"), d) == """
Dict{Vector{$(curmod_prefix)Bar}, Vector{$(curmod_prefix)Foo}} with 1 entry:
[Bar(1), Bar(2)] => [Foo(1), Foo(2)]"""
end
@testset "alignment for pairs" begin # (#22899)
@test replstr([1=>22,33=>4]) == "2-element Vector{Pair{$Int, $Int}}:\n 1 => 22\n 33 => 4"
# first field may have "=>" in its representation
@test replstr(Pair[(1=>2)=>3, 4=>5]) ==
"2-element Vector{Pair}:\n (1 => 2) => 3\n 4 => 5"
@test replstr(Any[Dict(1=>2)=> (3=>4), 1=>2]) ==
"2-element Vector{Any}:\n Dict(1 => 2) => (3 => 4)\n 1 => 2"
# left-alignment when not using the "=>" symbol
@test replstr(Any[Pair{Integer,Int64}(1, 2), Pair{Integer,Int64}(33, 4)]) ==
"2-element Vector{Any}:\n Pair{Integer, Int64}(1, 2)\n Pair{Integer, Int64}(33, 4)"
end
@testset "alignment for complex arrays" begin # (#34763)
@test replstr([ 1e-7 + 2.0e-11im, 2.0e-5 + 4e0im]) == "2-element Vector{ComplexF64}:\n 1.0e-7 + 2.0e-11im\n 2.0e-5 + 4.0im"
@test replstr([ 1f-7 + 2.0f-11im, 2.0f-5 + 4f0im]) == "2-element Vector{ComplexF32}:\n 1.0f-7 + 2.0f-11im\n 2.0f-5 + 4.0f0im"
end
@testset "display arrays non-compactly when size(⋅, 2) == 1" begin
# 0-dim
@test replstr(zeros(Complex{Int})) == "0-dimensional Array{Complex{$Int}, 0}:\n0 + 0im"
A = Array{Pair,0}(undef); A[] = 1=>2
@test replstr(A) == "0-dimensional Array{Pair, 0}:\n1 => 2"
# 1-dim
@test replstr(zeros(Complex{Int}, 2)) ==
"2-element Vector{Complex{$Int}}:\n 0 + 0im\n 0 + 0im"
@test replstr([1=>2, 3=>4]) == "2-element Vector{Pair{$Int, $Int}}:\n 1 => 2\n 3 => 4"
# 2-dim
@test replstr(zeros(Complex{Int}, 2, 1)) ==
"2×1 Matrix{Complex{$Int}}:\n 0 + 0im\n 0 + 0im"
@test replstr(zeros(Complex{Int}, 1, 2)) ==
"1×2 Matrix{Complex{$Int}}:\n 0+0im 0+0im"
@test replstr([1=>2 3=>4]) == "1×2 Matrix{Pair{$Int, $Int}}:\n 1=>2 3=>4"
@test replstr([1=>2 for x in 1:2, y in 1:1]) ==
"2×1 Matrix{Pair{$Int, $Int}}:\n 1 => 2\n 1 => 2"
# 3-dim
@test replstr(zeros(Complex{Int}, 1, 1, 1)) ==
"1×1×1 Array{Complex{$Int}, 3}:\n[:, :, 1] =\n 0 + 0im"
@test replstr(zeros(Complex{Int}, 1, 2, 1)) ==
"1×2×1 Array{Complex{$Int}, 3}:\n[:, :, 1] =\n 0+0im 0+0im"
end
@testset "arrays printing follows the :compact property when specified" begin
x = 3.141592653589793
@test showstr(x) == "3.141592653589793"
@test showstr([x, x], :compact => true) == "[3.14159, 3.14159]"
@test showstr([x, x]) == showstr([x, x], :compact => false) == "[3.141592653589793, 3.141592653589793]"
@test showstr([x x; x x], :compact => true) ==
"[3.14159 3.14159; 3.14159 3.14159]"
@test showstr([x x; x x]) == showstr([x x; x x], :compact => false) ==
"[3.141592653589793 3.141592653589793; 3.141592653589793 3.141592653589793]"
@test replstr([x, x], :compact => false) ==
"2-element Array{Float64, 1}:\n 3.141592653589793\n 3.141592653589793"
@test replstr([x, x]) == "2-element Vector{Float64}:\n 3.141592653589793\n 3.141592653589793"
@test replstr([x, x], :compact => true) == "2-element Vector{Float64}:\n 3.14159\n 3.14159"
@test replstr([x x; x x]) == replstr([x x; x x], :compact => true) ==
"2×2 Matrix{Float64}:\n 3.14159 3.14159\n 3.14159 3.14159"
@test showstr([x x; x x], :compact => false) ==
"[3.141592653589793 3.141592653589793; 3.141592653589793 3.141592653589793]"
end
@testset "Array printing with limited rows" begin
arrstr = let buf = IOBuffer()
function (A, rows)
show(IOContext(buf, :displaysize => (rows, 80), :limit => true), "text/plain", A)
String(take!(buf))
end
end
A = Int64[1]
@test arrstr(A, 4) == "1-element Vector{Int64}: …"
@test arrstr(A, 5) == "1-element Vector{Int64}:\n 1"
push!(A, 2)
@test arrstr(A, 5) == "2-element Vector{Int64}:\n ⋮"
@test arrstr(A, 6) == "2-element Vector{Int64}:\n 1\n 2"
push!(A, 3)
@test arrstr(A, 6) == "3-element Vector{Int64}:\n 1\n ⋮"
@test arrstr(zeros(4, 3), 4) == "4×3 Matrix{Float64}: …"
@test arrstr(zeros(4, 30), 4) == "4×30 Matrix{Float64}: …"
@test arrstr(zeros(4, 3), 5) == "4×3 Matrix{Float64}:\n ⋮ ⋱ "
@test arrstr(zeros(4, 30), 5) == "4×30 Matrix{Float64}:\n ⋮ ⋱ "
@test arrstr(zeros(4, 3), 6) == "4×3 Matrix{Float64}:\n 0.0 0.0 0.0\n ⋮ "
@test arrstr(zeros(4, 30), 6) ==
string("4×30 Matrix{Float64}:\n",
" 0.0 0.0 0.0 0.0 0.0 0.0 0.0 0.0 … 0.0 0.0 0.0 0.0 0.0 0.0 0.0\n",
" ⋮ ⋮ ⋱ ⋮ ")
end
module UnexportedOperators
function + end
function == end
end
@testset "Parseable printing of types" begin
@test repr(typeof(print)) == "typeof(print)"
@test repr(typeof(Base.show_default)) == "typeof(Base.show_default)"
@test repr(typeof(UnexportedOperators.:+)) == "typeof($(curmod_prefix)UnexportedOperators.:+)"
@test repr(typeof(UnexportedOperators.:(==))) == "typeof($(curmod_prefix)UnexportedOperators.:(==))"
anonfn = x->2x
modname = string(@__MODULE__)
anonfn_type_repr = "$modname.var\"$(typeof(anonfn).name.name)\""
@test repr(typeof(anonfn)) == anonfn_type_repr
@test repr(anonfn) == anonfn_type_repr * "()"
@test repr("text/plain", anonfn) == "$(typeof(anonfn).name.mt.name) (generic function with 1 method)"
mkclosure = x->y->x+y
clo = mkclosure(10)
@test repr("text/plain", clo) == "$(typeof(clo).name.mt.name) (generic function with 1 method)"
@test repr(UnionAll) == "UnionAll"
end
let x = TypeVar(:_), y = TypeVar(:_)
@test repr(UnionAll(x, UnionAll(y, Pair{x,y}))) == "Pair"
@test repr(UnionAll(y, UnionAll(x, Pair{x,y}))) == "Pair{_2, _1} where {_1, _2}"
@test repr(UnionAll(x, UnionAll(y, Pair{UnionAll(x,Ref{x}),y}))) == "Pair{Ref}"
@test repr(UnionAll(y, UnionAll(x, Pair{UnionAll(y,Ref{x}),y}))) == "Pair{Ref{_2}, _1} where {_1, _2}"
end
let x, y, x
x = TypeVar(:a)
y = TypeVar(:a)
z = TypeVar(:a)
@test repr(UnionAll(z, UnionAll(x, UnionAll(y, Tuple{x,y,z})))) == "Tuple{a1, a2, a} where {a, a1, a2}"
@test repr(UnionAll(z, UnionAll(x, UnionAll(y, Tuple{z,y,x})))) == "Tuple{a, a2, a1} where {a, a1, a2}"
end
let x = TypeVar(:_, Number), y = TypeVar(:_, Number)
@test repr(UnionAll(x, UnionAll(y, Pair{x,y}))) == "Pair{_1, _2} where {_1<:Number, _2<:Number}"
@test repr(UnionAll(y, UnionAll(x, Pair{x,y}))) == "Pair{_2, _1} where {_1<:Number, _2<:Number}"
@test repr(UnionAll(x, UnionAll(y, Pair{UnionAll(x,Ref{x}),y}))) == "Pair{Ref{_1} where _1<:Number, _1} where _1<:Number"
@test repr(UnionAll(y, UnionAll(x, Pair{UnionAll(y,Ref{x}),y}))) == "Pair{Ref{_2}, _1} where {_1<:Number, _2<:Number}"
end
is_juliarepr(x) = eval(Meta.parse(repr(x))) == x
@testset "unionall types" begin
X = TypeVar(gensym())
Y = TypeVar(gensym(), Ref, Ref)
x, y, z = TypeVar(:a), TypeVar(:a), TypeVar(:a)
struct TestTVUpper{A<:Integer} end
# named typevars
@test is_juliarepr(Ref{A} where A)
@test is_juliarepr(Ref{A} where A>:Ref)
@test is_juliarepr(Ref{A} where A<:Ref)
@test is_juliarepr(Ref{A} where Ref<:A<:Ref)
@test is_juliarepr(TestTVUpper{<:Real})
@test is_juliarepr(TestTVUpper{<:Integer})
@test is_juliarepr(TestTVUpper{<:Signed})
# typearg order
@test is_juliarepr(UnionAll(X, Pair{X,<:Any}))
@test is_juliarepr(UnionAll(X, Pair{<:Any,X}))
# duplicates
@test is_juliarepr(UnionAll(X, Pair{X,X}))
# nesting
@test is_juliarepr(UnionAll(X, Ref{Ref{X}}))
@test is_juliarepr(Union{T, Int} where T)
@test is_juliarepr(Pair{A, <:A} where A)
# renumbered typevars with same names
@test is_juliarepr(UnionAll(z, UnionAll(x, UnionAll(y, Tuple{x,y,z}))))
# shortened typevar printing
@test repr(Ref{<:Any}) == "Ref"
@test repr(Pair{1, <:Any}) == "Pair{1}"
@test repr(Ref{<:Number}) == "Ref{<:Number}"
@test repr(Pair{1, <:Number}) == "Pair{1, <:Number}"
@test repr(Ref{<:Ref}) == "Ref{<:Ref}"
@test repr(Ref{>:Ref}) == "Ref{>:Ref}"
@test repr(Pair{<:Any, 1}) == "Pair{<:Any, 1}"
yname = sprint(Base.show_unquoted, Y.name)
@test repr(UnionAll(Y, Ref{Y})) == "Ref{$yname} where Ref<:$yname<:Ref"
@test endswith(repr(TestTVUpper{<:Real}), "TestTVUpper{<:Real}")
@test endswith(repr(TestTVUpper), "TestTVUpper")
@test endswith(repr(TestTVUpper{<:Signed}), "TestTVUpper{<:Signed}")
# exception for tuples
@test is_juliarepr(Tuple)
@test is_juliarepr(Tuple{})
@test is_juliarepr(Tuple{<:Any})
end
@testset "showarg" begin
io = IOBuffer()
A = reshape(Vector(Int16(1):Int16(2*3*5)), 2, 3, 5)
@test summary(A) == "2×3×5 Array{Int16, 3}"
v = view(A, :, 3, 2:5)
@test summary(v) == "2×4 view(::Array{Int16, 3}, :, 3, 2:5) with eltype Int16"
@test Base.showarg(io, v, false) === nothing
@test String(take!(io)) == "view(::Array{Int16, 3}, :, 3, 2:5)"
r = reshape(v, 4, 2)
@test summary(r) == "4×2 reshape(view(::Array{Int16, 3}, :, 3, 2:5), 4, 2) with eltype Int16"
@test Base.showarg(io, r, false) === nothing
@test String(take!(io)) == "reshape(view(::Array{Int16, 3}, :, 3, 2:5), 4, 2)"
p = PermutedDimsArray(r, (2, 1))
@test summary(p) == "2×4 PermutedDimsArray(reshape(view(::Array{Int16, 3}, :, 3, 2:5), 4, 2), (2, 1)) with eltype Int16"
@test Base.showarg(io, p, false) === nothing
@test String(take!(io)) == "PermutedDimsArray(reshape(view(::Array{Int16, 3}, :, 3, 2:5), 4, 2), (2, 1))"
p = reinterpret(reshape, Tuple{Float32,Float32}, [1.0f0 3.0f0; 2.0f0 4.0f0])
@test summary(p) == "2-element reinterpret(reshape, Tuple{Float32, Float32}, ::Matrix{Float32}) with eltype Tuple{Float32, Float32}"
@test Base.showarg(io, p, false) === nothing
@test String(take!(io)) == "reinterpret(reshape, Tuple{Float32, Float32}, ::Matrix{Float32})"
r = Base.IdentityUnitRange(2:2)
B = @view ones(2)[r]
Base.showarg(io, B, false)
@test String(take!(io)) == "view(::Vector{Float64}, $(repr(r)))"
end
@testset "Methods" begin
m = which(sin, (Float64,))
io = IOBuffer()
show(io, "text/html", m)
s = String(take!(io))
@test occursin(" in Base.Math ", s)
end
module AlsoExportsPair
Pair = 0
export Pair
end
module TestShowType
export TypeA
struct TypeA end
using ..AlsoExportsPair
end
@testset "module prefix when printing type" begin
@test sprint(show, TestShowType.TypeA) == "$(@__MODULE__).TestShowType.TypeA"
b = IOBuffer()
show(IOContext(b, :module => @__MODULE__), TestShowType.TypeA)
@test String(take!(b)) == "$(@__MODULE__).TestShowType.TypeA"
b = IOBuffer()
show(IOContext(b, :module => TestShowType), TestShowType.TypeA)
@test String(take!(b)) == "TypeA"
using .TestShowType
@test sprint(show, TypeA) == "$(@__MODULE__).TestShowType.TypeA"
b = IOBuffer()
show(IOContext(b, :module => @__MODULE__), TypeA)
@test String(take!(b)) == "TypeA"
# issue #26354; make sure testing for symbol visibility doesn't cause
# spurious binding resolutions
show(IOContext(b, :module => TestShowType), Base.Pair)
@test !Base.isbindingresolved(TestShowType, :Pair)
@test String(take!(b)) == "Core.Pair"
show(IOContext(b, :module => TestShowType), Base.Complex)
@test Base.isbindingresolved(TestShowType, :Complex)
@test String(take!(b)) == "Complex"
end
@testset "typeinfo" begin
@test replstr([[Int16(1)]]) == "1-element Vector{Vector{Int16}}:\n [1]"
@test showstr([[Int16(1)]]) == "Vector{Int16}[[1]]"
@test showstr(Set([[Int16(1)]])) == "Set(Vector{Int16}[[1]])"
@test showstr([Float16(1)]) == "Float16[1.0]"
@test showstr([[Float16(1)]]) == "Vector{Float16}[[1.0]]"
@test replstr(Real[Float16(1)]) == "1-element Vector{Real}:\n Float16(1.0)"
@test replstr(Array{Real}[Real[1]]) == "1-element Vector{Array{Real}}:\n [1]"
# printing tuples (Issue #25042)
@test replstr(fill((Int64(1), zeros(Float16, 3)), 1)) ==
"1-element Vector{Tuple{Int64, Vector{Float16}}}:\n (1, [0.0, 0.0, 0.0])"
@testset "nested Any eltype" begin
x = Any[Any[Any[1]]]
# The element of x (i.e. x[1]) has an eltype which can't be deduced
# from eltype(x), so this must also be printed
@test replstr(x) == "1-element Vector{Any}:\n Any[Any[1]]"
end
# Issue #25038
A = [0.0, 1.0]
@test replstr(view(A, [1], :)) == "1×1 view(::Matrix{Float64}, [1], :) with eltype Float64:\n 0.0"
# issue #27680
@test showstr(Set([(1.0,1.0), (2.0,2.0), (3.0, 3.0)])) == (sizeof(Int) == 8 ?
"Set([(1.0, 1.0), (3.0, 3.0), (2.0, 2.0)])" :
"Set([(1.0, 1.0), (2.0, 2.0), (3.0, 3.0)])")
# issue #27747
let t = (x = Integer[1, 2],)
v = [t, t]
@test showstr(v) == "@NamedTuple{x::Vector{Integer}}[(x = [1, 2],), (x = [1, 2],)]"
@test replstr(v) == "2-element Vector{@NamedTuple{x::Vector{Integer}}}:\n (x = [1, 2],)\n (x = [1, 2],)"
end
# issue #25857
@test repr([(1,),(1,2),(1,2,3)]) == "Tuple{$Int, Vararg{$Int}}[(1,), (1, 2), (1, 2, 3)]"
# issues #25466 & #26256
@test replstr([:A => [1]]) == "1-element Vector{Pair{Symbol, Vector{$Int}}}:\n :A => [1]"
# issue #26881
@test showstr([keys(Dict('a' => 'b'))]) == "Base.KeySet{Char, Dict{Char, Char}}[['a']]"
@test showstr([values(Dict('a' => 'b'))]) == "Base.ValueIterator{Dict{Char, Char}}[['b']]"
@test replstr([keys(Dict('a' => 'b'))]) == "1-element Vector{Base.KeySet{Char, Dict{Char, Char}}}:\n ['a']"
@test showstr(Pair{Integer,Integer}(1, 2), :typeinfo => Pair{Integer,Integer}) == "1 => 2"
@test showstr([Pair{Integer,Integer}(1, 2)]) == "Pair{Integer, Integer}[1 => 2]"
@test showstr([(a=1,)]) == "[(a = 1,)]"
@test showstr(Dict{Integer,Integer}(1 => 2)) == "Dict{Integer, Integer}(1 => 2)"
@test showstr(Dict(true=>false)) == "Dict{Bool, Bool}(1 => 0)"
@test showstr(Dict((1 => 2) => (3 => 4))) == "Dict((1 => 2) => (3 => 4))"
# issue #27979 (displaying arrays of pairs containing arrays as first member)
@test replstr([[1.0]=>1.0]) == "1-element Vector{Pair{Vector{Float64}, Float64}}:\n [1.0] => 1.0"
# issue #28159
@test replstr([(a=1, b=2), (a=3,c=4)]) == "2-element Vector{NamedTuple{names, Tuple{$Int, $Int}} where names}:\n (a = 1, b = 2)\n (a = 3, c = 4)"
@test replstr(Vector[Any[1]]) == "1-element Vector{Vector}:\n Any[1]"
@test replstr(AbstractDict{Integer,Integer}[Dict{Integer,Integer}(1=>2)]) ==
"1-element Vector{AbstractDict{Integer, Integer}}:\n Dict(1 => 2)"
# issue #34343
@test showstr([[1], Int[]]) == "[[1], $Int[]]"
@test showstr([Dict(1=>1), Dict{Int,Int}()]) == "[Dict(1 => 1), Dict{$Int, $Int}()]"
# issue #42719, NamedTuple with @var_str
@test replstr((; var"a b"=1)) == """(var"a b" = 1,)"""
@test replstr((; var"#var#"=1)) == """(var"#var#" = 1,)"""
@test replstr((; var"a"=1, b=2)) == "(a = 1, b = 2)"
@test replstr((; a=1, b=2)) == "(a = 1, b = 2)"
# issue 48828, typeinfo missing for arrays with >2 dimensions
@test showstr(Float16[1.0 3.0; 2.0 4.0;;; 5.0 7.0; 6.0 8.0]) ==
"Float16[1.0 3.0; 2.0 4.0;;; 5.0 7.0; 6.0 8.0]"
end
@testset "#14684: `display` should print associative types in full" begin
d = Dict(1 => 2, 3 => 45)
td = TextDisplay(PipeBuffer())
display(td, d)
result = read(td.io, String)
@test occursin(summary(d), result)
# Is every pair in the string?
for el in d
@test occursin(string(el), result)
end
end
@testset "#43766: `display` trailing newline" begin
td = TextDisplay(PipeBuffer())
display(td, 1)
@test read(td.io, String) == "1\n"
show(td.io, 1)
@test read(td.io, String) == "1"
end
function _methodsstr(@nospecialize f)
buf = IOBuffer()
show(buf, methods(f))
return String(take!(buf))
end
@testset "show function methods" begin
@test occursin("methods for generic function \"sin\" from Base:\n", _methodsstr(sin))
end
@testset "show macro methods" begin
@test startswith(_methodsstr(getfield(Base,Symbol("@show"))), "# 1 method for macro \"@show\" from Base:\n")
end
@testset "show constructor methods" begin
@test occursin(" methods for type constructor:\n", _methodsstr(Vector))
end
@testset "show builtin methods" begin
@test startswith(_methodsstr(typeof), "# 1 method for builtin function \"typeof\" from Core:\n")
end
@testset "show callable object methods" begin
@test occursin("methods for callable object:\n", _methodsstr(:))
end
@testset "#20111 show for function" begin
K20111(x) = y -> x
@test startswith(_methodsstr(K20111(1)), "# 1 method for anonymous function")
end
@testset "show non-callable object" begin
@test "# 0 methods for callable object" == _methodsstr(1.0f0)
end
@generated f22798(x::Integer, y) = :x
@testset "#22798" begin
buf = IOBuffer()
show(buf, methods(f22798))
@test occursin("f22798(x::Integer, y)", String(take!(buf)))
end
@testset "Intrinsic printing" begin
@test sprint(show, Core.Intrinsics.cglobal) == "Core.Intrinsics.cglobal"
@test repr(Core.Intrinsics.cglobal) == "Core.Intrinsics.cglobal"
let io = IOBuffer()
show(io, MIME"text/plain"(), Core.Intrinsics.cglobal)
str = String(take!(io))
@test occursin("cglobal", str)
@test occursin("(intrinsic function", str)
end
@test string(Core.Intrinsics.add_int) == "add_int"
end
@testset "repr(mime, x)" begin
@test repr("text/plain", UInt8[1 2;3 4]) == "2×2 Matrix{UInt8}:\n 0x01 0x02\n 0x03 0x04"
@test repr("text/html", "raw html data") == "raw html data"
@test repr("text/plain", "string") == "\"string\""
@test repr("image/png", UInt8[2,3,4,7]) == UInt8[2,3,4,7]
@test repr("text/plain", 3.141592653589793) == "3.141592653589793"
@test repr("text/plain", 3.141592653589793, context=:compact=>true) == "3.14159"
@test repr("text/plain", context=:compact=>true) == "\"text/plain\""
@test repr(MIME("text/plain"), context=:compact=>true) == "MIME type text/plain"
end
@testset "#26799 BigInt summary" begin
@test Base.dims2string(tuple(BigInt(10))) == "10-element"
@test Base.inds2string(tuple(BigInt(10))) == "10"
@test summary(BigInt(1):BigInt(10)) == "10-element UnitRange{BigInt}"
@test summary(Base.OneTo(BigInt(10))) == "10-element Base.OneTo{BigInt}"
end
@testset "Tuple summary" begin
@test summary((1,2,3)) == "Tuple{$Int, $Int, $Int}"
@test summary((:a, "b", 'c')) == "Tuple{Symbol, String, Char}"
end
# Tests for code_typed linetable annotations
function compute_annotations(f, types)
src = code_typed(f, types, debuginfo=:source)[1][1]
ir = Core.Compiler.inflate_ir(src)
la, lb, ll = Base.IRShow.compute_ir_line_annotations(ir)
max_loc_method = maximum(length(s) for s in la)
return join((strip(string(a, " "^(max_loc_method-length(a)), b)) for (a, b) in zip(la, lb)), '\n')
end
@noinline leaffunc() = print()
@inline g_line() = leaffunc()
# Test that separate instances of the same function do not get merged
@inline function f_line()
g_line()
g_line()
g_line()
nothing
end
h_line() = f_line()
@test startswith(compute_annotations(h_line, Tuple{}), """
│╻╷ f_line
││╻ g_line
││╻ g_line""")
# Tests for printing Core.Compiler internal objects
@test repr(Core.Compiler.SSAValue(23)) == ":(%23)"
@test repr(Core.Compiler.SSAValue(-2)) == ":(%-2)"
@test repr(Core.Compiler.ReturnNode(23)) == ":(return 23)"
@test repr(Core.Compiler.ReturnNode()) == ":(unreachable)"
@test repr(Core.Compiler.GotoIfNot(true, 4)) == ":(goto %4 if not true)"
@test repr(Core.Compiler.PhiNode(Int32[2, 3], Any[1, Core.SlotNumber(3)])) == ":(φ (%2 => 1, %3 => _3))"
@test repr(Core.Compiler.UpsilonNode(Core.SlotNumber(3))) == ":(ϒ (_3))"
@test repr(Core.Compiler.PhiCNode(Any[1, Core.SlotNumber(3)])) == ":(φᶜ (1, _3))"
@test sprint(Base.show_unquoted, Core.Compiler.Argument(23)) == "_23"
@test sprint(Base.show_unquoted, Core.Compiler.Argument(-2)) == "_-2"
eval(Meta._parse_string("""function my_fun28173(x)
y = if x == 1
"HI"
elseif x == 2
r = 1
s = try
r = 2
Base.inferencebarrier(false) && error()
"BYE"
catch
r = 3
"CAUGHT!"
end
"\$r\$s"
else
"three"
end
return y
end""", "a"^80, 1, 1, :statement)[1]) # use parse to control the line numbers
let src = code_typed(my_fun28173, (Int,), debuginfo=:source)[1][1]
ir = Core.Compiler.inflate_ir(src)
src.debuginfo = Core.DebugInfo(src.debuginfo.def) # IRCode printing defaults to incomplete line info printing, so turn it off completely for CodeInfo too
let source_slotnames = String["my_fun28173", "x"],
repr_ir = split(repr(ir, context = :SOURCE_SLOTNAMES=>source_slotnames), '\n'),
repr_ir = "CodeInfo(\n" * join((l[4:end] for l in repr_ir), "\n") * ")" # remove line numbers
@test repr(src) == repr_ir
end
lines1 = split(repr(ir), '\n')
@test all(isspace, pop!(lines1))
Core.Compiler.insert_node!(ir, 1, Core.Compiler.NewInstruction(QuoteNode(1), Val{1}), false)
Core.Compiler.insert_node!(ir, 1, Core.Compiler.NewInstruction(QuoteNode(2), Val{2}), true)
Core.Compiler.insert_node!(ir, length(ir.stmts.stmt), Core.Compiler.NewInstruction(QuoteNode(3), Val{3}), false)
Core.Compiler.insert_node!(ir, length(ir.stmts.stmt), Core.Compiler.NewInstruction(QuoteNode(4), Val{4}), true)
lines2 = split(repr(ir), '\n')
@test all(isspace, pop!(lines2))
@test popfirst!(lines2) == "2 1 ── $(QuoteNode(1))"
let line1 = popfirst!(lines1)
line2 = popfirst!(lines2)
@test startswith(line1, "2 1 ── ")
@test startswith(line2, " │ ")
@test line2[12:end] == line2[12:end]
end
@test popfirst!(lines2) == " │ $(QuoteNode(2))"
@test pop!(lines2) == " └─── \$(QuoteNode(4))"
@test pop!(lines1) == "18 └─── return %21"
@test pop!(lines2) == " │ return %21"
@test pop!(lines2) == "18 │ \$(QuoteNode(3))"
@test lines1 == lines2
# verbose linetable
io = IOBuffer()
Base.IRShow.show_ir(io, ir, Base.IRShow.default_config(ir; verbose_linetable=true))
seekstart(io)
@test count(contains(r"@ a{80}:\d+ within `my_fun28173"), eachline(io)) == 10
# Test that a bad :invoke doesn't cause an error during printing
Core.Compiler.insert_node!(ir, 1, Core.Compiler.NewInstruction(Expr(:invoke, nothing, sin), Any), false)
io = IOBuffer()
Base.IRShow.show_ir(io, ir)
seekstart(io)
@test contains(String(take!(io)), "Expr(:invoke, nothing")
end
# Verify that extra instructions at the end of the IR
# don't throw errors in the printing, but instead print
# with as unnamed "!" BB.
let src = code_typed(gcd, (Int, Int), debuginfo=:source)[1][1]
ir = Core.Compiler.inflate_ir(src)
push!(ir.stmts.stmt, Core.Compiler.ReturnNode())
lines = split(sprint(show, ir), '\n')
@test all(isspace, pop!(lines))
@test pop!(lines) == " !!! ── unreachable::#UNDEF"
end
@testset "printing and interpolating nothing" begin
@test sprint(print, nothing) == "nothing"
@test string(nothing) == "nothing"
@test repr(nothing) == "nothing"
@test string(1, "", nothing) == "1nothing"
@test let x = nothing; "x = $x" end == "x = nothing"
@test let x = nothing; "x = $(repr(x))" end == "x = nothing"
# issue #27352 : No interpolating nothing into commands
@test_throws ArgumentError `/bin/foo $nothing`
@test_throws ArgumentError `$nothing`
@test_throws ArgumentError let x = nothing; `/bin/foo $x` end
end
struct X28004
value::Any
end
function Base.show(io::IO, x::X28004)
print(io, "X(")
show(io, x.value)
print(io, ")")
end
@testset """printing "Any" is not skipped with nested arrays""" begin
@test replstr(Union{X28004,Vector}[X28004(Any[X28004(1)])], :compact => true) ==
"1-element Vector{Union{X28004, Vector}}:\n X(Any[X(1)])"
end
# Issue 25589 - Underlines in cmd printing
replstrcolor(x) = sprint((io, x) -> show(IOContext(io, :limit => true, :color => true),
MIME("text/plain"), x), x)
@test occursin("\e[", replstrcolor(`curl abc`))
# issue #30303
@test repr(Symbol("a\$")) == "Symbol(\"a\\\$\")"
@test string(sin) == "sin"
@test string(:) == "Colon()"
@test string(Iterators.flatten) == "flatten"
@test Symbol(Iterators.flatten) === :flatten
@test startswith(string(x->x), "#")
# printing of bools and bool arrays
@testset "Bool" begin
@test repr(true) == "true"
@test repr(Number[true, false]) == "Number[true, false]"
@test repr([true, false]) == "Bool[1, 0]" == repr(BitVector([true, false]))
@test_repr "Bool[1, 0]"
end
@testset "Unions with Bool (#39590)" begin
@test repr([missing, false]) == "Union{Missing, Bool}[missing, 0]"
@test_repr "Union{Bool, Nothing}[1, 0, nothing]"
end
# issue #26847
@test_repr "Union{Missing, Float32}[1.0]"
# intersection of #45396 and #48822
@test_repr "Union{Missing, Rational{Int64}}[missing, 1//2, 2]"
# Don't go too far with #48822
@test_repr "Union{String, Bool}[true]"
# issue #30505
@test repr(Union{Tuple{Char}, Tuple{Char, Char}}[('a','b')]) == "Union{Tuple{Char}, Tuple{Char, Char}}[('a', 'b')]"
# issue #30927
Z = Array{Float64}(undef,0,0)
@test eval(Meta.parse(repr(Z))) == Z
@testset "show undef" begin
# issue #33204 - Parseable `repr` for `undef`
@test eval(Meta.parse(repr(undef))) == undef == UndefInitializer()
@test showstr(undef) == "UndefInitializer()"
@test occursin(repr(undef), replstr(undef))
@test occursin("initializer with undefined values", replstr(undef))
vec_undefined = Vector(undef, 2)
vec_initialisers = fill(undef, 2)
@test showstr(vec_undefined) == "Any[#undef, #undef]"
@test showstr(vec_initialisers) == "[$undef, $undef]"
@test replstr(vec_undefined) == "2-element Vector{Any}:\n #undef\n #undef"
@test replstr(vec_initialisers) == "2-element Vector{UndefInitializer}:\n UndefInitializer(): array initializer with undefined values\n UndefInitializer(): array initializer with undefined values"
end
# issue #31065, do not print parentheses for nested dot expressions
@test sprint(Base.show_unquoted, :(foo.x.x)) == "foo.x.x"
@testset "show_delim_array" begin
sdastr(f, n) = # sda: Show Delim Array
sprint((io, x) -> Base.show_delim_array(io, x, "[", ",", "]", false, f, n), Iterators.take(1:f+n, f+n))
@test sdastr(1, 0) == "[1]"
@test sdastr(1, 1) == "[1]"
@test sdastr(1, 2) == "[1, 2]"
@test sdastr(2, 2) == "[2, 3]"
@test sdastr(3, 3) == "[3, 4, 5]"
end
@testset "show Set" begin
s = Set{Int}(1:22)
str = showstr(s)
@test startswith(str, "Set([")
@test endswith(str, "])")
@test occursin(" … ", str)
str = replstr(s)
@test startswith(str, "Set{$Int} with 22 elements:\n")
@test endswith(str, "\n ⋮ ")
@test count(==('\n'), str) == 20
@test replstr(Set(['a'^100])) == "Set{String} with 1 element:\n \"aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa…"
end
@testset "Simple printing of StridedArray" begin
@test startswith(sprint(show, StridedArray), "StridedArray")
@test startswith(sprint(show, StridedVecOrMat), "StridedVecOrMat")
@test startswith(sprint(show, StridedVector), "Strided")
@test startswith(sprint(show, StridedMatrix), "Strided")
@test occursin("StridedArray", sprint(show, SubArray{T, N, A} where {T,N,A<:StridedArray}))
@test !occursin("Strided", sprint(show, Union{DenseArray, SubArray}))
@test !occursin("Strided", sprint(show, Union{DenseArray, Base.ReinterpretArray, Base.ReshapedArray, SubArray}))
end
@testset "0-dimensional Array. Issue #31481" begin
for x in (zeros(Int32), collect('b'), fill(nothing), BitArray(0))
@test eval(Meta.parse(repr(x))) == x
end
@test showstr(zeros(Int32)) == "fill(0)"
@test showstr(collect('b')) == "fill('b')"
@test showstr(fill(nothing)) == "fill(nothing)"
@test showstr(BitArray(0)) == "BitArray(0)"
@test replstr(zeros(Int32)) == "0-dimensional Array{Int32, 0}:\n0"
@test replstr(collect('b')) == "0-dimensional Array{Char, 0}:\n'b'"
@test replstr(fill(nothing)) == "0-dimensional Array{Nothing, 0}:\nnothing"
@test replstr(BitArray(0)) == "0-dimensional BitArray{0}:\n0"
# UndefInitializer
@test showstr(fill(undef)) == "fill($undef)"
@test replstr(fill(undef)) == "0-dimensional Array{UndefInitializer, 0}:\n$undef"
# `#undef` values
@test showstr(Array{String, 0}(undef)) == "Array{String, 0}($undef)"
@test replstr(Array{String, 0}(undef)) == "0-dimensional Array{String, 0}:\n$(Base.undef_ref_str)"
# "undef" with isbits type
@test startswith(showstr(Array{Int32, 0}(undef)), "fill(")
@test startswith(replstr(Array{Int32, 0}(undef)), "0-dimensional Array{Int32, 0}:\n")
end
# issue #31402, Print Symbol("true") as Symbol("true") instead of :true
@test sprint(show, Symbol(true)) == "Symbol(\"true\")"
@test sprint(show, Symbol("true")) == "Symbol(\"true\")"
@test sprint(show, Symbol(false)) == "Symbol(\"false\")"
@test sprint(show, Symbol("false")) == "Symbol(\"false\")"
# begin/end indices
@weak_test_repr "a[begin, end, (begin; end)]"
@test repr(Base.remove_linenums!(:(a[begin, end, (begin; end)]))) == ":(a[begin, end, (begin;\n end)])"
@weak_test_repr "a[begin, end, let x=1; (x+1;); end]"
@test repr(Base.remove_linenums!(:(a[begin, end, let x=1; (x+1;); end]))) ==
":(a[begin, end, let x = 1\n begin\n x + 1\n end\n end])"
@test_repr "a[(bla;)]"
@test_repr "a[(;;)]"
@weak_test_repr "a[x -> f(x)]"
@testset "Base.Iterators" begin
@test sprint(show, enumerate("test")) == "enumerate(\"test\")"
@test sprint(show, enumerate(1:5)) == "enumerate(1:5)"
@test sprint(show, enumerate([1,2,3])) == "enumerate([1, 2, 3])"
@test sprint(show, enumerate((1,1.0,'a'))) == "enumerate((1, 1.0, 'a'))"
@test sprint(show, zip()) == "zip()"
@test sprint(show, zip([1,2,3])) == "zip([1, 2, 3])"
@test sprint(show, zip(1:3, ('a','b','c'))) == "zip(1:3, ('a', 'b', 'c'))"
@test sprint(show, zip(1:3, ('a','b','c'), "abc")) == "zip(1:3, ('a', 'b', 'c'), \"abc\")"
end
@testset "skipmissing" begin
@test sprint(show, skipmissing("test")) == "skipmissing(\"test\")"
@test sprint(show, skipmissing(1:5)) == "skipmissing(1:5)"
@test sprint(show, skipmissing([1,2,missing])) == "skipmissing(Union{Missing, $Int}[1, 2, missing])"
@test sprint(show, skipmissing((missing,1.0,'a'))) == "skipmissing((missing, 1.0, 'a'))"
end
@testset "unicode in method table" begin
αsym = gensym(:α)
ℓsym = gensym(:ℓ)
eval(:(foo($αsym) = $αsym))
eval(:(bar($ℓsym) = $ℓsym))
@test contains(string(methods(foo)), "foo(α)")
@test contains(string(methods(bar)), "bar(ℓ)")
end
module M37012
export AValue, B2, SimpleU
struct AnInteger{S<:Integer} end
struct AStruct{N} end
const AValue{S} = Union{AStruct{S}, AnInteger{S}}
struct BStruct{T,S} end
const B2{S,T} = BStruct{T,S}
const SimpleU = Union{AnInteger, AStruct, BStruct}
end
@test Base.make_typealias(M37012.AStruct{1}) === nothing
@test isempty(Base.make_typealiases(M37012.AStruct{1})[1])
@test string(M37012.AStruct{1}) == "$(curmod_prefix)M37012.AStruct{1}"
@test string(Union{Nothing, Number, Vector}) == "Union{Nothing, Number, Vector}"
@test string(Union{Nothing, Number, Vector{<:Integer}}) == "Union{Nothing, Number, Vector{<:Integer}}"
@test string(Union{Nothing, AbstractVecOrMat}) == "Union{Nothing, AbstractVecOrMat}"
@test string(Union{Nothing, AbstractVecOrMat{<:Integer}}) == "Union{Nothing, AbstractVecOrMat{<:Integer}}"
@test string(M37012.BStruct{T, T} where T) == "$(curmod_prefix)M37012.B2{T, T} where T"
@test string(M37012.BStruct{T, S} where {T<:Unsigned, S<:Signed}) == "$(curmod_prefix)M37012.B2{S, T} where {T<:Unsigned, S<:Signed}"
@test string(M37012.BStruct{T, S} where {T<:Signed, S<:T}) == "$(curmod_prefix)M37012.B2{S, T} where {T<:Signed, S<:T}"
@test string(Union{M37012.SimpleU, Nothing}) == "Union{Nothing, $(curmod_prefix)M37012.SimpleU}"
@test string(Union{M37012.SimpleU, Nothing, T} where T) == "Union{Nothing, $(curmod_prefix)M37012.SimpleU, T} where T"
@test string(Union{AbstractVector{T}, T} where T) == "Union{AbstractVector{T}, T} where T"
@test string(Union{AbstractVector, T} where T) == "Union{AbstractVector, T} where T"
@test sprint(show, :(./)) == ":((./))"
@test sprint(show, :((.|).(.&, b))) == ":((.|).((.&), b))"
@test sprint(show, :(a'ᵀ)) == ":(a'ᵀ)"
@test sprint(show, :((+)')) == ":((+)')"
for s in (Symbol("'"), Symbol("'⁻¹"))
@test Base.isoperator(s)
@test !Base.isunaryoperator(s)
@test !Base.isbinaryoperator(s)
@test Base.ispostfixoperator(s)
end
@testset "method printing with non-standard identifiers ($mime)" for mime in (
MIME("text/plain"), MIME("text/html"),
)
_show(io, x) = show(io, MIME(mime), x)
@eval var","(x) = x
@test occursin("var\",\"(x)", sprint(_show, methods(var",")))
@eval f1(var"a.b") = 3
@test occursin("f1(var\"a.b\")", sprint(_show, methods(f1)))
@test sprint(_show, Method[]) == "0-element Vector{Method}"
italic(s) = mime == MIME("text/html") ? "<i>$s</i>" : s
@eval f2(; var"123") = 5
@test occursin("f2(; $(italic("var\"123\"")))", sprint(_show, methods(f2)))
@eval f3(; var"%!"...) = 7
@test occursin("f3(; $(italic("var\"%!\"...")))", sprint(_show, methods(f3)))
@eval f4(; var"...") = 9
@test_broken occursin("f4(; $(italic("var\"...\"")))", sprint(_show, methods(f4)))
end
@testset "printing of syntactic operators" begin
@test sprint(show, :(var"::" + var"$")) == ":(var\"::\" + (\$))"
@test sprint(show, :(!var"...")) == ":(!var\"...\")"
@test sprint(show, :(var"'ᵀ" - 1)) == ":(var\"'ᵀ\" - 1)"
@test sprint(show, :(::)) == ":(::)"
@test sprint(show, :?) == ":?"
@test sprint(show, :(var"?" + var"::" + var"'")) == ":(var\"?\" + var\"::\" + var\"'\")"
end
@testset "printing of function types" begin
s = sprint(show, MIME("text/plain"), typeof(sin))
@test s == "typeof(sin) (singleton type of function sin, subtype of Function)"
s = sprint(show, MIME("text/plain"), ModFWithParams.f_with_params)
@test endswith(s, "ModFWithParams.f_with_params")
s = sprint(show, MIME("text/plain"), ModFWithParams.f_with_params{2})
@test endswith(s, "ModFWithParams.f_with_params{2}")
s = sprint(show, MIME("text/plain"), UnionAll)
@test s == "UnionAll"
s = sprint(show, MIME("text/plain"), Function)
@test s == "Function"
end
@testset "printing inline n-dimensional arrays and one-column matrices" begin
@test replstr([Int[1 2 3 ;;; 4 5 6]]) == "1-element Vector{Array{$Int, 3}}:\n [1 2 3;;; 4 5 6]"
@test replstr([Int[1 2 3 ;;; 4 5 6;;;;]]) == "1-element Vector{Array{$Int, 4}}:\n [1 2 3;;; 4 5 6;;;;]"
@test replstr([fill(1, (20,20,20))]) == "1-element Vector{Array{$Int, 3}}:\n [1 1 … 1 1; 1 1 … 1 1; … ; 1 1 … 1 1; 1 1 … 1 1;;; 1 1 … 1 1; 1 1 … 1 1; … ; 1 1 … 1 1; 1 1 … 1 1;;; 1 1 … 1 1; 1 1 … 1 1; … ; 1 1 … 1 1; 1 1 … 1 1;;; … ;;; 1 1 … 1 1; 1 1 … 1 1; … ; 1 1 … 1 1; 1 1 … 1 1;;; 1 1 … 1 1; 1 1 … 1 1; … ; 1 1 … 1 1; 1 1 … 1 1;;; 1 1 … 1 1; 1 1 … 1 1; … ; 1 1 … 1 1; 1 1 … 1 1]"
@test replstr([fill(1, 5, 1)]) == "1-element Vector{Matrix{$Int}}:\n [1; 1; … ; 1; 1;;]"
@test replstr([fill(1, 5, 2)]) == "1-element Vector{Matrix{$Int}}:\n [1 1; 1 1; … ; 1 1; 1 1]"
@test replstr([[1;]]) == "1-element Vector{Vector{$Int}}:\n [1]"
@test replstr([[1;;]]) == "1-element Vector{Matrix{$Int}}:\n [1;;]"
@test replstr([[1;;;]]) == "1-element Vector{Array{$Int, 3}}:\n [1;;;]"
end
@testset "ncat and nrow" begin
@test_repr "[1;;]"
@test_repr "[1;;;]"
@test_repr "[1;; 2]"
@test_repr "[1;;; 2]"
@test_repr "[1;;; 2 3;;; 4]"
@test_repr "[1;;; 2;;;; 3;;; 4]"
@test_repr "T[1;;]"
@test_repr "T[1;;;]"
@test_repr "T[1;; 2]"
@test_repr "T[1;;; 2]"
@test_repr "T[1;;; 2 3;;; 4]"
@test_repr "T[1;;; 2;;;; 3;;; 4]"
end
@testset "Cmd" begin
@test sprint(show, `true`) == "`true`"
@test sprint(show, setenv(`true`, "A" => "B")) == """setenv(`true`,["A=B"])"""
@test sprint(show, setcpuaffinity(`true`, [1, 2])) == "setcpuaffinity(`true`, [1, 2])"
@test sprint(show, setenv(setcpuaffinity(`true`, [1, 2]), "A" => "B")) ==
"""setenv(setcpuaffinity(`true`, [1, 2]),["A=B"])"""
end
# Test that alignment takes into account unicode and computes alignment without
# color/formatting.
struct ColoredLetter; end
Base.show(io::IO, ces::ColoredLetter) = Base.printstyled(io, 'A'; color=:red)
struct ⛵; end
Base.show(io::IO, ces::⛵) = Base.print(io, '⛵')
@test Base.alignment(stdout, ⛵()) == (0, 2)
@test Base.alignment(IOContext(IOBuffer(), :color=>true), ColoredLetter()) == (0, 1)
@test Base.alignment(IOContext(IOBuffer(), :color=>false), ColoredLetter()) == (0, 1)
# spacing around dots in Diagonal, etc:
redminusthree = sprint((io, x) -> printstyled(io, x, color=:red), "-3", context=stdout)
@test Base.replace_with_centered_mark(redminusthree) == Base.replace_with_centered_mark("-3")
# `show` implementations for `Method`
let buf = IOBuffer()
# single line printing by default
show(buf, only(methods(sin, (Float64,))))
@test !occursin('\n', String(take!(buf)))
# two-line printing for rich display
show(buf, MIME("text/plain"), only(methods(sin, (Float64,))))
@test occursin('\n', String(take!(buf)))
end
@testset "basic `show_ir` functionality tests" begin
mktemp() do f, io
redirect_stdout(io) do
let io = IOBuffer()
for i = 1:10
# make sure we don't error on printing IRs at any optimization level
ir = only(Base.code_ircode(sin, (Float64,); optimize_until=i))[1]
@test try; show(io, ir); true; catch; false; end
compact = Core.Compiler.IncrementalCompact(ir)
@test try; show(io, compact); true; catch; false; end
end
end
end
close(io)
@test isempty(read(f, String)) # make sure we don't unnecessarily lean anything into `stdout`
end
end
@testset "IRCode: fix coloring of invalid SSA values" begin
# get some ir
function foo(i)
j = i+42
j == 1 ? 1 : 2
end
ir = only(Base.code_ircode(foo, (Int,)))[1]
# replace an instruction
add_stmt = ir.stmts[1]
inst = Core.Compiler.NewInstruction(Expr(:call, add_stmt[:stmt].args[1], add_stmt[:stmt].args[2], 999), Int)
node = Core.Compiler.insert_node!(ir, 1, inst)
Core.Compiler.setindex!(add_stmt, node, :stmt)
# the new node should be colored green (as it's uncompacted IR),
# and its uses shouldn't be colored at all (since they're just plain valid references)
str = sprint(; context=:color=>true) do io
show(io, ir)
end
@test contains(str, "\e[32m%6 =")
@test contains(str, "%1 = %6")
# if we insert an invalid node, it should be colored appropriately
Core.Compiler.setindex!(add_stmt, Core.Compiler.SSAValue(node.id+1), :stmt)
str = sprint(; context=:color=>true) do io
show(io, ir)
end
@test contains(str, "%1 = \e[31m%7")
end
@testset "issue #46947: IncrementalCompact double display of just-compacted nodes" begin
# get some IR
foo(i) = i == 1 ? 1 : 2
ir = only(Base.code_ircode(foo, (Int,)))[1]
instructions = length(ir.stmts)
lines_shown(obj) = length(findall('\n', sprint(io->show(io, obj))))
@test lines_shown(ir) == instructions
# insert a couple of instructions
let inst = Core.Compiler.NewInstruction(Expr(:identity, 1), Nothing)
Core.Compiler.insert_node!(ir, 2, inst)
end
let inst = Core.Compiler.NewInstruction(Expr(:identity, 2), Nothing)
Core.Compiler.insert_node!(ir, 2, inst)
end
let inst = Core.Compiler.NewInstruction(Expr(:identity, 3), Nothing)
Core.Compiler.insert_node!(ir, 4, inst)
end
instructions += 3
@test lines_shown(ir) == instructions
# compact the IR, ensuring we always show the same number of lines
# (the instructions + a separator line)
compact = Core.Compiler.IncrementalCompact(ir)
@test lines_shown(compact) == instructions + 1
state = Core.Compiler.iterate(compact)
while state !== nothing
@test lines_shown(compact) == instructions + 1
state = Core.Compiler.iterate(compact, state[2])
end
@test lines_shown(compact) == instructions + 1
ir = Core.Compiler.complete(compact)
@test lines_shown(compact) == instructions + 1
end
@testset "#46424: IncrementalCompact displays wrong basic-block boundaries" begin
# get some cfg
function foo(i)
j = i+42
j == 1 ? 1 : 2
end
ir = only(Base.code_ircode(foo, (Int,)))[1]
# at every point we should be able to observe these three basic blocks
function verify_display(ir)
str = sprint(io->show(io, ir))
@test contains(str, "1 ─ %1 = ")
@test contains(str, r"2 ─ \s+ return 1")
@test contains(str, r"3 ─ \s+ return 2")
end
verify_display(ir)
# insert some instructions
for i in 1:3
inst = Core.Compiler.NewInstruction(Expr(:call, :identity, i), Int)
Core.Compiler.insert_node!(ir, 2, inst)
end
# compact
compact = Core.Compiler.IncrementalCompact(ir)
verify_display(compact)
# Compact the first instruction
state = Core.Compiler.iterate(compact)
# Insert some instructions here
for i in 1:2
inst = Core.Compiler.NewInstruction(Expr(:call, :identity, i), Int, Int32(1))
Core.Compiler.insert_node_here!(compact, inst)
verify_display(compact)
end
while state !== nothing
state = Core.Compiler.iterate(compact, state[2])
verify_display(compact)
end
# complete
ir = Core.Compiler.complete(compact)
verify_display(ir)
end
@testset "IRCode: CFG display" begin
# get a cfg
function foo(i)
j = i+42
j == 1 ? 1 : 2
end
ir = only(Base.code_ircode(foo, (Int,)))[1]
cfg = ir.cfg
str = sprint(io->show(io, cfg))
@test contains(str, r"CFG with \d+ blocks")
@test contains(str, r"bb 1 \(stmt.+\) → bb.*")
end
@testset "IncrementalCompact: correctly display attach-after nodes" begin
# set some IR
function foo(i)
j = i+42
return j
end
ir = only(Base.code_ircode(foo, (Int,)))[1]
# insert a bunch of nodes, inserting both before and after instruction 1
inst = Core.Compiler.NewInstruction(Expr(:call, :identity, 1), Int)
Core.Compiler.insert_node!(ir, 1, inst)
inst = Core.Compiler.NewInstruction(Expr(:call, :identity, 2), Int)
Core.Compiler.insert_node!(ir, 1, inst)
inst = Core.Compiler.NewInstruction(Expr(:call, :identity, 3), Int)
Core.Compiler.insert_node!(ir, 1, inst, true)
inst = Core.Compiler.NewInstruction(Expr(:call, :identity, 4), Int)
Core.Compiler.insert_node!(ir, 1, inst, true)
# at every point we should be able to observe these instructions (in order)
function verify_display(ir)
str = sprint(io->show(io, ir))
lines = split(str, '\n')
patterns = ["identity(1)",
"identity(2)",
"add_int",
"identity(3)",
"identity(4)",
"return"]
line_idx = 1
pattern_idx = 1
while pattern_idx <= length(patterns) && line_idx <= length(lines)
# we test pattern-per-pattern, in order,
# so that we skip e.g. the compaction boundary
if contains(lines[line_idx], patterns[pattern_idx])
pattern_idx += 1
end
line_idx += 1
end
@test pattern_idx > length(patterns)
end
verify_display(ir)
compact = Core.Compiler.IncrementalCompact(ir)
verify_display(compact)
state = Core.Compiler.iterate(compact)
while state !== nothing
verify_display(compact)
state = Core.Compiler.iterate(compact, state[2])
end
ir = Core.Compiler.complete(compact)
verify_display(ir)
end
let buf = IOBuffer()
Base.show_tuple_as_call(buf, Symbol(""), Tuple{Function,Any})
@test String(take!(buf)) == "(::Function)(::Any)"
end
module Issue49382
abstract type Type49382 end
end
using .Issue49382
(::Type{Issue49382.Type49382})() = 1
@test sprint(show, methods(Issue49382.Type49382)) isa String
# Showing of bad SlotNumber in Expr(:toplevel)
let lowered = Meta.lower(Main, Expr(:let, Expr(:block), Expr(:block, Expr(:toplevel, :(x = 1)), :(y = 1))))
ci = lowered.args[1]
@assert isa(ci, Core.CodeInfo)
@test !isempty(ci.slotnames)
@assert ci.code[1].head === :toplevel
ci.code[1].args[1] = :($(Core.SlotNumber(1)) = 1)
# Check that this gets printed as `_1 = 1` not `y = 1`
@test contains(sprint(show, ci), "_1 = 1")
end
# Pointers should be reprable
@test is_juliarepr(pointer([1]))
@test is_juliarepr(Ptr{Vector{Complex{Float16}}}(UInt(0xdeadbeef)))
# Toplevel MethodInstance with undef :uninferred
let topmi = ccall(:jl_new_method_instance_uninit, Ref{Core.MethodInstance}, ());
topmi.specTypes = Tuple{}
topmi.def = Main
@test contains(repr(topmi), "Toplevel MethodInstance")
end