Skip to content

Commit 97cebb1

Browse files
committed
Use StyledStrings Faces for stacktrace printing
This allows for the faces used (e.g. the file path) to be user-customised, which provides an escape from themes that make bright black invisible.
1 parent 31b45a2 commit 97cebb1

File tree

2 files changed

+45
-27
lines changed

2 files changed

+45
-27
lines changed

base/errorshow.jl

+44-26
Original file line numberDiff line numberDiff line change
@@ -100,10 +100,39 @@ function showerror(io::IO, ex, bt; backtrace=true)
100100
end
101101
end
102102

103+
function stacktrace_path(file::Union{Nothing, String}, line::Union{Nothing, Int64})
104+
realfile = if !isnothing(file) && file != "" && !startswith(String(file), "REPL")
105+
fixup_stdlib_path(String(file))
106+
end
107+
pathstr = file
108+
if !isnothing(pathstr)
109+
stacktrace_expand_basepaths() && (pathstr = something(find_source_file(file), file))
110+
stacktrace_contract_userdir() && (pathstr = contractuser(pathstr))
111+
end
112+
linestr, llen = if !isnothing(line) && line > 0
113+
string(something(pathstr, ""), ':', line), ncodeunits(string(line))
114+
else
115+
something(pathstr, ""), 0
116+
end
117+
if !isnothing(realfile)
118+
flen = ncodeunits(basename(realfile))
119+
AnnotatedString(linestr,
120+
[(1:ncodeunits(linestr), :link => Filesystem.uripath(realfile)),
121+
(1:ncodeunits(linestr)-flen-llen-1, :face => :julia_stacktrace_location),
122+
(ncodeunits(linestr)-flen-llen:ncodeunits(linestr)-llen-1, :face => :julia_stacktrace_filename),
123+
(ncodeunits(linestr)-llen:ncodeunits(linestr), :face => :julia_stacktrace_fileline)])
124+
else
125+
AnnotatedString(linestr, [(1:ncodeunits(linestr), :face => :julia_stacktrace_location)])
126+
end
127+
end
128+
129+
stacktrace_path(location::LineNumberNode) =
130+
stacktrace_path(if !isnothing(location.file) String(location.file) end, location.line)
131+
103132
function showerror(io::IO, ex::LoadError, bt; backtrace=true)
104133
!isa(ex.error, LoadError) && print(io, "LoadError: ")
105134
showerror(io, ex.error, bt, backtrace=backtrace)
106-
print(io, "\nin expression starting at $(ex.file):$(ex.line)")
135+
print(io, "\nin expression starting at ", stacktrace_path(ex.file, ex.line))
107136
end
108137
showerror(io::IO, ex::LoadError) = showerror(io, ex, [])
109138

@@ -618,7 +647,7 @@ end
618647
const update_stackframes_callback = Ref{Function}(identity)
619648

620649
const STACKTRACE_MODULECOLORS = Iterators.Stateful(Iterators.cycle([:magenta, :cyan, :green, :yellow]))
621-
const STACKTRACE_FIXEDCOLORS = IdDict(Base => :light_black, Core => :light_black)
650+
const STACKTRACE_FIXEDCOLORS = IdDict(Base => :julia_stacktrace_basemodule, Core => :julia_stacktrace_basemodule)
622651

623652
function show_full_backtrace(io::IO, trace::Vector; print_linebreaks::Bool)
624653
num_frames = length(trace)
@@ -703,9 +732,9 @@ function show_reduced_backtrace(io::IO, t::Vector)
703732
cycle_length = repeated_cycle[1][2]
704733
repetitions = repeated_cycle[1][3]
705734
popfirst!(repeated_cycle)
706-
printstyled(io,
707-
"--- the above ", cycle_length, " lines are repeated ",
708-
repetitions, " more time", repetitions>1 ? "s" : "", " ---", color = :light_black)
735+
repmsg = string("--- the above ", cycle_length, " lines are repeated ",
736+
repetitions, " more time", repetitions>1 ? "s" : "", " ---")
737+
print(io, AnnotatedString(repmsg, [(1:ncodeunits(repmsg), :face => :julia_stacktrace_repetition)]))
709738
if i < length(displayed_stackframes)
710739
println(io)
711740
stacktrace_linebreaks() && println(io)
@@ -758,41 +787,30 @@ function print_stackframe(io, i, frame::StackFrame, n::Int, ndigits_max, modulec
758787
digit_align_width = ndigits_max + 2
759788

760789
# frame number
761-
print(io, " ", lpad("[" * string(i) * "]", digit_align_width))
762-
print(io, " ")
790+
frameindex = lpad('[' * string(i) * ']', digit_align_width)
791+
print(io, ' ', AnnotatedString(frameindex, [(1:ncodeunits(frameindex), :face => :julia_stacktrace_frameindex)]), ' ')
763792

764793
StackTraces.show_spec_linfo(IOContext(io, :backtrace=>true), frame)
765794
if n > 1
766-
printstyled(io, " (repeats $n times)"; color=:light_black)
795+
repmsg = " (repeats $n times)"
796+
print(io, AnnotatedString(repmsg, [(2:ncodeunits(repmsg), :face => :julia_stacktrace_repetition)]))
767797
end
768798
println(io)
769799

770800
# @ Module path / file : line
771801
print_module_path_file(io, modul, file, line; modulecolor, digit_align_width)
772802

773803
# inlined
774-
printstyled(io, inlined ? " [inlined]" : "", color = :light_black)
804+
print(io, if inlined AnnotatedString("[inlined]", [(1:9, :face => :julia_stacktrace_inlined)]) else "" end)
775805
end
776806

777-
function print_module_path_file(io, modul, file, line; modulecolor = :light_black, digit_align_width = 0)
778-
printstyled(io, " " ^ digit_align_width * "@", color = :light_black)
779-
780-
# module
807+
function print_module_path_file(io, modul, file, line; modulecolor = :bright_black, digit_align_width = 0)
808+
print(io, ' ' ^ digit_align_width, AnnotatedString("@", [(1:1, :face => :julia_stacktrace_location)]))
781809
if modul !== nothing && modulecolor !== nothing
782-
print(io, " ")
783-
printstyled(io, modul, color = modulecolor)
810+
mstr = string(modul)
811+
print(io, " ", AnnotatedString(mstr, [(1:ncodeunits(mstr), :face => modulecolor)]))
784812
end
785-
786-
# filepath
787-
file = fixup_stdlib_path(file)
788-
stacktrace_expand_basepaths() && (file = something(find_source_file(file), file))
789-
stacktrace_contract_userdir() && (file = contractuser(file))
790-
print(io, " ")
791-
dir = dirname(file)
792-
!isempty(dir) && printstyled(io, dir, Filesystem.path_separator, color = :light_black)
793-
794-
# filename, separator, line
795-
printstyled(io, basename(file), ":", line; color = :light_black, underline = true)
813+
print(io, ' ', stacktrace_path(file, line))
796814
end
797815

798816
function show_backtrace(io::IO, t::Vector)

base/path.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -627,7 +627,7 @@ function uripath(path::String)
627627
s, r"[^A-Za-z0-9\-_.~]+" => percent_escape)
628628
string("file://", gethostname(), '/',
629629
join(map(encode_uri_component,
630-
split(path, Filesystem.path_separator, keepempty=false)),
630+
split(path, path_separator_re, keepempty=false)),
631631
'/'))
632632
end
633633

0 commit comments

Comments
 (0)