Installation
The first step is to get the latest version of procmail. When this article was started the last version was 3.11pre7.
After getting the source code for the program, in order to install it it will be necessary to uncompress and unpack it using the following command: tar -xzvf procmail.tar.gz
The next step is to edit the files Makefile and config.h. Since this article is only an introduction and to keep things simple in this first introduction, we will not explain the various configuration options of these files. The interested reader can consult the manual pages and the documentation provided with the sources.
Nevertheless, it is convenient to mention the existence of the option BASENAME in the Makefile. With this option we indicate the directory base where procmail will be installed. From the basename directory we indicate will hang other directories as bin and man.
Finally, to compile the package we must execute the command make install.
Procmail can be installed for the whole system and then be called through some rule of sendmail(8) or it can be by some user for his personal use. In the later case, the user will indicate the use of procmail in his .forward, where a line like the following should be present: |IFS=' ' && exec /home/juan/procmail/bin/procmail -f- || exit 75 #juan
In this line I assumed that the user juan has installed procmail in his HOME. For this particular line the BASENAME that should have been specified during compilation is /home/juan/procmail.
Basic Functions
Procmail reads from its standard input. It checks the file .procmailrc which is a configuration file that the user should have in his HOME. This file defines certain rules that will tell procmail what to do after reading a message. It can be directed to check for certain strings in the header of the message, in order to decide whether to save the message, ignore it, reply automatically, etc..
Procmail also let us deal automatically with the arriving mail or the mail stored in a file.
Configuration
The configuration file used is .procmailrc and it must be placed in HOME.
Every line started with # is considered a comment.
Lines started with :0 or :0: indicate the beginning of a new rule which tells procmail what to do with a message.
The lines started with * indicate a condition to be satisfied for a rule to be applied. This is the mechanism procmail uses to select which messages require processing with the rule and which do not.
The remaining lines, that is, those not starting with a :, or * are considered commands, in other words the action to be applied by procmail to the message satisfying the condition. Some possible actions or commands are deleting a message, forwarding a message, saving a message...
The first thing to be indicated in a .procmailrc are some of the environment variables. Here are some of the variables I would recommend to define in your .procmailrc
MAILDIR
Indicates the directory where procmail saves the files with mail messages. This variable usually points to $HOME/mail or $HOME/Mail. Whether it is one or the other is a matter of the kind of mail reader being used.
LOGFILE
Indicates the name of a log file where procmail leaves record of all the transactions carried out.
SENDMAIL
Indicates where to find sendmail, which will be used to automatically reply messages.
FORMAIL
Indicates where to find formail. This program is distributed together with procmail and its purpose is to modify the mail headers or to reformat a message before sending or storing a message.
DEFAULT
File where a message is saved if procmail was not able to apply any of the rules defined.
Anywhere in .procmailrc one can define an environment variable. If the variable is written without the symbol = followed by a value, such a variable is removed.
Regarding rules, they come in two groups: Rules that consider the message delivered after applying their action and rules that do not.
The first rules are simple, after applying their action then assume no further rules should apply to this message and therefore deliver it.
The second type of rules, those that do not consider the message delivered, upon applying their action is very useful when one wishes the possibility of applying multiple rules and actions to the same message before delivery.
The general syntax for a rule is the following:
:0 [options] [ : [exclusion file] ]
* condition 1
* condition 2
.
.
.
* condition N
command
Lets go by parts analyzing this construction. First every rule starts with a :0, after it may follow any of these options: H The condition is applied to the mail header.
B The condition is searched in the body of the message.
D When the condition is investigated, upper and lower cases are distinguishable.
A This rule is applied only if the previous one was.
a Like A, except that the execution of the previous rule must have been carried out without errors.
E This rule is executed if the previous one was not.
e This rule will be executed if the previous one was executed but exit with some error.
h The message header is passed to the command.
b The message body is passed to the command.
f The command is interpreted as a filter.
c Generate a carbon copy cc of the message. When executing a rule that considers the message delivered with this flag, then one accomplishes a work around its delivery, and further rules can be applied to the carbon copy of the message.
w Wait until the command finish executing to receive its exit code.
W As the previous option but in case of error does not emit any messages.
i Ignore possible typing errors.
r Writes the message such as it is. It does not test whether it ends with a blank line.
By default, if no options are specified, the condition is tested on the mail header (option H). The command receives in its standard input both header and body of the message (options h and b). There is no distinction between upper and lower case.
After :0 and the possible options, a second : may follow, if it does it indicates that the destination file where the message is going to be written should be blocked to avoid two processes writing simultaneously over the file. Optionally one can indicate the exclusion file which is used as a lock.
Next follow the conditions, one per line and preceded by a * character. Conditions are usually written as regular expressions in order to find character string within the header or body of a message. Regular expressions use the following symbols among others: ^ Beginning of a line.
$ End of a line . Any character except a carriage return
* Zero or more times.
+ One or more times.
? Zero or more times.
[a-z] Range of characters, in this example from a to z.
[^a-z]Any character outside the range a to z.
a|b Either 'a' or 'b'
After the conditions come a single command. If the first character of the command is any of the following then a special behavior onsets: ! The message is redirected to all the mail addresses indicated.
| If this symbol is continued by an executable then it is run whenever the condition is satisfied. If nothing follows the symbol the full text of the message goes to the standard output. If instead the name of a variable follows |, then the result of the indicated command is saved into that variable.
Mailing Lists
A situation where procmail can be very useful is in the management of our mail. Let us suppose that we are subscribed to three Linux mailing lists. Each list is identified by the address of origin, for instance we could have the following addresses
l-linux@calvo.teleco.ulpgc.es
linux@nuclecu.unam.mx
linux-security@redhat.com
Under normal circumstances the messages from these mailing lists will arrive together, at the same mail box, and if nothing is done they will remain mixed up. It would be much easier if as the mail were arriving it were sorted and stored in appropriate files.
Procmail can solve this problem easily. We could use a file .procmailrc with the following simple rules to sort our mail from the Linux mailing lists:
:0
* ^From.*l-linux@calvo.teleco.ulpgc.es
l-linux
:0
* ^From.*linux@nuclecu.unam.mx
linux-mx
:0
* ^From.*linux-security@redhat.com
linux-security
Let us examine carefully one of the rules. If you understand the inner workings of a rule, then is easy to understand the rest because the basic mechanism is always the same.
First one finds a :0 string that indicates the beginning of a new rule. There are no further options so procmail will take the default options for this rule: upper and lower case are indistinguishable, the condition applies only to the mail header, the command receives both header and body of the message.
In the next line one finds the condition, that as it was previously mentioned is always identifiable because its first character is a *. The condition is the following regular expression: ^From.*linux@nuclecu.unam.mx
The substring ^From tells procmail to search for those lines beginning by the substring From .
The following .* character means any number of characters. We saw before that in regular expressions a "." character is equivalent to any character and that * refers to zero or more. Then .* means that after the initial From there may be zero or more characters.
Next should come linux@nuclecu.unam.mx, which is the address from where the messages originate.
Thinking a bit about the regular expressions, you will see that the following lines would be recognize by this rule:
From: linux@nuclecu.unam.mx
From:linux@nuclecu.unam.mx
FROM linux@nuclecu.unam.mx
With this rule one can already distinguish messages coming from this address and the others. So now that one has the message what shall be done with it?.
The next line is the command (or action), and it indicates what to do with the message. In this case it is supposed to be delivered to the file linux-mx where it should be stored. In case of not indicating the absolute path to the file, by default it will be considered relative to the environment variable $MAILDIR.
Clearly messages arriving from different lists can now be distributed to various files according to their origin (From field).
Automatic Reply
Another situation where procmail will be useful is for automatic replies. It comes often handy, for example, if you wish to send your public PGP key automatically to anyone that requests it by E-Mail
To do this, write a rule that will consider as a petition for our PGP public key any message with the string PGP in its subject. This rule can be written as:
0:
* ^Subject.*PGP
| (formail -r ; cat $HOME/key.asc) | sendmail -t
This same idea can be applied to write the typical program that lets people know we are off on vacation and that we will reply to their E-Mail after coming back:
0:
| (formail -r; cat $HOME/vacations.txt) | sendmail -t
In the last case there is no condition since all messages should be sent the same notice.
Avoiding Infinite Loops in Automated Replies
In the previous examples no attempt was made to handle possible infinite loops that sometimes occur when mail is answered automatically.
If a message whose origin is our own E-Mail address would be replied to then the reply would come back to us. This message would be replied once more and so on in an infinity loop. In order to avoid it, when answering the message one should add an extra line in the header to indicate that message was already answered. Using the option -A of formail: formail -r -A"X-Loop: dir@email.es"
Where dir@email.es would be your own E-Mail address. This way when generating the header for the reply the line X-Loop is added so that later on one can check for it with a new rule:
:0
* !^X-Loop: dir@email.es
| (formail -r -A"X-Loop: dir@email.es" ;
cat $HOME/vacation.txt) | sendmail -t
This rule prevents an infinite loop because any message containing in the header a line X-Loop will not satisfy the condition and as a result will not be answered by procmail.
Decoding Files
Another interesting rule for our .procmailrc is that which decodifies the incoming mail encoded with uuencode(1) automatically. The rule could be given as:
:0 B
* ^begin 644 .*
{
MAILDIR=$HOME/files
:0
| uudecode
}
Here it is explicitly indicated with the option B that the rule's condition should only be tested on the body of the message.
If the rule finds a line beginning with the string "begin 644" it means that it has found the beginning of a file encoded with uuencode(1) so it establishes the environment variable MAILDIR, which is equivalent to changing the directory pointed by that variable. From that moment on, all the print or display actions will be performed taking the base directory indicated. In our case we are interested on saving the received messages in $HOME/files.
Next there is a rule, without condition, that only pipes the message to uudecode(1) for decodification. The original file will go to $HOME/files.
Conclusion
Hopefully after this brief introduction to procmail, it is clear that procmail is extremely versatile and it can help to manage your mail easy and conveniently. I recommend to experiment with the regular expressions and the rules, adapt them to your needs because procmail's possibilities go well beyond what I have been able to discuss in this short introduction.
|