2 """Combine logs from multiple bitcoin nodes as well as the test_framework log.
4 This streams the combined log output to stdout. Use combine_logs.py > outputfile
5 to write to an outputfile."""
8 from collections
import defaultdict
, namedtuple
15 # Matches on the date format at the start of the log event
16 TIMESTAMP_PATTERN
= re
.compile(r
"^\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}\.\d{6}")
18 LogEvent
= namedtuple('LogEvent', ['timestamp', 'source', 'event'])
21 """Main function. Parses args, reads the log files and renders them as text or html."""
23 parser
= argparse
.ArgumentParser(usage
='%(prog)s [options] <test temporary directory>', description
=__doc__
)
24 parser
.add_argument('-c', '--color', dest
='color', action
='store_true', help='outputs the combined log with events colored by source (requires posix terminal colors. Use less -r for viewing)')
25 parser
.add_argument('--html', dest
='html', action
='store_true', help='outputs the combined log as html. Requires jinja2. pip install jinja2')
26 args
, unknown_args
= parser
.parse_known_args()
28 if args
.color
and os
.name
!= 'posix':
29 print("Color output requires posix terminal colors.")
32 if args
.html
and args
.color
:
33 print("Only one out of --color or --html should be specified")
36 # There should only be one unknown argument - the path of the temporary test directory
37 if len(unknown_args
) != 1:
38 print("Unexpected arguments" + str(unknown_args
))
41 log_events
= read_logs(unknown_args
[0])
43 print_logs(log_events
, color
=args
.color
, html
=args
.html
)
45 def read_logs(tmp_dir
):
48 Delegates to generator function get_log_events() to provide individual log events
49 for each of the input log files."""
51 files
= [("test", "%s/test_framework.log" % tmp_dir
)]
52 for i
in itertools
.count():
53 logfile
= "{}/node{}/regtest/debug.log".format(tmp_dir
, i
)
54 if not os
.path
.isfile(logfile
):
56 files
.append(("node%d" % i
, logfile
))
58 return heapq
.merge(*[get_log_events(source
, f
) for source
, f
in files
])
60 def get_log_events(source
, logfile
):
61 """Generator function that returns individual log events.
63 Log events may be split over multiple lines. We use the timestamp
64 regex match as the marker for a new log event."""
66 with
open(logfile
, 'r') as infile
:
73 # if this line has a timestamp, it's the start of a new log event.
74 time_match
= TIMESTAMP_PATTERN
.match(line
)
77 yield LogEvent(timestamp
=timestamp
, source
=source
, event
=event
.rstrip())
79 timestamp
= time_match
.group()
80 # if it doesn't have a timestamp, it's a continuation line of the previous log.
83 # Flush the final event
84 yield LogEvent(timestamp
=timestamp
, source
=source
, event
=event
.rstrip())
85 except FileNotFoundError
:
86 print("File %s could not be opened. Continuing without it." % logfile
, file=sys
.stderr
)
88 def print_logs(log_events
, color
=False, html
=False):
89 """Renders the iterator of log events into text or html."""
91 colors
= defaultdict(lambda: '')
93 colors
["test"] = "\033[0;36m" # CYAN
94 colors
["node0"] = "\033[0;34m" # BLUE
95 colors
["node1"] = "\033[0;32m" # GREEN
96 colors
["node2"] = "\033[0;31m" # RED
97 colors
["node3"] = "\033[0;33m" # YELLOW
98 colors
["reset"] = "\033[0m" # Reset font color
100 for event
in log_events
:
101 print("{0} {1: <5} {2} {3}".format(colors
[event
.source
.rstrip()], event
.source
, event
.event
, colors
["reset"]))
107 print("jinja2 not found. Try `pip install jinja2`")
109 print(jinja2
.Environment(loader
=jinja2
.FileSystemLoader('./'))
110 .get_template('combined_log_template.html')
111 .render(title
="Combined Logs from testcase", log_events
=[event
._asdict
() for event
in log_events
]))
113 if __name__
== '__main__':