1
/* Copyright (c) 2002-2013 Pigeonhole authors, see the included COPYING file
8
#include "str-sanitize.h"
11
#include "safe-mkstemp.h"
12
#include "mail-user.h"
14
#include "sieve-common.h"
15
#include "sieve-stringlist.h"
16
#include "sieve-binary.h"
17
#include "sieve-code.h"
18
#include "sieve-message.h"
19
#include "sieve-extensions.h"
20
#include "sieve-commands.h"
21
#include "sieve-actions.h"
22
#include "sieve-validator.h"
23
#include "sieve-generator.h"
24
#include "sieve-interpreter.h"
25
#include "sieve-dump.h"
26
#include "sieve-result.h"
28
#include "sieve-ext-variables.h"
30
#include "sieve-extprograms-common.h"
35
* "filter" <program-name: string> [<arguments: string-list>]
39
static bool cmd_filter_generate
40
(const struct sieve_codegen_env *cgenv,
41
struct sieve_command *ctx);
43
const struct sieve_command_def cmd_filter = {
46
-1, /* We check positional arguments ourselves */
49
sieve_extprogram_command_validate,
59
static bool cmd_filter_operation_dump
60
(const struct sieve_dumptime_env *denv, sieve_size_t *address);
61
static int cmd_filter_operation_execute
62
(const struct sieve_runtime_env *renv, sieve_size_t *address);
64
const struct sieve_operation_def cmd_filter_operation = {
65
"FILTER", &filter_extension,
67
cmd_filter_operation_dump,
68
cmd_filter_operation_execute
75
static bool cmd_filter_generate
76
(const struct sieve_codegen_env *cgenv, struct sieve_command *cmd)
78
sieve_operation_emit(cgenv->sblock, cmd->ext, &cmd_filter_operation);
80
/* Emit is_test flag */
81
sieve_binary_emit_byte(cgenv->sblock, ( cmd->ast_node->type == SAT_TEST ));
83
/* Generate arguments */
84
if ( !sieve_generate_arguments(cgenv, cmd, NULL) )
87
/* Emit a placeholder when the <arguments> argument is missing */
88
if ( sieve_ast_argument_next(cmd->first_positional) == NULL )
89
sieve_opr_omitted_emit(cgenv->sblock);
98
static bool cmd_filter_operation_dump
99
(const struct sieve_dumptime_env *denv, sieve_size_t *address)
101
unsigned int is_test = 0;
103
/* Read is_test flag */
104
if ( !sieve_binary_read_byte(denv->sblock, address, &is_test) )
107
sieve_code_dumpf(denv, "FILTER (%s)", (is_test ? "test" : "command"));
108
sieve_code_descend(denv);
110
/* Dump optional operands */
111
if ( sieve_action_opr_optional_dump(denv, address, NULL) != 0 )
114
if ( !sieve_opr_string_dump(denv, address, "program-name") )
117
return sieve_opr_stringlist_dump_ex(denv, address, "arguments", "");
124
static int cmd_filter_get_tempfile
125
(const struct sieve_runtime_env *renv)
127
struct sieve_instance *svinst = renv->svinst;
128
struct mail_user *mail_user = renv->scriptenv->user;
132
path = t_str_new(128);
133
mail_user_set_get_temp_prefix(path, mail_user->set);
134
fd = safe_mkstemp(path, 0600, (uid_t)-1, (gid_t)-1);
136
sieve_sys_error(svinst, "filter action: "
137
"safe_mkstemp(%s) failed: %m", str_c(path));
141
/* We just want the fd, unlink it */
142
if (unlink(str_c(path)) < 0) {
143
/* Shouldn't happen.. */
144
sieve_sys_error(svinst, "filter action: "
145
"unlink(%s) failed: %m", str_c(path));
146
if ( close(fd) < 0 ) {
147
sieve_sys_error(svinst, "filter action: "
148
"close(%s) failed after error: %m", str_c(path));
156
static int cmd_filter_operation_execute
157
(const struct sieve_runtime_env *renv, sieve_size_t *address)
159
const struct sieve_extension *this_ext = renv->oprtn->ext;
160
unsigned int is_test = 0;
161
struct sieve_stringlist *args_list = NULL;
162
enum sieve_error error = SIEVE_ERROR_NONE;
163
string_t *pname = NULL;
164
const char *program_name = NULL;
165
const char *const *args = NULL;
173
/* The is_test flag */
175
if ( !sieve_binary_read_byte(renv->sblock, address, &is_test) ) {
176
sieve_runtime_trace_error(renv, "invalid is_test flag");
177
return SIEVE_EXEC_BIN_CORRUPT;
180
/* Optional operands */
182
if ( sieve_action_opr_optional_read(renv, address, NULL, &ret, NULL) != 0 )
187
if ( (ret=sieve_extprogram_command_read_operands
188
(renv, address, &pname, &args_list)) <= 0 )
191
program_name = str_c(pname);
192
if ( args_list != NULL &&
193
sieve_stringlist_read_all(args_list, pool_datastack_create(), &args) < 0 ) {
194
sieve_runtime_trace_error(renv, "failed to read args operand");
195
return args_list->exec_status;
204
sieve_runtime_trace(renv, SIEVE_TRLVL_ACTIONS, "filter action");
205
sieve_runtime_trace_descend(renv);
206
sieve_runtime_trace(renv, SIEVE_TRLVL_ACTIONS,
207
"execute program `%s'", str_sanitize(program_name, 128));
210
if ( (tmp_fd=cmd_filter_get_tempfile(renv)) < 0 ) {
215
struct sieve_extprogram *sprog = sieve_extprogram_create
216
(this_ext, renv->scriptenv, renv->msgdata, "filter", program_name, args,
219
if ( sprog != NULL && sieve_extprogram_set_input_mail
220
(sprog, sieve_message_get_mail(renv->msgctx)) >= 0 ) {
221
struct ostream *outdata =
222
o_stream_create_fd(tmp_fd, 0, FALSE);
223
sieve_extprogram_set_output(sprog, outdata);
224
o_stream_unref(&outdata);
226
ret = sieve_extprogram_run(sprog);
232
sieve_extprogram_destroy(&sprog);
236
struct istream *newmsg;
238
sieve_runtime_trace(renv, SIEVE_TRLVL_ACTIONS,
239
"executed program successfully");
241
newmsg = i_stream_create_fd(tmp_fd, (size_t)-1, TRUE);
243
if ( (ret=sieve_message_substitute(renv->msgctx, newmsg)) >= 0 ) {
244
sieve_runtime_trace(renv, SIEVE_TRLVL_ACTIONS,
247
sieve_runtime_critical(renv, NULL, "filter action",
248
"filter action: failed to substitute message");
251
i_stream_unref(&newmsg);
255
if ( close(tmp_fd) < 0 ) {
257
(renv->svinst, "filter action: close(temp_file) failed: %m");
262
if ( error == SIEVE_ERROR_NOT_FOUND ) {
263
sieve_runtime_error(renv, NULL,
264
"filter action: program `%s' not found",
265
str_sanitize(program_name, 80));
267
sieve_extprogram_exec_error(renv->ehandler,
268
sieve_runtime_get_full_command_location(renv),
269
"filter action: failed to execute to program `%s'",
270
str_sanitize(program_name, 80));
273
sieve_runtime_trace(renv, SIEVE_TRLVL_ACTIONS,
274
"filter action: program indicated false result");
279
sieve_interpreter_set_test_result(renv->interp, ( ret > 0 ));
281
return SIEVE_EXEC_OK;