printer.py 3.29 KB
Newer Older
yuguo960516's avatar
yuguo960516 committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
import pandas as pd
from collections import namedtuple


class Printer(object):
    def __init__(self, field_names, print_format="table", persistent_file=None):
        assert print_format in ("table", "normal")

        self.field_names_ = field_names
        self.format_ = print_format
        self.records_ = []
        self.handlers_ = dict()
        self.str_lens_ = dict()
        self.title_printed_ = False

        if persistent_file is not None:
            self.csv_ = open(persistent_file, "a")
        else:
            self.csv_ = None

        self.Record = None

    def __def__(self):
        if self.csv_ is not None:
            self.csv_.close()

    def finish(self):
        err = f"{len(self.field_names_)} vs. {len(self.handlers_)}"
        assert len(self.field_names_) == len(self.handlers_), err
        err = f"{len(self.field_names_)} vs. {len(self.str_lens_)}"
        assert len(self.field_names_) == len(self.str_lens_), err
        for fname in self.field_names_:
            assert fname in self.handlers_, f"{fname} handler not register"
            assert fname in self.str_lens_, f"{fname} str_len not register"

        self.Record = namedtuple("Record", self.field_names_)
        # DEBUG(zwx):
        # dummy = self.Record(*(["-"] * len(self.field_names_)))
        # df = pd.DataFrame(dummy)
        # if self.persistent_file_ is not None:
        #     df.to_csv(self.persistent_file_, mode='a', header=True)

    def record(self, *args, **kwargs):
        assert self.Record is not None
        r = self.Record(*args, **kwargs)
        self.records_.append(r)

    def register_handler(self, field, handler):
        assert callable(handler)
        self.handlers_[field] = handler

    def register_str_len(self, field, str_len):
        assert isinstance(str_len, int)
        self.str_lens_[field] = str_len

    def reset_records(self):
        self.records_ = []

    def print_table_title(self):
        fields = ""
        sep = ""

        for fname in self.field_names_:
            str_len = self.str_lens_[fname]
            fields += "| {} ".format(fname.ljust(str_len))
            sep += f"| {'-' * str_len} "

        fields += "|"
        sep += "|"
        print(fields)
        print(sep)
        self.title_printed_ = True

    def reset_title_printed(self):
        self.title_printed_ = False

    def print(self):
        df = pd.DataFrame(self.records_)
        fields = []
        for fname in self.field_names_:
            assert fname in self.handlers_
            handler = self.handlers_[fname]
            field_value = handler(df[fname])
            fields.append(field_value)

        if self.format_ == "table":
            if not self.title_printed_:
                self.print_table_title()

            record = ""
            for i, str_len in enumerate(self.str_lens_.values()):
                record += "| {} ".format(str(fields[i]).ljust(str_len))

            record += "|"
            print(record)

        elif self.format_ == "normal":
            record = ""

            for i, fval in enumerate(fields):
                fname = self.field_names_[i]
                record += f"{fname}: {fval}, "

            print(record)
        else:
            raise ValueError

        if self.csv_ is not None:
            df.to_csv(self.csv_, header=False)

        self.reset_records()