aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@rspamd.com>2023-01-21 12:15:12 +0000
committerGitHub <noreply@github.com>2023-01-21 12:15:12 +0000
commit4445807558c29aa5c376b19fcb81528141ec3571 (patch)
tree83f343a8904e4684cc82c7d7746583565d38a2f1
parentea4d03b6f98f373e9c1fa4f96af69b60765e6194 (diff)
parent42e1c8d507f72adc2a7390ac4baf1248067cf40b (diff)
downloadrspamd-4445807558c29aa5c376b19fcb81528141ec3571.tar.gz
rspamd-4445807558c29aa5c376b19fcb81528141ec3571.zip
Merge pull request #4377 from amishmm/humanreport
[Feature] rspamc: add -R option for human readable report
-rw-r--r--doc/rspamc.113
-rw-r--r--doc/rspamc.1.md4
-rw-r--r--src/client/rspamc.cxx199
3 files changed, 195 insertions, 21 deletions
diff --git a/doc/rspamc.1 b/doc/rspamc.1
index 338f8053d..0f823e64b 100644
--- a/doc/rspamc.1
+++ b/doc/rspamc.1
@@ -149,6 +149,19 @@ Bind to specified ip address
.RS
.RE
.TP
+.B \-R, \-\-human
+Output human readable report.
+The first line of the output contains the message score and three
+threshold scores, in this format:
+.IP
+.nf
+\f[C]
+ score/greylist/addheader/reject,action=N:ACTION,spam=0|1,skipped=0|1
+\f[]
+.fi
+.RS
+.RE
+.TP
.B \-j, \-\-json
Output formatted JSON
.RS
diff --git a/doc/rspamc.1.md b/doc/rspamc.1.md
index afb256792..335c22513 100644
--- a/doc/rspamc.1.md
+++ b/doc/rspamc.1.md
@@ -86,6 +86,10 @@ requires input.
-b *host:port*, \--bind=*host:port*
: Bind to specified ip address
+-R, \--human
+: Output human readable report. The first line of the output contains the message score and three threshold scores, in this format:
+: score/greylist/addheader/reject,action=N:ACTION,spam=0|1,skipped=0|1
+
-j, \--json
: Output formatted JSON
diff --git a/src/client/rspamc.cxx b/src/client/rspamc.cxx
index 973bb58a4..5bb75ef38 100644
--- a/src/client/rspamc.cxx
+++ b/src/client/rspamc.cxx
@@ -71,6 +71,7 @@ static gboolean pass_all;
static gboolean tty = FALSE;
static gboolean verbose = FALSE;
static gboolean print_commands = FALSE;
+static gboolean humanreport = FALSE;
static gboolean json = FALSE;
static gboolean compact = FALSE;
static gboolean headers = FALSE;
@@ -134,6 +135,7 @@ static GOptionEntry entries[] =
"Bind to specified ip address", nullptr},
{"commands", 0, 0, G_OPTION_ARG_NONE, &print_commands,
"List available commands", nullptr},
+ {"human", 'R', 0, G_OPTION_ARG_NONE, &humanreport, "Output human readable report", nullptr},
{"json", 'j', 0, G_OPTION_ARG_NONE, &json, "Output json reply", nullptr},
{"compact", '\0', 0, G_OPTION_ARG_NONE, &compact, "Output compact json reply", nullptr},
{"headers", 0, 0, G_OPTION_ARG_NONE, &headers, "Output HTTP headers",
@@ -824,6 +826,71 @@ add_options(GQueue *opts)
}
static void
+print_indented_line(FILE *out, std::string_view line, size_t maxlen, size_t indent)
+{
+ if (maxlen < 1) {
+ return;
+ }
+
+ std::string_view s;
+ for (size_t pos = 0; pos < line.size(); pos += s.size()) {
+ s = line.substr(pos, pos ? (maxlen-indent) : maxlen);
+ if (indent && pos) {
+ fmt::print(out, "{:>{}}", " ", indent);
+ }
+ fmt::print(out, "{}\n", s);
+ }
+}
+
+static void
+rspamc_symbol_human_output(FILE *out, const ucl_object_t *obj)
+{
+ auto first = true;
+ auto score = 0.0;
+ const char *desc = nullptr;
+
+ const auto *key = ucl_object_key(obj);
+ const auto *val = ucl_object_lookup(obj, "score");
+ if (val != nullptr) {
+ score = ucl_object_todouble(val);
+ }
+
+ val = ucl_object_lookup(obj, "description");
+ if (val != nullptr) {
+ desc = ucl_object_tostring(val);
+ }
+
+ auto line = fmt::format("{:>4.1f} {:<22} ", score, key);
+ if (desc != nullptr) {
+ line += desc;
+ }
+
+ val = ucl_object_lookup(obj, "options");
+ if (val != nullptr && ucl_object_type(val) == UCL_ARRAY) {
+ ucl_object_iter_t it = nullptr;
+ const ucl_object_t *cur;
+
+ line += fmt::format("{}[", desc == nullptr ? "" : " ");
+
+ while ((cur = ucl_object_iterate (val, &it, true)) != nullptr) {
+ if (first) {
+ line += fmt::format("{}", ucl_object_tostring(cur));
+ first = false;
+ }
+ else {
+ line += fmt::format(",{}", ucl_object_tostring(cur));
+ }
+ }
+ line += ']';
+ }
+ else if (desc == nullptr) {
+ line += '\n';
+ }
+
+ print_indented_line(out, line, 78, 28);
+}
+
+static void
rspamc_symbol_output(FILE *out, const ucl_object_t *obj)
{
auto first = true;
@@ -835,7 +902,7 @@ rspamc_symbol_output(FILE *out, const ucl_object_t *obj)
fmt::print(out, "({:.2f})", ucl_object_todouble(val));
}
val = ucl_object_lookup(obj, "options");
- if (val != nullptr && val->type == UCL_ARRAY) {
+ if (val != nullptr && ucl_object_type(val) == UCL_ARRAY) {
ucl_object_iter_t it = nullptr;
const ucl_object_t *cur;
@@ -858,40 +925,71 @@ rspamc_symbol_output(FILE *out, const ucl_object_t *obj)
static void
rspamc_metric_output(FILE *out, const ucl_object_t *obj)
{
- double score = 0, required_score = 0;
int got_scores = 0;
+ bool is_spam = false, is_skipped = false;
+ double score = 0, required_score = 0, greylist_score =0, addheader_score = 0;
auto print_protocol_string = [&](const char *ucl_name, const char *output_message) {
auto *elt = ucl_object_lookup(obj, ucl_name);
if (elt) {
- fmt::print(out, "{}: {}\n", output_message,
- emphasis_argument(ucl_object_tostring(elt)));
+ if (humanreport) {
+ fmt::print(out, ",{}={}", output_message, emphasis_argument(ucl_object_tostring(elt)));
+ }
+ else {
+ fmt::print(out, "{}: {}\n", output_message, emphasis_argument(ucl_object_tostring(elt)));
+ }
}
};
- fmt::print(out, "[Metric: default]\n");
+ if (!humanreport) {
+ fmt::print(out, "[Metric: default]\n");
+ }
const auto *elt = ucl_object_lookup(obj, "required_score");
-
if (elt) {
required_score = ucl_object_todouble(elt);
got_scores++;
}
elt = ucl_object_lookup(obj, "score");
-
if (elt) {
score = ucl_object_todouble(elt);
got_scores++;
}
+ /* XXX: greylist_score is not yet in checkv2 */
+ elt = ucl_object_lookup(obj, "greylist_score");
+ if (elt) {
+ greylist_score = ucl_object_todouble(elt);
+ }
+
+ /* XXX: addheader_score is not yet in checkv2 */
+ elt = ucl_object_lookup(obj, "addheader_score");
+ if (elt) {
+ addheader_score = ucl_object_todouble(elt);
+ }
+
+ if (humanreport) {
+ fmt::print(out,
+ "{}/{}/{}/{}",
+ emphasis_argument(score, 2),
+ emphasis_argument(greylist_score, 2),
+ emphasis_argument(addheader_score, 2),
+ emphasis_argument(required_score, 2));
+ }
+
elt = ucl_object_lookup(obj, "action");
if (elt) {
auto act = rspamd_action_from_str_rspamc(ucl_object_tostring(elt));
if (act.has_value()) {
if (!tty) {
- print_protocol_string("action", "Action");
+ if (humanreport) {
+ fmt::print(out, ",action={}:{}", act.value(), ucl_object_tostring(elt));
+ }
+ else {
+ print_protocol_string("action", "Action");
+ }
}
else {
/* Colorize action type */
@@ -915,26 +1013,63 @@ rspamc_metric_output(FILE *out, const ucl_object_t *obj)
colorized_action = fmt::format(fmt::emphasis::bold, ucl_object_tostring(elt));
break;
}
- fmt::print(out, "Action: {}\n", colorized_action);
+
+ if (humanreport) {
+ fmt::print(out, ",action={}:{}", act.value(), colorized_action);
+ }
+ else {
+ fmt::print(out, "Action: {}\n", colorized_action);
+ }
}
- fmt::print(out, "Spam: {}\n", emphasis_argument(act.value() < METRIC_ACTION_GREYLIST ?
- "true" : "false"));
+ is_spam = act.value() < METRIC_ACTION_GREYLIST ? true : false;
+ if (!humanreport) {
+ fmt::print(out, "Spam: {}\n", is_spam ? "true" : "false");
+ }
}
else {
- print_protocol_string("action", "Action");
+ if (humanreport) {
+ fmt::print(out, ",action={}:{}", METRIC_ACTION_NOACTION, ucl_object_tostring(elt));
+ }
+ else {
+ print_protocol_string("action", "Action");
+ }
}
}
- print_protocol_string("subject", "Subject");
+ if (!humanreport) {
+ print_protocol_string("subject", "Subject");
+ }
+
+ if (humanreport) {
+ /* XXX: why checkv2 does not provide "is_spam"? */
+ elt = ucl_object_lookup(obj, "is_spam");
+ if (elt) {
+ is_spam = ucl_object_toboolean(elt);
+ }
+
+ elt = ucl_object_lookup(obj, "is_skipped");
+ if (elt) {
+ is_skipped = ucl_object_toboolean(elt);
+ }
- if (got_scores == 2) {
+ fmt::print(out, ",spam={},skipped={}\n", is_spam ? 1 : 0, is_skipped ? 1 : 0);
+ }
+ else if (got_scores == 2) {
fmt::print(out,
"Score: {} / {}\n",
emphasis_argument(score, 2),
emphasis_argument(required_score, 2));
}
+ if (humanreport) {
+ fmt::print(out, "Content analysis details: ({} points, {} required)\n\n",
+ emphasis_argument(score, 2),
+ emphasis_argument(required_score, 2));
+ fmt::print(out, " pts rule name description\n");
+ fmt::print(out, "---- ---------------------- --------------------------------------------------\n");
+ }
+
elt = ucl_object_lookup(obj, "symbols");
if (elt) {
@@ -949,9 +1084,12 @@ rspamc_metric_output(FILE *out, const ucl_object_t *obj)
sort_ucl_container_with_default(symbols, "name");
for (const auto *sym_obj : symbols) {
- rspamc_symbol_output(out, sym_obj);
+ humanreport ? rspamc_symbol_human_output(out, sym_obj) : rspamc_symbol_output(out, sym_obj);
}
}
+ if (humanreport) {
+ fmt::print(out, "\n");
+ }
}
static void
@@ -988,8 +1126,10 @@ rspamc_symbols_output(FILE *out, ucl_object_t *obj)
}
};
- print_protocol_string("message-id", "Message-ID");
- print_protocol_string("queue-id", "Queue-ID");
+ if (!humanreport) {
+ print_protocol_string("message-id", "Message-ID");
+ print_protocol_string("queue-id", "Queue-ID");
+ }
const auto *elt = ucl_object_lookup(obj, "urls");
@@ -1003,7 +1143,14 @@ rspamc_symbols_output(FILE *out, ucl_object_t *obj)
emitted = (char *)ucl_object_emit(elt, UCL_EMIT_JSON);
}
- fmt::print(out, "Urls: {}\n", emitted);
+ if (humanreport) {
+ if (emitted && strcmp(emitted, "[]")) {
+ print_indented_line(out, fmt::format("Domains found: {}", emitted), 78, 4);
+ }
+ }
+ else {
+ fmt::print(out, "Urls: {}\n", emitted);
+ }
free(emitted);
}
@@ -1018,11 +1165,21 @@ rspamc_symbols_output(FILE *out, ucl_object_t *obj)
emitted = (char *)ucl_object_emit(elt, UCL_EMIT_JSON);
}
- fmt::print(out, "Emails: {}\n", emitted);
+ if (humanreport) {
+ if (emitted && strcmp(emitted, "[]")) {
+ print_indented_line(out, fmt::format("Emails found: {}", emitted), 78, 4);
+ }
+ }
+ else {
+ fmt::print(out, "Emails: {}\n", emitted);
+ }
free(emitted);
}
print_protocol_string("error", "Scan error");
+ if (humanreport) {
+ return;
+ }
elt = ucl_object_lookup(obj, "messages");
if (elt && elt->type == UCL_OBJECT) {
@@ -1646,13 +1803,13 @@ rspamc_client_cb(struct rspamd_client_connection *conn,
}
else {
if (cmd.need_input && !json) {
- if (!compact) {
+ if (!compact && !humanreport) {
fmt::print(out, "Results for file: {} ({:.3} seconds)\n",
emphasis_argument(cbdata->filename), diff);
}
}
else {
- if (!compact && !json) {
+ if (!compact && !json && !humanreport) {
fmt::print(out, "Results for command: {} ({:.3} seconds)\n",
emphasis_argument(cmd.name), diff);
}