/* GNU Mailutils -- a suite of utilities for electronic mail
Copyright (C) 1999-2021 Free Software Foundation, Inc.
GNU Mailutils is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3, or (at your option)
any later version.
GNU Mailutils is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with GNU Mailutils. If not, see . */
#ifdef HAVE_CONFIG_H
# include
#endif
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define D_DEFAULT "TPt"
int keep_going;
int compile_only;
char *mbox_url;
int sieve_debug;
int verbose;
char *script;
int expression_option;
int dry_run;
static int sieve_print_locus = 1; /* Should the log messages include the
locus */
static int no_program_name;
static mu_list_t env_list;
static mu_list_t var_list;
static int
sieve_setenv (void *item, void *data)
{
char *str = item;
mu_sieve_machine_t mach = data;
int rc = mu_sieve_set_environ (mach, str, str + strlen (str) + 1);
if (rc)
mu_error (_("can't set environment item %s: %s"),
str, mu_strerror (rc));
return 0;
}
static int
sieve_setvar (void *item, void *data)
{
char *str = item;
mu_sieve_machine_t mach = data;
mu_sieve_variable_initialize (mach, str, str + strlen (str) + 1);
return 0;
}
static void
modify_debug_flags (mu_debug_level_t set, mu_debug_level_t clr)
{
mu_debug_level_t lev;
mu_debug_get_category_level (mu_sieve_debug_handle, &lev);
mu_debug_set_category_level (mu_sieve_debug_handle, (lev & ~clr) | set);
}
static void
set_debug_level (const char *arg)
{
for (; *arg; arg++)
{
switch (*arg)
{
case 'T':
modify_debug_flags (MU_DEBUG_LEVEL_UPTO(MU_DEBUG_TRACE9),
MU_DEBUG_LEVEL_MASK(MU_DEBUG_ERROR));
break;
case 'P':
modify_debug_flags (MU_DEBUG_LEVEL_MASK(MU_DEBUG_PROT), 0);
break;
case 'g':
modify_debug_flags (MU_DEBUG_LEVEL_MASK(MU_DEBUG_TRACE1), 0);
break;
case 't':
modify_debug_flags (MU_DEBUG_LEVEL_MASK(MU_DEBUG_TRACE4), 0);
break;
case 'i':
modify_debug_flags (MU_DEBUG_LEVEL_MASK(MU_DEBUG_TRACE9), 0);
break;
default:
mu_error (_("%c is not a valid debug flag"), *arg);
}
}
}
static void
cli_compile_and_dump (struct mu_parseopt *po, struct mu_option *opt,
char const *arg)
{
compile_only = 2;
}
static void
cli_debug (struct mu_parseopt *po, struct mu_option *opt, char const *arg)
{
set_debug_level (arg);
}
static void
cli_email (struct mu_parseopt *po, struct mu_option *opt, char const *arg)
{
int rc = mu_set_user_email (arg);
if (rc)
mu_parseopt_error (po, _("invalid email: %s"), mu_strerror (rc));
}
static void
assign (struct mu_parseopt *po, struct mu_option *opt, char const *arg,
mu_list_t *plist, char const *what)
{
char *p = strchr (arg, '=');
if (p == NULL)
mu_parseopt_error (po, _("malformed %s: %s"), what, arg);
else
{
char *str;
str = mu_strdup (arg);
str[p - arg] = 0;
if (!*plist)
{
mu_list_create (plist);
mu_list_set_destroy_item (*plist, mu_list_free_item);
}
mu_list_append (*plist, str);
}
}
static void
cli_env (struct mu_parseopt *po, struct mu_option *opt, char const *arg)
{
assign (po, opt, arg, &env_list, _("environment setting"));
}
static void
cli_var (struct mu_parseopt *po, struct mu_option *opt, char const *arg)
{
assign (po, opt, arg, &var_list, _("variable assignment"));
}
static struct mu_option sieve_options[] = {
{ "dry-run", 'n', NULL, MU_OPTION_DEFAULT,
N_("do not execute any actions, just print what would be done"),
mu_c_bool, &dry_run },
{ "no-actions", 0, NULL, MU_OPTION_ALIAS },
{ "keep-going", 'k', NULL, MU_OPTION_DEFAULT,
N_("keep on going if execution fails on a message"),
mu_c_bool, &keep_going },
{ "compile-only", 'c', NULL, MU_OPTION_DEFAULT,
N_("compile script and exit"),
mu_c_bool, &compile_only },
{ "dump", 'D', NULL, MU_OPTION_DEFAULT,
N_("compile script, dump disassembled sieve code to terminal and exit"),
mu_c_string, NULL, cli_compile_and_dump },
{ "mbox-url", 'f', N_("MBOX"), MU_OPTION_DEFAULT,
N_("mailbox to sieve (defaults to user's mail spool)"),
mu_c_string, &mbox_url },
{ "ticket", 't', N_("TICKET"), MU_OPTION_DEFAULT,
N_("ticket file for user authentication"),
mu_c_string, &mu_ticket_file },
{ "debug", 'd', N_("FLAGS"), MU_OPTION_ARG_OPTIONAL,
N_("debug flags (defaults to \"" D_DEFAULT "\")"),
mu_c_string, NULL, cli_debug, D_DEFAULT },
{ "verbose", 'v', NULL, MU_OPTION_DEFAULT,
N_("log all actions"),
mu_c_bool, &verbose },
{ "line-info", 0, N_("BOOL"), MU_OPTION_DEFAULT,
N_("print source location along with action logs (default)"),
mu_c_bool, &sieve_print_locus },
{ "email", 'e', N_("ADDRESS"), MU_OPTION_DEFAULT,
N_("override user email address"),
mu_c_string, NULL, cli_email },
{ "expression", 'E', NULL, MU_OPTION_DEFAULT,
N_("treat SCRIPT as Sieve program text"),
mu_c_bool, &expression_option },
{ "no-program-name", 0, NULL, MU_OPTION_DEFAULT,
N_("do not prefix diagnostic messages with the program name"),
mu_c_int, &no_program_name },
{ "environment", 0, N_("NAME=VALUE"), MU_OPTION_DEFAULT,
N_("set sieve environment value"),
mu_c_string, NULL, cli_env },
{ "variable", 0, N_("NAME=VALUE"), MU_OPTION_DEFAULT,
N_("set sieve variable"),
mu_c_string, NULL, cli_var },
MU_OPTION_END
}, *options[] = { sieve_options, NULL };
int
mu_compat_printer (void *data, mu_log_level_t level, const char *buf)
{
fputs (buf, stderr);
return 0;
}
static int
cb_debug (void *data, mu_config_value_t *val)
{
if (mu_cfg_assert_value_type (val, MU_CFG_STRING))
return 1;
set_debug_level (val->v.string);
return 0;
}
static int
cb_email (void *data, mu_config_value_t *val)
{
int rc;
if (mu_cfg_assert_value_type (val, MU_CFG_STRING))
return 1;
rc = mu_set_user_email (val->v.string);
if (rc)
mu_error (_("invalid email: %s"), mu_strerror (rc));
return rc;
}
static struct mu_cfg_param sieve_cfg_param[] = {
{ "keep-going", mu_c_bool, &keep_going, 0, NULL,
N_("Do not abort if execution fails on a message.") },
{ "mbox-url", mu_c_string, &mbox_url, 0, NULL,
N_("Mailbox to sieve (defaults to user's mail spool)."),
N_("url") },
{ "ticket", mu_c_string, &mu_ticket_file, 0, NULL,
N_("Ticket file for user authentication."),
N_("ticket") },
{ "debug", mu_cfg_callback, NULL, 0, cb_debug,
N_("Debug flags. Argument consists of one or more of the following "
"flags:\n"
" g - main parser traces\n"
" T - mailutils traces (sieve.trace9)\n"
" P - network protocols (sieve.prot)\n"
" t - sieve trace (MU_SIEVE_DEBUG_TRACE)\n"
" i - sieve instructions trace (MU_SIEVE_DEBUG_INSTR)."),
N_("arg: string") },
{ "verbose", mu_c_bool, &verbose, 0, NULL,
N_("Log all executed actions.") },
{ "line-info", mu_c_bool, &sieve_print_locus, 0, NULL,
N_("Print source locations along with action logs (default).") },
{ "email", mu_cfg_callback, NULL, 0, cb_email,
N_("Set user email address."),
N_("arg: string") },
{ NULL }
};
static char *sieve_capa[] = {
"debug",
"mailbox",
"locking",
"logging",
"mailer",
"sieve",
NULL
};
static struct mu_cli_setup cli = {
options,
sieve_cfg_param,
N_("GNU sieve -- a mail filtering tool."),
N_("SCRIPT"),
NULL,
N_("Sieve-specific debug levels:\n\
\n\
trace1 - print parse tree before optimization\n\
trace2 - print parse tree after optimization\n\
trace3 - print parser traces\n\
trace4 - print tests and actions being executed\n\
trace9 - print each Sieve instruction being executed\n\
\n\
Compatibility debug flags:\n\
g - main parser traces\n\
T - mailutils traces (same as --debug-level=sieve.trace0-trace1)\n\
P - network protocols (same as --debug-level=sieve.=prot)\n\
t - sieve trace (same as --debug-level=sieve.=trace4)\n\
i - sieve instructions trace (same as --debug-level=sieve.=trace9)\n")
};
static void
_sieve_action_log (mu_sieve_machine_t mach,
const char *action, const char *fmt, va_list ap)
{
size_t uid = 0;
mu_message_t msg;
mu_stream_t stream;
mu_sieve_get_diag_stream (mach, &stream);
msg = mu_sieve_get_message (mach);
mu_message_get_uid (msg, &uid);
mu_stream_printf (stream, "\033s<%d>\033%c<%d>", MU_LOG_NOTICE,
sieve_print_locus ? 'O' : 'X', MU_LOGMODE_LOCUS);
mu_stream_printf (stream, _("%s on msg uid %lu"),
action, (unsigned long) uid);
if (fmt && strlen (fmt))
{
mu_stream_printf (stream, ": ");
mu_stream_vprintf (stream, fmt, ap);
}
mu_stream_printf (stream, "\n");
mu_stream_unref (stream);
}
static int
sieve_message (mu_sieve_machine_t mach)
{
int rc;
mu_stream_t instr;
mu_message_t msg;
mu_attribute_t attr;
rc = mu_stdio_stream_create (&instr, MU_STDIN_FD, MU_STREAM_SEEK);
if (rc)
{
mu_error (_("cannot create stream: %s"), mu_strerror (rc));
return EX_SOFTWARE;
}
rc = mu_stream_to_message (instr, &msg);
mu_stream_unref (instr);
if (rc)
{
mu_error (_("cannot create message from stream: %s"),
mu_strerror (rc));
return EX_SOFTWARE;
}
mu_message_get_attribute (msg, &attr);
mu_attribute_unset_deleted (attr);
rc = mu_sieve_message (mach, msg);
if (rc)
/* FIXME: switch (rc)...*/
return EX_SOFTWARE;
return mu_attribute_is_deleted (attr) ? 1 : EX_OK;
}
static int
sieve_mailbox (mu_sieve_machine_t mach)
{
int rc;
mu_mailbox_t mbox = NULL;
/* Create and open the mailbox. */
if ((rc = mu_mailbox_create_default (&mbox, mbox_url)) != 0)
{
if (mbox)
mu_error (_("could not create mailbox `%s': %s"),
mbox_url, mu_strerror (rc));
else
mu_error (_("could not create default mailbox: %s"),
mu_strerror (rc));
goto cleanup;
}
/* Open the mailbox read-only if we aren't going to modify it. */
if (mu_sieve_is_dry_run (mach))
rc = mu_mailbox_open (mbox, MU_STREAM_READ);
else
rc = mu_mailbox_open (mbox, MU_STREAM_RDWR);
if (rc != 0)
{
if (mbox)
{
mu_url_t url = NULL;
mu_mailbox_get_url (mbox, &url);
mu_error (_("cannot open mailbox %s: %s"),
mu_url_to_string (url), mu_strerror (rc));
}
else
mu_error (_("cannot open default mailbox: %s"),
mu_strerror (rc));
mu_mailbox_destroy (&mbox);
goto cleanup;
}
/* Process the mailbox */
rc = mu_sieve_mailbox (mach, mbox);
cleanup:
if (mbox && !dry_run)
{
int e;
/* A message won't be marked deleted unless the script executed
succesfully on it, so we always do an expunge, it will delete
any messages that were marked DELETED even if execution failed
on a later message. */
if ((e = mu_mailbox_expunge (mbox)) != 0)
{
if (mbox)
mu_error (_("expunge on mailbox `%s' failed: %s"),
mbox_url, mu_strerror (e));
else
mu_error (_("expunge on default mailbox failed: %s"),
mu_strerror (e));
}
if (e && !rc)
rc = e;
}
mu_sieve_machine_destroy (&mach);
mu_mailbox_close (mbox);
mu_mailbox_destroy (&mbox);
/* FIXME: switch (rc) ... */
return rc ? EX_SOFTWARE : EX_OK;
}
int
main (int argc, char *argv[])
{
mu_sieve_machine_t mach;
int rc;
/* Native Language Support */
MU_APP_INIT_NLS ();
mu_auth_register_module (&mu_auth_tls_module);
mu_cli_capa_register (&mu_cli_capa_sieve);
mu_sieve_debug_init ();
mu_register_all_formats ();
mu_cli (argc, argv, &cli, sieve_capa, NULL, &argc, &argv);
if (dry_run)
verbose++;
if (no_program_name)
{
mu_stream_t errstr;
mu_log_tag = NULL;
rc = mu_stdstream_strerr_create (&errstr, MU_STRERR_STDERR, 0, 0,
NULL, NULL);
if (rc == 0)
{
mu_stream_destroy (&mu_strerr);
mu_strerr = errstr;
}
}
if (argc == 0)
{
mu_error (_("script must be specified"));
exit (EX_USAGE);
}
else if (argc == 1)
{
if (expression_option)
script = mu_strdup (argv[0]);
else if (strcmp (argv[0], "-") == 0)
{
mu_stream_t mstr;
mu_off_t size;
int rc;
rc = mu_memory_stream_create (&mstr, MU_STREAM_RDWR);
if (rc)
{
mu_diag_funcall (MU_DIAG_ERROR, "mu_memory_stream_create",
NULL, rc);
exit (EX_SOFTWARE);
}
rc = mu_stream_copy (mstr, mu_strin, 0, &size);
if (rc)
{
mu_diag_funcall (MU_DIAG_ERROR, "mu_stream_copy", NULL, rc);
exit (EX_SOFTWARE);
}
rc = mu_stream_seek (mstr, 0, MU_SEEK_SET, NULL);
if (rc)
{
mu_diag_funcall (MU_DIAG_ERROR, "mu_stream_seek", NULL, rc);
exit (EX_SOFTWARE);
}
script = mu_alloc (size + 1);
rc = mu_stream_read (mstr, script, size, NULL);
if (rc)
{
mu_diag_funcall (MU_DIAG_ERROR, "mu_stream_read", NULL, rc);
exit (EX_SOFTWARE);
}
script[size] = 0;
mu_stream_destroy (&mstr);
expression_option = 1;
}
else
script = mu_tilde_expansion (argv[0], MU_HIERARCHY_DELIMITER, NULL);
}
else
{
mu_error (_("only one SCRIPT can be specified"));
exit (EX_USAGE);
}
/* Sieve interpreter setup. */
rc = mu_sieve_machine_create (&mach);
if (rc)
{
mu_error (_("cannot initialize sieve machine: %s"), mu_strerror (rc));
return EX_SOFTWARE;
}
mu_sieve_set_environ (mach, "location", "MS");
mu_sieve_set_environ (mach, "phase", "post");
mu_list_foreach (env_list, sieve_setenv, mach);
mu_list_destroy (&env_list);
if (var_list)
{
mu_sieve_require_variables (mach);
mu_list_foreach (var_list, sieve_setvar, mach);
mu_list_destroy (&var_list);
}
if (verbose)
mu_sieve_set_logger (mach, _sieve_action_log);
if (expression_option)
{
struct mu_locus_point pt;
pt.mu_file = "stdin";
pt.mu_line = 1;
pt.mu_col = 0;
rc = mu_sieve_compile_text (mach, script, strlen (script), &pt);
}
else
rc = mu_sieve_compile (mach, script);
if (rc)
return EX_CONFIG;
/* We can finish if it's only a compilation check. */
if (compile_only)
{
if (compile_only == 2)
{
mu_sieve_set_dbg_stream (mach, mu_strout);
mu_sieve_disass (mach);
}
return EX_OK;
}
mu_sieve_set_dry_run (mach, dry_run);
if (mbox_url && strcmp (mbox_url, "-") == 0)
rc = sieve_message (mach);
else
rc = sieve_mailbox (mach);
return rc;
}