Skip to content

Commit 5331115

Browse files
committed
Implement table log monitor
1 parent 59d2c23 commit 5331115

5 files changed

+171
-4
lines changed

Project.toml

+1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ version = "0.1.0"
55

66
[deps]
77
Logging = "56ddb016-857b-54e1-b83d-db4d58db5568"
8+
PrettyTables = "08abe8d2-0d0c-5749-adfa-8a2ac140af0d"
89
ProgressMeter = "92933f4c-e287-5a05-a399-4b506db050ca"
910

1011
[compat]

src/StickyMessages.jl

+20-3
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,14 @@ function StickyMessages(io::IO; ansi_codes=io isa Base.TTY &&
4242
sticky
4343
end
4444

45+
function firstline!(sticky::StickyMessages, label)
46+
idx = findfirst(m -> m[1] == label, sticky.messages)
47+
idx === nothing && throw(KeyError(label))
48+
idx == 1 && return
49+
sticky.messages[idx], sticky.messages[1] = sticky.messages[1], sticky.messages[idx]
50+
return
51+
end
52+
4553
# Count newlines in a message or sequence of messages
4654
_countlines(msg::String) = sum(c->c=='\n', msg)
4755
_countlines(messages) = length(messages) > 0 ? sum(_countlines, messages) : 0
@@ -93,7 +101,7 @@ function showsticky(io, prev_nlines, messages)
93101
nothing
94102
end
95103

96-
function Base.push!(sticky::StickyMessages, message::Pair)
104+
function Base.push!(sticky::StickyMessages, message::Pair; first=true)
97105
if !sticky.ansi_codes
98106
write(sticky.io, message[2])
99107
return
@@ -103,9 +111,18 @@ function Base.push!(sticky::StickyMessages, message::Pair)
103111
prev_nlines = _countlines(sticky.messages)
104112
idx = findfirst(m->m[1] == label, sticky.messages)
105113
if idx === nothing
106-
push!(sticky.messages, label=>text)
114+
if first
115+
insert!(sticky.messages, 1, label=>text)
116+
else
117+
push!(sticky.messages, label=>text)
118+
end
107119
else
108-
sticky.messages[idx] = label=>text
120+
if first
121+
sticky.messages[idx] = sticky.messages[1]
122+
sticky.messages[1] = label=>text
123+
else
124+
sticky.messages[idx] = label=>text
125+
end
109126
end
110127
showsticky(sticky.io, prev_nlines, sticky.messages)
111128
end

src/TableMonitor.jl

+108
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,108 @@
1+
struct Row
2+
id
3+
names::Vector{String}
4+
values::Vector{String}
5+
end
6+
7+
struct TableMonitor
8+
sticky_messages::StickyMessages
9+
io::IO
10+
rows::Vector{Row}
11+
flushed::Vector{Bool}
12+
end
13+
14+
TableMonitor(sticky_messages::StickyMessages, io::IO = sticky_messages.io) =
15+
TableMonitor(sticky_messages, io, [], [])
16+
17+
function tablelines(
18+
table::TableMonitor,
19+
oldrows = nothing,
20+
tobeflushed = ();
21+
bottom_line = false,
22+
)
23+
data = permutedims(mapreduce(row -> row.values, vcat, table.rows))
24+
highlighters = ()
25+
if oldrows !== nothing
26+
data = vcat(data, permutedims(mapreduce(row -> row.values, vcat, oldrows)))
27+
right = cumsum(Int[length(row.values) for row in oldrows])
28+
left = insert!(right[1:end-1], 1, 0)
29+
# Dim unchanged rows:
30+
highlighters = Highlighter(
31+
(data, i, j) -> i == 2 && !any(k -> left[k] < j <= right[k], tobeflushed);
32+
foreground = :dark_gray,
33+
)
34+
end
35+
text = sprint(context = table.io) do io
36+
pretty_table(
37+
io,
38+
data,
39+
mapreduce(row -> row.names, vcat, table.rows),
40+
PrettyTableFormat(top_line = false, bottom_line = bottom_line);
41+
highlighters = highlighters,
42+
)
43+
end
44+
return split(text, "\n")
45+
end
46+
47+
function draw!(table::TableMonitor, oldrows = nothing, tobeflushed = ())
48+
lines = tablelines(table, oldrows, tobeflushed)
49+
formatted = join(reverse!(lines[1:3]), "\n") # row, hline, header
50+
if oldrows !== nothing
51+
print(table.io, '\n', lines[4])
52+
end
53+
push!(table.sticky_messages, _tablelabel => formatted; first = true)
54+
end
55+
56+
const _tablelabel = gensym("TableMonitor")
57+
58+
Base.push!(table::TableMonitor, (id, namedtuple)::Pair{<:Any,<:NamedTuple}) = push!(
59+
table,
60+
Row(id, collect(string.(keys(namedtuple))), collect(string.(values(namedtuple)))),
61+
)
62+
63+
function Base.push!(table::TableMonitor, row::Row)
64+
idx = findfirst(x -> x.id == row.id, table.rows)
65+
if idx !== nothing
66+
if table.flushed[idx]
67+
tobeflushed = ()
68+
oldrows = nothing
69+
table.flushed[idx] = false
70+
else
71+
tobeflushed = findall(!, table.flushed)
72+
oldrows = copy(table.rows)
73+
table.flushed .= [i != idx for i in 1:length(table.rows)]
74+
end
75+
table.rows[idx] = row
76+
draw!(table, oldrows, tobeflushed)
77+
else
78+
push!(table.flushed, false)
79+
push!(table.rows, row)
80+
draw!(table)
81+
end
82+
return table
83+
end
84+
85+
function flush_table!(table::TableMonitor)
86+
pop!(table.sticky_messages, _tablelabel)
87+
88+
lines = tablelines(table; bottom_line = true)[1:end-1]
89+
bottom = pop!(lines)
90+
for line in reverse!(lines)
91+
print(table.io, '\n', line)
92+
end
93+
print(table.io, '\n', bottom)
94+
95+
fill!(table.flushed, true)
96+
end
97+
98+
function Base.pop!(table::TableMonitor, id)
99+
idx = findfirst(x -> x.id == id, table.rows)
100+
idx === nothing && return
101+
102+
if !table.flushed[idx]
103+
flush_table!(table)
104+
end
105+
106+
deleteat!(table.flushed, idx)
107+
return deleteat!(table.rows, idx)
108+
end

src/TerminalLogger.jl

+38-1
Original file line numberDiff line numberDiff line change
@@ -29,12 +29,16 @@ struct TerminalLogger <: AbstractLogger
2929
message_limits::Dict{Any,Int}
3030
sticky_messages::StickyMessages
3131
bars::Dict{Any,Progress}
32+
was_table::Base.RefValue{Bool}
33+
table::TableMonitor
3234
end
3335
function TerminalLogger(stream::IO=stderr, min_level=ProgressLevel;
3436
meta_formatter=default_metafmt, show_limited=true,
3537
right_justify=0)
38+
sticky_messages = StickyMessages(stream)
3639
TerminalLogger(stream, min_level, meta_formatter,
37-
show_limited, right_justify, Dict{Any,Int}(), StickyMessages(stream), Dict{Any,Progress}())
40+
show_limited, right_justify, Dict{Any,Int}(), sticky_messages, Dict{Any,Progress}(),
41+
Ref(false), TableMonitor(sticky_messages))
3842
end
3943

4044
shouldlog(logger::TerminalLogger, level, _module, group, id) =
@@ -147,6 +151,7 @@ function handle_progress(logger, message, id, progress)
147151
end
148152

149153
if progress == "done" || progress >= 1
154+
flush_table(logger)
150155
pop!(logger.sticky_messages, id)
151156
println(logger.stream, msg)
152157
else
@@ -160,6 +165,36 @@ function handle_progress(logger, message, id, progress)
160165
end
161166
end
162167

168+
struct Unspecified end
169+
170+
function maybe_handle_table(logger, id; table = Unspecified(), _...)
171+
if table isa NamedTuple
172+
push!(logger.table, id => table)
173+
logger.was_table[] = true
174+
return true
175+
elseif table === nothing
176+
pop!(logger.table, id)
177+
logger.was_table[] = true
178+
return true
179+
end
180+
flush_table(logger)
181+
return false
182+
end
183+
184+
function flush_table(logger)
185+
if logger.was_table[]
186+
flush_table!(logger.table)
187+
188+
# To "connect" flushed rows with the current row and header in
189+
# the sticky messages, `TableMonitor` do not print newline
190+
# after the message. Thus, we need to print a newline when
191+
# switching to non-table message:
192+
println(logger.stream)
193+
194+
logger.was_table[] = false
195+
end
196+
end
197+
163198
function handle_message(logger::TerminalLogger, level, message, _module, group, id,
164199
filepath, line; maxlog=nothing, progress=nothing,
165200
sticky=nothing, kwargs...)
@@ -174,6 +209,8 @@ function handle_message(logger::TerminalLogger, level, message, _module, group,
174209
return
175210
end
176211

212+
maybe_handle_table(logger, id; kwargs...) && return
213+
177214
substr(s) = SubString(s, 1, length(s)) # julia 0.6 compat
178215

179216
# Generate a text representation of the message and all key value pairs,

src/TerminalLoggers.jl

+4
Original file line numberDiff line numberDiff line change
@@ -10,11 +10,15 @@ import Logging:
1010
using ProgressMeter:
1111
Progress, update!
1212

13+
using PrettyTables:
14+
Highlighter, PrettyTableFormat, pretty_table
15+
1316
export TerminalLogger
1417

1518
const ProgressLevel = LogLevel(-1)
1619

1720
include("StickyMessages.jl")
21+
include("TableMonitor.jl")
1822
include("TerminalLogger.jl")
1923

2024
end # module

0 commit comments

Comments
 (0)