${COURIER_HOME} is the main installation directory, everything
lives under here. 
${COURIER_HOME}/ARCH is a soft link to ${ARCH} 
${COURIER_HOME}/${ARCH}/bin - binary files 
${COURIER_HOME}/${ARCH}/lib - library files 
${COURIER_HOME}/bin - mostly soft links to arch/bin 
${COURIER_HOME}/lib - mostly soft links to arch/lib 
${COURIER_HOME}/man - manual pages 
${COURIER_HOME}/local/config - local configuration 
${COURIER_HOME}/local/msgs - mail queue - messages and control files
${COURIER_HOME}/local/msgq - mail queue - control files, sorted by
next scheduled delivery attempt time 
${COURIER_HOME}/local/tmp - temporary directory
All directories can be softlinks, or mount points, EXCEPT that
local/msg[sq] and local/tmp must be on the same
filesystem.
In local/msg[sq] the userid of the control and the message file
are that of the user who created the message.
courierctl scripts - these scripts start and stop courierd,
and performs other administrative functions.
courierctl.startscripts - these scripts start courierd. They do NOT start any input module daemons, that must be done separately. Note that this command only begins the startup process. courierd may fail to start for some reason, you will need to check your system logs to verify that courierd is properly running.
courierctl.stop- this command stops courierd. courierd is stopped cold, interrupting any pending deliveries.
courierctl.restart- this command restarts courierd by sending it a SIGHUP. courierd will suspend starting any new deliveries, then wait for all current deliveries to finish, then restart itself. This action is necessary after making certain configuration changes.
courierd - the main scheduling daemon. The heart of Courier is just a
raw scheduler that figures out when to attempt to deliver messages that are in
the queue. There will actually be two courierd processes running, at
the same time. courierd is started and stopped by
courierctl.
courierd scheduling daemon is transport neutral, and external
modules are used to submit messages to courierd, and attempt to deliver
messages at regular intervals.  The term 'input module' refers to an
external module that submits messages to courierd, and 'output module' refers
to an external module that delivers messages. The term 'transport module', or
a plain 'module' refers to both an input module and an output module for the
same platform, which usually come together. For example, Courier typically
receives and sends mail via ESMTP, however, it is possible to adjust the
default configuration so that ESMTP is only used to receive mail. For sending,
mail gets delivered using some other protocol.
Output module are started by courierd, and communicate with
courierd via pipes. Output module are notified which messages must be
delivered, and to which addresses. Courier may send multiple messages to the
same output module, without waiting for an acknowledgement that an attempted
delivery for previous messages has been made.
E-mail addresses that appear in the message's headers, and which are listed in the message's delivery envelope, are stored in a "canonical" format. E-mail messages that come from an input module have their addresses rewritten from the transport-specific format to the canonical format. E-mail messages which are sent to be delivered to an output module have their addresses rewritten to the canonical format.
The canonical format is the format for E-mail addresses that is used by Courier's system. Therefore, messages that are delivered to local mailboxes do not undergo header or address rewriting.
Messages are submitted to the Courier server by running the ${COURIER_HOME}/bin/submit, which takes care of rewriting addresses from the transport format to canonical format. When an output module receives a message to deliver from Courier's queue, the E-mail addresses in the message will be in the canonical format, and the output module is responsible for rewriting addresses in the headers into the transport-specific format. The output module can take advantage of a set of library functions which automate this process.
On platforms that support dynamically-opened shared libraries, Courier can be compiled to use shared libraries. This allows rewriting functions to be modified, and have the changes take effect without fully shutting down Courier. Otherwise, they can be compiled as regular, statically linked libraries.
submit programsubmit is used to submit a message into Courier's scheduling queue.
submit is not invoked directly. submit does not have any
global read or execute permissions, only user and group permissions, therefore
it must be invoked by a process that executes at least as group mail.
submit provides the functionality most transports will need in order
to submit messages into the queue. Submit accepts several options, and one
argument: the name of the input module submitting the message. submit
also reads the environment, therefore the environment variables must be
scrubbed before invoking submit. submit communicates via
standard input and output with the module to read the identify of the sender
and the recipients of an E-mail message, and the message contents.
submit enforces message size limits and may reject the sender or any
recipient. Options are:
-expn=address- do not accept any message, just attempt to 'expand' this address.
-vrfy=address- do not accept any message, just attempt to 'verify' this address.
-verp- use VERP-encoded envelope sender address.
submit opens the shared libraries, reads the E-mail addresses of the
sender and the recipient, reads the message itself, and rewrites all addresses
to the canonical format by running the module's address rewriting function in
the shared library. submit may perform other adjustments, such as
adding the Date: or the Message-ID: header, or adding MIME
headers. submit also checks if any recipient is a system-defined
alias, and adds the alias's list of addresses to the recipient list.
submit will also try to remove duplicate E-mail addresses from the
recipient list, wherever possible.  Wherever possible, submit
will reject E-mail addresses that it knows are not valid.
mailq - display current mail queue for the running user (or the
entire queue, if invoked as root). Since individual control files are owned by
the originating user. mailq simply prints out the contents of every control
file it can open.
-s - sort messages by starting time. With many messages, this can eat up memory, and CPU cycles. The list is not sorted, by default.
reformime - standalone program that can be used by module to
implement some header rewriting. I am borrowing the reformime tool from the
maildrop package, which contains complete documentation for reformime.
That's it. Other program modules come as a part of plug-in transport module. For example, the SMTP transport doesn't have to be installed, resulting in Courier being used to just deliver mail on a single system. Or, only the SMTP output module can be installed, so that the machine will be able to send mail via SMTP, but not receive it. Or, only the SMTP input module can be installed, so that the machine can receive mail via SMTP, but not send it (which also means that bounces can't be delivered, so they double-bounce to postmaster).
submit program also reads the following variables from the
environment:
SPAMDATAFILTER=pathname - run this spam filter. After receiving
this message, run the indicated program. In addition to the environment
variables inherited by submit, submit sets MODULE to the name of the input
module, before running the spam filter. Please note that the program is run
with the user/group id of the submit process!
MIME=option - sets options for mime rewriting. The options are:
none - no mime rewriting whatsoever; 7bit - convert 8bit mime sections to
quoted-printable; 8bit - convert quoted-printable to 8bit.
NOADDMSGID - if set, submit will not add a Message-Id:
header, if the message does not have it.
NOADDDATE - if set, submit will not add a Date: header,
if the message doesn't have it.
SIZELIMIT - maximum size of a message in bytes. Larger messages
are rejected. If this environment variable is not set, Courier will read the
control/sizelimit file. If control/sizelimit does not exist,
Courier will use "0". "0" is used to specify that there is no maximum message
size limit.