]> mj.ucw.cz Git - libucw.git/blobdiff - ucw/table-types.c
tableprinter: size units are now static
[libucw.git] / ucw / table-types.c
index 902e2829b75c057165e0e019eb6771b5ac27d259..1894a9a6daddf673b929461bc69ee7b3eeac75a9 100644 (file)
+/*
+ *     UCW Library -- Table printer types
+ *
+ *     (c) 2014 Robert Kessl <robert.kessl@economia.cz>
+ */
+
 #include <ucw/lib.h>
-#include <ucw/config.h>
 #include <ucw/table-types.h>
 #include <ucw/fastbuf.h>
-#include <ucw/config.h>
 #include <ucw/table.h>
 #include <time.h>
 #include <stdio.h>
 #include <stdlib.h>
+#include <inttypes.h>
+#include <errno.h>
+
+/** xt_size **/
 
-void table_col_size_name(struct table *tbl, const char *col_name, u64 val)
+static struct unit_definition xtype_units_size[] = {
+  [SIZE_UNIT_BYTE] = { "", 1LLU, 1 },
+  [SIZE_UNIT_KILOBYTE] = { "KB", 1024LLU, 1 },
+  [SIZE_UNIT_MEGABYTE] = { "MB", 1024LLU * 1024LLU, 1 },
+  [SIZE_UNIT_GIGABYTE] = { "GB", 1024LLU * 1024LLU * 1024LLU, 1 },
+  [SIZE_UNIT_TERABYTE] = { "TB", 1024LLU * 1024LLU * 1024LLU * 1024LLU, 1 },
+  { 0, 0, 0 }
+};
+
+static const char *xt_size_format(void *src, u32 fmt, struct mempool *pool)
 {
-  int col = table_get_col_idx(tbl, col_name);
-  table_col_size(tbl, col, val);
+  u64 curr_val = *(u64*) src;
+
+  if(fmt == XTYPE_FMT_RAW) {
+    return mp_printf(pool, "%"PRIu64, curr_val);
+  }
+
+  uint out_units = SIZE_UNIT_BYTE;
+
+  if(fmt == XTYPE_FMT_DEFAULT) {
+    curr_val = curr_val;
+    out_units = SIZE_UNIT_BYTE;
+  } else if(fmt == XTYPE_FMT_PRETTY) {
+    curr_val = curr_val;
+    out_units = SIZE_UNIT_BYTE;
+  } else if((fmt & SIZE_UNITS_FIXED) != 0) {
+    curr_val = curr_val / xtype_units_size[fmt & ~SIZE_UNITS_FIXED].num;
+    out_units = fmt & ~SIZE_UNITS_FIXED;
+  }
+
+  return mp_printf(pool, "%"PRIu64"%s", curr_val, xtype_units_size[out_units].unit);
 }
 
-void table_col_size(struct table *tbl, int col, u64 val)
+static const char *xt_size_fmt_parse(const char *opt_str, u32 *dest, struct mempool *pool)
 {
-  ASSERT_MSG(col < tbl->column_count && col >= 0, "Table column %d does not exist.", col);
-  ASSERT(tbl->columns[col].type == COL_TYPE_ANY || COL_TYPE_SIZE == tbl->columns[col].type);
-
-  tbl->last_printed_col = col;
-  tbl->row_printing_started = 1;
-
-  static u64 unit_div[] = {
-    [UNIT_BYTE] = (u64) 1,
-    [UNIT_KILOBYTE] = (u64) 1024LU,
-    [UNIT_MEGABYTE] = (u64) (1024LU * 1024LU),
-    [UNIT_GIGABYTE] = (u64) (1024LU * 1024LU * 1024LU),
-    [UNIT_TERABYTE] = (u64) (1024LU * 1024LU * 1024LU * 1024LU)
-  };
-
-  static const char *unit_suffix[] = {
-    [UNIT_BYTE] = "",
-    [UNIT_KILOBYTE] = "KB",
-    [UNIT_MEGABYTE] = "MB",
-    [UNIT_GIGABYTE] = "GB",
-    [UNIT_TERABYTE] = "TB"
-  };
-
-  // FIXME: do some rounding?
-  uint out_type = 0;
-  if(tbl->column_order[col].output_type == CELL_OUT_UNINITIALIZED) {
-    val = val / unit_div[UNIT_BYTE];
-    out_type = 0;
-  } else {
-    val = val / unit_div[tbl->column_order[col].output_type];
-    out_type = tbl->column_order[col].output_type;
-  }
-
-  table_col_printf(tbl, col, "%lu%s", val, unit_suffix[out_type]);
-}
+  if(opt_str == NULL) {
+    return "NULL is not supported as a column argument.";
+  }
 
-#define FORMAT_TIME_SIZE 20    // Minimum buffer size
+  if(strlen(opt_str) == 0 || strcmp(opt_str, "B") == 0 || strcmp(opt_str, "Bytes") == 0) {
+    *dest = SIZE_UNIT_BYTE | SIZE_UNITS_FIXED;
+    return NULL;
+  }
 
-void table_col_timestamp_name(struct table *tbl, const char * col_name, u64 val)
-{
-  int col = table_get_col_idx(tbl, col_name);
-  table_col_size(tbl, col, val);
+  int unit_idx = xtype_unit_parser(opt_str, xtype_units_size);
+  if(unit_idx == -1) {
+    return mp_printf(pool, "Unknown option '%s'", opt_str);
+  }
+
+  *dest = unit_idx | SIZE_UNITS_FIXED;
+  return NULL;
 }
 
-void table_col_timestamp(struct table *tbl, int col, u64 val)
+static const char *xt_size_parse(const char *str, void *dest, struct mempool *pool)
 {
-  ASSERT_MSG(col < tbl->column_count && col >= 0, "Table column %d does not exist.", col);
-  ASSERT(tbl->columns[col].type == COL_TYPE_ANY || COL_TYPE_TIMESTAMP == tbl->columns[col].type);
+  errno = 0;
+  char *units_start = NULL;
+  u64 parsed = strtoul(str, &units_start, 10);
+  if(str == units_start) {
+    return mp_printf(pool, "Invalid value of size: '%s'.", str);
+  }
+
+  if(errno == EINVAL) {
+    return "Error occured during parsing of size.";
+  }
+  if(errno == ERANGE) {
+    return "Error: size value either too large or too small.";
+  }
+
+  if(*units_start == 0) {
+    *(u64*) dest = (u64) parsed;
+    return NULL;
+  }
+
+  int unit_idx = xtype_unit_parser(units_start, xtype_units_size);
+  if(unit_idx == -1) {
+    return mp_printf(pool, "Invalid units: '%s'.", units_start);
+  }
+
+  *(u64*) dest = parsed * xtype_units_size[unit_idx].num;
+  return NULL;
+}
+
+TABLE_COL_BODY(size, u64)
+
+const struct xtype xt_size = {
+  .size = sizeof(u64),
+  .name = "size",
+  .parse = xt_size_parse,
+  .format = xt_size_format,
+  .parse_fmt = xt_size_fmt_parse
+};
 
+/** xt_timestamp **/
+
+#define FORMAT_TIME_SIZE 20    // Minimum buffer size
+
+static const char *xt_timestamp_format(void *src, u32 fmt, struct mempool *pool)
+{
   char formatted_time_buf[FORMAT_TIME_SIZE] = { 0 };
 
-  time_t tmp_time = (time_t)val;
+  u64 tmp_time_u64 = *(u64*)src;
+  time_t tmp_time = (time_t) tmp_time_u64;
   struct tm t = *gmtime(&tmp_time);
-
-  switch (tbl->column_order[col].output_type) {
-  case TIMESTAMP_EPOCH:
-  case CELL_OUT_UNINITIALIZED:
-    sprintf(formatted_time_buf, "%lu", val);
+  switch (fmt) {
+  case XTYPE_FMT_DEFAULT:
+  case XTYPE_FMT_RAW:
+    sprintf(formatted_time_buf, "%"PRIu64, tmp_time_u64);
     break;
-  case TIMESTAMP_DATETIME:
+  case XTYPE_FMT_PRETTY:
     strftime(formatted_time_buf, FORMAT_TIME_SIZE, "%F %T", &t);
     break;
   default:
-    abort();
+    ASSERT(0);
     break;
   }
 
-  table_col_printf(tbl, col, "%s", formatted_time_buf);
+  return mp_printf(pool, "%s", formatted_time_buf);
+}
+
+static const char *xt_timestamp_fmt_parse(const char *opt_str, u32 *dest, struct mempool *pool)
+{
+  if(opt_str == NULL) {
+    return "NULL is not supported as a column argument.";
+  }
+
+  if(strcasecmp(opt_str, "timestamp") == 0 || strcasecmp(opt_str, "epoch") == 0) {
+    *dest = TIMESTAMP_EPOCH;
+    return NULL;
+  } else if(strcasecmp(opt_str, "datetime") == 0) {
+    *dest = TIMESTAMP_DATETIME;
+    return NULL;
+  }
+
+  return mp_printf(pool, "Invalid column format option: '%s'.", opt_str);
 }
 
+static const char *xt_timestamp_parse(const char *str, void *dest, struct mempool *pool)
+{
+  errno = 0;
+  char *parse_end = NULL;
+  u64 parsed = strtoul(str, &parse_end, 10);
+  if(str == parse_end) {
+    return mp_printf(pool, "Invalid value of timestamp: '%s'.", str);
+  }
+  if(errno == EINVAL) {
+    return "Error occured during parsing of size.";
+  }
+
+  if(errno == ERANGE) {
+    return "Error: size value either too large or too small.";
+  }
+
+  if(*parse_end == 0) {
+    *(u64*) dest = (u64) parsed;
+    return NULL;
+  }
+
+  struct tm parsed_time;
+  parse_end = strptime(str, "%F %T", &parsed_time);
+  if(parse_end == NULL) {
+    return mp_printf(pool, "Invalid value of timestamp: '%s'.", str);
+  }
+  if(*parse_end != 0) {
+    return mp_printf(pool, "Invalid value of timestamp: '%s'.", str);
+  }
+
+  time_t tmp_time = mktime(&parsed_time);
+  *(u64*)dest = (u64) tmp_time;
+
+  return NULL;
+}
+
+TABLE_COL_BODY(timestamp, u64)
+
+const struct xtype xt_timestamp = {
+  .size = sizeof(u64),
+  .name = "timestamp",
+  .parse = xt_timestamp_parse,
+  .format = xt_timestamp_format,
+  .parse_fmt = xt_timestamp_fmt_parse
+};