Previous   Next   Contents       (Exim 4.10 Specification)

38. Adding a local scan function to Exim

In these days of email worms, viruses, and ever-increasing spam, some sites want to apply a lot of checking to messages before accepting them. You can do a certain amount through string expansions and the condition condition in the ACL that runs after the SMTP DATA command (see chapter 37), but this has its limitations. To allow for more general checking that can be customized to a site's own requirements, there is the possibility of linking Exim with a private message scanning function, written in C. If you want to run code that is written in something other than C, you can of course use a little C stub to call it.

Unlike the ACL checks, which apply only to incoming SMTP messages, a local scan function is run for every incoming message. It can therefore be used to control non-SMTP messages from local processes.

Exim applies a timeout to calls of this function, and there is an option called local_scan_timeout for setting it. The default is 5 minutes. Zero means ``no timeout''. If the timeout is exceeded, the incoming message is rejected with a temporary error if it is an SMTP message. For a non-smtp message, the message is dropped and Exim ends with a non-zero code. The incident is logged on the main and reject logs.

To make use of the local scan function feature, you must tell Exim where your function is before building Exim, by setting LOCAL_SCAN_SOURCE in your Local/Makefile. A recommended place to put it is in the Local directory, so you might set

  LOCAL_SCAN_SOURCE=Local/local_scan.c

for example. The function must be called local_scan(). It is called by Exim after it has received a message, when the success return code is about to be sent. For SMTP input, this is after all the ACLs have been run. The return code from your function controls whether the message is actually accepted or not. There is a commented template function (that just accepts the message) in the file src/local_scan.c.

38.1. API for local_scan()

You must include this line near the start of your code:

  #include "local_scan.h"

This header file defines a number of variables and other values, and the prototype for the function itself. Exim is coded to use unsigned char values almost exclusively, and one of the things this header defines is a shorthand for unsigned char called uschar. It also contains the following macro definitions, to simplify casting character strings and pointers to character strings:

  #define CS   (char *)
  #define CSS  (char **)
  #define US   (unsigned char *)
  #define USS  (unsigned char **)

The function prototype for local_scan() is:

  extern int local_scan(int fd, uschar **return_text);

The arguments are as follows:

The function must return an int value which is one of the following macros:

If the message is not being received by interactive SMTP, failures are reported by writing to stderr or by sending an email, as configured by the -oe command line options.

38.2. Available Exim variables

The header local_scan.h gives you access to a number of Exim variables. These are the only ones that are guaranteed to be maintained from release to release:


uschar *sender_address

The envelope sender address. For bounce messages this is the empty string.


header_line *header_list

A pointer to a chain of header lines. The header_line structure is discussed below.


header_line *header_last

A pointer to the last of the header lines.


uschar *interface_address

The IP address of the interface that received the message, as a string. This is NULL for locally submitted messages.


int interface_port

The port on which this message was received.


uschar *received_protocol

The name of the protocol by which the message was received.


int recipients_count

The number of accepted recipients.


recipient_item *recipients_list

The list of accepted recipients, held in a vector of length recipients_count. The recipient_item structure is discussed below. You can add additional recipients by calling receive_add_recipient() (see below). You can delete recipients by removing them from the vector and adusting the value in recipients_count. In particular, by setting recipients_count to zero you remove all recipients. If you then return the value LOCAL_SCAN_ACCEPT, the message is accepted, but immediately blackholed. To replace the recipients, set recipients_count zero and then call receive_add_recipient() as often as needed.


uschar *sender_host_address

The IP address of the sending host, as a string. This is NULL for locally-submitted messages.


uschar *sender_host_authenticated

The name of the authentication mechanism that was used, or NULL if the message was not received over an authenticated SMTP connection.


uschar *sender_host_name

The name of the sending host, if known.


int sender_host_port

The port on the sending host.



38.3. Structure of header lines

The header_line structure contains the members listed below. You can add additional header lines by calling the header_add() function (see below). You can cause header lines to be ignored (deleted) by setting their type to *.


struct header_line *next

A pointer to the next header line, or NULL for the last line.


int type

A code identifying certain headers that Exim recognizes. The codes are printing characters, and are documented in chapter 48 of this manual. Notice in particular that any header line whose type is * is not transmitted with the message. This flagging is used for header lines that have been rewritten, or are to be removed (for example, Envelope-sender: header lines.) Effectively, * means ``deleted''.


int slen

The number of characters in the header line, including the terminating and any internal newlines.


uschar *text

A pointer to the text of the header. It always ends with a newline, followed by a zero byte. Internal newlines are preserved.



38.4. Structure of recipient items

The recipient_item structure contains these members:


uschar *address

This is a pointer to the recipient address as it was received.


int pno

This is used in later Exim processing when top level addresses are created by the one_time option. It is not relevant at the time local_scan() is run and must always contain -1 at this stage.


uschar *errors_to

If this value is not NULL, bounce messages caused by failing to deliver to the recipient are sent to the address it contains. In other words, it overrides the envelope sender for this one recipient. (Compare the errors_to generic router option.) When local_scan() is called, this field is NULL for all recipients.



38.5. Available Exim functions

The header local_scan.h gives you access to a number of Exim functions. These are the only ones that are guaranteed to be maintained from release to release:


pid_t child_open_exim(int *fd)
int child_close(pid_t pid, int timeout)

These two functions provide you with a means of submitting a new message to Exim. (Of course, you can also call /usr/sbin/sendmail yourself if you want, but this packages it all up for you.) The first function creates a pipe, forks a subprocess that is running

  exim -t -oem -oi -f <>

and returns to you (via the int * argument) a file descriptor for the pipe that is connected to the standard input. The yield of the function is the PID of the subprocess. You can then write a message to the file descriptor, with recipients in To:, Cc:, and/or Bcc: header lines. When you have finished, call child_close() with the PID as the first argument, and a timeout in seconds as the second. A value of zero means wait as long as it takes (which is usually fine in this circumstance). The return value is as follows:


uschar *expand_string(uschar *string)

This is an interface to Exim's string expansion code. The return value is the expanded string, or NULL if there was an expansion failure.


void header_add(int type, char *format, ...)

This function allows you to add additional header lines. The first argument is the type, and should normally be a space character. The second argument is a format string and any number of substitution arguments as for sprintf(). You may include internal newlines if you want, and you must ensure that the string ends with a newline.


void log_write(unsigned int selector, int which, char *format, ...)

This function writes to Exim's log files. The first argument should be zero (it is concerned with log_selector). The second argument can be LOG_MAIN or LOG_REJECT or the inclusive ``or'' of both of them. The remaining arguments are a format and relevant insertion arguments. The string should not contain any newlines, not even at the end.


void receive_add_recipient(uschar *address, int pno)

This function adds an additional recipient to the message. The first argument is the recipient address. The second argument must always be -1.

This function does not allow you to specify a private errors_to address (as described with the structure of recipient_item above), because it pre-dates the addition of that field to the structure. However, it is easy to add such a value afterwards. For example:

  receive_add_recipient(monitor@mydom.example, -1);
  recipients_list[recipients_count-1].errors_to = 
    US"postmaster@mydom.example";


void *store_get(int)

This function accesses Exim's internal store manager. It gets a new chunk of memory whose size is given by the argument. Exim bombs out if it ever runs out of memory.


uschar *string_copy(uschar *string)
uschar *string_copyn(uschar *string, int length)
uschar *string_sprintf(char *format, ...)

These three functions create strings using Exim's dynamic store facilities. The first makes a copy of an entire string. The second copies up to a maximum number of characters, indicated by the second argument. The third uses a format and insertion arguments to create a new string. In each case, the result is a pointer to a new string.




Previous  Next  Contents       (Exim 4.10 Specification)