4.4 posftfix的性能控制 
之所以对postfix的性能进行控制,是为了在遇到邮件风暴时保证postfix可以正常运行。通常,我们可以通过对下列postfix参数的配置来调节postfix的性能,这些参数都是通过mail.cf配置文件进行配置的,修改以后不要忘了运行postfix reload命令来使配置生效。 

1. 进程数限制 
可以通过default_process_limit 参数来控制postfix系统同时可以运行的最 
大进程数目。缺省值是50个。  

2. 对同一目标主机的并发连接限制 
当向同一目标主机发出SMTP连接时,postfix初始化发出两个SMTP连接, 
如果投递成功则增加并发的SMTP连接数目,遇到拥塞时又减少并发连接的数目。postfix中通过以下的参数对同一目标主机的并发连接进行控制: 
* initial_destination_concurrency:控制对同一目标主机的初始化并发连接数目。缺省值为2。 
* default_destination_concurrency_limit:控制初始化连接后对同一目标主机的最大并发连接数目。缺省值为10。 
* local_destination_concurrency_limit:控制对同一本地收件人的最大同时投递的邮件数目。缺省值为2,因为对本地同一收件人投递邮件时投递工作只能一个接一个的进行,所以设得在大也没用。 

3. 对同一封邮件的收件人数目限制 
通过default_destination_recipient_limit参数来控制postfix的投递代理(如 
smtp进程)可以将同一封邮件发送给多少个收件人。缺省值为50。也可以用明确指出该投递代理的参数来覆盖该缺省值。如用smtpd_recipient_limit来指定smtp投递代理可以将同一封邮件发送给多少个收件人,该参数的缺省值为1000。 
     
4. 推迟投递控制 
通过defer_transports参数,我们可以推迟投递该参数指定的邮件直到postfix明确的提出投递要求。下面我们看一个例子: 
有一个小型的局域网,用户都将邮件发送给局域网内部的一台postfix 
邮件服务器,然后通过在该服务器上拨号将邮件发送出去。这时我们可以这样指定该参数的值: 
   defer_transports = smtp 
该语句表示postfix推迟投递所有的邮件直到执行sendmail -q命令,这样 
我们就可以在ppp的脚本中加上sendmail -q,以便在拨号成功后让postfix开始投递邮件。 

邮件系统postfix安装与配置(二) 

http://LinuxAid.com.cn 01-01-12 11:26 318p bye2000 
-------------------------------------------------------------------------------- 
  

5. 关于延迟邮件的再投递控制 
可以通过以下的几个参数实现对延迟邮件的再投递控制: 
queue_run_delay:设置队列管理进行扫描deferred邮件队列的频率,缺省值为1000秒。 
maximal_queue_lifetime:设置postfix在放弃投递而返回不可投递信息前,被延迟邮件再deferred邮件队列中的生存时间。 
minimal_backoff_time:当一封邮件投递失败后,邮件队列将在一段时间内忽视该邮件的存在,也就是我们前面讲的时间邮票。该参数就是用来设置最小的时间邮票。缺省值为1000秒。 
maximal_backoff_time:设置最大的时间邮票。 

6. 对拒绝服务攻击的处理 
postfix对每一个SMTP会话都设置一个错误计数器,当该客户端的请求未 
被接受或违反那UCE规则时,该计数器就增1。随着计数器的增加,postfix将采取不同的措施来防止恶意用户的拒绝服务攻击。 
smtpd_error_sleep_time:当该错误计数器的值还很小时,postfix将暂停 
smtpd_error_sleep_time指定的时间,然后向客户端报告一个错误。该参数的缺省值为5秒。 
    smtpd_soft_error_limit:当错误计数器的值超过该参数指定的值时,postfix在响应该客户端请求前将沉睡一段时间。缺省值为10。 
smtpd_hard_error_limit:当错误计数器的值超过该参数指定的值时,postfix 
中断同该客户端的连接。缺省值为100。 

4.5 postfix对使用资源的控制 
通过特定的postfix配置参数,我们可以实现postfix运行时对所消耗的资源的灵活控制。可以通过以下几个方面来控制postfix消耗的资源: 

1. 限制内存中的对象的大小 
要控制对内存资源的消耗,必须控制内存中对象的大小。可以用以下的参数来进行对象大小的控制: 
line_length_limit:控制读入数据时每一行的大小,如果太长则强行将其分割成更短的行,太长的行在投递时再重组。缺省值为2048 bytes。 
header_size_limit:限制信头长度。缺省值为102400bytes。 
message_size_limit:限制postfix队列文件的大小。缺省值为10240000 bytes。 
queue_minfree:邮件队列中可用的空间大小。缺省为无限制。建议该值最好时message_size_limit的数倍以便于处理大邮件。 
bounce_size_limit:限制某一邮件不可投递时,返回给发件人不可投递报告的大小,缺省值为50000 bytes。 

2. 限制内存中对象的数目 
qmgr_message_recipient_limit:设置内存中收件人地址的最大数目。缺省值为10000。 
qmgr_message_active_limit:设置active邮件队列中邮件数目的最大值。缺省值为1000。 
duplicate_filter_limit:设置需要local和cleanup后台程序记住的收件人地址的最大数目。缺省值为1000。 

    3.限制等待一个外部命令完成的时间 
command_time_limit:设置local程序等待一个外部命令完成的时间。缺省值为1000秒。 

4. 限制文件锁定的操作时间 
   deliver_lock_attempts:设置锁定一个文件的最大尝试次数。缺省值为5次。 
   deliver_lock_delay:设置如果锁定一个文件失败后再次尝试的等待时间,缺省值为1秒。 
    
5. 控制错误恢复 
 在某些情况下(如高负载),postfix的某个进程可能会死掉,这时master进 
程会试图重新启动该进程,我们可以通过下面的参数来控制这种行为: 
fork_attempts:试图重启动一个进程的最大尝试次数。缺省值为5次。 
fork_delay:每两次尝试之间的等待时间,缺省值为1秒。 
transport_retry_time:队列管理进程每两次尝试连接一个不正常的投递代理进程之间的等待时间。缺省为60秒。 

4.6 postfix中的地址操作 

1. 将地址改写为标准格式 
在cleanup进程进行表查询之前,它首选请求trivial-rewrite进程将新 
邮件地址改写成标准的user@fully.qualified.domain格式。改写的目的是为了减少查询表中的条目,从而提供查询的效率。trivial-rewrite进程可以实现以下的地址改写: 
* 将包含源路由信息的地址如@hosta,@hostb:user@site写成user@site,因为postfix不支持包含源路由信息的地址格式。 
* 将形如user%domain的地址改写成user@domain的形式。该特性可通过allow_percent_hack参数进行控制,其缺省值为yes。 
* 将只包含user的地址改写成user@$myorigin。该特性可通过append_at_myorigin参数进行控制,其缺省值为yes。最好不要改动其缺省值,因为大多数的postfix进程更擅长处理地址形如user@$myorigin的邮件。 
* 将形如user@host的地址改写成user@host.$mydomain。该特性可通过append_dot_mydomain参数进行控制,其缺省值为yes。 
* 将形如user@site.的地址改写成 user@site,也即除去了最后的点号。 

2. 规范地址映射 
在cleanup进程将一封新邮件存入incoming邮件队列之前,cleanup进程 
将根据查询规范表来进行地址改写,从而使之更具可读性。主要是替换形如Firstname.Lastname 风格的地址以及清除无效的域。缺省postfix是不进行规范地址改写的,你可以通过指定canonical_maps参数的值来使其生效。如: 
     canonical_maps = hash:/etc/postfix/canonical  
     也可以分别为收件人和发件人地址分别指定不同的改写规范,这时参数sender_canonical_maps和recipient_canonical_maps的优先级比canonical_maps高。如: 
     sender_canonical_maps = hash:/etc/postfix/sender_canonical  
     recipient_canonical_maps = hash:/etc/postfix/recipient_canonical 

3. 地址欺骗 
就是将形如user@host.domain的地址改写成user@domain或user@other.domain,  
好像是从其他的邮件服务器发出的一样。缺省该功能是被禁止的,可以用参数masquerade_domains使其生效,如: 
    masquerade_domains = $mydomain 
    也可以通过masquerade_exceptions参数对特定的用户不进行地址欺骗,如: 
masquerade_exceptions = root 
注意:地址欺骗只对发件人地址有作用。 

4. 虚拟地址映射 
在运用了规范地址映射和地址欺骗之后,cleanup将使用虚拟表映射将邮件 
重定向到所有的收件人。注意,该操作仅仅作用于信头上的收件人地址。这样,我们就可以将发送到虚拟域的邮件投递到真实用户的邮箱。系统的别名数据库同样可以起到相同的作用。缺省该功能是被禁止的,可以用参数virtual_maps使其生效,如: 
     virtual_maps = hash:/etc/postfix/virtual 
      
5. Relocated数据库查询 
    Relocated表格提供如何将邮件发送给在系统中没有帐号的用户。缺省该功能是被禁止的,可以用参数relocated_maps使其生效,如: 
    relocated_maps = hash:/etc/postfix/relocated 
     
6. 别名数据库查询 
当邮件在本地投递时,local投递代理进程会在别名数据库(linux下为 
/etc/aliases)中查询收件人的别名。该操作不会作用于邮件信头中的地址。可以用alias_maps指定使用的别名数据库。缺省地,该功能是有效的,如: 
    alias_maps = hash:/etc/aliases  
可以通过alias_database参数控制别名数据库的路径,如: 
alias_database = hash:/etc/aliases 

五、 postfix的配置实例 
 5.1为拨号用户配置postfix 
假设有一个小公司使用拨号上网,公司通过ISP的SMTP服务器(假设其域名为mail.isp.com,ip地址为201.110.1.100)发送邮件;公司内部的员工通过公司的SMTP服务器(运行postfix)收发邮件,下面我们一起来配置公司内部的SMTP服务器。 

1. 因为没有自己的固定ip和域名,所以必须指定ISP的SMTP服务器作 
为邮件网关(智能主机)。可以用如下的参数指定: 
   relayhost = [201.110.1.100] 
注意:relayhost的值可以是domain、host、host:port、[address]或[address]。 

2. 当接收到新邮件时,postfix就会尝试投递该邮件。如果将该公司内部的 
SMTP服务器设置为按需拨号,也就是一有程序请求外联就拨号,则会增加上网的费用。这时我们可以通过postfix的defer_transports参数推迟投递新邮件直到postfix提出明确的要求,如指定: 
    defer_transports = smtp 
这时如果我们在ppp的脚本(如/etc/ppp/ip-up.local)加上如下命令,则postfix只在拨号成功后投递新邮件: 
/usr/sbin/sendmail 

3. 因为我们是将邮件转发到邮件网关(201.110.1.100)而不是自己进行投 
递,所以我们没有必要使用DNS,因此我们通过如下的参数取消DNS查询: 
disable_dns_lookups = yes 

4. 为了保证我们能收到回信,我们必须进行域伪装。 
masquerade_domains = isp.com 

下面就是我们的配置文件mail.cf: 

#指定邮件网关 
relayhost =  [201.110.1.100] 
# 在拨号成功后才投递邮件 
defer_transports = smtp 
#取消DNS查询 
disable_dns_lookups = yes 
#一般常规配置 
queue_directory = /var/spool/postfix 
program_directory = /usr/libexec/postfix 
command_directory = /usr/sbin 
daemon_directory = /usr/libexec/postfix 
mail_owner = postfix 
default_privs = nobody 
mail_spool_directory = /var/spool/mail 
mailbox_command = /usr/bin/procmail 
local_destination_concurrency_limit = 2 
default_destination_concurrency_limit = 10 
debug_peer_level = 2 
debugger_command=PATH=/usr/bin:/usr/X11R6/bin,xxgdb$ 
                 daemon_directory/$ process_name $process_id & sleep 5 
    # 假设本地网络为192.168.1.1/24 
mynetworks = 192.168.1.1/24 
# host specific information 
myhostname = yourhost.isp.com 
mydomain = local.isp.com 
myorigin = $mydomain 
where do we receive mail and who do we accept/receive mail for? 
inet_interfaces = all 
mydestination = $myhostname, localhost.$mydomain, $mydomain 
default_transport = smtp 
masquerade_domains = isp.com 

需要注意的是:这里的$mydomain、$myorigin、$mydestination不能为isp.com, 因为你如果设定为isp.com,内部SMTP服务器就会认为你的邮件是转发给它的,故而在本地尝试投递邮件,结果只会返回“unknown user”的错误。其次,该配置只实现了将邮件通过SMTP发送到ISP的SMTP服务器的手段,缺乏从ISP的邮件服务器取信的方法,这一点就只能通过如outlook或foxmail等的mail客户端软件来实现了。 

5.2为中小型企业用户配置postfix 
 假设有一家数千名员工的公司,该公司通过租用专线上网。现在公司决定 
通过postfix来建立自己的邮件系统。在这里我们假设该公司的域为some.com, 邮件服务器的域名mail.some.com,地址为202.200.180.2,DNS服务器的域名为dns.some.com,地址为202.200.180.1。 
1. 配置DNS服务器,设置MX记录指向mail.some.com。相关的配置文件 
为/var/named/some.com(假设其zone文件就叫some.com, 有关DNS配置的内容请参看本书的相关章节)的内容如下: 

@       IN      SOA  dns.some.com. root.dns.some.com ( 
                       2000011307  ; serial 
                       28800       ; refresh, seconds 
                       14400              ; retry, seconds 
                       3600000              ; expire, seconds 
                       86400            ; minimum, seconds 
                        ) 

@             IN    NS      dns.some.com. 
@             IN    A       202.200.180.1 
@             IN    MX      10  mail.some.com.  

localhost     IN    A        127.0.0.1 
dns         IN     A        202.200.180.1 
mail         IN    A        202.200.180.2 
host1        IN     A       202.200.180.3 
host2        IN     A       202.200.180.4 

2. 配置postfix,其配置文件及相关的解释如下: 

#设置一般的路径信息 
queue_directory = /var/spool/postfix 
command_directory = /usr/sbin 
daemon_directory = /usr/libexec/postfix 
mail_spool_directory = /var/spool/mail 

#设置邮件及邮件队列的所有者为postfix 
mail_owner = postfix 

#设置邮件服务器的主机名 
myhostname = mail.some.com 

#设置mydomain、myorigin和mydomain参数 
mydomain = some.com 
myorigin =$mydomain  
mydestination = $mydomain 

#设置postfix服务监听的端口 
inet_interfaces = all 

#设置本地收件人的用户名查询手段,缺省是查询/etc/passwd文件 
#和别名数据库 
local_recipient_maps = $alias_maps unix:passwd.byname 
alias_maps = hash:/etc/aliases 
alias_database = hash:/etc/aliases 

#设置最终的本地投递代理程序,在这里我们使用流行的procmail 
mailbox_command = /usr/bin/procmail 

#设置该值为$mydomain以便客户端的连接 
relay_domains = $mydomain 
mynetworks = 202.200.180.0/24 

#设置向用户显示的主机名和版本信息 
smtpd_banner = $myhostname ESMTP $mail_name 

#对于并发进程的限制,保持系统缺省值就可以满足要求了。 
local_destination_concurrency_limit = 2 
default_destination_concurrency_limit = 10 

#如果你不知道你在做什么,最好不要改变下面的设置 
debug_peer_level = 2 
debugger_command = 
         PATH=/usr/bin:/usr/X11R6/bin 
         xxgdb $daemon_directory/$process_name $process_id & sleep 5 

3. 在RedHat中我们通常使用imap作为pop3服务器,可以通过rpm -q imap 
命令查看系统有没有安装imap。如果没有安装则插入linux光盘,用rpm -ivh imap-4.5-4.rpm 进行安装。 

4. 缺省地,pop3服务器是由inet 启动的,所以必须去掉/etc/inetd.conf文 
件中有关pop3的一行注释。如下所示: 
  pop-3  stream  tcp    nowait  root   /usr/sbin/ipop3d    ipop3d 
  
5. 重新启动inet服务器,启动postfix: 
#/etc/rc.d/init.d/inet restart 
#postfix start 

5.3在防火墙内部配置postfix 
    假设一公司通过租用专线上网,公司内部使用192.168.0.0的私有ip, 然后通过防火墙(双宿主主机)的ip欺骗上网,公司的邮件服务器(mail.some.com)也在内部网中,也使用私有ip。我们假设在防火墙上进行了端口转发,可以将Internet对防火墙25端口的请求包转发到内部的邮件服务器上,并且运行DNS服务的防火墙的MX记录指向防火墙本身。 
在这个例子中,我想着重说明的是有关映射文件的用法。main.cf配置文件和相关的解释如下所示: 


    #表明自己的身份 
myhostname = mail.some.com 
mydomain = some.com 
mydestination = $mydomain, $myhostname, localhost.$mydomain 
myorigin = $mydomain 
#让postfix监听所有接口 
inet_interfaces = all 

#通过mynetworks参数接受内部网用户的SMTP连接请求 
mynetworks = 192.168.0.0/8   

#向postfix管理员报告的信息量 
notify_classes = resource, software, bounce, policy 

#如果客户端的ip地址符合$maps_rbl_domains参数中列出的则拒绝之 
maps_rbl_domains = rbl.maps.vix.com, dul.maps.vix.com 

#对可以连接的客户端进行严格的限制 
smtpd_client_restrictions =  
             #客户端ip符合$mynetworks定义的范围则接受连接 
permit_mynetworks,  
        #根据access的查询结果判断客户端连接的合法性 
              check_client_access hash:/etc/postfix/access,  
     #拒绝ip符合$maps_rbl_domains定义范围的连接 
               reject_maps_rbl,  
    #如果客户端在DNS中没有记录则拒绝连接,要慎用 
               reject_unknown_hostname 

#通过发件人的地址进行限制 
smtpd_sender_restrictions =  
    permit_mynetworks,  
 check_sender_access hash:/etc/postfix/access 
     
#设置虚拟主机数据库,别忘了执行"postmap virtual"进行格式转换 
virtual_maps = hash:/etc/postfix/virtual  
     
#对无系统帐号的邮件进行转发设置,如离开公司的员工 
relocated_maps = hash:/etc/postfix/relocated 
    
#设置别名数据库 
alias_maps = hash:/etc/postfix/aliases 
     
# 我们使用smtp投递代理 
default_transport = smtp 
     
# 一些常规设置 
mail_owner = postfix 
default_privs = nobody 
  
#设置路径信息 
queue_directory = /var/spool/postfix  
program_directory = /usr/libexec/postfix  
command_directory = /usr/sbin  
daemon_directory = /usr/libexec/postfix  
mail_spool_directory = /var/spool/mail  
mailbox_command = /usr/bin/procmail 
    
#并发连接设置 
local_destination_concurrency_limit = 2  
default_destination_concurrency_limit = 10  
  
然后,我们执行以下命令: 

#进入postfix配置目录 
cd /etc/postfix 
#用newaliases初始化别名数据库 
newaliases 
#用postmap分别建立virtual、access和relocated查询数据库 
postmap virtual 
postmap access 
postmap relocated 
#启动postfix 
/etc/rc.d/init.d/postfix start 

现在我们来看看virtual、access和reloacted几个查询文件的格式,下面是这几个文件的示例和注释: 

    #virtual文件示例 
#假设在这个例子中我们有个虚拟域为other.com 
other.com 
      
    #access 文件示例 
#如果符合前面的条件则进行后面操作,可以有三种操作: 
#1. [45]XX $messag:拒绝接受并且向客户端显示预定义的信息 
#2. REJECT:拒绝接受,不显示信息 
#3. OK允许连接 
    ispy99@noman.com.cn     550 Go away 
    friend.com     OK 
    202.192     REJECT 
     
    #relocated 文件示例 
    #该文件主要是将发给无系统帐号的邮件进行转发 
    who@some.com     onetwo@newone.com 
六、 postfix中的命令行工具及其它 
下面我们来看一看postfix的命令行工具,通过这些工具的使用可能会使你 
对postfix的管理更简单。 

6.1 sendmail兼容的命令行工具 

1. mailq  对邮件队列文件进行列表。表中的每一个条目包含有以下信息: 
队列文件ID、邮件的大小、到达的时间、发件人、收件人和投递延迟的原因(如果投递有延迟的话)。该命令主要是与showq后台程序通信来获取队列文件的相关信息。该命令无参数。 

2. newaliases 该工具进行别名数据库的初始化。如果没有指定数据库的类 
型,则使用系统默认的数据库类型(在linux下为hash)。该命令可以不带参数执行。 

6.2 postfix自带的命令行工具 

1. postcat  打印邮件队列文件的内容。后面界要显示的队列文件名,可以 
带一个-v的参数进行冗余显示。 

2. postconf  打印配置参数设置后的值或postfix的其他信息。 
             -d 打印配置参数的缺省值。 
             -m 列出所有支持的查询表类型。 
                不带参数则打印配置参数设置后的值。 

3. postmap  建立postfix查询数据库。在linux下可以直接跟上原始文件 
而不带任何参数来建立该数据库。 

6.3 postfix的日志 
     postfix的日志文件位于/etc/log/maillog, 文件中包含有postfix的启动信息、出错信息以及同其他SMTP服务器的会话等等。如下所示: 

Sep 10 05:54:17 mail postfix/smtpd[5072]: disconnect from unknown[204.140.244.150] 
Sep 10 06:06:00 mail postfix/qmgr[467]: 50D403DF8: from=;, size=6591 (queue active) 
Sep 10 06:11:06 mail postfix/smtp[5085]: connect to bjmx2.163.net[202.108.255.241]: read timeout (port 25)
Sep 10 06:16:07 mail postfix/smtp[5085]: connect to bjmx3.163.net[202.108.255.242]: read timeout (port 25)
Sep 10 06:21:08 mail postfix/smtp[5085]: connect to bjmx1.163.net[202.108.255.240]: read timeout (port 25)

6.4 在postfix中使用MySQL数据库 
     Scott Cotton 和 Joshua Marcus写了一段可以在postfix中添加mysql映射类型的代码,从而我们可以将postfix查询的别名数据库等数据存储在mysql数据库中,让postfix进行mysql查询来得到结果。这样做将有助于提供postfix的运行效率, 有其对需要不断对映射数据的站点特别有用。 

   1.为postfix添加识别mysql数据库映射的功能 
a. 由于这段代码使用了mysql客户端库,所以我们必须安装mysql的开包。 
可以到www.redhat.com等linux相关站点下载mysql开发包,也可以从某些linux的资源光盘中取得mysql的开发包,如MySQL-client-3.22.30-1.i386.rpm。 
   b. 安装该开发包: 
   rpm -ivh MySQL-client-3.22.30-1.i386.rpm 
c. 下载postfix的源代码包,根据本章“3.1源代码包的安装”的提示进行 
安装,但是注意在执行make命令之前先执行以下命令: 
make -f Makefile.init makefiles 'CCARGS=-DHAS_MYSQL -I /usr/include/mysql'  
                          'AUXLIBS=/usr/lib/mysql/libmysqlclient.a -lm' 
    
5. 配置postfix使用mysql数据库映射 
我们以alias_maps进行说明。在main.cf中指定: 
   alias_maps = mysql:/etc/postfix/mysql-aliases.cf 

6. 编辑mysql-aliases.cf 

#首先指定登录到mysql服务器的用户名和密码 
user = your_user_name 
password = your_password 

#连接的数据库名称 
dbname = your_database_name 

#查询的表名 
table = mytable 

#添加表的字段名称 
#forward_addr为转发地址 
#alias为别名数据 
select_field=forward_addr 
where_field=alias 
#添加附加的查询条件 
additional_conditions=and status='paid' 

#指定要连接的MySQL服务器 
hosts=your.mysql.server 

   这样,当发生一个查询的时候,postfix是以这样的SQL语句进行查询的: 
   select forward_addr from mytable where alias='$lookup' and status='paid' 

   以上只是一个例子。在实际的运用中您可以指定多个mysql数据库,使用多个数据表格。