diff options
author | Karel Zak | 2014-03-19 16:33:09 +0100 |
---|---|---|
committer | Karel Zak | 2014-04-03 12:29:16 +0200 |
commit | 3e542c765412904331fddb722b52fcdf31997d44 (patch) | |
tree | d1aa9ff7c70cd3e987297688cf35b136621dab0f /libsmartcols/src/table_print.c | |
parent | libsmartcols: add table struct, cleanup refcounting (diff) | |
download | kernel-qcow2-util-linux-3e542c765412904331fddb722b52fcdf31997d44.tar.gz kernel-qcow2-util-linux-3e542c765412904331fddb722b52fcdf31997d44.tar.xz kernel-qcow2-util-linux-3e542c765412904331fddb722b52fcdf31997d44.zip |
libsmartcols: add table code
Signed-off-by: Karel Zak <kzak@redhat.com>
Diffstat (limited to 'libsmartcols/src/table_print.c')
-rw-r--r-- | libsmartcols/src/table_print.c | 577 |
1 files changed, 577 insertions, 0 deletions
diff --git a/libsmartcols/src/table_print.c b/libsmartcols/src/table_print.c new file mode 100644 index 000000000..694b9c9ca --- /dev/null +++ b/libsmartcols/src/table_print.c @@ -0,0 +1,577 @@ +/* + * table.c - functions handling the data at the table level + * + * Copyright (C) 2010-2014 Karel Zak <kzak@redhat.com> + * + * This file may be redistributed under the terms of the + * GNU Lesser General Public License. + */ +#include <stdio.h> +#include <stdlib.h> +#include <unistd.h> +#include <string.h> +#include <termios.h> +#include <ctype.h> + +#include "nls.h" +#include "mbsalign.h" +#include "widechar.h" +#include "ttyutils.h" +#include "carefulputc.h" +#include "smartcolsP.h" + +#define is_last_column(_tb, _cl) \ + list_entry_is_last(&(_cl)->cl_columns, &(_tb)->tb_columns) + +static void print_data(struct libscols_table *tb, + struct libscols_column *cl, + struct libscols_line *ln, /* optional */ + struct libscols_cell *ce, /* optional */ + char *data) +{ + size_t len = 0, i, width; + char *buf; + const char *color = NULL; + + assert(tb); + assert(cl); + + if (!data) + data = ""; + + /* raw mode */ + if (tb->flags & SCOLS_FL_RAW) { + fputs_nonblank(data, stdout); + if (!is_last_column(tb, cl)) + fputc(' ', stdout); + return; + } + + /* NAME=value mode */ + if (tb->flags & SCOLS_FL_EXPORT) { + fprintf(stdout, "%s=", scols_cell_get_data(&cl->header)); + fputs_quoted(data, stdout); + if (!is_last_column(tb, cl)) + fputc(' ', stdout); + return; + } + + if (tb->is_term) { + if (ce && !color) + color = ce->color; + if (ln && !color) + color = ln->color; + if (!color) + color = cl->color; + } + + /* note that 'len' and 'width' are number of cells, not bytes */ + buf = mbs_safe_encode(data, &len); + data = buf; + if (!data) + data = ""; + + if (!len || len == (size_t) -1) { + len = 0; + data = NULL; + } + width = cl->width; + + if (is_last_column(tb, cl) && len < width) + width = len; + + /* truncate data */ + if (len > width && (cl->flags & SCOLS_FL_TRUNC)) { + if (data) + len = mbs_truncate(data, &width); + if (!data || len == (size_t) -1) { + len = 0; + data = NULL; + } + } + if (data) { + if (!(tb->flags & SCOLS_FL_RAW) && (cl->flags & SCOLS_FL_RIGHT)) { + size_t xw = cl->width; + if (color) + fputs(color, stdout); + fprintf(stdout, "%*s", (int) xw, data); + if (color) + fputs(UL_COLOR_RESET, stdout); + if (len < xw) + len = xw; + } + else { + if (color) + fputs(color, stdout); + fputs(data, stdout); + if (color) + fputs(UL_COLOR_RESET, stdout); + } + } + for (i = len; i < width; i++) + fputc(' ', stdout); /* padding */ + + if (!is_last_column(tb, cl)) { + if (len > width && !(cl->flags & SCOLS_FL_TRUNC)) { + fputc('\n', stdout); + for (i = 0; i <= (size_t) cl->seqnum; i++) { + struct libscols_column *x = scols_table_get_column(tb, i); + printf("%*s ", -((int)x->width), " "); + } + } else + fputc(' ', stdout); /* columns separator */ + } + + free(buf); +} + +static char *line_get_ascii_art(struct libscols_table *tb, + struct libscols_line *ln, + char *buf, size_t *bufsz) +{ + const char *art; + size_t len; + + assert(ln); + + if (!ln->parent) + return buf; + + buf = line_get_ascii_art(tb, ln->parent, buf, bufsz); + if (!buf) + return NULL; + + if (list_entry_is_last(&ln->ln_children, &ln->parent->ln_branch)) + art = " "; + else + art = tb->symbols->vert; + + len = strlen(art); + if (*bufsz < len) + return NULL; /* no space, internal error */ + + memcpy(buf, art, len); + *bufsz -= len; + return buf + len; +} + +static char *line_get_data(struct libscols_table *tb, + struct libscols_line *ln, + struct libscols_column *cl, + char *buf, size_t bufsz) +{ + const char *data; + struct libscols_symbols *sym; + struct libscols_cell *ce; + char *p = buf; + + assert(tb); + assert(ln); + assert(cl); + assert(cl->seqnum <= tb->ncols); + + memset(buf, 0, bufsz); + + ce = scols_line_get_cell(ln, cl->seqnum); + data = ce ? scols_cell_get_data(ce) : NULL; + if (!data) + return NULL; + + if (!(cl->flags & SCOLS_FL_TREE)) { + strncpy(buf, data, bufsz); + buf[bufsz - 1] = '\0'; + return buf; + } + + /* + * Tree stuff + */ + if (ln->parent) { + p = line_get_ascii_art(tb, ln->parent, buf, &bufsz); + if (!p) + return NULL; + } + + sym = tb->symbols; + + if (!ln->parent) + snprintf(p, bufsz, "%s", data); /* root node */ + else if (list_entry_is_last(&ln->ln_children, &ln->parent->ln_branch)) + snprintf(p, bufsz, "%s%s", sym->right, data); /* last chaild */ + else + snprintf(p, bufsz, "%s%s", sym->branch, data); /* any child */ + + return buf; +} + +/* + * Prints data, data maybe be printed in more formats (raw, NAME=xxx pairs) and + * control and non-printable chars maybe encoded in \x?? hex encoding. + */ +static void print_line(struct libscols_table *tb, + struct libscols_line *ln, char *buf, size_t bufsz) +{ + struct libscols_column *cl; + struct libscols_iter itr; + + assert(ln); + + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_column(tb, &itr, &cl) == 0) + print_data(tb, cl, ln, + scols_line_get_cell(ln, cl->seqnum), + line_get_data(tb, ln, cl, buf, bufsz)); + fputc('\n', stdout); +} + +static void print_header(struct libscols_table *tb, char *buf, size_t bufsz) +{ + struct libscols_column *cl; + struct libscols_iter itr; + + assert(tb); + + if (!tb->first_run || + (tb->flags & SCOLS_FL_NOHEADINGS) || + (tb->flags & SCOLS_FL_EXPORT) || + list_empty(&tb->tb_lines)) + return; + + /* set width according to the size of data + */ + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_column(tb, &itr, &cl) == 0) { + strncpy(buf, scols_cell_get_data(&cl->header), bufsz); + buf[bufsz - 1] = '\0'; + print_data(tb, cl, NULL, &cl->header, buf); + } + fputc('\n', stdout); +} + +static void print_table(struct libscols_table *tb, char *buf, size_t bufsz) +{ + struct libscols_line *ln; + struct libscols_iter itr; + + assert(tb); + + print_header(tb, buf, bufsz); + + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_line(tb, &itr, &ln) == 0) + print_line(tb, ln, buf, bufsz); +} + +static void print_tree_line(struct libscols_table *tb, + struct libscols_line *ln, + char *buf, size_t bufsz) +{ + struct list_head *p; + + print_line(tb, ln, buf, bufsz); + + if (list_empty(&ln->ln_branch)) + return; + + /* print all children */ + list_for_each(p, &ln->ln_branch) { + struct libscols_line *chld = + list_entry(p, struct libscols_line, ln_children); + print_tree_line(tb, chld, buf, bufsz); + } +} + +static void print_tree(struct libscols_table *tb, char *buf, size_t bufsz) +{ + struct libscols_line *ln; + struct libscols_iter itr; + + assert(tb); + + print_header(tb, buf, bufsz); + + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_line(tb, &itr, &ln) == 0) { + if (ln->parent) + continue; + print_tree_line(tb, ln, buf, bufsz); + } +} + +/* + * This function counts column width. + * + * For the SCOLS_FL_NOEXTREMES columns is possible to call this function two + * times. The first pass counts width and average width. If the column + * contains too large fields (width greater than 2 * average) then the column + * is marked as "extreme". In the second pass all extreme fields are ignored + * and column width is counted from non-extreme fields only. + */ +static void count_column_width(struct libscols_table *tb, + struct libscols_column *cl, + char *buf, + size_t bufsz) +{ + struct libscols_line *ln; + struct libscols_iter itr; + int count = 0; + size_t sum = 0; + + assert(tb); + assert(cl); + + cl->width = 0; + + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_line(tb, &itr, &ln) == 0) { + char *data = line_get_data(tb, ln, cl, buf, bufsz); + size_t len = data ? mbs_safe_width(data) : 0; + + if (len == (size_t) -1) /* ignore broken multibyte strings */ + len = 0; + if (len > cl->width_max) + cl->width_max = len; + + if (cl->is_extreme && len > cl->width_avg * 2) + continue; + else if (cl->flags & SCOLS_FL_NOEXTREMES) { + sum += len; + count++; + } + if (len > cl->width) + cl->width = len; + } + + if (count && cl->width_avg == 0) { + cl->width_avg = sum / count; + + if (cl->width_max > cl->width_avg * 2) + cl->is_extreme = 1; + } + + /* check and set minimal column width */ + if (scols_cell_get_data(&cl->header)) + cl->width_min = mbs_safe_width(scols_cell_get_data(&cl->header)); + + /* enlarge to minimal width */ + if (cl->width < cl->width_min && !(cl->flags & SCOLS_FL_STRICTWIDTH)) + cl->width = cl->width_min; + + /* use relative size for large columns */ + else if (cl->width_hint >= 1 && cl->width < (size_t) cl->width_hint + && cl->width_min < (size_t) cl->width_hint) + + cl->width = (size_t) cl->width_hint; +} + +/* + * This is core of the scols_* voodo... + */ +static void recount_widths(struct libscols_table *tb, char *buf, size_t bufsz) +{ + struct libscols_column *cl; + struct libscols_iter itr; + size_t width = 0; /* output width */ + int trunc_only; + int extremes = 0; + + /* set basic columns width + */ + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_column(tb, &itr, &cl) == 0) { + count_column_width(tb, cl, buf, bufsz); + width += cl->width + (is_last_column(tb, cl) ? 0 : 1); + extremes += cl->is_extreme; + } + + if (!tb->is_term) + return; + + /* reduce columns with extreme fields + */ + if (width > tb->termwidth && extremes) { + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_column(tb, &itr, &cl) == 0) { + size_t org_width; + + if (!cl->is_extreme) + continue; + + org_width = cl->width; + count_column_width(tb, cl, buf, bufsz); + + if (org_width > cl->width) + width -= org_width - cl->width; + else + extremes--; /* hmm... nothing reduced */ + } + } + + if (width < tb->termwidth) { + /* try to found extreme column which fits into available space + */ + if (extremes) { + /* enlarge the first extreme column */ + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_column(tb, &itr, &cl) == 0) { + size_t add; + + if (!cl->is_extreme) + continue; + + /* this column is tooo large, ignore? + if (cl->width_max - cl->width > + (tb->termwidth - width)) + continue; + */ + + add = tb->termwidth - width; + if (add && cl->width + add > cl->width_max) + add = cl->width_max - cl->width; + + cl->width += add; + width += add; + + if (width == tb->termwidth) + break; + } + } + if (width < tb->termwidth) { + /* enalarge the last column */ + struct libscols_column *cl = list_entry( + tb->tb_columns.prev, struct libscols_column, cl_columns); + + if (!(cl->flags & SCOLS_FL_RIGHT) && tb->termwidth - width > 0) { + cl->width += tb->termwidth - width; + width = tb->termwidth; + } + } + } + + /* bad, we have to reduce output width, this is done in two steps: + * 1/ reduce columns with a relative width and with truncate flag + * 2) reduce columns with a relative width without truncate flag + */ + trunc_only = 1; + while (width > tb->termwidth) { + size_t org = width; + + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_column(tb, &itr, &cl) == 0) { + if (width <= tb->termwidth) + break; + if (cl->width_hint > 1 && !(cl->flags & SCOLS_FL_TRUNC)) + continue; /* never truncate columns with absolute sizes */ + if (cl->flags & SCOLS_FL_TREE) + continue; /* never truncate the tree */ + if (trunc_only && !(cl->flags & SCOLS_FL_TRUNC)) + continue; + if (cl->width == cl->width_min) + continue; + + /* truncate column with relative sizes */ + if (cl->width_hint < 1 && cl->width > 0 && width > 0 && + cl->width > cl->width_hint * tb->termwidth) { + cl->width--; + width--; + } + /* truncate column with absolute size */ + if (cl->width_hint > 1 && cl->width > 0 && width > 0 && + !trunc_only) { + cl->width--; + width--; + } + + } + if (org == width) { + if (trunc_only) + trunc_only = 0; + else + break; + } + } + +/* + fprintf(stderr, "terminal: %d, output: %d\n", tb->termwidth, width); + + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_column(tb, &itr, &cl) == 0) { + fprintf(stderr, "width: %s=%zd [hint=%d, avg=%zd, max=%zd, extreme=%s]\n", + cl->name, cl->width, + cl->width_hint > 1 ? (int) cl->width_hint : + (int) (cl->width_hint * tb->termwidth), + cl->width_avg, + cl->width_max, + cl->is_extreme ? "yes" : "not"); + } +*/ + return; +} +static size_t strlen_line(struct libscols_line *ln) +{ + size_t i, sz = 0; + + assert(ln); + + for (i = 0; i < ln->ncells; i++) { + struct libscols_cell *ce = scols_line_get_cell(ln, i); + const char *data = ce ? scols_cell_get_data(ce) : NULL; + + sz += data ? strlen(data) : 0; + } + + return sz; +} + +/* + * @tb: table + * + * Prints the table to stdout + */ +int scols_print_table(struct libscols_table *tb) +{ + char *line; + size_t line_sz; + struct libscols_line *ln; + struct libscols_iter itr; + + assert(tb); + if (!tb) + return -1; + + if (tb->first_run) { + tb->is_term = isatty(STDOUT_FILENO); + + if (tb->is_term && !tb->termwidth) + tb->termwidth = get_terminal_width(); + if (tb->termwidth <= 0) + tb->termwidth = 80; + } + + line_sz = tb->termwidth; + + scols_reset_iter(&itr, SCOLS_ITER_FORWARD); + while (scols_table_next_line(tb, &itr, &ln) == 0) { + size_t sz = strlen_line(ln); + if (sz > line_sz) + line_sz = sz; + } + + line_sz++; /* make a space for \0 */ + line = malloc(line_sz); + if (!line) + return -ENOMEM; + + if (tb->first_run && + !((tb->flags & SCOLS_FL_RAW) || (tb->flags & SCOLS_FL_EXPORT))) + recount_widths(tb, line, line_sz); + + if (tb->flags & SCOLS_FL_TREE) + print_tree(tb, line, line_sz); + else + print_table(tb, line, line_sz); + + free(line); + tb->first_run = FALSE; + return 0; +} |