进程和线程编程 |
目 录
1. 进程和线程编程
1. 原始管道
1. pipe()
2. dup()
3. dup2()
4. popen()和pclose()
2. 命名管道
1. 创建FIFO
2. 操作FIFO
3. 阻塞FIFO
3. 消息队列
1. msgget()
2. msgsnd()
3. msgrcv()
4. msgctl()
4. 信号量
1. semget()
2. semop()
3. semctl()
5. 共享内存
1. shmget()
2. shmat()
3. shmctl()
4. shmdt()
6. 线程
1. 线程同步
2. 使用信号量协调程序
3. 代码例子
1. newthread
2. exitthead
3. getchannel
4. def
5. release
6. redezvous
7. unbouded
进程和线程编程
看一下UNIX系统中的进程和Mach的任务和线程之间的关系。在UNIX系统中,一个进程包括一个可执行的程序和一系列的资源,例如文件描述符表和地址空间。在Mach中,一个任务仅包括一系列的资源;线程处理所有的可执行代码。一个Mach的任务可以有任意数目的线程和它相关,同时每个线程必须和某个任务相关。和某一个给定的任务相关的所有线程都共享任务的资源。这样,一个线程就是一个程序计数器、一个堆栈和一系列的寄存器。所有需要使用的数据结构都属于任务。一个UNIX系统中的进程在Mach中对应于一个任务和一个单独的线程。
[目录]
原始管道
使用C语言创建管道要比在shell下使用管道复杂一些。如果要使用C语言创建一个简单的管道,可以使用系统调用pipe()。它接受一个参数,也就是一个包括两个整数的数组。如果系统调用成功,此数组将包括管道使用的两个文件描述符。创建一个管道之后,一般情况下进程将产生一个新的进程。
可以通过打开两个管道来创建一个双向的管道。但需要在子进程中正确地设置文件描述必须在系统调用fork()中调用pipe(),否则子进程将不会继承文件描述符。当使用半双工管道时,任何关联的进程都必须共享一个相关的祖先进程。因为管道存在于系统内核之中,所以任何不在创建管道的进程的祖先进程之中的进程都将无法寻址它。而在命名管道中却不是这样。
[目录]
pipe()
系统调用:pipe();
原型:intpipe(intfd[2]);
返回值:如果系统调用成功,返回0
如果系统调用失败返回-1:errno=EMFILE(没有空闲的文件描述符)
EMFILE(系统文件表已满)
EFAULT(fd数组无效)
注意fd[0]用于读取管道,fd[1]用于写入管道。
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
main()
{
intfd[2];
pipe(fd);
..
}
一旦创建了管道,我们就可以创建一个新的子进程:
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
main()
{
intfd[2];
pid_t childpid;
pipe(fd);
if((childpid=fork())==-1)
{
perror("fork");
exit(1);
}..
}
如果父进程希望从子进程中读取数据,那么它应该关闭fd1,同时子进程关闭fd0。反之,如果父进程希望向子进程中发送数据,那么它应该关闭fd0,同时子进程关闭fd1。因为文件描述符是在父进程和子进程之间共享,所以我们要及时地关闭不需要的管道的那一端。单从技术的角度来说,如果管道的一端没有正确地关闭的话,你将无法得到一个EOF。
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
main()
{
intfd[2];
pid_t childpid;
pipe(fd);
if((childpid=fork())==-1)
{
perror("fork");
exit(1);
}
if(childpid==0)
{
/*Child process closes up in put side of pipe*/
close(fd[0]);
}
else
{
/*Parent process closes up out put side of pipe*/
close(fd[1]);
}..
}
正如前面提到的,一但创建了管道之后,管道所使用的文件描述符就和正常文件的文件描述符一样了。
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
intmain(void)
{
intfd[2],nbytes;
pid_tchildpid;
charstring[]="Hello,world!/n";
charreadbuffer[80];
pipe(fd);
if((childpid=fork())==-1)
{
perror("fork");
exit(1);
}
if(childpid==0)
{
/*Child process closes up in put side of pipe*/
close(fd[0]);
/*Send"string"through the out put side of pipe*/
write(fd[1],string,strlen(string));
exit(0);
}
else
{
/*Parent process closes up out put side of pipe*/
close(fd[1]);
/*Readinastringfromthepipe*/
nbytes=read(fd[0],readbuffer,sizeof(readbuffer));
printf("Receivedstring:%s",readbuffer);
}
return(0);
}
一般情况下,子进程中的文件描述符将会复制到标准的输入和输出中。这样子进程可以使用exec()执行另一个程序,此程序继承了标准的数据流。
[目录]
dup()
系统调用:dup();
原型:intdup(intoldfd);
返回:如果系统调用成功,返回新的文件描述符
如果系统调用失败,返回-1:errno=EBADF(oldfd不是有效的文件描述符)
EBADF(newfd超出范围)
EMFILE(进程的文件描述符太多)
注意旧文件描述符oldfd没有关闭。虽然旧文件描述符和新创建的文件描述符可以交换使用,但一般情况下需要首先关闭一个。系统调用dup()使用的是号码最小的空闲的文件描述符。
再看下面的程序:
..
childpid=fork();
if(childpid==0)
{
/*Close up standard input of the child*/
close(0);
/*Dup licate the input side of pipe to stdin*/
dup(fd[0]);
execlp("sort","sort",NULL);
.
}
因为文件描述符0(stdin)被关闭,所以dup()把管道的输入描述符复制到它的标准输入中。这样我们可以调用execlp(),使用sort程序覆盖子进程的正文段。因为新创建的程序从它的父进程中继承了标准输入/输出流,所以它实际上继承了管道的输入端作为它的标准输入端。现在,最初的父进程送往管道的任何数据都将会直接送往sort函数。
[目录]
dup2()
系统调用:dup2();
原型:intdup2(intoldfd,intnewfd);
返回值:如果调用成功,返回新的文件描述符
如果调用失败,返回-1:errno=EBADF(oldfd不是有效的文件描述符)
EBADF(newfd超出范围)
EMFILE(进程的文件描述符太多)
注意dup2()将关闭旧文件描述符。
使用此系统调用,可以将close操作和文件描述符复制操作集成到一个系统调用中。另外,此系统调用保证了操作的自动进行,也就是说操作不能被其他的信号中断。这个操作将会在返回系统内核之前完成。如果使用前一个系统调用dup(),程序员不得不在此之前执行一个close()操作。请看下面的程序:
..
childpid=fork();
if(childpid==0)
{
/*Close stdin,dup licate the input side of pipe to stdin*/
dup2(0,fd[0]);
execlp("sort","sort",NULL);
..
}
[目录]
popen()和pclose()
如果你认为上面创建和使用管道的方法过于繁琐的话,你也可以使用下面的简单的方法:
库函数:popen()和pclose();
原型:FILE*popen(char*command,char*type);
返回值:如果成功,返回一个新的文件流。
如果无法创建进程或者管道,返回NULL。
此标准的库函数通过在系统内部调用pipe()来创建一个半双工的管道,然后它创建一个子进程,启动shell,最后在shell上执行command参数中的命令。管道中数据流的方向是由第二个参数type控制的。此参数可以是r或者w,分别代表读或写。但不能同时为读和写。在Linux系统下,管道将会以参数type中第一个字符代表的方式打开。所以,如果你在参数type中写入rw,管道将会以读的方式打开。
虽然此库函数的用法很简单,但也有一些不利的地方。例如它失去了使用系统调用pipe()时可以有的对系统的控制。尽管这样,因为可以直接地使用shell命令,所以shell中的一些通配符和其他的一些扩展符号都可以在command参数中使用。
使用popen()创建的管道必须使用pclose()关闭。其实,popen/pclose和标准文件输入/输出流中的fopen()/fclose()十分相似。
库函数:pclose();
原型:intpclose(FILE*stream);
返回值:返回系统调用wait4()的状态。
如果stream无效,或者系统调用wait4()失败,则返回-1。
注意此库函数等待管道进程运行结束,然后关闭文件流。库函数pclose()在使用popen()创建的进程上执行wait4()函数。当它返回时,它将破坏管道和文件系统。
在下面的例子中,用sort命令打开了一个管道,然后对一个字符数组排序:
#include<stdio.h>
#defineMAXSTRS5
intmain(void)
{
intcntr;
FILE*pipe_fp;
char*strings[MAXSTRS]={"echo","bravo","alpha",
"charlie","delta"};
/*Createonewaypipelinewithcalltopopen()*/
if((pipe_fp=popen("sort","w"))==NULL)
{
perror("popen");
exit(1);
}
/*Processingloop*/
for(cntr=0;cntr<MAXSTRS;cntr++){
fputs(strings[cntr],pipe_fp);
fputc('/n',pipe_fp);
}
/*Closethepipe*/
pclose(pipe_fp);
return(0);
}
因为popen()使用shell执行命令,所以所有的shell扩展符和通配符都可以使用。此外,它还可以和popen()一起使用重定向和输出管道函数。再看下面的例子:
popen("ls~scottb","r");
popen("sort>/tmp/foo","w");
popen("sort|uniq|more","w");
下面的程序是另一个使用popen()的例子,它打开两个管道(一个用于ls命令,另一个用于
sort命令):
#include<stdio.h>
intmain(void)
{
FILE*pipein_fp,*pipeout_fp;
charreadbuf[80];
/*Createonewaypipelinewithcalltopopen()*/
if((pipein_fp=popen("ls","r"))==NULL)
{
perror("popen");
exit(1);
}
/*Createonewaypipelinewithcalltopopen()*/
if((pipeout_fp=popen("sort","w"))==NULL)
{
perror("popen");
exit(1);
}
/*Processingloop*/
while(fgets(readbuf,80,pipein_fp))
fputs(readbuf,pipeout_fp);
/*Closethepipes*/
pclose(pipein_fp);
pclose(pipeout_fp);
return(0);
}
最后,我们再看一个使用popen()的例子。此程序用于创建一个命令和文件之间的管道:
#include<stdio.h>
intmain(intargc,char*argv[])
{
FILE*pipe_fp,*infile;
charreadbuf[80];
if(argc!=3){
fprintf(stderr,"USAGE:popen3[command][filename]/n");
exit(1);
}
/*Open up input file*/
if((infile=fopen(argv[2],"rt"))==NULL)
{
perror("fopen");
exit(1);
}
/*Create one way pipe line with call topopen()*/
if((pipe_fp=popen(argv[1],"w"))==NULL)
{
perror("popen");
exit(1);
}
/*Processingloop*/
do{
fgets(readbuf,80,infile);
if(feof(infile))break;
fputs(readbuf,pipe_fp);
}while(!feof(infile));
fclose(infile);
pclose(pipe_fp);
return(0);
}
下面是使用此程序的例子:
popen3sortpopen3.c
popen3catpopen3.c
popen3morepopen3.c
popen3catpopen3.c|grepmain
[目录]
命名管道
命名管道和一般的管道基本相同,但也有一些显著的不同:
*命名管道是在文件系统中作为一个特殊的设备文件而存在的。
*不同祖先的进程之间可以通过管道共享数据。
*当共享管道的进程执行完所有的I/O操作以后,命名管道将继续保存在文件系统中以便以后使用。
一个管道必须既有读取进程,也要有写入进程。如果一个进程试图写入到一个没有读取进程的管道中,那么系统内核将会产生SIGPIPE信号。当两个以上的进程同时使用管道时,这一点尤其重要。
[目录]
创建FIFO
可以有几种方法创建一个命名管道。头两种方法可以使用shell。
mknodMYFIFOp
mkfifoa=rwMYFIFO
上面的两个命名执行同样的操作,但其中有一点不同。命令mkfifo提供一个在创建之后直接改变FIFO文件存取权限的途径,而命令mknod需要调用命令chmod。
一个物理文件系统可以通过p指示器十分容易地分辨出一个FIFO文件。
$ls-lMYFIFO
prw-r--r--1rootroot0Dec1422:15MYFIFO|
请注意在文件名后面的管道符号“|”。
我们可以使用系统调用mknod()来创建一个FIFO管道:
库函数:mknod();
原型:intmknod(char*pathname,mode_tmode,dev_tdev);
返回值:如果成功,返回0
如果失败,返回-1:errno=EFAULT(无效路径名)
EACCES(无存取权限)
ENAMETOOLONG(路径名太长)
ENOENT(无效路径名)
ENOTDIR(无效路径名)
下面看一个使用C语言创建FIFO管道的例子:
mknod("/tmp/MYFIFO",S_IFIFO|0666,0);
在这个例子中,文件/tmp/MYFIFO是要创建的FIFO文件。它的存取权限是0666。存取权限
也可以使用umask修改:
final_umask=requested_permissions&~original_umask
一个常用的使用系统调用umask()的方法就是临时地清除umask的值:
umask(0);
mknod("/tmp/MYFIFO",S_IFIFO|0666,0);
另外,mknod()中的第三个参数只有在创建一个设备文件时才能用到。它包括设备文件的
主设备号和从设备号。
}
}
[目录]
操作FIFO
FIFO上的I/O操作和正常管道上的I/O操作基本一样,只有一个主要的不同。系统调用open用来在物理上打开一个管道。在半双工的管道中,这是不必要的。因为管道在系统内核中,而不是在一个物理的文件系统中。在我们的例子中,我们将像使用一个文件流一样使用管道,也就是使用fopen()打开管道,使用fclose()关闭它。
请看下面的简单的服务程序进程:
#include<stdio.h>
#include<stdlib.h>
#include<sys/stat.h>
#include<unistd.h>
#include<linux/stat.h>
#defineFIFO_FILE"MYFIFO"
intmain(void)
{
FILE*fp;
charreadbuf[80];
/*CreatetheFIFOifitdoesnotexist*/
umask(0);
mknod(FIFO_FILE,S_IFIFO|0666,0);
while(1)
{
fp=fopen(FIFO_FILE,"r");
fgets(readbuf,80,fp);
printf("Receivedstring:%s/n",readbuf);
fclose(fp);
return(0);
因为FIFO管道缺省时有阻塞的函数,所以你可以在后台运行此程序:
$fifoserver&
再来看一下下面的简单的客户端程序:
#include<stdio.h>
#include<stdlib.h>
#defineFIFO_FILE"MYFIFO"
intmain(int argc,char* argv[])
{
FILE*fp;
if(argc!=2){
printf("USAGE:fifoclient[string]/n");
exit(1);
}
if((fp=fopen(FIFO_FILE,"w"))==NULL){
perror("fopen");
exit(1);
}
fputs(argv[1],fp);
fclose(fp);
return(0);
}
[目录]
阻塞FIFO
一般情况下,FIFO管道上将会有阻塞的情况发生。也就是说,如果一个FIFO管道打开供读取的话,它将一直阻塞,直到其他的进程打开管道写入信息。这种过程反过来也一样。如果你不需要阻塞函数的话,你可以在系统调用open()中设置O_NONBLOCK标志,这样可以取消缺省的阻塞函数。
[目录]
消息队列
在UNIX的SystemV版本,AT&T引进了三种新形式的IPC功能(消息队列、信号量、以及共享内存)。但BSD版本的UNIX使用套接口作为主要的IPC形式。Linux系统同时支持这两个版本。
[目录]
msgget()
系统调用msgget()
如果希望创建一个新的消息队列,或者希望存取一个已经存在的消息队列,你可以使用系统调用msgget()。
系统调用:msgget();
原型:intmsgget(key_t key,int msgflg);
返回值:如果成功,返回消息队列标识符
如果失败,则返回-1:errno=EACCESS(权限不允许)
EEXIST(队列已经存在,无法创建)
EIDRM(队列标志为删除)
ENOENT(队列不存在)
ENOMEM(创建队列时内存不够)
ENOSPC(超出最大队列限制)
系统调用msgget()中的第一个参数是关键字值(通常是由ftok()返回的)。然后此关键字值将会和其他已经存在于系统内核中的关键字值比较。这时,打开和存取操作是和参数msgflg中的内容相关的。
IPC_CREAT如果内核中没有此队列,则创建它。
IPC_EXCL当和IPC_CREAT一起使用时,如果队列已经存在,则失败。
如果单独使用IPC_CREAT,则msgget()要么返回一个新创建的消息队列的标识符,要么返回具有相同关键字值的队列的标识符。如果IPC_EXCL和IPC_CREAT一起使用,则msgget()要么创建一个新的消息队列,要么如果队列已经存在则返回一个失败值-1。IPC_EXCL单独使用是没有用处的。
下面看一个打开和创建一个消息队列的例子:
intopen_queue(key_t keyval)
{
intqid;
if((qid=msgget(keyval,IPC_CREAT|0660))==-1)
{
return(-1);
}
return(qid);
}
[目录]
msgsnd()
系统调用msgsnd()
一旦我们得到了队列标识符,我们就可以在队列上执行我们希望的操作了。如果想要往队列中发送一条消息,你可以使用系统调用msgsnd():
系统调用:msgsnd();
原型:intmsgsnd(int msqid,struct msgbuf*msgp,int msgsz,int msgflg);
返回值:如果成功,0。
如果失败,-1:errno=EAGAIN(队列已满,并且使用了IPC_NOWAIT)
EACCES(没有写的权限)
EFAULT(msgp地址无效)
EIDRM(消息队列已经删除)
EINTR(当等待写操作时,收到一个信号)
EINVAL(无效的消息队列标识符,非正数的消息类型,或
者无效的消息长度)
ENOMEM(没有足够的内存复制消息缓冲区)
系统调用msgsnd()的第一个参数是消息队列标识符,它是由系统调用msgget返回的。第二个参数是msgp,是指向消息缓冲区的指针。参数msgsz中包含的是消息的字节大小,但不包括消息类型的长度(4个字节)。
参数msgflg可以设置为0(此时为忽略此参数),或者使用IPC_NOWAIT。
如果消息队列已满,那么此消息则不会写入到消息队列中,控制将返回到调用进程中。如果没有指明,调用进程将会挂起,直到消息可以写入到队列中。
下面是一个发送消息的程序:
intsend_message(int qid,struct mymsgbuf *qbuf)
{
intresult,length;
/*The length is essentially the size of the structure minus sizeof(mtype)*/
length=sizeof(structmymsgbuf)-sizeof(long);
if((result=msgsnd(qid,qbuf,length,0))==-1)
{
return(-1);
}
return(result);
}
这个小程序试图将存储在缓冲区qbuf中的消息发送到消息队列qid中。下面的程序是结合了上面两个程序的一个完整程序:
#include<stdio.h>
#include<stdlib.h>
#include<linux/ipc.h>
#include<linux/msg.h>
main()
{
intqid;
key_t msgkey;
struct mymsgbuf{
longmtype;/*Message type*/
intrequest;/*Work request number*/
doublesalary;/*Employee's salary*/
}msg;
/*Generateour IPC key value*/
msgkey=ftok(".",'m');
/*Open/createthequeue*/
if((qid=open_queue(msgkey))==-1){
perror("open_queue");
exit(1);
}
/*Load up the message with a r bitrary test data*/
msg.mtype=1;/*Messagetypemustbeapositivenumber!*/
msg.request=1;/*Dataelement#1*/
msg.salary=1000.00;/*Data element #2(my yearly salary!)*/
/*Bombsaway!*/
if((send_message(qid,&msg))==-1){
perror("send_message");
exit(1);
}
}
在创建和打开消息队列以后,我们将测试数据装入到消息缓冲区中。最后调用send_messag把消息发送到消息队列中。现在在消息队列中有了一条消息,我们可以使用ipcs命令来查看队列的状态。下面讨论如何从队列中获取消息。可以使用系统调用msgrcv():
[目录]
msgrcv()
系统调用:msgrcv();
原型:intmsgrcv(intmsqid,structmsgbuf*msgp,intmsgsz,longmtype,intmsgflg);
返回值:如果成功,则返回复制到消息缓冲区的字节数。
如果失败,则返回-1:errno=E2BIG(消息的长度大于msgsz,没有MSG_NOERROR)
EACCES(没有读的权限)
EFAULT(msgp指向的地址是无效的)
EIDRM(队列已经被删除)
EINTR(被信号中断)
EINVAL(msgqid无效,或者msgsz小于0)
ENOMSG(使用IPC_NOWAIT,同时队列中的消息无法满足要求)
很明显,第一个参数用来指定将要读取消息的队列。第二个参数代表要存储消息的消息缓冲区的地址。第三个参数是消息缓冲区的长度,不包括mtype的长度,它可以按照如下的方法计算:
msgsz=sizeof(structmymsgbuf)-sizeof(long);
第四个参数是要从消息队列中读取的消息的类型。如果此参数的值为0,那么队列中最长时间的一条消息将返回,而不论其类型是什么。
如果调用中使用了IPC_NOWAIT作为标志,那么当没有数据可以使用时,调用将把ENOMSG返回到调用进程中。否则,调用进程将会挂起,直到队列中的一条消息满足msgrcv()的参数要求。如果当客户端等待一条消息的时候队列为空,将会返回EIDRM。如果进程在等待消息的过程中捕捉到一个信号,则返回EINTR。
下面就是一个从队列中读取消息的程序:
intread_message(int qid,long type,struct mymsgbuf*qbuf)
{
intresult,length;
/*The length is essentially the size of the structure minus sizeof(mtype)*/
length=sizeof(structmymsgbuf)-sizeof(long);
if((result=msgrcv(qid,qbuf,length,type,0))==-1)
{
return(-1);
}
return(result);
}
在成功地读取了一条消息以后,队列中的这条消息的入口将被删除。
参数msgflg中的MSG_NOERROR位提供一种额外的用途。如果消息的实际长度大于msgsz,同时使用了MSG_NOERROR,那么消息将会被截断,只有与msgsz长度相等的消息返回。一般情况下,系统调用msgrcv()会返回-1,而这条消息将会继续保存在队列中。我们可以利用这个特点编制一个程序,利用这个程序可以查看消息队列的情况,看看符合我们条件的消息是否已经到来:
intpeek_message(int qid,long type)
{
intresult,length;
if((result=msgrcv(qid,NULL,0,type,IPC_NOWAIT))==-1)
{
if(errno==E2BIG)
return(TRUE);
}
return(FALSE);
}
在上面的程序中,我们忽略了缓冲区的地址和长度。这样,系统调用将会失败。尽管如此,我们可以检查返回的E2BIG值,它说明符合条件的消息确实存在。
[目录]
msgctl()
系统调用msgctl()
下面我们继续讨论如何使用一个给定的消息队列的内部数据结构。我们可以使用系统调用msgctl ( )来控制对消息队列的操作。
系统调用: msgctl( ) ;
调用原型: int msgctl ( int msgqid, int cmd, struct msqid_ds *buf );
返回值: 0 ,如果成功。
- 1,如果失败:errno = EACCES (没有读的权限同时cmd 是IPC_STAT )
EFAULT (buf 指向的地址无效)
EIDRM (在读取中队列被删除)
EINVAL (msgqid无效, 或者msgsz 小于0 )
EPERM (IPC_SET或者IPC_RMID 命令被使用,但调用程序没有写的权限)
下面我们看一下可以使用的几个命令:
IPC_STAT
读取消息队列的数据结构msqid_ds,并将其存储在b u f指定的地址中。
IPC_SET
设置消息队列的数据结构msqid_ds中的ipc_perm元素的值。这个值取自buf参数。
IPC_RMID
从系统内核中移走消息队列。
我们在前面讨论过了消息队列的数据结构(msqid_ds)。系统内核中为系统中的每一个消息队列保存一个此数据结构的实例。通过使用IPC_STAT命令,我们可以得到一个此数据结构的副本。下面的程序就是实现此函数的过程:
int get_queue_ds( int qid, struct msgqid_ds *qbuf )
{
if( msgctl( qid, IPC_STAT, qbuf) == -1)
{
return(-1);
}
return(0);
}
如果不能复制内部缓冲区,调用进程将返回-1。如果调用成功,则返回0。缓冲区中应该包括消息队列中的数据结构。
消息队列中的数据结构中唯一可以改动的元素就是ipc_perm。它包括队列的存取权限和关于队列创建者和拥有者的信息。你可以改变用户的id、用户的组id以及消息队列的存取权限。
下面是一个修改队列存取模式的程序:
int change_queue_mode(int qid, char *mode )
{
struct msqid_ds tmpbuf;
/* Retrieve a current copy of the internal data structure */
get_queue_ds( qid, &tmpbuf);
/* Change the permissions using an old trick */
sscanf(mode, "%ho", &tmpbuf.msg_perm.mode);
/* Update the internal data structure */
if( msgctl( qid, IPC_SET, &tmpbuf) == -1)
{
return(-1);
}
return(
}
我们通过调用get_queue_ds来读取队列的内部数据结构。然后,我们调用sscanf( )修改数据结构msg_perm中的mode 成员的值。但直到调用msgctl()时,权限的改变才真正完成。在这里msgctl()使用的是IPC_SET命令。
最后,我们使用系统调用msgctl ( )中的IPC_RMID命令删除消息队列:
int remove_queue(int qid )
{
if( msgctl( qid, IPC_RMID, 0) == -1)
{
return(-1);
}
return(0);
}
};
[目录]
信号量
信号量是一个可以用来控制多个进程存取共享资源的计数器。它经常作为一种锁定机制来防止当一个进程正在存取共享资源时,另一个进程也存取同一资源。下面先简要地介绍一下信号量中涉及到的数据结构。
1.内核中的数据结构semid_ds
和消息队列一样,系统内核为内核地址空间中的每一个信号量集都保存了一个内部的数据结构。数据结构的原型是semid_ds。它是在linux/sem.h中做如下定义的:
/*One semid data structure for each set of semaphores in the system.*/
structsemid_ds{
structipc_permsem_perm;/*permissions..seeipc.h*/
time_tsem_otime;/*last semop time*/
time_tsem_ctime;/*last change time*/
structsem*sem_base;/*ptr to first semaphore in array*/
structwait_queue*eventn;
structwait_queue*eventz;
structsem_undo*undo;/*undo requestson this array*/
ushortsem_nsems;/*no. of semaphores in array*/
};
sem_perm是在linux/ipc.h定义的数据结构ipc_perm的一个实例。它保存有信号量集的存取权限的信息,以及信号量集创建者的有关信息。
sem_otime最后一次semop()操作的时间。
sem_ctime最后一次改动此数据结构的时间。
sem_base指向数组中第一个信号量的指针。
sem_undo数组中没有完成的请求的个数。
sem_nsems信号量集(数组)中的信号量的个数。
2.内核中的数据结构sem
在数据结构semid_ds中包含一个指向信号量数组的指针。此数组中的每一个元素都是一个
数据结构sem。它也是在linux/sem.h中定义的:
/*One semaphore structure for each semaphore in the system.*/
structsem{
shortsempid;/*pid of las toperation*/
ushortsemval;/*current value*/
ushortsemncnt;/*num procs awaiting increase in semval*/
ushortsemzcnt;/*num procs awaiting semval=0*/
};
sem_pid最后一个操作的PID(进程ID)。
sem_semval信号量的当前值。
sem_semncnt等待资源的进程数目。
sem_semzcnt等待资源完全空闲的进程数目。
[目录]
semget()
我们可以使用系统调用semget()创建一个新的信号量集,或者存取一个已经存在的信号量集:
系统调用:semget();
原型:intsemget(key_t key,int nsems,int semflg);
返回值:如果成功,则返回信号量集的IPC标识符。如果失败,则返回-1:errno=EACCESS(没有权限)
EEXIST(信号量集已经存在,无法创建)
EIDRM(信号量集已经删除)
ENOENT(信号量集不存在,同时没有使用IPC_CREAT)
ENOMEM(没有足够的内存创建新的信号量集)
ENOSPC(超出限制)
系统调用semget()的第一个参数是关键字值(一般是由系统调用ftok()返回的)。系统内核将此值和系统中存在的其他的信号量集的关键字值进行比较。打开和存取操作与参数semflg中的内容相关。IPC_CREAT如果信号量集在系统内核中不存在,则创建信号量集。IPC_EXCL当和IPC_CREAT一同使用时,如果信号量集已经存在,则调用失败。如果单独使用IPC_CREAT,则semget()要么返回新创建的信号量集的标识符,要么返回系统中已经存在的同样的关键字值的信号量的标识符。如果IPC_EXCL和IPC_CREAT一同使用,则要么返回新创建的信号量集的标识符,要么返回-1。IPC_EXCL单独使用没有意义。参数nsems指出了一个新的信号量集中应该创建的信号量的个数。信号量集中最多的信号量的个数是在linux/sem.h中定义的:
#defineSEMMSL32/*<=512maxnumofsemaphoresperid*/
下面是一个打开和创建信号量集的程序:
intopen_semaphore_set(key_t keyval,int numsems)
{
intsid;
if(!numsems)
return(-1);
if((sid=semget(mykey,numsems,IPC_CREAT|0660))==-1)
{
return(-1);
}
return(sid);
}
};
[目录]
semop()
系统调用:semop();
调用原型:int semop(int semid,struct sembuf*sops,unsign ednsops);
返回值:0,如果成功。-1,如果失败:errno=E2BIG(nsops大于最大的ops数目)
EACCESS(权限不够)
EAGAIN(使用了IPC_NOWAIT,但操作不能继续进行)
EFAULT(sops指向的地址无效)
EIDRM(信号量集已经删除)
EINTR(当睡眠时接收到其他信号)
EINVAL(信号量集不存在,或者semid无效)
ENOMEM(使用了SEM_UNDO,但无足够的内存创建所需的数据结构)
ERANGE(信号量值超出范围)
第一个参数是关键字值。第二个参数是指向将要操作的数组的指针。第三个参数是数组中的操作的个数。参数sops指向由sembuf组成的数组。此数组是在linux/sem.h中定义的:
/*semop systemcall takes an array of these*/
structsembuf{
ushortsem_num;/*semaphore index in array*/
shortsem_op;/*semaphore operation*/
shortsem_flg;/*operation flags*/
sem_num将要处理的信号量的个数。
sem_op要执行的操作。
sem_flg操作标志。
如果sem_op是负数,那么信号量将减去它的值。这和信号量控制的资源有关。如果没有使用IPC_NOWAIT,那么调用进程将进入睡眠状态,直到信号量控制的资源可以使用为止。如果sem_op是正数,则信号量加上它的值。这也就是进程释放信号量控制的资源。最后,如果sem_op是0,那么调用进程将调用sleep(),直到信号量的值为0。这在一个进程等待完全空闲的资源时使用。
[目录]
semctl()
系统调用:semctl();
原型:int semctl(int semid,int semnum,int cmd,union semunarg);
返回值:如果成功,则为一个正数。
如果失败,则为-1:errno=EACCESS(权限不够)
EFAULT(arg指向的地址无效)
EIDRM(信号量集已经删除)
EINVAL(信号量集不存在,或者semid无效)
EPERM(EUID没有cmd的权利)
ERANGE(信号量值超出范围)
系统调用semctl用来执行在信号量集上的控制操作。这和在消息队列中的系统调用msgctl是十分相似的。但这两个系统调用的参数略有不同。因为信号量一般是作为一个信号量集使用的,而不是一个单独的信号量。所以在信号量集的操作中,不但要知道IPC关键字值,也要知道信号量集中的具体的信号量。这两个系统调用都使用了参数cmd,它用来指出要操作的具体命令。两个系统调用中的最后一个参数也不一样。在系统调用msgctl中,最后一个参数是指向内核中使用的数据结构的指针。我们使用此数据结构来取得有关消息队列的一些信息,以及设置或者改变队列的存取权限和使用者。但在信号量中支持额外的可选的命令,这样就要求有一个更为复杂的数据结构。
系统调用semctl()的第一个参数是关键字值。第二个参数是信号量数目。
参数cmd中可以使用的命令如下:
·IPC_STAT读取一个信号量集的数据结构semid_ds,并将其存储在semun中的buf参数中。
·IPC_SET设置信号量集的数据结构semid_ds中的元素ipc_perm,其值取自semun中的buf参数。
·IPC_RMID将信号量集从内存中删除。
·GETALL用于读取信号量集中的所有信号量的值。
·GETNCNT返回正在等待资源的进程数目。
·GETPID返回最后一个执行semop操作的进程的PID。
·GETVAL返回信号量集中的一个单个的信号量的值。
·GETZCNT返回这在等待完全空闲的资源的进程数目。
·SETALL设置信号量集中的所有的信号量的值。
·SETVAL设置信号量集中的一个单独的信号量的值。
参数arg代表一个semun的实例。semun是在linux/sem.h中定义的:
/*arg for semctl systemcalls.*/
unionsemun{
intval;/*value for SETVAL*/
structsemid_ds*buf;/*buffer for IPC_STAT&IPC_SET*/
ushort*array;/*array for GETALL&SETALL*/
structseminfo*__buf;/*buffer for IPC_INFO*/
void*__pad;
val当执行SETVAL命令时使用。buf在IPC_STAT/IPC_SET命令中使用。代表了内核中使用的信号量的数据结构。array在使用GETALL/SETALL命令时使用的指针。
下面的程序返回信号量的值。当使用GETVAL命令时,调用中的最后一个参数被忽略:
intget_sem_val(intsid,intsemnum)
{
return(semctl(sid,semnum,GETVAL,0));
}
下面是一个实际应用的例子:
#defineMAX_PRINTERS5
printer_usage()
{
int x;
for(x=0;x<MAX_PRINTERS;x++)
printf("Printer%d:%d/n/r",x,get_sem_val(sid,x));
}
下面的程序可以用来初始化一个新的信号量值:
void init_semaphore(int sid,int semnum,int initval)
{
union semunsemopts;
semopts.val=initval;
semctl(sid,semnum,SETVAL,semopts);
}
注意系统调用semctl中的最后一个参数是一个联合类型的副本,而不是一个指向联合类型的指针。
[目录]
共享内存
共享内存就是由几个进程共享一段内存区域。这可以说是最快的IPC形式,因为它无须任何的中间操作(例如,管道、消息队列等)。它只是把内存段直接映射到调用进程的地址空间中。这样的内存段可以是由一个进程创建的,然后其他的进程可以读写此内存段。
每个系统的共享内存段在系统内核中也保持着一个内部的数据结构shmid_ds。此数据结构是在linux/shm.h中定义的,如下所示:
/* One shmid data structure for each shared memory segment in the system. */
struct shmid_ds {
struct ipc_perm shm_perm; /* operation perms */
int shm_segsz; /* size of segment (bytes) */
time_t shm_atime; /* last attach time */
time_t shm_dtime; /* last detach time */
time_t shm_ctime; /* last change time */
unsigned short shm_cpid; /* pid of creator */
unsigned short shm_lpid; /* pid of last operator */
short shm_nattch; /* no. of current attaches */
/* the following are private */
unsigned short shm_npages; /* size of segment (pages) */
unsigned long *shm_pages; /* array of ptrs to frames -> SHMMAX */
struct vm_area_struct *attaches; /* descriptors for attaches */
};
shm_perm 是数据结构ipc_perm的一个实例。这里保存的是内存段的存取权限,和其他的有关内存段创建者的信息。
shm_segsz 内存段的字节大小。
shm_atime 最后一个进程存取内存段的时间。
shm_dtime 最后一个进程离开内存段的时间。
shm_ctime 内存段最后改动的时间。
shm_cpid 内存段创建进程的P I D。
shm_lpid 最后一个使用内存段的进程的P I D。
shm_nattch 当前使用内存段的进程总数。
[目录]
shmget()
系统调用:shmget();
原型:int shmget(key_t key,int size,int shmflg);
返回值:如果成功,返回共享内存段标识符。如果失败,则返回-1:errno=EINVAL(无效的内存段大小)
EEXIST(内存段已经存在,无法创建)
EIDRM(内存段已经被删除)
ENOENT(内存段不存在)
EACCES(权限不够)
ENOMEM(没有足够的内存来创建内存段)
系统调用shmget()中的第一个参数是关键字值(它是用系统调用ftok()返回的)。其他的操作都要依据shmflg中的命令进行。
·IPC_CREAT如果系统内核中没有共享的内存段,则创建一个共享的内存段。
·IPC_EXCL当和IPC_CREAT一同使用时,如果共享内存段已经存在,则调用失败。
当IPC_CREAT单独使用时,系统调用shmget()要么返回一个新创建的共享内存段的标识符,要么返回一个已经存在的共享内存段的关键字值。如果IPC_EXCL和IPC_CREAT一同使用,则要么系统调用新创建一个共享的内存段,要么返回一个错误值-1。IPC_EXCL单独使用没有意义。
下面是一个定位和创建共享内存段的程序:
int open_segment(key_t keyval,int segsize)
{
int shmid;
if((shmid=shmget(keyval,segsize,IPC_CREAT|0660))==-1)
{
return(-1);
}
return(shmid);
}
一旦一个进程拥有了一个给定的内存段的有效IPC标识符,它的下一步就是将共享的内存段映射到自己的地址空间中。
[目录]
shmat()
系统调用: shmat();
原型:int shmat ( int shmid, char *shmaddr, int shmflg);
返回值:如果成功,则返回共享内存段连接到进程中的地址。如果失败,则返回- 1:errno = EINVAL (无效的IPC ID 值或者无效的地址)
ENOMEM (没有足够的内存)
EACCES (存取权限不够)
如果参数a d d r的值为0,那么系统内核则试图找出一个没有映射的内存区域。我们推荐使用这种方法。你可以指定一个地址,但这通常是为了加快对硬件设备的存取,或者解决和其他程序的冲突。
下面的程序中的调用参数是一个内存段的I P C标识符,返回内存段连接的地址:
char *attach_segment(int shmid)
{
return(shmat(shmid, 0, 0));
}
一旦内存段正确地连接到进程以后,进程中就有了一个指向该内存段的指针。这样,以后就可以使用指针来读取此内存段了。但一定要注意不能丢失该指针的初值。
[目录]
shmctl()
系统调用:shmctl ( ) ;
原型:int shmctl( int shmqid, int cmd, struct shmid_ds *buf );
返回值: 0 ,如果成功。
-1,如果失败:errno = EACCES (没有读的权限,同时命令是IPC_STAT)
EFAULT(buf指向的地址无效,同时命令是IPC_SET和IPC_STAT )
EIDRM (内存段被移走)
EINVAL (shmqid 无效)
EPERM (使用IPC_SET 或者IPC_RMID 命令,但调用进程没有写的权限)
IPC_STAT 读取一个内存段的数据结构shmid_ds,并将它存储在buf参数指向的地址中。
IPC_SET 设置内存段的数据结构shmid_ds中的元素ipc_perm的值。从参数buf中得到要设置的值。
IPC_RMID 标志内存段为移走。
命令IPC_RMID并不真正从系统内核中移走共享的内存段,而是把内存段标记为可移除。进程调用系统调用shmdt()脱离一个共享的内存段。
[目录]
shmdt()
系统调用:shmdt();
调用原型:int shmdt ( char *shmaddr );
返回值:如果失败,则返回- 1:errno = EINVAL (无效的连接地址)
当一个进程不在需要共享的内存段时,它将会把内存段从其地址空间中脱离。但这不等于将共享内存段从系统内核中移走。当进程脱离成功后,数据结构shmid_ds中元素shm_nattch将减1。当此数值减为0以后,系统内核将物理上把内存段从系统内核中移走。
[目录]
线程
线程通常叫做轻型的进程。虽然这个叫法有些简单化,但这有利于了解线程的概念。因为线程和进程比起来很小,所以相对来说,线程花费更少的CPU资源。进程往往需要它们自己的资源,但线程之间可以共享资源,所以线程更加节省内存。Mach的线程使得程序员可以编写并发运行的程序,而这些程序既可以运行在单处理器的机器上,也可以运行在多处理器的机器中。另外,在单处理器环境中,当应用程序执行容易引起阻塞和延迟的操作时,线程可以提高效率。
用子函数pthread_create创建一个新的线程。它有四个参数:一个用来保存线程的线程变量、一个线程属性、当线程执行时要调用的函数和一个此函数的参数。例如:
pthread_ta_thread ;
pthread_attr_ta_thread_attribute ;
void thread_function(void *argument);
char * some_argument;
pthread_create( &a_thread, a_thread_attribute, (void *)&thread_function,
(void *) &some_argument);
线程属性只指明了需要使用的最小的堆栈大小。在以后的程序中,线程的属性可以指定其他的值,但现在大部分的程序可以使用缺省值。不像UNIX系统中使用fork系统调用创建的进程,它们和它们的父进程使用同一个执行点,线程使用在pthread_create中的参数指明要开始执行的函数。
现在我们可以编制第一个程序了。我们编制一个多线程的应用程序,在标准输出中打印“Hello Wo r l d”。首先我们需要两个线程变量,一个新线程开始执行时可以调用的函数。我们还需要指明每一个线程应该打印的信息。一个做法是把要打印的字符串分开,给每一个线程一个字符串作为开始的参数。请看下面的代码:
void print_message_function( void *ptr );
main( )
{
pthread_t thread1, thread2;
char *message1 = "Hello";
char *message2 = "Wo r l d " ;
pthread_create( &thread1, pthread_attr_default,
(void*)&print_message_function, (void*) message1);
pthread_create(&thread2, pthread_attr_default,
(void*)&print_message_function, (void*) message2);
exit( 0 ) ;
}
void print_message_function( void *ptr )
{
char *message;
message = (char *) ptr;
printf("%s ", message);
}
程序通过调用pthread_create创建第一个线程,并将“Hello”作为它的启动参数。第二个线程的参数是“World”。当第一个线程开始执行时,它使用参数“Hello”执行函数print_message_function。它在标准输出中打印“Hello”,然后结束对函数的调用。线程当离开它的初始化函数时就将终止,所以第一个线程在打印完“Hello”后终止。当第二个线程执行时,它打印“World”然后终止。但这个程序有两个主要的缺陷。
首先也是最重要的是线程是同时执行的。这样就无法保证第一个线程先执行打印语句。所以你很可能在屏幕上看到“World Hello”,而不是“Hello World”。请注意对exit的调用是父线程在主程序中使用的。这样,如果父线程在两个子线程调用打印语句之前调用exit,那么将不会有任何的打印输出。这是因为exit函数将会退出进程,同时释放任务,所以结束了所有的线程。任何线程(不论是父线程或者子线程)调用exit 都会终止所有其他线程。如果希望线程分别终止,可以使用pthread_exit函数。
我们可以使用一个办法弥补此缺陷。我们可以在父线程中插入一个延迟程序,给子线程足够的时间完成打印的调用。同样,在调用第二个之前也插入一个延迟程序保证第一个线程在第二个线程执行之前完成任务。
void print_message_function( void *ptr );
main ( )
{
pthread_t thread1, thread2;
char *message1 = "Hello”;
char *message2 = "Wo r l d " ;
pthread_create( &thread1, pthread_attr_default,
(void *) &print_message_function, (void *) message1);
sleep (10) ;
pthread_create(&thread2, pthread_attr_default,
(void *) &print_message_function, (void *) message2);
sleep ( 10 ) ;
exit (0) ;
}
void print_message_function( void *ptr )
{
char *message;
message = (char *) ptr;
printf("%s", message);
pthread_exit(0) ;
}
这样是否达到了我们的要求了呢?不尽如此,因为依靠时间的延迟执行同步是不可靠的。这里遇到的情形和一个分布程序和共享资源的情形一样。共享的资源是标准的输出设备,分布计算的程序是三个线程。
其实这里还有另外一个错误。函数sleep和函数e x i t一样和进程有关。当线程调用sleep时,整个的进程都处于睡眠状态,也就是说,所有的三个线程都进入睡眠状态。这样我们实际上没有解决任何的问题。希望使一个线程睡眠的函数是pthread_delay_np。例如让一个线程睡眠2秒钟,用如下程序:
struct timespec delay;
delay.tv_sec = 2;
delay.tv_nsec = 0;
pthread_delay_np( &delay );
}
[目录]
线程同步
POSIX提供两种线程同步的方法,mutex和条件变量。mutex是一种简单的加锁的方法来控制对共享资源的存取。我们可以创建一个读/写程序,它们共用一个共享缓冲区,使用mutex来控制对缓冲区的存取。
void reader_function(void);
void writer_function(void);
char buf f e r ;
int buffer_has_item = 0;
pthread_mutex_t mutex;
struct timespec delay;
main( )
{
pthread_t reader;
delay.tv_sec = 2;
delay.tv_nsec = 0;
pthread_mutex_init(&mutex, pthread_mutexattr_default);
pthread_create( &reader, pthread_attr_default, (void*)&reader_function,
N U L L ) ;
writer_function( )
void writer_function(void)
{
while( 1 )
{
pthread_mutex_lock( &mutex );
if ( buffer_has_item == 0 )
{
buffer = make_new_item();
buffer_has_item = 1;
}
pthread_mutex_unlock( &mutex );
pthread_delay_np( &delay );
}
}
void reader_function(void)
{
while( 1 )
{
pthread_mutex_lock( &mutex );
if ( buffer_has_item == 1)
{
consume_item( buffer );
buffer_has_item = 0;
}
pthread_mutex_unlock( &mutex );
pthread_delay_np( &delay );
}
}
在上面的程序中,我们假定缓冲区只能保存一条信息,这样缓冲区只有两个状态,有一条信息或者没有信息。使用延迟是为了避免一个线程永远占有mutex。
但mutex的缺点在于它只有两个状态,锁定和非锁定。POSIX的条件变量通过允许线程阻塞和等待另一个线程的信号方法,从而弥补了mutex的不足。当接受到一个信号时,阻塞线程将会被唤起,并试图获得相关的mutex的锁。
[目录]
使用信号量协调程序
我们可以使用信号量重新看一下上面的读/写程序。涉及信号量的操作是semaphore_up、semaphore_down、semaphore_init、semaphore_destroy和semaphore_decrement。所有这些操作都只有一个参数,一个指向信号量目标的指针。
void reader_function(void);
void writer_function(void);
char buffer ;
Semaphore writers_turn;
Semaphore readers_turn;
main( )
{
pthread_t reader;
semaphore_init( &readers_turn );
semaphore_init( &writers_turn );
/* writer must go first */
semaphore_down( &readers_turn );
pthread_create( &reader, pthread_attr_default,
(void *)&reader_function, NULL);
w r i t e r _ f u n c t i o n ( ) ;
}
void writer_function(void)
{
w h i l e ( 1 )
{
semaphore_down( &writers_turn );
b u ffer = make_new_item();
semaphore_up( &readers_turn );
}
}
void reader_function(void)
{
w h i l e ( 1 )
{
semaphore_down( &readers_turn );
consume_item( buffer );
semaphore_up( &writers_turn );
}
}
这个例子也没有完全地利用一般信号量的所有函数。我们可以使用信号量重新编写“Hello world” 的程序:
void print_message_function( void *ptr );
Semaphore child_counter;
Semaphore worlds_turn;
main( )
{
pthread_t thread1, thread2;
char *message1 = "Hello";
char *message2 = "Wo r l d " ;
semaphore_init( &child_counter );
semaphore_init( &worlds_turn );
semaphore_down( &worlds_turn ); /* world goes second */
semaphore_decrement( &child_counter ); /* value now 0 */
semaphore_decrement( &child_counter ); /* value now -1 */
/*
* child_counter now must be up-ed 2 times for a thread blocked on it
* to be released
*
* /
pthread_create( &thread1, pthread_attr_default,
(void *) &print_message_function, (void *) message1);
semaphore_down( &worlds_turn );
pthread_create(&thread2, pthread_attr_default,
(void *) &print_message_function, (void *) message2);
semaphore_down( &child_counter );
/* not really necessary to destroy since we are exiting anyway */
semaphore_destroy ( &child_counter );
semaphore_destroy ( &worlds_turn );
e x i t ( 0 ) ;
}
void print_message_function( void *ptr )
{
char *message;
message = (char *) ptr;
printf("%s ", message);
fflush(stdout);
semaphore_up( &worlds_turn );
semaphore_up( &child_counter );
p t h r e a d _ e x i t ( 0 ) ;
}
信号量c h i l d _ c o u n t e r用来强迫父线程阻塞,直到两个子线程执行完p r i n t f语句和其后的semaphore_up( &child_counter )语句才继续执行。
Semaphore.h
#ifndef SEMAPHORES
#define SEMAPHORES
#include
#include
typedef struct Semaphore
{
int v;
pthread_mutex_t mutex;
pthread_cond_t cond;
}
S e m a p h o r e ;
int semaphore_down (Semaphore * s);
int semaphore_decrement (Semaphore * s);
int semaphore_up (Semaphore * s);
void semaphore_destroy (Semaphore * s);
void semaphore_init (Semaphore * s);
int semaphore_value (Semaphore * s);
int tw_pthread_cond_signal (pthread_cond_t * c);
int tw_pthread_cond_wait (pthread_cond_t * c, pthread_mutex_t * m);
int tw_pthread_mutex_unlock (pthread_mutex_t * m);
int tw_pthread_mutex_lock (pthread_mutex_t * m);
void do_error (char *msg);
# e n d i f
Semaphore.c
#include "semaphore.h"
/ *
* function must be called prior to semaphore use.
*
* /
v o i d
semaphore_init (Semaphore * s)
{
s->v = 1;
if (pthread_mutex_init (&(s->mutex), pthread_mutexattr_default) == -1)
do_error ("Error setting up semaphore mutex");
if (pthread_cond_init (&(s->cond), pthread_condattr_default) == -1)
do_error ("Error setting up semaphore condition signal");
* function should be called when there is no longer a need for
* the semaphore.
*
* /
v o i d
semaphore_destroy (Semaphore * s)
{
if (pthread_mutex_destroy (&(s->mutex)) == -1)
do_error ("Error destroying semaphore mutex");
if (pthread_cond_destroy (&(s->cond)) == -1)
do_error ("Error destroying semaphore condition signal");
}
/ *
* function increments the semaphore and signals any threads that
* are blocked waiting a change in the semaphore.
*
* /
i n t
semaphore_up (Semaphore * s)
{
int value_after_op;
tw_pthread_mutex_lock (&(s->mutex));
( s - > v ) + + ;
value_after_op = s->v;
tw_pthread_mutex_unlock (&(s->mutex));
tw_pthread_cond_signal (&(s->cond));
return (value_after_op);
}
/ *
* function decrements the semaphore and blocks if the semaphore is
* <= 0 until another thread signals a change.
*
* /
i n t
semaphore_down (Semaphore * s)
{
int value_after_op;
tw_pthread_mutex_lock (&(s->mutex));
while (s->v <= 0)
{
tw_pthread_cond_wait (&(s->cond), &(s->mutex));
}
( s - > v ) - - ;
value_after_op = s->v;
tw_pthread_mutex_unlock (&(s->mutex));
return (value_after_op);
}
/ *
* function does NOT block but simply decrements the semaphore.
* should not be used instead of down -- only for programs where
* multiple threads must up on a semaphore before another thread
* can go down, i.e., allows programmer to set the semaphore to
* a negative value prior to using it for synchronization.
*
* /
i n t
semaphore_decrement (Semaphore * s)
{
int value_after_op;
tw_pthread_mutex_lock (&(s->mutex));
s - > v - - ;
value_after_op = s->v;
tw_pthread_mutex_unlock (&(s->mutex));
return (value_after_op);
}
/ *
* function returns the value of the semaphore at the time the
* critical section is accessed. obviously the value is not guarenteed
* after the function unlocks the critical section. provided only
* for casual debugging, a better approach is for the programmar to
* protect one semaphore with another and then check its value.
* an alternative is to simply record the value returned by semaphore_up
* or semaphore_down.
*
* /
i n t
semaphore_value (Semaphore * s)
{
/* not for sync */
int value_after_op;
tw_pthread_mutex_lock (&(s->mutex));
value_after_op = s->v;
tw_pthread_mutex_unlock (&(s->mutex));
return (value_after_op);
}
/* -------------------------------------------------------------------- */
/* The following functions replace standard library functions in that */
/* they exit on any error returned from the system calls. Saves us */
/* from having to check each and every call above. */
/* -------------------------------------------------------------------- */
i n t
tw_pthread_mutex_unlock (pthread_mutex_t * m)
{
int return_value;
if ((return_value = pthread_mutex_unlock (m)) == -1)
do_error ("pthread_mutex_unlock");
return (return_value);
}
i n t
tw_pthread_mutex_lock (pthread_mutex_t * m)
{
int return_value;
if ((return_value = pthread_mutex_lock (m)) == -1)
do_error ("pthread_mutex_lock");
return (return_value);
}
i n t
tw_pthread_cond_wait (pthread_cond_t * c, pthread_mutex_t * m)
{
int return_value;
if ((return_value = pthread_cond_wait (c, m)) == -1)
do_error ("pthread_cond_wait");
return (return_value);
}
i n t
tw_pthread_cond_signal (pthread_cond_t * c)
{
int return_value;
if ((return_value = pthread_cond_signal (c)) == -1)
do_error ("pthread_cond_signal");
return (return_value);
}
/ *
* function just prints an error message and exits
*
* /
v o i d
do_error (char *msg)
{
perror (msg);
exit (1);
}
[目录]
代码例子
[目录]
newthread
/***********************************************************************
Case study source code from the book `The Linux A to Z'
by Phil Cornes. Published by Prentice Hall, 1996.
Copyright (C) 1996 Phil Cornes
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave , Cambridge , MA 02139 , USA .
***********************************************************************/
new_thread(int (*start_addr)(void), int stack_size)
{
struct context *ptr;
int esp;
/* 1 */
if (!(ptr = (struct context *)malloc(sizeof(struct context))))
return 0;
/* 2 */
if (!(ptr->stack = (char *)malloc(stack_size)))
return 0;
/* 3 */
esp = (int)(ptr->stack+(stack_size-4));
*(int *)esp = (int)exit_thread;
*(int *)(esp-4) = (int)start_addr;
*(int *)(esp-8) = esp-4;
ptr->ebp = esp-8;
/* 4 */
if (thread_count++)
{
/* 5 */
ptr->next = current->next;
ptr->prev = current;
current->next->prev = ptr;
current->next = ptr;
}
else
{
/* 6 */
ptr->next = ptr;
ptr->prev = ptr;
current = ptr;
switch_context(&main_thread, current);
}
return 1;
}
[目录]
exitthead
/***********************************************************************
Case study source code from the book `The Linux A to Z'
by Phil Cornes. Published by Prentice Hall, 1996.
Copyright (C) 1996 Phil Cornes
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave , Cambridge , MA 02139 , USA .
***********************************************************************/
static exit_thread(void)
{
struct context dump, *ptr;
/* 1 */
if (--thread_count)
{
/* 2 */
ptr = current;
current->prev->next = current->next;
current->next->prev = current->prev;
current = current->next;
free(ptr->stack);
free(ptr);
switch_context(&dump, current);
}
else
{
/* 3 */
free(current->stack);
free(current);
switch_context(&dump, &main_thread);
}
}
[目录]
getchannel
/***********************************************************************
Case study source code from the book `The Linux A to Z'
by Phil Cornes. Published by Prentice Hall, 1996.
Copyright (C) 1996 Phil Cornes
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave , Cambridge , MA 02139 , USA .
***********************************************************************/
get_channel(int number)
{
struct channel *ptr;
/* 1 */
for (ptr = channel_list; ptr; ptr = ptr->link)
if (ptr->number==number)
return((int)ptr);
/* 2 */
if (!(ptr = (struct channel *)malloc(sizeof(struct channel))))
return 0;
/* 3 */
ptr->number = number;
ptr->message_list = 0;
ptr->message_tail = 0;
ptr->sr_flag = 0;
ptr->link = channel_list;
channel_list = ptr;
return((int)ptr);
}
[目录]
def
/***********************************************************************
Case study source code from the book `The Linux A to Z'
by Phil Cornes. Published by Prentice Hall, 1996.
Copyright (C) 1996 Phil Cornes
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave , Cambridge , MA 02139 , USA .
***********************************************************************/
#include <string.h>
struct context /* One structure for each thread */
{
int ebp; /* Base pointer (stack frame pointer) store */
char *stack; /* Pointer to memory block for thread stack */
struct context *next; /* Round robin circular list pointer */
struct context *prev; /* Round robin circular list pointer */
};
struct channel /* One structure for each communication channel */
{
int number; /* Channel number */
int sr_flag; /* 0=no queue, 1=send queued, 2=receive queued */
struct channel *link; /* Link to next channel in list */
struct message *message_list; /* Head of message queue */
struct message *message_tail; /* Tail of message queue */
};
struct message /* One structure for each pending send/receive */
{
int size; /* Size of message in bytes */
char *addr; /* Pointer to start of message */
struct message *link; /* Link to next message in queue */
struct context *thread; /* Which thread blocks on this struct */
};
static struct context main_thread; /* Storage for main() details */
static struct context *current; /* Currently executing thread */
static int thread_count = 0; /* Number of threads to schedule */
static struct channel *channel_list = 0; /* List of all channels */
static int switch_context(struct context *, struct context *);
static int exit_thread(void);
static int rendezvous(struct channel *, char *, int, int);
[目录]
release
/***********************************************************************
Case study source code from the book `The Linux A to Z'
by Phil Cornes. Published by Prentice Hall, 1996.
Copyright (C) 1996 Phil Cornes
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave , Cambridge , MA 02139 , USA .
***********************************************************************/
release(void)
{
/* 1 */
if (thread_count<=1)
return 0;
/* 2 */
current = current->next;
switch_context(current->prev, current);
return 1;
}
static switch_context(struct context *from, struct context *to)
{
/* 3 */
__asm__
(
"movl 8(%ebp),%eax/n/t"
"movl %ebp,(%eax)/n/t"
"movl 12(%ebp),%eax/n/t"
"movl (%eax),%ebp/n/t"
);
}
[目录]
redezvous
/***********************************************************************
Case study source code from the book `The Linux A to Z'
by Phil Cornes. Published by Prentice Hall, 1996.
Copyright (C) 1996 Phil Cornes
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave , Cambridge , MA 02139 , USA .
***********************************************************************/
send(int chan, char *addr, int size)
{
/* 1 */
return(rendezvous((struct channel *)chan, addr, size, 1));
}
receive(int chan, char *addr, int size)
{
/* 2 */
return(rendezvous((struct channel *)chan, addr, size, 2));
}
static int rendezvous(struct channel *chan, char *addr,
int size, int sr_flag)
{
struct message *ptr;
int nbytes;
/* 3 */
if (sr_flag==3-chan->sr_flag)
{
/* 4 */
ptr = chan->message_list;
chan->message_list = ptr->link;
ptr->thread->next = current->next;
ptr->thread->prev = current;
current->next->prev = ptr->thread;
current->next = ptr->thread;
++thread_count;
/* 5 */
nbytes = (size<ptr->size)?size:ptr->size;
ptr->size = nbytes;
/* 6 */
if (sr_flag==1)
memcpy(ptr->addr, addr, nbytes);
else
memcpy(addr, ptr->addr, nbytes);
/* 7 */
if (!chan->message_list)
chan->sr_flag = 0;
return(nbytes);
}
else
{
/* 8 */
ptr = (struct message *)malloc(sizeof(struct message));
if (!chan->message_list)
chan->message_list = ptr;
else
chan->message_tail->link = ptr;
chan->message_tail = ptr;
ptr->link = 0;
ptr->size = size;
ptr->addr = addr;
chan->sr_flag = sr_flag;
ptr->thread = current;
current->prev->next = current->next;
current->next->prev = current->prev;
/* 9 */
if (--thread_count)
{
current = current->next;
switch_context(ptr->thread, current);
}
else
switch_context(ptr->thread, &main_thread);
/* 10 */
nbytes = ptr->size;
free(ptr);
return(nbytes);
}
}
[目录]
unbouded
/***********************************************************************
Case study source code from the book `The Linux A to Z'
by Phil Cornes. Published by Prentice Hall, 1996.
Copyright (C) 1996 Phil Cornes
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave , Cambridge , MA 02139 , USA .
***********************************************************************/
int buffer(void);
int start(void);
int ch_desc1, ch_desc2;
main(void)
{
ch_desc1 = get_channel(1);
ch_desc2 = get_channel(2);
new_thread(start, 1024);
}
start(void)
{
int i, n;
new_thread(buffer, 1024);
for (i = 0; i<10, ++i)
{
send(ch_desc1, &i, sizeof(int));
release();
}
for (i = 0; i<10, ++i)
{
receive(ch_desc2, &n, sizeof(int));
printf("i=%d n=%d/n", i, n);
release();
}
}
buffer(void)
{
int i;
receive(ch_desc1, &i, sizeof(int));
new_thread(buffer, 1024);
send(ch_desc2, &i, sizeof(int));
}
网络服务 |
目 录
1. 网络服务
1. wu-ftpd
2. smbd
3. apache虚拟服务器
4. apache+php3+mysql
5. apache+apacheJserv
6. apache+tomcat
7. java环境(中文)
8. sendmaild
1. sendmail扮演的角色
2. 如何运行sendmail
3. sendmail.cf
4. 邮件中枢与分发代理
5. 宏
6. 规则
7. 规则集0
9. qmail
10. router
11. cvs
12. bind
13. openssh
1. 简单使用
2. 配置文件
3. scp
4. 使用key做验证
5. rsync与ssh配合使用
6. cvs
14. Khttpd
网络服务
[目录]
wu-ftpd
chmod 777 /home/ftpd/incoming
chown ftp /home/ftpd/incoming
chgrp ftp /home/ftpd/incoming
改etc下的ftpaccess,我的是这样:
loginfails 2
class local real,guest,anonymous *.domain 0.0.0.0
class remote real,guest,anonymous *
limit local 20 Any /etc/msgs/msg.toomany
limit remote 100 SaSu|Any1800-0600 /etc/msgs/msg.toomany
limit remote 60 Any /etc/msgs/msg.toomany
readme README* login
readme README* cwd=*
message /welcome.msg login
message .message cwd=*
compress yes local remote
tar yes local remote
# allow use of private file for SITE GROUP and SITE GPASS?
private yes
# passwd-check <none|trivial|rfc822> [<enforce|warn>]
passwd-check rfc822 warn
log commands real
log transfers anonymous,real inbound,outbound
shutdown /etc/shutmsg
# all the following default to "yes" for everybody
delete no guest,anonymous # delete permission?
overwrite no guest,anonymous # overwrite permission?
rename no guest,anonymous # rename permission?
chmod no anonymous # chmod permission?
umask no anonymous # umask permission?
# specify the upload directory information
upload /home/ftp /pub no
upload /home/ftp /pub1 no
upload /home/ftp /incoming yes
upload /home/ftp /pub1/incoming yes
upload /home/ftp /bin no
upload /home/ftp /etc no
# directory aliases... [note, the ":" is not required]
alias inc: /incoming
# cdpath
cdpath /incoming
cdpath /pub
cdpath /
# path-filter...
path-filter anonymous /etc/pathmsg ^[-A-Za-z0-9_/.]*$ ^/. ^-
path-filter guest /etc/pathmsg ^[-A-Za-z0-9_/.]*$ ^/. ^-
# specify which group of users will be treated as "guests".
guestgroup ftponly
email user@hostname
[目录]
smbd
更改/etc/smb.conf,主要要确定下面的内容:
workgroup=MYWKGP
smb passwd file=/etc/smbpasswd
security=user (使用unix用户安全)
encrypt passwords= yes
然后就可以添加目录了,首先添加宿主目录:
(其实就是把对应的行的注释去掉)
[homes]
comment = home directory
browseable = no
writeable= yes
还可以再加一个共享目录:
[public]
comment = public directory
path=/public
public=yes
writeable=no
read only=yes
printable=no
最后当然是加入用户,比如想把fake用户当成windows 系统中的winfake,执行
smbadduser fake:winfake
输入访问口令,以后在windows 98中只要提供winfake和对应口令
就可以访问/home/fake和/public目录
有时windows 95/98会出现一些兼容问题,研究encrypt passwords=那一行吧
==================================================================
最好用swat来设置吧,把/etc/inet.conf最后swat前的注释去掉
然后重启inetd,就可以通过任意一个机器用浏览器来配置了
输入地址http://samba_server:901/即可
95/98/nt都可以的说,还好这里什么机器都有,嘿嘿,都可以互相
访问的,包括打印,95/98网络登录时要用samba里的用户名哦
[global]
workgroup = IPC
netbios name = DATASERVER
server string = Data Server - Linux
encrypt passwords = Yes
log file = /var/log/samba/log.%m
max log size = 50
#这三行设置超时,还没搞清楚,不过这样马马虎虎能工作,一定要设
#否则mount上的设备umount不掉,一直busy的说,:PP
change notify timeout = 30
deadtime = 1
keepalive = 0
socket options = TCP_NODELAY SO_RCVBUF=8192 SO_SNDBUF=8192
dns proxy = No
#这三行是umask的设置,看情况设置吧
create mask = 0755
force create mode = 0755
force directory mode = 0755
#用户的home目录,根据用户不同而变化,本来就有的,不是我加的说
[homes]
comment = Home Directories
read only = No
browseable = No
[printers]
comment = All Printers
path = /var/spool/samba
print ok = Yes
browseable = No
#我自己加的一个共享目录
[public]
comment = Public Software
path = /home/ftp/pub
guest ok = Yes
[目录]
apache虚拟服务器
多域名单IP的:
1. 首先要解决域名的问题,你可以用nslookup <你的域名>看看该域名是不是指向了你IP, *.yeah.net本来就是做的虚拟主机, 所以解析出来都是指向www.yeah.net (202.96.152.198), 通常情况下, apache启动时会检查你用作虚拟主机的域名。
2. 改httpd.conf, 加<virtualhost> 例如:
<VirtualHost I007.com> (写IP也可以)
ServerAdmin [email protected]
DocumentRoot /home/I007/public_html
ServerName I007.com (这里只能写域名)
</VirtualHost>
多IP的:
1. 最好每个IP还是能对应一个域名
2. 加载ip_alias模块 (/sbin/insmod ip_alias)
3. 加IP (/sbin/ifconfig eth?:? ???.???.???.???)
4. 改httpd.conf, 和上面差不多
可以看出, 两者可以共存,以上均做过实验, 没有问题..
[目录]
apache+php3+mysql
*安装apache-devel rpm包.
*安装MySQL相关的包.
rpm -ivh MySQL-3.22.25-1.i386.rpm
rpm -ivh MySQL-client-3.22.25-1.i386.rpm
rpm -ivh MySQL-devel-3.22.25-1.i386.rpm
*下载php3源文件php-3.0.tar.gz
*解开源文件
tar xzvf php-3.0.tar.gz
*进入php-3.0源文件目录
./configure --with-mysql=/usr --with-apxs=/usr/sbin/apxs --with-config-file
-path=/etc/httpd (还有其他选项,可以运行 ./configure --help 看看)
*建一个modules目录
mkdir modules
*进入regex目录
运行 make r (这一步可能不必要)
运行 make lib
*退回上一级目录
运行 make install
* cp modules/libphp3.so /etc/httpd/modules/
* cp php3.ini-dist /etc/httpd/php3.ini
* 修改/etc/httpd/conf/httpd.conf 加入
LoadModule php3_module modules/libphp3.so
AddModule mod_php3.c
修改/etc/httpd/conf/srm.conf 加入
AddType application/x-http-php3 .php3
*重起apache
/etc/rc.d/init.d/httpd restart
*测试
查看mysqld是否已经运行.如果没有,先运行mysqld
将php3源文件下tests目录下的mysql文件拷贝到 /home/httpd/html目录下
lynx localhost/mysql.php3
如果有问题,可以将mysql.php3中的127.0.0.1改为localhost
[目录]
apache+apacheJserv
=============================================================================
环境:mdk7
首先使用BJLC光盘上的jdk1.2pre2
具体装过程略
rpm: apache-1.3.9 apache-devel jsdk-2.0
tgz: Apache-Jserv1.1 gnujsp-1.0.0
解开Jserv的src
./configure --enable-EAPI --with-JSDK=/usr/lib/classes/jsdk.jar /
--with-apxs=/usr/sbin/apxs
应该没有问题,很顺利编译通过
再/etc/httpd/conf/httpd.conf
在最后(一定要在最后否则有很多小麻烦)
Include ./jserv/jserv.conf
重新启动apache
http://localhost/servlet/Hello
如果能看到应该就差不多了
试http://localhost/jserv/
一般是403 forbidden
如果是forbidden
修改
jserv.conf
搜索Location
里面有一段allow
改成allow localhost.localdomain
好像就可以了
用localhost或者是127.0.0.1都不行
反正这是我碰到的倒霉事希望大家能一次成功
到这apache-jserv就算是完成了,也就是有了
一个servlet容器并且有了在apache里面的模块
能够同这个容器进行通信,通信协议最好用
apjv12,反正所由地方都设成apjv12就没有问题
否则很可能出现不匹配的情况,最后导致
500错误,在log中显示malformed fata
前面提到servlet容器建好了
Jserv调通了,还有两个很有用的log文件
在我的mdk7里面是
/usr/logs/里面的mod_jserv.log和jserv.log
这两个文件在调试中极有用
它们的权限要求nobody可以写
所以我的做法是
chmod 664 *
chgrp nobody *
大牛是不是这样做,我不知道
可能这样会有安全性问题
前面如果你看到了jserv的状态页点击apjv12://localhost:8007
应该能够看到各个zone的配置
如果看不到应该收到所谓selfservlet一个选项
没有打开的提示,打开就好了好像是在jserv.properties里面
接下来是搞gnujsp
解开了会发现jsp是完全用java写成的例程
主要是两个gnujsp-...jar和servlet-...jar
反正我用的是gnujsp10和servlet-2.0-plus
拷到/usr/servlet/jsp(这是随便给的)
编写gjsp.properties文件
关键是
servlet.gnujsp.class=org.gjt.jsp.JspServlet
servlet.gnijsp.initArgs=checkclass=true,/ //这样编译好的jspservlet可以利用
pagebase=/home/httpd/html,/
scratchdir=/usr/servlets/jsp/scratch,/ //这里要有文章
debug=true,/ //先打开,调好了再关掉
compiler=builtin-javac -classpath / //这里builtin-javac需要jdk1.2的tools.jar
%classpath%:%scratchdir%:/usr/servlets/jsp/gnujsp.jar:/
/usr/servlets/jsp/servlet.jar:/usr/servlets/jsp/beans / //指定编译用的classpath
-d %scratchdir% -deprication %source% //就是编译一个普通的java文件一样
首先要注意你选择的scratchdir一定要nobody可写,其次使用builtin的javac
需要在jserv.properties加上
wrapper.class...=/usr/jdk1.2/lib/tools.jar (这是我安装的路径)
到这里jsp的zone就差不多了
现在开始在jserv.properties里面指明zones
很简单不废话了
最后是在jserv.conf里面作mount,照着例子写
也不难,前面有人贴过了
最后就可以试试自己的jsp了
首先先从http://localhost/jserv/
里面看一看jsp的zone配制出来没有
验证一下参数都正确
====================================================================================================
1.先确认apachejserv已能正常工作
2.把gnujsp10.jar,servlet-2.0-plus.jar拷到/home/httpd/sevlets目录下
3.修改jserv.conf,加入
ApJServAction .jsp /servlets/org.gjt.jsp.JspServlet
ApjServMount /java /root(这个目录主要是用来存放jsp生成的java
和class文件,可视具体情况而定)
4.修改jserv.properties,加入
wrapper.classpath=/usr/jdk/lib/tools.jar
wrapper.classpath=/home/httpd/servlets/gnujsp10.jar
wrapper.classpath=/home/httpd/servlets/servlet-2.0-plus.jar
5.修改zone.properties,加入
repositories=/home/httpd/java/
servlet.gnujsp.code=org.gjt.jsp.JspServlet
servlet.org.gjt.jsp.JspServlet.initArgs=checkclass=true
servlet.org.gjt.jsp.JspServlet.initArgs=scratchdir=/home/httpd/java
servlet.org.gjt.jsp.JspServlet.initArgs=compiler=builtin-javac -encoding %enco
ding% -classpath %classpath% -d %scratchdir% -deprecation %source%
[目录]
apache+tomcat
编译得到mod_jserv.so和ApacheJServ.jar
在tomcat3.0的etc子目录里面有tomcat.conf,在httpd.conf里面Include一下
启动tomcat(./tomcat.sh start)然后启动httpd,看看成功了没有。
如果不行,我碰到的一个问题是log file有问题,直接指定成下面的样子
ApJServLogFile /var/log/httpd/mod_jserv.log
就过了。
最后实验http://localhost:8080/一下确信tomcat跑了。然后实验http://localhost:8080/examples/jsp和http://localhost/examples/jsp
[目录]
java环境(中文)
好象有许多人都碰到了这个问题, 我以前也碰到过, 在网上找了一些解决方法看了看, 在自己的机器上试了试, 都没有解决问题, 趁着这几天把 Debian的中文问题搞好了, 顺便把这个也弄一弄, 终于搞明白了一些东西. 错误之处还请大家指教. 这里主要指的是 Swing 中显示的中文, 关于其他中文操作问题,暂不作讨论.
其实说起来 Java 的中文问题非常简单, 有两个重要因素, 一个是编译时的编码(encoding), 另一个是字体. 这两个缺一不可.
一, 编译时的编码:
指的是编译 Java 的源程序时指定的编码格式, 即使用
javac -encoding MyEncoding MyJavaFile.java
编译 MyJavaFile 时 -encoding 所指定的 MyEncoding. 通常中文的编码有大陆和新加坡使用的 GB2312 以及 台湾和香港使用的 Big5, 普通英文用的是ISO_8859_1. 这里只关注 GB2312. 如果在编译的命令行中指定了编码, 那么就使用指定的编码格式, 如果没有指定, 将使用系统缺省的编码格式, 在 windows的中文简体版下都是 GB2312, 而 Linux 下不同的终端可以有不同的编码, 可以直接设置环境变量 LC_ALL LANG 等指定相应的不同类型信息的编码, 比如数字,日期, 货币, 姓名等等. 可以用 locale 命令来看当前终端所有的编码. 在Java 程序中可以使用 System.getProperty("file.encoding"); (注 1) 来获取当前的编码格式. 好了, 当编译的时候, 源码中的中文字符串都会按照指定的或者缺省的编码格式映射为相应的 UNICODE , 而不是看做简单的 ASCII 字符.所以编码格式就是处理源码中的各种各样的字符, 与运行时无关, 所以如果你的源文件中没有直接需要使用中文的地方, 比如菜单, 按钮, 标签什么的, 就不需要指定编码了.
二, 字体的选择:
必须使用能够显示中文的字体, 好象有些废话. 但还是有必要说一说. AWT 可以在中文系统中直接显示中文, 但是在不是中文系统但是有中文字体的系统中不能, 需要修改一下 $JAVA_HOME/jre/lib/font.properties,加入中文字体, 至于怎么加入, 就不谈了. 而 Swing 不需要这样, 只要你的系统中有中文字体就可以使用这样的字体来显示中文. 怎么知道一种字体能不能显示中文呢? 看下面的程序片段: (注 2)
Font[] fonts = GraphicsEnvironment.getLocalGraphicsEnvironment().getAllFFonts();
System.out.println("Chinese Fonts: /n");
for (int i = 0; i < fonts.length; i ++)
{
if (fonts[i].canDisplayUpTo("/u4e00") > 0) // 注 3
{
System.out.println(fonts[i].getFontName());
}
}
好了, 知道了系统中有没有中文字体, 然后选择一个就是了. 如果选择了不能显示中文的字体, 通常的情况是显示方框.
好了, Swing 的中文显示介绍完了, 输入和编辑中文就和显示差不多, 选一个字体就可以了, 具体得看你而 Linux 的具体设置了, 这里就不多说了.
从以上两点来看, 其实最重要的还是字体的选择. 我的网站上有 ScreenShot和测试程序, 欢迎来访问. http://SuperMMX.dhs.org
注 1: 据 jdk1.2.2 的文档, 并没有这个属性, 很奇怪, 但是确实可用.
注 2: jdk 1.2 以上自带的字体支持 Unicode, 但是不支持 Unicode 汉字. 所以打出来的结果包含了 jdk 自带的字体.
注 3: UNICODE 中, UNICODE 汉字从 /u3400 到 /u9FFF 中间, /uF900 到 /uFAFF也有一些, 但是 GB2312 和 Big5 的汉字和字符都是在 /u4E00 到 /u9FFF中间.
参考文献:
jdk tools doc
一篇没有作者名字, 也没有出处的英文文章.
[目录]
sendmaild
最简提示:
主要的问题是如何配置sendmail.cf.
如果你用的是redhat,执行:
cd /usr/lib/sendmail-cf/cf
m4 redhat.mc > sendmail.cf
cp sendmail.cf /etc/sendmail.cf
sendmail -bd(或者reboot)
上面的方法配置之后服务器发信没问题,而远端机器却不能发信,是因为缺省redhat.mc生成的sendmail.cf禁止了客户服务,
这样:找到下面一行(sendmail.cf):
R$* $#error $@ 5.7.1 $: "550 relay denied"
改成
R$- $@ ok
就行了,不过当心有人用你的系统发垃圾邮件
[目录]
sendmail扮演的角色
1. sendmail扮演的角色
sendmail程序扮演着多种不同的角色,最主要的角色就是传递电子邮件。它监听来自网络的电子邮件,传送电子邮件到另一台机器,通过本地传送,将本地信件传给本地程序。它能够在邮件中附加上文件,也能够使用管道将邮件发给其它程序。它能够维护一个邮件队列,有序地将邮件发送出去,还可以理解接收者的邮件别名,将其发送到真正的目标用户当中去。
1.1在文件系统中的角色
sendmail程序的角色(位置)在本地文件系统中,就象一个倒置的树,如图3-1。当sendmail运行时,首先读取/etc/sendmail.cf配置文件。在这个配置文件中,能够指出sendmail所需要的其它文件与目录的位置。
图3-1 sendmail.cf的层次结构
安全起见,在配置文件/etc/sendmail.cf中指定的文件、目录名一般都使用绝对路径,例如:使用/var/spool/mqueue,而不使用mqueue。在我们讲解这些文件之前,首先运行如下命令以收集一个文件列表:
% grep = / /etc/sendmail.cf
注:如果你使用的是V8.7以上版本的sendmail,则你需要查找的应该改为”/[^0-9].*/”.
这个命令将会出现类似如下的结果:
O AliasFile=/etc/aliases
#O ErrorHeader=/etc/sendmail.oE
O HelpFile=/usr/lib/sendmail.hf
O QueueDirectory=/var/spool/mqueue
O StatusFile=/etc/sendmail.st
#O UserDatabaseSpec=/etc/userdb
#O ServiceSwitchFile=/etc/service.switch
#O HostsFile=/etc/hosts
#O SafeFileEnvironment=/arch
Mlocal,P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10/30, R=20/40,
Mprog, P=/bin/sh, F=lsDFMoeu, S=10/30, R=20/40, D=$z:/,
请注意,有些行是以字母O开始,有些行是以字母M开始,有些行是以#开始的。以字母O开始的行是配置选项。字母O后面紧跟着选项的名字。选项的值指出了sendmail使用的文件。例如,AliasFile定义了本地的aliases数据库文件。以M开始的行则定义了分发代理。以#开始的行就是注释。
首先我们考察以O开始的选项行。然后再分析以M开始的分发代理选项。
1.1.1 aliases文件
aliasing就是将接收者的名字转换另一个名字。一种情况下是将一些通用名字(如root、webmaster)转换成真正的用户名。另一种情况下是将一个名字转换成多个名字的列表(如使用邮件列表)。
Aliases文件在sendmail.cf文件中的AliasFile选项中指定,例如:
O AliasFile=/etc/aliases
以下是一个aliases文件的简要实例:
# Mandatory aliases.
postmaster: root
MAILER-DAEMON: postmaster
# The five forms of aliases
John_Adams: adamj
xpres: ford,carter,bush
oldlist: :include: /usr/local/oldguys
nobody: /dev/null
ftphelp: |/usr/local/bin/sendhelp
你的aliases文件可能更长、更复杂的,不过,以上这个示例也显示了aliases所有可能的构成。
以#开始的行,是注释行。空行被忽略不计。第一行就是一个注释行,它指出了2、3两行是每一个aliases文件被必须强制拥有的。所有的别名格式都是一样的,一个名字(别名)和一个要改为成的名字(原名)。别名在“:”号的左边,原名在“:”号的右边。名字是不区分大小写。例如:POSTMASTER,Postmaster以及postmaster都是相同的。
如果信封上列出的接收者名字是本地用户的话,sendmail程序就会查找aliases文件。如果sendmail发现接收者名字如果与aliases文件中的“:”号左边的名字相匹配,就将接收者名字替换为“:”号右边的名字。例如:发给本地的postmaster的信,会被转变成为发给root的信。
在一个名字被替换后,会使用这个被替换后的名字继续查找,直到没有匹配的名字为止。如MAILER-DAEMON首先被转变为postmatser,然后postmaster又被改被为root。由于没有一个以root开始的别名项,所以转换过程到此结束,信件最后被传送到root的信箱中去。
任何一个aliases文件必须有一个将postmaster转变成实际用户的别名项。因为当邮件出现问题时,总是会生成一个错误报告的信,发给postmaster。所以最好将这样的信发给邮件的系统管理员。
当电子邮件被退回时,将会发给MAILER-DAEMON。所以这个别名是必须的。没有这个设置的话,退回的信将会在发件人与收件人之间不断来回传送。
Aliases文件中有五种类型:
John_Adams: adamj
xpres: ford,carter,reagan,bush
oldlist: :include: /usr/local/oldguys
nobody: /dev/null
ftphelp: |/usr/local/bin/sendhelp
首先我们看一下第一行,这一行的格式与我们前面的例子相似,这一句让sendmail程序将发给John_Adams的信都给真实用户adamj。
xpres那一行则象大家演示了如何将一个名字扩展到一组名字,发给xpres的信件将扩展为ford、carter、reagan、bush,并使用这些名字进行aliases处理,直到无匹配为止。然后将信件的副本分发给每一个。
而oldlist这行,则象大家演示了如何从一个文件中读取一组名字,本例中就是让sendmail将发给oldlist的信息扩展为在/usr/local/oldguys文件里的用户列表。请记住它的格式,在文件绝对路径名前需加上“:include:”。
nobody这一行,象大家演示了如何用文件名代替别名。发到这个邮件中的内容将添加到所指定的文件后面去。这里指定的是/dev/null。这样就是指发到nobody的信件将会被简单地丢弃。
最后一行,ftphelp那一行象大家演示了如何用程序名代替别名。字符“|”使sendmail将这个邮件信息通过管道发给所指定的程序。
aliases文件可能变得非常复杂。它能够用于解决许多特殊的问题。aliases文件的更多的内容将在第24章:别名中详述。
1.1.2 邮件队列目录
在很多情况下,都可能使一个电子邮件临时无法发送,例如:远程主机已经down了,或临时出现了磁盘错误。为了确保邮件最终能够发送成功,sendmail将会把它们存到邮件队列目录中,直到发送成功为止。
配置文件中的选项QueueDirectory用于指定sendmail的邮件队列目录:
O QueueDirectory=/var/spool/mqueue
这里指出的目录名必须是全路径名。
如果你用足够的权限,看一看队列目录。如果没有邮件等待发送的话,它们可能是空的。如果它们不是空的,那么可能包括形如以下的文件:
dfQAA07038 dfMAA08000 qfQAA07038 qfMAA08000
当一个邮件信息进入了邮件队列,将分成两个部分,每一个部分都保存在一个文件中。头信息存在一个文件名是以qf开头的文件中。邮件内容部分则存在一个文件名是以df开头的文件中。
上例中,有两个邮件在邮件队列中。其中一个被标识为QAA07038,而另一个被标识为MAA08000。
队列文件的格式与处理方法,我们将在第23章:邮件队列中详细说明。
1.2本地分发的角色
sendmail的另外一个角色则是分发电子邮件信息给本地用户。一个本地用户在本地系统上有一个邮箱。分发本地邮件,就是将其附加到这个用户的邮箱中。
通常,sendmail不是直接将邮件信息直接放到文件中去。在上一节中,我们看到,只有指定sendmail程序将邮件附加到一个文件中时,才这样做的。但这是一个例外,不是规则,sendmail调用其他程序执行分发。被调用的程序叫做分发代理。
在你的sendmail.cf文件中,有两行用来定义本地分发代理,其中一个用于在本地系统中分发邮件:
Mlocal,P=/bin/mail, F=lsDFMAw5:/|@rm, s=10, R=20/40,
Mprog,P=/bin/sh, F=lsDFMeu, S=10, R=20/40, D=$z:/,
程序/bin/mail用来将邮件附加到用户的邮箱中。程序/bin/sh用来运行其它程序来处理分发。
1.2.1 分发到邮箱
配置文件中,以Mlocal开始的行定义了邮件如何附加到用户的邮箱文件中去。通常是使用/bin/mail程序,也可以使用deliver或mail.local程序。
在UNIX系统中,用户的邮箱是一个单独文件,其中个邮件信息。通常UNIX系统约定(但不是唯一的可能)每一个在邮箱文件中邮件信息以一个五字节长的“From ”(4个字母、一个空格)开始,并以一个空行结束。
Sendmail程序并不知道也不关心用户的邮箱文件是什么样的。而只关心将邮件添加到邮箱文件中的程序名称。例如:/bin/mail。以M开头的配置行定义了分发代理,详细的介绍可以参看第6章:Mail中枢和分发代理,以及第30章:分发代理。
1.2.2 借助程序分发
在1.1小节中的aliases文件示例中的ftphelp行,是以字符“|”开始的程序名作为Mail的目的地:
ftphelp: |/usr/local/bin/sendhelp
在此情况下,发送到ftphelp中的mail,经过别名转换到|/usr/local/bin/sendhelp中。以字符“|”开始的目的地地址告诉sendmail启动这个程序,而非添加到一个文件中去。这主要是实现使用一个邮件程序对接收的邮件作一些有用的处理。
Sendmail程序不直接运行邮件的分发程序。而是运行一个shell,并告诉shell运行这个程序。这个shell的名字在配置文件中以Mprog开始的行中定义:
Mprog,P=/bin/sh, F=lsDFMeu, S=10, R=20/40, D=$z:/,
在这个例子是,指定的shell是/bin/sh。也可以使用/bin/ksh或smrsb。
1.3网络传输角色
sendmail还有一个角色就是负责将邮件传送到另一台机器。当sendmail确定接收者不在本地系统中时,邮件将传送出去。下列是典型的配置文件中定义负责将邮件传送到其它机器的分发代理:
MsmtpP=[IPC], F=mDFMuX, S=11/31, R=21, E=
, L=990,
MuucpP=/usr/bin/uux, F=DFMhuUd, S=12, R=22/42, M=10000000,
而在实际的配置文件中可能会有一些不同。上面例子中,smtp可以写为ether或ddn或其它的一些东西。而uucp可以写为suucp或uucp-dom。一个重要的知识点就是有一些分发代理处理本地分发,另外一些处理跨越网络的分发。
1.1.1 TCP/IP
sendmail程序内在拥有在同一种网络中传输邮件的能力,那就是使用TCP/IP;下列的行就是用来指示sendmail去处理:
MsmtpP=[IPC], F=mDFMux, S=11/31, R=21, E=
, L=990
其中[IPC]可以写作[TCP],它们是完全等价的。
当sendmail程序在TCP/IP网络中传输mail时,首先发送“信封”上的发信人主机名到另一个站点。如果这个站点认可这个发信人的主机名是合法的,本地的sendmail程序将发送“信封”上的收信人列表。这个站点针对每一个收件人确定接受或拒绝。如果一些收件人被接受,则本地的sendmail程序发送出邮件信息(信头和信体)。
1.1.2 UUCP
在配置文件中设置sendmail如何通过UUCP传输邮件的行如下所示:
MuucpP=/usr/bin/uux, F=DFMhuUd, S=12, R=22/42, M=10000000,
这行告诉sendmail程序使用/bin/uux来通过UUCP网络传输邮件。
1.1.3 其它协议
sendmail程序还可通过其它网络协议传输mail。你可以从我们前面做过grep操作的输出结果中发现它们,它们看上去象:
Mfax, P=/usr/local/lib/fax/mailfax, F=DFMhu, S=14, R=24, M=100000,
Mmail11, P=/usr/etc/maill11, F=nsFx, S=15, R=25, A=mail11 $g $x $h $u
Mmac,P=/usr/bin/macmail, F=CDFMmpsu, R=16, S=16, A=macmail –t $u
Mfax行定义了使用sendmail发送FAX的途径。FAX通过电话线传输文件的图形影象。在这个配置中,程序/usr/lib/fax/mailfax,将一个邮件文件的图形影象FAX出去。
Mmail11行定义了使用mail11程序在DEC网络上传输邮件,一般应用于DEC系统。
Mmac行定义了在Macintosh电脑系统的AppleTalk网络上传输邮件。
在所有的这些例子中,sendmail通过专用的服务程序在网络上发送电子邮件。记住,sendmail本身只能直接在基于TCP/IP的网络上工作。
1.4 Daemon角色
就象sendmail能够在基于TCP/IP的网络上传输电子邮件一样,它也能够接收来自于网络的电子邮件。为了实现这个,就必须运行在daemon(守候进程)模式。Daemon是一个运行在后台,不受终端约束的程序。
作为一个daemon,sendmail通常在系统启动时就运行。当一个电子邮件发送到你的机器时,远程机器将与运行在你机器上的sendmail daemon“商谈”。
想观察你的系统如何将sendmail运行在daemon模式下,你可执行以下任何一条命令:
% grep sendmail /etc/rc* (BSD系统)
% grep sendmail /etc/init.d/* (SysV系统)
% grep sendmail /etc/*rc (HP-UX系统)
一个典型的输出是:
/etc/rc.local:if [-f /usr/lib/sendmail –a –f /etc/sendmail.cf]; then
/etc/rc.local: /usr/lib/sendmail –bd –q1h; echo –n ‘ sendmail’
上面的第二行是sendmail在系统启动时运行的命令。
/usr/lib/sendmail –bd –q1h
命令选项-bd是使sendmail运行在daemon模式下。命令选项-q1h是让sendmail每小时唤醒一次,处理队列。
[目录]
如何运行sendmail
2.如何运行sendmail
一种运行sendmail的方法是,直接在命令行上附加接收者参数。例如:下面命令就是发一个邮件消息给george。
% /usr/lib/sendmail george
你也可以同时给出多个接收者。例如,发一个邮件给george,truman和teddy:
% /usr/lib/sendmail george,truman,teddy
sendmail程序可以接受两种不同的命令行参数。不以“-”字符开始的参数是接收者,如上面的george。而以“-”字符开始的参数则是影响sendmail运行的开发选项。所有的命令行开关选项的解释,
标记 说明
-b 设置运行模式
-v 运行于冗长模式
-d 运行于调试模式
2.1适宜的模式(-b)
sendmail程序的“-b”参数可以衍生出许多功能。例如:使sendmail显示队列的内容,使sendmail重建别名数据库。本章将介绍一些比较常用到的子参数。
格式 说明
-ba 使用ARPAnet(灰皮书书)协议
-bD 以守候进程模式运行,但不fork
-bd 以守候进程模式运行
-bH 清除固有的主机状态
-bh 打印固有的主机状态
-bi 重建别名数据库
-bm 成为一个邮件发送者
-bp 打印邮件队列内容
-bs 在标准输入处运行SMTP
-bt 测试模式:仅解析地址
-bv 检验:不收集、分发
-bz 冻结配置文件
选项可以使sendmail执行时象其它名字。每个名字可以是一个硬连接,一个符号连接,或sendmail的一份拷贝。
hoststat -bh 打印固有主机状态
mailq -bp 显示邮件队列内容
newaliases -bi 重建别名数据库
purgestat -bH 清除固有主机状态
smtpd -bd 以守侯进程运行
2.1.1 守候进程模式 (-bd)
sendmail程序能够在后台以守侯进程模式运行,监听来自其它机器的邮件。Sendmail当第一次以守侯进程模式运行时,程序只读取配置文件一次,然后就一直运行,不再读取配置文件了。也就是说,执行后将不会发现配置文件的变化。
当你对配置文件sendmail.cf做了任何修改,都需要kill掉sendmail进程,然后重新启动它。但当你kill掉这个守候进程前,必须知道如何正确地重新启动它。这些信息是/etc/sendmail/pid或一个系统rc文件。
在BSD系列的UNIX系统,守候进程通常使用以下命令启动:
/usr/lib/sendmail –bd –q1h
命令行开关选项“-bd”指定sendmail以守候进程方式运行。“-q”选项告诉sendmail多久去查看一次待处理邮件队列。“-q1h”就是将其设置为1小时。
在你的机器上启动sendmail的命令,可能与我们这儿给出的是不同的。如果你管理许多不同的UNIX系统,你无须知道每一种是如何运行的。
2.1.1.1 kill并重启sendmail v8.7
在sendmail 8.7以后的版本中,kill并重启sendmail变得比较简单。一个单一的命令将会完成这一工作:
% kill –HUP `head –1 /etc/sendmail.pid`
这个单一的命令与下一个小节中的两个命令的效果完全相同。
2.1.1.2 kill并重启sendmail v8.6
当你要以守候进程方式启动sendmail,你须确认没有一个已运行的sendmail守候进程。在8.6版sendmail中,可以在/etc/sendmail.pid文件的第一行中找到进程ID号pid。你可以执行以下命令来kill掉sendmail:
% kill `head –1 /etc/sendmail.pid`
当你kill掉当前运行的守候进程,你可以使用以下命令来重新运行sendmail:
% `tail –1 /etc/sendmail.pid`
2.1.1.3 kill并重启老版本sendmail
在老版本的sendmail中,你必须使用ps来获得sendmail的进程ID号pid。在BSD UNIX和System V UNIX使用ps的方法不尽相同。
针对BSD UNIX来说,ps命令的将得到类似以下的输出:
% ps ax | grep sendmail | grep –v grep
99 ? IN 0:07 /usr/lib/sendmail -bd –q1h
% kill 99
这个输出的最左边的就是进程ID号。
针对System V UNIX系统来说,ps命令的参数及命令输出都不尽相同:
% ps ae | grep sendmail
99 ?0:01 sendmail
% kill 99
在老版本的sendmail中,你必须通过查看rc文件来获知如何重启sendmail。
2.1.1.4 如果你忘记了kill守候进程
如果你在重启sendmail时,忘了kill掉原来的进程的话,你就会看到类似以下列出的错误消息,每5秒显示一次。
Getrequests: cannot bind: Addredd already in use
Getrequests: cannot bind: Addredd already in use
Getrequests: cannot bind: Addredd already in use
Getrequests: cannot bind: Addredd already in use
Getrequests: cannot bind: Addredd already in use
Getrequests: cannot bind: Addredd already in use
Getrequests: cannot bind: Addredd already in use
Getrequests: cannot bind: Addredd already in use
Getrequests: cannot bind: Addredd already in use
Getrequests: cannot bind: Addredd already in use
Getrequests: cannot bind: Addredd already in use
Opendaemonsocket: Server SMTP socket wedged:exiting
2.1.2 显示邮件队列模式 (-bp)
sendmail程序能够显示邮件队列的内容。可以通过两种方法实现:一种是运行mailq,另一种是运行带开关选项“-bp”的sendmail。无论你使用哪种方法,邮件队列将会显示出来。如果这个队列是空的,sendmail将打印出:
Mail queue is empty
如果有一个邮件正在等待队列中,那么将输出更多的消息,其中包含类似于下列的信息:
) Mail Queue (1 requests)
--Q-ID--- --Size-- ----Q-Time
-------------------Sender/Recipient-------------------
GAA29775* 702 Thu Mar 12 16:51<[email protected]>
Deferred:Host fbi.dc.gov is down
<[email protected]>
在此,带开关选项“-bp”的sendmail显示了仅有一个邮件信息在队列中。如果有多个,那么每一个邮件都会像这样列出来。每一个至少有两行输出。
第一行显示邮件和发送者的细节信息。GAA29775是这个邮件在队列中的标记。“*”号则代表由于这个邮件正在被处理,所以已被锁定。“702”是邮件体的字节数。这里的时间则是邮件被放入队列的时间。地址则显示了发送者的名字。
第二行则可能显示出错的原因,这个邮件就是因为暂时无法分发,所以才暂存在队列中的。
第三行则可能显示接收者地址。
如果你想更完整、清楚地了解这里的输出,参见第23章:队列。
2.1.3 重建别名库模式 (-bi)
由于sendmail有可能需要在存放上千条别名记录的aliases文件中寻找别名,为了提高效率,可以使用dbm或db格式来存储。使用这种数据库格式来存储将大大提高检索速度。
尽管sendmail可以在aliases文件改变时自动更新数据库,但它并不总能及时完成。你可以通过运行newaliases命令或带“-bi”开关选项的sendmail来完成,以下这两个命令是相同的:
% newaliases
% /usr/lib/sendmail -bi
稍过一会儿,将显示出统计信息:
/etc/aliases: 859 aliases, longest 615 bytes, 28096 bytes total
这一行表示数据库已成功重建。从8.6版以后,就可以存在多个别名文件,所以每一行都是以别名文件名开始的。然后是显示处理了的别名,最大的一项的长度,总长度,同时有多少个出错也会显示出来。
关于aliases文件更详细的信息,参见第24章:别名。
2.1.4 校验模式 (-bv)
带开关选项“-bv”的sendmail是一个简单方便的检查别名的工具。它能够在别名中递归地查找,并显示出最终的用户名称。
假设aliases文件中有如下别名设置:
animals: farmanimals, wildanimals
bill-eats: redmeat
birds: farmbirds, wildbirds
bob-eats: seafood,whitemeat
farmanimals: pig, cow
farmbirds: chicken, turkey
fish: cod, tuna
redmeat: animals
seafood: fish,shellfish
shellfish: crab, lobster
ted-eats: bob-eats, bill-eats
whitemeat: birds
wildanimals: deer, boar
wildbirds: quail
虽然你也可以通过演算得知ted-eats最终的用户名称,但远不如使用sendmail来帮你完成那样方便。使用sendmail也将更加准确,而且对于很大很复杂的aliases文件来说,更加显得重要。
另外,sendmail –bv还有一个附加的功能,那就是可以检验别名是否真的可分发。假设在aliases文件中包含以下一行:
root: fred, larry
假定fred是一个拥有本地机器帐户的系统管理员,但用户larry已经离开,帐号已经被删除。你可以运行sendmail –bv检查所有的用户是否有效:
% /usr/lib/sendmail –bv root
这个命令将在aliases文件检查root用户,由于larry不存在,输出将会如下所示:
larry … User unknow
fred … deliverable: mailer local, user fred
2.2冗长模式(-v)
命令行开关选项“-v”,将使sendmail运行在冗长(verbose)模式下。在这个模式下,sendmail将会打印出转发邮件的每一步的详细说明。为了观察运行在冗长模式下sendmail的运行情况,可以执行:
% /usr/lib/sendmail –v you < sendstuff
邮件传送在本地进行,输出如下所示:
you … Connecting to loca…
you … Sent
当sendmail通过TCP/IP网络传送邮件到其它机器,它将使用一个叫SMTP(简单邮件传输协议)。为了观察使用SMTP的情况,我们再次运行sendmail程序,但这次,我们使用一个不在本地的E-mail地址代替“you”:
% /usr/lib/sendmail –v [email protected] < sendstuff
这个命令的输出看起来类似:
[email protected] … Connecting to remote.domain via smtp …
220-remote.Domain Sendmail 8.6.12/8.5 ready at
Fri, 13 Dec 1996 06:36:12 –0800
220 ESMTP spoken here
>>> EHLO here.us.edu
250-remote.domain Hello here.us.edu,pleased to meet you
250-EXPN
250-SIZE
250 HELP
>>> MAIL From:<[email protected]>
250 <[email protected]> … Sender ok
>>> RCPT To:<[email protected]>
250 <[email protected]> … Recipient ok
>>> DATA
354 Enter mail, end with “.” on a line by itself
>>> .
250 GAA20115 Message accepted for delivery
[email protected] … Sent (GAA20115 Message accepted for delivery)
Closing connection to remote.domain
>>> QUIT
221 remote.domain closing connection
以数字开头的行和以字符串“<<<”开头的行组成了SMTP的会话过程。我们马上说谈论一下它们。其它行显示本地sendmail尝试做的操作和成功完成的操作:
[email protected] … Connecting to remote.domain via smtp …
…
[email protected] … Sent (GAA20115 Message accepted for delivery)
Closing connection to remote.domain
第一行显示使用网络发送信件到远程主机remote.domain上。最后两行显示邮件已经发送成功。
在SMTP会话中,以“<<<”开始的行显示本地机器对远程机器的交谈。而来自远程机器的应答行则以数字开始的行。现在我们来看一下会话过程。
220-remote.Domain Sendmail 8.6.12/8.5 ready at
Fri, 13 Dec 1996 06:36:12 –0800
220 ESMTP spoken here
一旦sendmail与远程机器连接上后,sendmail就等待远程机器初始化会话。远程机器说它准备好发送,详细的主机名。如果远程主机也运行了sendmail,也将说sendmail的名字与版本。还有就是日期与时间。
第二行也以220开始,“ESMTP spoken here”的含义是远程站点能够使用扩展的SMTP协议。如果远程机器跑的sendmail是8.7或以上版本,ESMTP将可能会出现在第一行。
如果sendmail等待接收这个初始化信息太久,就会打印“Connection timed out”信息,并将这个邮件放入邮件队列中。
接着,本地sendmail发送EHLO(以>>>开始),传送扩展的HELLO信息,和本地主机名:
>>> EHLO here.us.edu
250-remote.domain Hello here.us.edu,pleased to meet you
250-EXPN
250-SIZE
250 HELP
在EHLO中的E说明本地sendmail也是使用ESMTP的。远程主机以250开始的回执ESMTP支持的服务列表。
如果本地机器发送EHLO消息时,传送的是短主机名(如here)就可能会遇到一个问题。远程主机无法得知这个短主机名的位置,因为它不在远程主机的域remote.domain中。这也就是为什么sendmail一直使用完整的主机名来表示。一个完整的主机名是由主机名加上一个点,然后再加上DNS域名项。
如果到现在为止,一切正常的话,本地机器将说明邮件的发件人:
>>> MAIL From:<[email protected]>
250 <[email protected]> … Sender ok
在此,发件人的地址是远程机器认可的。
下一步,本地机器将说明收件人的名字:
>>> RCPT To:<[email protected]>
250 <[email protected]> … Recipient ok
如果远程主机上并无用户you的话,远程主机将返回“User unknown”错误,在此,收件人OK。注意这里的OK不一定能确保地址是完好的。只是确认了这个地址是可接受的。
当信封信息发送完成后,sendmail程序将试图发送信件信息(包括信头和信体)。
>>> DATA
354 Enter mail, end with “.” on a line by itself
>>> .
DATA告诉远程主机准备好了。当远程主机指示发送信息时,本地主机照做。最后的一个点用来标记一个邮件结束。这是SMTP的规定。因为邮件消息可能包含多行,而用一个小点开始也是合法,所以sendmail将会把这些小点转换成两个再发送出去。例如,假定当我们要发送以下文件时:
My results matched yours at first:
126.71
125.72
…
126.79
But then the numbers suddenly jumped high, looking like
Noise saturated the line.
为了防止以小点开始的行照成疑义,sendmail将会在以小点开始的行,插入一个附加的小点,所以实际传输的内容如下所示:
My results matched yours at first:
126.71
125.72
.…
126.79
But then the numbers suddenly jumped high, looking like
Noise saturated the line.
而另一方收到邮件后,再将这个附加的小点去掉,还原成原来的邮件内容。
远程主机上的sendmail将显示队列管理附于的标识:
250 GAA20115 Message accepted for delivery
>>> QUIT
221 remote.domain closing connection
本地主机上的sendmail发送QUIT,说明全部工作完成。远程主机返回应答信息确认。
注意,-v参数在发送信息到远程主机上时十分有用。它能够显示出SMTP会话过程,以帮助我们了解邮件转发的过程,也有利于我们排错。
2.3调试模式(-d)
sendmail程序也能产生并输出调试信息。要使用调试模式运行sendmail的话,就需要使用-d参数。这个参数将产生比-v参数更多、更详细的信息。输入以下命令行,用自己的帐户名代替you:
% /usr/lib/sendmail –d you < /dev/null
这个命令行产生很冗长的处理信息。我们在此不打算说明这些输出信息,在此,只需记住在sendmail程序在调试模式下运行会产生大量的信息。
同时,也会产生大量的调试信息,你可以修改、显示这些调试信息。你可以在-d参数后加上一个数字,输出将会限制在只输出指定类的调试信息。
输入以下命令用自己的帐户名代替you:
% /usr/lib/sendmail –d40 you < /dev/null
在此,-d40是调试第40类的信息。这类的信息是关于邮件队列的。以下是一个输出实例:
>>>>> queueing GAA14008 (new id) queueall=1 >>>>>
queueing 95688=you:
mailer 4 (local), host ‘’
user ‘you’, ruser ‘<null>’
next=0, alias 95460, uid 0, gid 0
flags=6008<QPRIMARY,QPINGONFAILURE,QPINGONDELAY>
owner=(none), home=”/home/you”, fullname=”Your FullName”
orcpt=”(none)”, statmta=(none), rstatus=(none)
<<<<< done queueing GAA14008 <<<<<
对于一个类,还可以指定一个级别,这个级别是用来调整输出的量。一个低级别将产生较少的输出,一个高级别将产生更多、更复杂的输出。它们的格式是在-d参数之后加上:
category.level
例如:
% /usr/lib/sendmail –d0.1 -bp
参数-d0指示sendmail产生通用的调试信息。而级别1则让sendmail的输出减少到最小限度。这个值可以省略,因为.1是缺省值。参数-bp让sendmail打印出邮件队列内容。输出看起来如下所示:
Version 8.8.4
Compiled with:LOG NAMED_BIND NDBM NETINET NETUNIX NIS SCANF
XDEBUG
= = = = = = = = = = = SYSTEM IDENTITY (after readcf) = = = = = = = = = = =
(short domain name) $w = here
(canonical domain name) $j =here.us.edu
(subdomain name) $m =us.edu
(node name) $k = here
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
Mail queue is empty
在此,开关选项“-d0.1”让sendmail打印出版本号、一些关于编译的信息,以及你的主机名。现在你提高级别看一下:
% /usr/lib/sendmail –d0.11 -bp
这时将显示如下信息:
Version 8.8.4
Compiled with:LOG NAMED_BIND NDBM NETINET NETUNIX NIS SCANF
XDEBUG
OS Defines: HASFLOCK HASGETUSERSHELL HASINITGROUPS HASLSTAT
HASSETREUID HASSETSID HASSETVBUF HASUNAME IDENTPROTO
IP_SRCROUTE
Config file: /etc/sendmail.cf
Pid file: /etc/sendmail.pid
Canonical name: here.us.edu
UUCP nodename: here
a.k.a.: [123.45.67.89]
= = = = = = = = = = = SYSTEM IDENTITY (after readcf) = = = = = = = = = = =
(short domain name) $w = here
(canonical domain name) $j =here.us.edu
(subdomain name) $m =us.edu
(node name) $k = here
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
Mail queue is empty
[目录]
sendmail.cf
3. sendmail.cf文件
在sendmail的配置文件sendmail.cf中的文本,有的形如MODEM的噪声,有的形如Dithers的咒语一样:
R$+@$=w ? sendmail.cf文件
{$/{{.+ ? modem的噪声
!@#!@@! ? Dithers的咒语
对于生手而言,要构建象下面的配置文件,一定是一件恐怖的事情:
R$+@$=W $@$1@$H user@thishost -< user@hub
R$=W!$+$@$2@$H thishost!user -< user@hub
R@$=W:$+ $@@$H:$2 @thishost:something
R$+%$=W $@$<#$1@$2 user%thishost
不过,如果回想起以前学习C语言时,你是否会对这个表达式感到恐惧呢?
# define getc(p)(--(p) -<_cnt<=0? ((int)*(p)-<ptr++):_filbuf(p))
就象任何一个新语言一样,学习sendmail.cf文件中所使用的语言需要时间和实践。在本章中,我们将介绍这个语言,不会在此只是对其作一个入门性的概述。
3.1概要
sendmail.cf文件是sendmail每次启动时要读取的配置文件。它包含了sendmail启动时必须的信息。它列出了所有重要文件的位置,指定了这些文件的缺省权限。包含了一些影响sendmail行为的选项。更重要的是,它还包含了地址重写(rewriting addresses)规则。
Sendmail.cf文件是按行组织的。一个配置命令行,均是由字符开头的,而且每行只有一个命令:
V7 ? 正确的
V7 ? 不正确,前面多了个空格
V7 Fw/etc/mxhosts ? 不正确,一行中有两个命令
Fw/etc/mxhosts ? 正确
每一个配置命令行,是由命令加上特定参数构成的。例如,命令V的参数是一个数字,而F命令的参数则是字母w,再加上绝对路径。
命令 说明
V 定义配置文件的版本(从8.6版开始才要求)
M 定义一个邮件传送代理
D 定义一个宏
R 定义一个地址重写规则
S 声明一个规则集
C 定义一宏集
F 从一个文件与管道中定义一宏集
O 定义一个选项
H 定义一个信头
P 定义传送优先级
T 声明受托用户(在8.1版忽略,8.7版重用)
K 声明一个key字数据库(从8.1版开始使用)
E 定义一个环境变量(从8.7版开始使用)
L 包括扩展的负载均衡支持
有一些配置命令,象V只会在sendmail.cf文件中出现一次,而有些象R命令就会多次在配置文件sendmail.cf中出现。
空行、以“#”开始的行将被处理为注释行而被忽略。一个以tab开始的行,则说明是上一行的继续,如:
# a commnet ? 注释行
V7
/Brekeley ? V7命令的继续
-
tab
除了一个命令,一个空行,一个空格,一个tab,或一个#字符以外,其它情况都是错误的。如果sendmail程序发现这种情况,将打印出如下警告,并忽略此行,然后接着读取后面的配置:
sendmail.cf : line 15:unknown control line “v6”
在这,sendmail找到了一行以v开始的行。由于小写的v不是一个有效的命令,sendmail将发出警告。而行号则指出了这个错误所在行。
下面几个小节将对每一种命令进行简单的实例说明。而这些命令的作用将会在本教程中阐述。所以如果在本节中无法完全理解的话,也不必担心。因为这里所有神秘的东西,在本书结束时都将变得十分清晰明白。
3.2最小配置
最小的配置的文件可以是空文件。你可以使用以下命令来创建这个文件:
% cp /dev/null client.cf
我们会慢慢地往这个文件中添加配置。将其命名为client.cf是为了避免覆盖了系统中的sendmail.cf文件。
现在,我们再运行sendmail,测试这个新配置文件的有效性:
% ./sendmail –Cclient.cf –bt >/dev/null
%
命令行开关选项-C用于指定sendmail使用一个指定的配置文件。而开关选项-bt则告诉sendmail运行在rule-testing模式下。注意,sendmail读取你的空配置文件,运行,没有任何提示。同时注意当在第2章编译完后无法运行sendmail,但现在你可以了。那是因为当时你没有配置文件,而现在有一个了(尽管这个文件是空的)。
3.2.1 版本
为了防止旧版本的sendmail因读取新版本的配置文件而破坏,在sendmail 8.1开始引入了一个V命令,这个命令的格式如下:
V7
编辑文件client.cf,然后加上这一行。“V”必须位于行首。后面跟的版本号必须是7,才能够使所有8.8版的sendmail.cf中的新功能生效。数字7是sendmail.cf的语法,表示有7个主要的变化。
3.2.2 注释
注释语句能够帮助其他人理解你的配置文件。同时,他们也能通过注释记起你几个月前的修改。注释语句对sendmail的执行速度影响极小,所以你无需担心这一问题。前面我们说过,以#开始的行,被sendmail认为是注释行,整行将被忽略。例如:
# This is a comment
另外,注释语句也可以在命令之后,如:
V7#This is another comment
增加一些注释语句到你的sendmail.cf文件中,使其成为:
# This is a comment
V7# This is another comment
Sendmail程序读取这个配置文件也将没有任何提示。
% ./sendmail –Cclient.cf –bt >/dev/null
%
3.3快速导览
你将发现,在配置文件中的其它命令远比V命令来得复杂。这儿,我们现在对每个命令作一个快速的导览,仅够你能够对它们有一个初步的了解。
3.3.1 邮件传送代理
通常情况下,sendmail程序不自己传送邮件,而是调用一个程序来完成。M命令就是定义一个邮件传送代理。例如,就象以前我们看到过的:
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40,
这将告诉sendmail,本地的邮件使用/bin/mail传送。这一行中的参数将在第6章:邮件中枢与传送代理,第30章:传送代理中详细说明。
3.3.2 宏
义一个值,然后可以在多次使用,提高sendmail.cf的可维护性。D命令用于定义宏。一个宏名可以是一个单字母或用大括号包起来的多个字符。定义一次后,就可以在其它地方使用。
DRmail.us.edu ? 一个单字符
D{REMOTE}mail.us.edu ? 用大括号包含起来的多个字符(从8.7版开始)
在这,R和{REMOTE}都是宏名,值是mail.us.edu。这个值可以在任何地方用$R和${REMOTE}访问。宏将在第7章:宏中介绍,更详细的在第31章:定义宏中。
3.3.3 规则
sendmail.cf文件的核心是一系列的地址重写规则。这是非常重要的,因为地址必须符合多种标准。R命令用来定义这些规则:
R$- $@ $1 @ $R user -< user @ remote
邮件地址将与最左边的规则($-)进行比较。如果与这个规则匹配,它们将根据右边的规则($@ $1 @ $R)进行重写。而在最右边的文本则是注释(注意,这里不需要使用#号)。
如果使用多字符的宏名,并用#号标出注释的话,将使语句减少一些神秘:
R$- # If a plain user name
$@ $1 @ ${REMOTE} # append “@” remote host
3.3.4 规则集(Rule Sets)
由于地址重写可能需要好几步,规则可以组织成为规则集,用S命令开始规则集:
S3该命令定义规则集3,从sendmail 8.7版开始,规则集也可以使用字符来命令,如:
SHubset
该命令定义Hubset规则集,这种方式的命名,sendmail将会自己对其编号。
所有的跟在S命令后的R命令(规则)组成规则集。一个规则集结束于定义另一个规则集的S命令。
3.3.5宏集(Class Macros)
用D命令定义的宏只能有一个值,但这通常是不够的。我们经常需要定义一个拥有多个值的宏,然后就像数组一样来组织这些值。C命令定义一个宏集。一个宏集就像一个数组一样,能够有多个项。宏集的名字是一个单字母,从8.7版开始,也可以用大括号包含多个字符作为名字,例如:
CW localhost fontserver ? 一个单字符作为名字—W
C{MY_NAMES} localhost fontserver ? 多个字符作名字---{MY_NAMES}
在这里,每一个宏集都包含两个值:localhost和fontserver。我们可以通过表达式$=W和$={MY_NAMES}来访问这些宏集。有关于宏集的更多信息,可以参考第12章和第32章。
3.3.6 文件宏集(File Class Macros)
为了管理更加容易,我们常将比较长的信息或经常变化的信息存放到一个文件中去。命令F可以定义一个文件宏集。这个宏集的值是这个文件的内容,如:
FW/etc/mynames
F{MY_NAMES}/etc/mynames
在此,文件宏集W和{MY_NAMES}将从文件/etcmynames中获得它们的值。
文件宏集也可以从一个程序的输出中得值。它定义为:
FM|/bin/shownames
F{MY_NAMES}|/bin/shownames
在此,sendmail将运行程序/bin/shownames,这个程序的输出将成为文件宏集的值。
3.3.7 选项
选项将告诉sendmail程序许多有用的和必要的事情。它们指定key文件的位置,设置超时时间,以及定义sendmail在出错时如何处理。它可以调整sendmail,以使它符合你特定的需要。
命令O用来设置这些选项。以下就是一个例子:
OQ/var/spool/mqueue
O QueueDirectory=/var/spool/mqueue
在此,Q选项定义邮件队列文件为/var/spool/mqueue。
3.3.8 信头
邮件消息由两个部分组成:一个是信头部分,另一个是在空行后的主体部分。主体部分可以包括任何内容。而信头部分,则需要严格按照标准。H命令用来指定信头的格式:
Hreceived:$?sfrom $s $.by Sj ($v/$Z) $?r with $r$. Id $I$?u for $u$.; $b
这个特定的H命令告诉sendmail收到后,必须在每一封信的信头加上这一行。
3.3.9 优先级
并不是所有的邮件具有相同的优先级。邮件列表的信应该在只有一个收件人的信的后面发送。P命令用于设定邮件的优先级。这个优先级用于邮件队列处理时决定邮件的顺序。
Pjuck=-100
这个P命令告诉sendmail,信头中用juck的信最后处理。关于这个命令的更详细内容,参见第14章和第35章。
3.3.10 受托用户
为了使某些软件(如UUCP)能够正确地生效,就必须能够告诉sendmail邮件是谁发来的。这要求软件与From:中指出的用户运行在不同的uid上。T命令列出这些用户的受托用户。所有包含在信头的其他用户将收到一个警告。
Troot daemon uucp
这个T命令指出有三个用户是受托用户。它们是root(UNIX系统中的上帝),daemon(sendmail通常以伪用户daemon身份运行),以及uucp。
3.3.11 key数据库
一些认证信息,例如UUCP主机列表等,最好在sendmail.cf外维护。外部的数据库(叫作keyed数据库)提供了更快的访问速度。Key数据库在8.6版中引入了几种格式,如:
Kuucp hash /etc/mail/uucphosts
这个K命令声明一个key数据库叫uucp,类型是hash,文件为/etc/mail/uucphost。
3.3.12 环境变量
sendmail程序的安全是十分重要的。一种破坏的的方法是运行时伪造一些环境变量。为了阻止它,8.0版的sendmail将在启动时去除所有的环境变量。然后根据预先设置的值进行设置。设置环境变量使用E命令。
EPOSTGRESHOME=/home/postgres
在此,环境变量POSTGRESHOME被赋予值/home/postgres。这个程序使用postgres数据库访问信息。
[目录]
邮件中枢与分发代理
6. 邮件中枢与分发代理
使用一个强大的中心机器处理所有的邮件,比让网络中所有的工作站自己处理自己的邮件好得多。这样的一个中心机器就叫作邮件中枢(Mail hub),它就象美国联邦快运公司处理包裹一样工作。在过去,当你通过联邦快运公司将包裹从旧金山寄到巴黎,包裹将首先被送到孟菲斯的田纳西,甚至你想寄到洛杉机,它也是先送到孟菲斯(参见图6-1)。这是因为孟菲斯是联邦快运公司的处理中心。所有的包,不管它们从哪来,到哪去,都先送到处理中心(中枢,hub),然后再从那里发送出来。
这种处理方法的好处就是,联邦外运公司只有处理中心需要知道如何发往全世界(当然,联邦快运是有多个处理中心的),而其它地方的分支机构只要知道如何发往处理中心就可以了。
类似地,你的工作站也可以将自己看成分支机构(客户机),将处理邮件的中心机器看成处理中心。这个邮件中枢将为整个机构处理邮件,这样做有许多好处:
1) 所有发往这个机构的信都发给邮件中枢,而不直接发往客户机。由邮件中枢来处理有几个优点:客户机无需运行sendmail守候进程来侦听邮件。而且客户机的名字无需告知外部,因此更容易安全地将客户机与Internet隔离。
2) 所有从机构往外发的邮件都先发给邮件中枢,然后由邮件中枢转发到最终目的地,而不是由客户机直接向外发送。这个机制可以使客户机无须一直观察Internet的变化;所有的待发邮件都保存在邮件中枢中,而不是客户机处,这样管理起来更简单;而且一个单一的sendmail.cf文件就可为所有的客户机服务。
3) 所有发出去的邮件都将被修改,使得其看起来象来自邮件中枢的邮件。这样回复的邮件就不会直接返回到每一个客户机。这样做,所有的邮件好象都来自一台机器,一台很大的机器。
4) 所有发往本地用户的邮件都将传送到邮件中枢,存放在spool中。而无需每台机器都建一个spool目录来存放收到的邮件。这样做的优点是所有本地的邮件都由一个机器处理,所有本地机器都无需进行处理。而且这样管理spool也更加容易。
但如果你的机构的电脑网络是不同体系结构的电脑组成的,或散在许多网络中,使用邮件中枢就不是太好了:
1) 如果一个机构经常有数量巨大的邮件收发,那么这台邮件中枢就可能负载过大,以使所有的邮件都被缓存,一直等待发送。如果负载厉害的话,甚至可能使本地邮件转发也会出现这样的现象,那是十分可怕的。
2) 为了让邮件中枢正常工作,它需要知道系统中所有用户的登录名,也就是意味着需要一个主/etc/passwd文件,它是所有主机上的/etc/passwd的集合,或者使用NIS系统来构建。
3) 因为所有的邮件都通过邮件中枢传送,而不是直接送给接收者,这就意味着一定会延迟。对于一些小的机构,这个延迟可以忽略不计,而让邮件的总量上去时,这会成为一个严重的问题。
4) 如果一个客户机是直接使用UUCP连接,或是与多个网络连接,那样配置文件会变得十分复杂。
4.1 client.cf文件
这个教程的目的是产生一个简单的邮件中枢的sendmail.cf文件。为了区别与前面几个章节中的sendmail.cf文件区分开来,在此我们把它称为client.cf文件。我们将在后面的几个小节中逐段地生成这个文件,并做一些相应的说明。本章中生成的文件将完成两个任务:
1) 它将指示sendmail运行为邮件中枢,发送所有给其它机器的mail服务器。
2) 它将使所有外出的邮件表现为从邮件中枢发出的一样。
在这章中,我们主要处理第一个任务:如何让客户机上的邮件送到邮件中枢。
4.2 定义一个邮件分发代理
sendmail除了在TCP/IP网络转发邮件外,并不能自己处理邮件传送。而是调用一个程序来实现。这个程序就叫做分发代理。哪个代理程序处理哪类邮件分发任务在sendmail.cf文件中定义。
加入你的client.cf文件中的第一项是定义负责将信件转发给邮件中枢的转发代理程序。最初,sendmail只需要转发代理程序的名字与路径。设置形如:
# This is a comment
V7 # this is another comment
# Delivery agent definition to forward mail to hub
Mhub, P=[IPC], A=IPC $h
这是一个最小的配置文件。将这些行加入到client.cf文件中。配置的第二行定义了一个分发代理。它包含三个部分,每个部分使用逗号隔开。
Mhub
以M开始的行用于定义邮件分发代理。紧跟着在后面的hub是分发代理的标识名。
这个标识名供你以后写的配置文件部分参考。
P=[IPC]
P=用于设置路径。这一句配置用来指出处理邮件分发的程序的全路径名,在这里使用的是[IPC]。([IPC]是一个Sendmail内部使用的专有名称,说明它有能力在TCP/IP网络上完成邮件分发工作。)而对于其它情况下,则设置为其转发程序的名称,如/bin/mail等。
A=IPC $h
A=后面跟着参数列表,用来指定P=指定的程序运行时的参数。第0个参数就是程序的名称(在此就是IPC,注意没有包含括号)。而其它的参数就是在A=后面指定的,这此例中仅有一个:$h。这个$h是一个宏,表示接收者的主机名。我们会在第7章中更详细地说明宏。它通常是关于分发代理定义的最后一项。
4.2.1 测试client.cf文件
在命令行下执行:
% /usr/lib/sendmail –oQ/tmp –Cclient.cf –bp
注意,你需要在命令行中加上-oQ/tmp,它将阻止sendmail改变mqueue目录。如果你忽略这个开关,sendmail会出现混乱、错误:
cannot chdir((null)):Bad file number
而-Cclient.cf开关则告诉sendmail使用当前目录中的client.cf文件,而不是系统中的/etc/sendmail.cf文件。而-bp开关则是让sendmail打印出队列的内容。以上命令行将产生如下所示的输出:
No local mailer defined
然而,缺少一个local mailer并没有什么大不了的。Sendmail程序在输出它后,就结束了,说明没有发现其它错误。
4.3 本地分发代理
当你运行了sendmail,它曾抱怨没发现本地分发代理定义。为了让它更快乐,我们就将下面的定义加入到client.cf文件中去。你可以使用以下命令,从/etc/sendmail.cf中抄过来:
% grep “^Mlocal” /etc/sendmail.cf << client.cf
注意,^M是真实的两个字符:^和M,而不是CTRL-M。现在,
我们启动编辑器,然后调入client.cf文件,你将看到类似下面的内容:
# This is a comment
V7 # this is another comment
# Delivery agent definition to forward mail to hub
Mhub, P=[IPC], A=IPC $h
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40, A=mail –d $u
你将注意到有三个新的等式,它比我们原来的hub等式复杂。这个新的M配置命令声明一个符号名,就象hub一样。在这里,我们用的名字是local。尽管,local定义对于一个真正在运行的sendmail.cf配置来说是十分重要的,而在此呢,我们仅仅是为了让sendmail不抱怨。
这个新的分发代理定义是由六个部分组成的(每个部分使用逗号隔开),一个符号名和五个等号。其中F=、S=和R=是新出现的。而关于M,P=和A=都在介绍hub那一节中介绍过了。
M 所有的邮件分发代理的定义都是从M开始的,如:
Mhub, P=[IPC],A=IPC $h
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40, A=mail –d $u
分发代理的符号名就紧跟在M命令之后,中间没有空格。在以上的例子中,符号名是hub和local。名叫hub的分发代理用于将邮件转发给邮件中枢。而local分发代理则将邮件分发给本机的用户。
P=在等号后面指出邮件分发程序的全路径名:
Mhub, P=[IPC],A=IPC $h
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40, A=mail –d $u
你的程序名可能与这里不同,但通常,local邮件分发代理都是将邮件放入用户的邮件spool文件中。
A= 在等号后面指定这个邮件分发代理运行时的参数。
Mhub, P=[IPC],A=IPC $h
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40, A=mail –d $u
注意,在local中使用了$u宏,而hub则使用了$h宏。$u宏包含接收者的名字(如bob),而$h宏则是包含接收主机的名字(如here.us.edu)。宏将在下一章说明。习惯中,这个部分一般放在最后。
以下三个部分是在local定义中新出现的,在hub的定义中并未使用。
F= 在等号后面指定确定的标记,用来告诉sendmail更多关于分发代理的东西。每一个标记都是单个字符(要么设置,要么不设置)。
Mhub, P=[IPC],A=IPC $h
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40, A=mail –d $u
这儿有许多标志可供选择。这些标志将在第30章分发代理中说明,而在后面的几章中也会说明一些。
S= 在等号后指定使用哪一个规则集来重写发送者地址:
Mhub, P=[IPC],A=IPC $h
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40, A=mail –d $u
由于不同的分发代理使用的地址格式不尽相同,所以有时需要重写发送者地址。
例如,[IPC]代理使用[email protected]的格式,而uucp代理则使用host!user格式。
再此,指定了分发代理使用第10规则集来重写发送者地址。我们将在第8章:地址和规则中详细地说明。
R= 在等号后面指定使用哪一个规则集来重写发送者地址:
Mhub, P=[IPC],A=IPC $h
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40, A=mail –d $u
同样的道理,我们需要重写接收者地址。在此,指定使用规则集20重写信封地址,使用规则集40重写信头地址。
而从sendmail 8.7版开始,引入了一个新的定义:
T= 这用来指定一些相关的信息,如:
Mlocal, …, T=DNS/RFC822/X-Unix
第一个信息是MTA使用的(这儿是DNS,因为sendmail是使用DNS来查寻地址的);接着在“/”之后的第二个信息是地址使用的(这儿是RFC822,也可以是X.400);最后是错误消息类型(这儿是X-Unix,说明/bin/mail将产生Unix的错误)。
4.3.1 略过规则集
由于我们有时并不需要转换规则集,这很简单,我们只需将S=和R=后面的规则集做如下的修改就可以了。
Mhub, P=[IPC],A=IPC $h
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=0, R=0, A=mail –d $u
规则集设为0,代表不使用规则集。
4.3.2 增加注释
注释对于每一个配置文件都是十分重要的部分,它将提醒你现在在做什么,以及以前做过什么。现在编辑client.cf文件,拿掉两行原来的注释,加入一行新的注释,使其内容为:
V7
# Delivery agent definition to forward mail to hub
Mhub, P=[IPC], A=IPC $h
# Sendmail requires this,but we won’t use it.
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40, A=mail –d $u
我们拿掉的那两行注释语句是没有意义的两名,它们只是演示用的,并无法起到注释的作用。
4.3.3 增加注释
现在,我们采用不同的方式运行sendmail:
% /usr/lib/sendmail –d0.15 –Cclient.cf –bt
-d0.5是一个调试开关,它告诉sendmail显示你定义的分发代理是如何处理的。而-bt使得sendmail以规则测试模式来运行。以上命令将产生如下的输入。
mailer 0 (prog):P=/bin/sh S=0/0 R=0/0 M=0 U=0:0 F=Dlos L=0 E=
T=DNS/RFC822/X-Unix A=sh –c $u
mailer 1 (*file*):P=[FILE] S=0/0 R=0/0 M=0 U=0:0 F=DEFMPlosu L=0 E=
T=DNS/RFC822/X-Unix A=FILE
mailer 2 (*include*):P=/dev/null S=0/0 R=0/0 M=0 U=0:0 F=su L=0 E=
T=
这个输出,将彻底地解释client.cf文件,通过M=、U=、L=和E=提示出来。
在前面的输出中,大家可以发现有几个等式并未在定义中定义。如hub的定义中只包含了A=和P=:
Mhub, P=[IPC], A=IPC $h
而当sendmail看到这个定义,将使其复杂化,加上了E=(换行符),T=
mailer 3 (hub):P=[IPC] s=0/0 R=0/0 M=0 U=0:0 F= L=0 E=
T=
注意,当F=0时,就表示空列表,没有定义标记。
4.4 为Mhub增加缺少的部分
最后一步,我们为Mhub增加它缺少的那部分:F=、S=、R=和T=。
编辑client.cf文件,然后增加Mhub那么,使其成为:
# Delivery agent definition to forward mail to hub
Mhub, P=[IPC], S=0, R=0, F=mDFMuXa, T=DNS/RFC822/SMTP, A=IPC $h
在此,我们让S=和R=都为0.S=用来指定重写发送者地址的规则集,而R=用来指定重写接收者的地址规则集。然而在此不需要地址重写,因此被赋予0值。
而T=部分则与local的定义类似,唯一不同的是,它使用SMTP代替了X-Unix,即local是报告UNIX的错误消息,而hub则是SMTP错误消息。关于这一点,我们在后面会专门说明。
而标志列表F=mDFMuXa,是最典型的设置。你可以根据自己的需要修改。所有的可用的标志在第30章中有详细说明。以下是一个概括性的说明表。
标志说明
m 这个代替能够同时为超过一个用户分发
D 在信头中包括Date:(日期)
F 在信头中包括From:(信从哪来)
M 在信头中包括Message-ID:(消息ID编号)
u 保持接收者姓名
X 遇到单独的点,变为“..”(两个点)
A 运行扩展的SMTP协议
关于邮件分发代替的定义,就简单地说到这里。
[目录]
宏
5. 宏
在sendmail.cf文件中有一个十分重要的组成部分,就是那些使用代表文本的符号。这与bsh和csh中的变量的使用极为类似:
REMOTE=mailhost (bsh)
set REMOTE=mailhost (csh)
D{REMOTE}mailhost(sendmail.cf)
以上几个语句都是定义了一个名为REMOTE的变量,被为其赋值为mailhost。
而要使用变量中存储的值,可以使用以下表达式:
$REMOTE (bsh)
$REMOTE (csh)
${REMOTE}(sendmail.cf)
也就是说,以上表达式将得到存储在变量REMOTE是值,在这个例子中就是文本字符mailhost。一经定义了REMOTE的值为mailhost,你就可以在任何地方使用以上表达式来代替文本字符mailhost。
5.1 概要
宏能极大地简化您的工作。它允许你在一个集中的地方定义一些文本符号。而你只需修改这里的文本字符,则改变将自己传播到文件中其它部分。例如,假设我们在sendmail.cf文件中定义了:
D{REMOTE}mailhost
如果你在sendmail.cf文件的其它地方都使用${REMOTE},则只需简单修改定义D{REMOTE}mailhost中的字符串,就将使所有表达式${REMOTE}的值改变。
宏定义的语法格式如下:
DXtext
必须使用字符D开始。这个字母D的后面紧跟着宏的名字(在这就是X),注意中间并没有空格隔开。接着在宏名的后面紧跟着值,注意在宏名与值之间也没有用空格隔开。这个值从宏名开始,直到本行结束。
宏名可以由一个字符组成,也可以由多个字符组成。如果你使用多个字符做为宏名,就必须使用“{ }”把它们括起来。如果是使用单个字符做宏名,则也应该使用“{ }”包括起来,而在V8.7之前也可以不使用“{ }”。
DRmailhost (sendmail V8.7以前版本)
D{R}mailhost
D{REMOTE}mailhost
通常,当读取配置文件的时候就会自动展开宏。因此,你必须在使用宏之间定义它们:
D{ROLE} son
S${ROLE}(此时的值是son)
D{ROLE}mother
S${ROLE}(此时的值是mother)
在这里,ROLE首先被赋值为son,当处理到第一个S命令时,sendmail会将${ROLE}替换为前面定义的值:son,因此得到:
Sson
而后,配置文件又将ROLE的值修改为了为mother,所以在处理到第二个S命令时,sendmail会将${ROLE}替换为ROLE的值mother,因此得到:
Smother
不过请注意这是一个十分不好的风格,通常,为了不产生混乱,每个宏应该只被定义一次。
5.2 定义宏
在上一章里,我们粗略了解了分发代理hub和local本地分发代理的定义。我们可以看到,在这两个分发代理的定义中的最后一部分“A=”都使用到了宏。
V7
# Delivery agent definition to forward mail to hub
Mhub, P=[IPC],S=0,R=0,F=mDFMuXa,T=DNS/RFC822/SMTP, A=IPC $h
# Sendmail requires this,but we won’t use it.
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40, A=mail –d $u
宏有两种:一种是你自己定义的,而另一种则是sendmail定义的。你自己定义的宏,宏名一定要以大写字母开始;如果宏名是以小写字母开始的,如h、u,则是sendmail定义的宏。
你已经看到过一个以大写字母开始的宏:
D{REMOTE}mailhost
在你的网络中,邮件中枢机器会使用了mailhost或类似的(如mailrelay)的别名,然而有时则不存在这样的别名。此时你必须使用它的实际机器名(如mail.us.edu)。现在,我们编辑client.cf,然后加入第一个宏{REMOTE}。
V7
# Defined macros
D{REMOTE}mailhost # The name of the mail hub
# Delivery agent definition to forward mail to hub
Mhub, P=[IPC],S=0,R=0,F=mDFMuXa,T=DNS/RFC822/SMTP, A=IPC $h
# Sendmail requires this,but we won’t use it.
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=10, R=20/40, A=mail –d $u
在此,我们在client.cf文件中新增了三行。第一行是一个注释语句,第三行则是一个空行,有来可视地分隔开宏定义部分与分发代理定义部分。第二行是新的宏定义。正如注释中说的,这个{ROMOTE}宏将包含邮件将转发到的机器名。
现在我们花一些时间来测试一下这个新的client.cf文件:
% ./sendmail –Cclient.cf –bt </dev/null
sendmail程序将读取并分析client.cf文件。由于在这个配置文件中并没有错误,所以sendmail将不会打印错误信息。
5.3预定义宏
在sendmail中,有一些内置的宏。你已经在分发代理的定义中看到过了u(接收者的用户名)和h(接收者的主机名)。它们无须在配置文件中定义,它们是sendmail定义的。这种宏,我们称之为预定义宏。下表中列出部分预定义宏:
宏名 描述
n 发送者错误消息标志符
v 当前运行的sendmail的版本
w 短主机名
j 规范的主机名
m 域名
k UUCP节点名
b RFC1123格式的日期
_ 身份鉴别信息
opMode 当前操作模式(在V8.7版之后才有)
在运行sendmail时,只需加上-d35.9参数,就可以显示出所有宏的定义:
% ./sendmail –d35.9 –Cclient.cf –bt </dev/null
尽管client.cf文件如此小,但这个命令的输出惊人地长:
define(* as $*)
define(+ as $+)
define(- as $-)
define(= as $=)
define(~ as $~)
define(# as $#)
define(@ as $@)
define(: as $:)
define(> as $>)
define(? as $?)
define(| as $|)
define(. as $.)
define( [ as $[)
define(] as $])
define(( as $()
define() as $))
define(& as $&)
define(0 as $0)
define(1 as $1)
define(2 as $2)
define(3 as $3)
define(4 as $4)
define(5 as $5)
define(6 as $6)
define(7 as $7)
define(8 as $8)
define(9 as $9)
define(n as $MAILER-DAEMON)
define(v as 8.8.4)
define(w as here.us.edu)
define(j as here.us.edu)
define(m as us.edu)
define(k as here)
define(b as Fri,13 Dec 1996 07:11:47 –0700 (PDT))
define(_ as you@localhost)
define(opMode as t)
redefine(w as here)
define(REMOTE as mailhost)
不同版本的sendmail的输出将不同。例如,l、o和e在V8.7以前的版本中存在,而在8.7以后的版本中就没有了。前面27行显示出了sendmail为operators(操作者,将在下一章中具体说明)预留的宏。而后面的11行则是我们现在所感兴趣的。这11行中,前10行都是senmail程序预定义的,最后1行则是用户自己定义的。
这些输出可以显示出另一个概念。一些内部宏是在读取配置文件之前定义的,你可以在client.cf文件中改变。例如我们要改为w,则只需在配置文件中加入一行:
Dwmyhost.my.domain
使用了这行配置文件后,宏w的值就从here.us.edu变成了myhost.my.domain了。
注意,在以上输出中并没有显示h和u宏。这些宏,尽管在内部定义了,但它们是直到邮件发送后才被赋予实际的值的。你不能够在你的配置文件中修改它们,因为它们在配置文件被读取之后才定义的。
最后,请注意只有V8的sendmail会为宏设置适当的缺省值。所有老版本的sendmail需要你手动地在client.cf文件中设置e、l、n、o以及q的缺省值。
5.3.1 主机名
本地主机名由两部分组成。主机名部分就是机器名,不包括“.”(如here)。而域名部分则至少有由“.”隔开的两个部分(如us.edu)。而全规范名(fully qualified)则是由主机名加上“.”,然后再加上域名构成。(例如:here.us.edu)。全规范名才能唯一标识一台机器。
本地机器的主机名也需要规范化。一台机器可以有多个名字(如mailhost或printserver等等),但只能有一个是正式的。你可以执行hostname命令来获取这个正式的名字:
% hostname
here.us.edu
下面我们就来看一下sendmail是如何解释你的本地主机名。我们以-d0.1的参数来执行sendmail:
% ./sendmail –Cclient.cf –d0.1 –bt </dev/null
version 8.8.4
Compiled with:LOG NAMED_BIND NDBM NETINET NETUNIX NIX SCANF XDEBUG
= = = = = = = = = SYSTEM IDENTITY (after readcf) = = = = = = = = =
(short domain name) $w = here
(canonical domain name) $j = here.us.edu
(subdomain name) $m = us.edu
(node name) $k = here
= = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = =
这里经常容易出错。例如,如果你并未与Internet连接,你可能就不会有一个合法的域名。或者你可能运行NIS而使得在hosts.byname映射中只有短主机名。一个很重要的事情就是观察以上输出中,sendmail是否正确地发现了你的短主机名和规范主机名:
(short domain name) $w = here
(canonical domain name) $j = here.us.edu
(subdomain name) $m = us.edu
如果这里有错误,你需要修复引起这个错误的系统问题。在系统级,调查以下问题:
1)/etc/hosts文件。你可能在这个文件中仅列出了短主机名。对于sendmail 8.7以前的版本,它需要你在这个文件的最开始一行列出全规范主机名。而从sendmail 8.7开始,新版的sendmail只需要在这个文件中存在全规范主机名就可以了。
2)nis映射。确认主机名能够被DNS查寻。
3)nsswitch.conf或svc.conf文件。确定正确的主机名查寻方法在此列出。
如果你无法发现,或没有足够的权限去做系统级的事情,你可以在你的client.cf文件中指定。例如,我们假设全规范名显示成了:
(canonical domain name) $j = here
为了修复这个错误,你可以在client.cf文件中重新定义宏j,也就是在client.cf中加入:
Dj$w,$m # The local official domain name
这个定义使得全规范名变成了主机名加“.”再加域名。这时,你可以再让sendmail显示一下,肯定会出现:
(canonical domain name) $j = here.us.edu
[目录]
规则
6. 地址与规则
在研究到规则的内部工作原理之前,我们需要新建一个虚构的网络以提供一个公用的环境,以支持我们要谈论的mail地址。
6.1 一个虚构的网络
我们拥有一个如图8-1所示的网络。它包括三个站点(每个云图表示一个站点),三个站点之间使用高速网络相连(有线连接)。每个站点都是一个由许多独立的计算机组成的一个DNS域。每一个域的设置都不相同,但从用户角度来看,发给一封信给本域内的用户与发一封信给其它域用户并没有什么区别。
图8-1 在我们这个虚拟网络中的域分布
一个域名是从右往左解释的。例如,acme.com是指acme是顶级域名com(供商业组织使用)的一部分。类似的,edu是供教育机构使用的顶级域名,gov是供政府组织使用的顶级域名。
一个域可以包括许多机器。每一台域内的机器都有一个由主机名、“.”、域名组成的全规范域名。例如,sec.acme.com就是一个全规范域名,它包括主机名sec,一个点,以及域名acme.com。
主机和域名是不区分大小写。就是说,sec.acme.com与下面几种写法完全等价:
SEC.Acme.COM
sEc.aCmE.cOm
6.1.1 域dc.gov
图8-2显示了dc.gov域的内部结构。它是由三台主机组成(在真实世界可能会由更多台主机组成)。这三台主机互相之间使用私有网络(有线连接)连接起来,以保证其安全性。仅有一台主机:fbi.dc.gov拥有与外部网络的连接。所有从外面进来的邮件,将首先到达fbi.dc.gov,然后由fbi.dc.gov转发给内部网内的适当的主机上。例如:用户george最终还是在他自己的机器wash.dc.gov上收到他的邮件,即便邮件是从fbi.dc.gov转给他的。在这种情况下,fbi.dc.gov所称为网关(gateway),因为它就象是一个在内部网与外部网之间的一个门;它也可以被称为转发器(forwarder),因为它从外部网接收所有给内部机器的信件,然后转发给他们。
图8-2 整个dc.gov域只有fbi.dc.gov可到达
不论网关(gateway)是否存在,在.dc.gov域中的用户仍然是在他们自己的机器上收到邮件。在.dc.gov域中,内部邮件通过内部网络直接从一台机器发往另一台机器,网关(gateway)并不关心。但发往外部网的邮件则必须先发给网关(gateway)。
网关(gateway)只会转发那些以域名为地址的邮件(如wash.dc.gov)。不幸地,因为网关(gateway)并不知道其它机器上有什么用户,所以象[email protected]这样的邮件,很可能会被网关(gateway)退回去。
6.1.2 域acme.com
在图8-3中显示了.acme.com域的内部结构。它与.dc.gov域不同,在这个域中所有的主机都是直接与外部网络(Internet)相连的。所有的机器都可以直接从外部网接到邮件。例如,用户tim同样是在他自己的机器boss.acme.com上接收和阅读邮件,但与.dc.gov域不同的是,这里不需要网关(gateway)。
图8-3 acme.com域中的所有主机都可以访问外部网
就象fbi.dc.gov,机器sec.acme.com将接收到发往所有地址是acme.com的邮件(如[email protected])。但是与网关fbi.dc.gov不同,sec.acme.com知道整个域中的用户情况。所以,发往[email protected]的邮件将被正确地转发给主机boss.acme.com。
6.1.3 域uc.edu
在图4中显示了第三种设置方法。在这个域中,所有的邮件(无论内部的还是外部的)都将发往mail.us.edu。与前面两个例子不同的是,所有的邮件都不再发到其它机器上,而是存放在mail.us.edu这台机器的spool目录中。其它机器通过mount这个目录来阅读他的邮件(或使用POP协议收信)。
图8-4 只有mail.us.edu接收整个域的邮件
在这个系统中,邮件地址仅需要包含本地接收的名字,一个“@”符,再加上域名(如[email protected])。这样不再需要知道每个用户使用哪台机器。
以上三种实现方法各有优点,也各有缺点。我们并不完全赞成某个,也不完全否决某个。在本章中,假设我们的机器在us.edu域中。我们选择这个域是因为其sendmail配置文件最简单。
6.1.4 UUCP和主机通路
在这个虚构的网络中,我们假定有两台机器通过拔号连接到acme.com域中(如图8-5)。在UUCP中,一台机器需要知道确切的连接方法。例如,sonya想从sec.acme.com中取得邮件,必须知道它还需要通过主机lady。
图8-5 UUCP连接到sec.acme.com
6.2 为什么需要规则(Rules)
规则(Rules)是在sendmail.cf文件中用来重写(修改)邮件地址,检查地址错误,选择邮件转发代理的。地址需要重写,是因为很多情况下,你需要根据转发代理的不同,命名不同的地址格式。如图8-6所示:
图8-6 规则修改了地址,检测到错误,选择分发代理
如果uuhost是一个通过拔号与你连接的主机,那么邮件就可能通过UUCP发送出去,因此地址friend@uuhost就需要转换成为UUCP的格式:uuhost!friend。
而由于地址@neighbor未指定接收邮件的用户名,所以将被认为是一个错误的格式。
分发代理是sendmail程序用来完成实际的邮件传送的。规则(Rules)通过检查每一个接收的地址,并根据地址为其选择分发代理。例如,对于[email protected]这个接收者地址而言,规则(Rules)检测到here.us.edu就是本机,因此选择本地(local)分发代理程序将最终分发到root的邮箱中。
关于分发代理和宏我们已经在前两章中说明了。这一章将讲解规则集(Rules Set)是如何处理各种地址的。
6.3 规则集(Rule Set)
一连串成组的规则将构成规则集。每一个规则集就象是一个子程序。在sendmail.cf文件中,我们使用S命令来说明规则集。例如:
S0
这条语句说明了规则集0。规则集是从0开始往下编号的。规则集0-5是sendmail内部定义来用于特殊用途的,参见表8-1。我们将在后面详细地展开说明。在配置文件中并不需要按规则集编号来编写,也就是说,完全可以先定义规则集S5,然后再定义S2、S7等。Sendmail能够自己来排序。
表8-1 规则集的用途
规则集 目的
0 邮件分发代理解析
1 处理发送者地址
2 处理接收者地址
3 预处理所有地址
4 后处理(Postprocess)所有地址
5 非别名本地用户重写
到现在为止,client.cf文件内尚未包含规则集定义。我们执行以下命令再启动sendmail,观察不存在规则的效果:
% ./sendmail –Cclient.cf –bt
ADDRESS TEST MODE (ruleset 3 NOT automatically invoked)
Enter <ruleset> <address>
>
命令行参数“-bt”令sendmail运行在地址测试模式。在这种模式下,sendmail等待你输入一个规则集和一个地址。然后显示规则集如何重写这个地址。在“>”提示符后,输入一个规则集编号,一个空格,再跟上Mail地址,例如:
> 0 [email protected]
rewrite:ruleset 0 input:gw @ wash . dc . gov
rewrite:ruleset 0 returns:gw @ wash . dc . gov
>
在地址测试模式下,所有的输出都将以“rewrite:”开始,以与其它的调试信息区别开来。而“input:”的意思是sendmail将其放入工作区,“return”则显示了通过规则集重写后产生的地址。
我们将[email protected]送给sendmail的地址,首先会被基于两个分隔字符集分成几个部分。这两个分隔字符集均是内部定一个的,一个可以在配置文件中修改,另一个则不能修改:
1). : @ [ ] 你可以修改这个分隔字符集
2)() <> , ; “
你不可以修改这个分隔字符集
因为“.”和“@”都属于分隔字符集,所以地址[email protected]被分成七个部分。我们还发现,返回的地址没有任何变化,因为我们没有定义任何规则。
如果没有定义规则集,那么结果将与输入一样。就好象一个只包含return语句的C语言的子程序一样。它们什么也不做,也不会报错。
如果你想退出这个规则测试模式,输入Ctrl+D。
6.4 规则(Rules)
一个规则集可以包含零个或多个单独的规则。规则以R命令开头,每个规则均由三个部分组成:
S0
Rlhsrhs comment
在第一行,S0,说明了这是规则集0的开始,所有在这行之后的以R开头的规则定义都属于这个规则集。一个新的规则集将以一个新的S语句开始(S后应该是一个不同的规则集编号)。
每一个R开头的行表示一个规则,在一个规则集可以有许许多多的规则,但我们在些假设在规则集0中只有一个规则。
每一个规则包含三个不同的不分,每个部分之间使用tab键分开。你可以在一个部分中包含空格键,但一定要使用tab键将各个部分分开。
规则的最左边部分是LHS,中间部分是RHS。这组成了一个规则。最后一部分是注释语句。它必须与RHS之间用一个或多个tab分开。
LHS和RHS组成了一个if-then语句,如果LHS是真值,那么就执行RHS,如果LHS是假的,那么将跳到一下个规则:
6.5 工作区(workspace)
管理LHS是真还是假,都需要进行比较。当对地址根据规则集进行重写处理时,sendmail将首先将它们标记化,存入内部的缓冲区,这就是工作区,workspace。
“gw” “@” “wash” “.” “dc” “gov”存储在工作区内。
当检测LHS时,也会被标记化,然后将其与工作区内的标记(地址标记化后的结果)进行比较。如果相同,就说明匹配,LHS的结果就true(真)。现在,我们临时在client.cf中添加一个规则来讲解这一过程。
S0
Rleft.side new.stuff
别忘了LHS与RHS之间要使用tab键隔开。接着我们执行以下命令,使其以规则测试(地址测试)的方式运行:
% ./sendmail –Cclient.cf –bt
ADDRESS TEST MODE(ruleset 3 NOT automatically invoked)
Enter <ruleset> <address>
>
在这一模式下,我们可以用“=S”命令查看所设置的规则:
> =S=0
Rleft . side new . stuff
>
这个显示说明,sendmail发现规则集0中仅包含一个规则。
现在,我们使用规则集0,以及一个邮件地址:
> 0 [email protected]
rewrite:releset 0 input:gw @ wash . dc .gov
rewrite:releset 0 return:gw @ wash . dc .gov
地址并没有被重写,是因为工作区的内容与LHS并不匹配。
[email protected] 工作区内容
Rleft.side new.stuff 规则
不匹配
现在,我们在提示符状态下,输入与LHS完全相同的字符:
> 0 left.side
rewrite:releset 0 input:left . side
rewrite:releset 0 return:new . stuff
>
一个惊人的事情发生了。规则重写了地址。地址left.side被规则集0重写为一个新的地址:new.stuff了。这个转变是因为输入的地址与LHS匹配,使得LHS值为真了。
left.side 工作区内容
Rleft.side new.stuff 规则
匹配 -----> 执行new.stuff
我们再做一个实验,输入大写的LEFT.SIDE,看一看:
> 0 LEFT.SIDE
rewrite:releset 0 input:LEFT . SIDE
rewrite:releset 0 return:new . stuff
>
我们发现,这样仍然认为工作区与LHS是匹配的,说明这个判断是不区分大小写的。
6.6 地址通过规则的流程
当规则集中包含多个规则时,整个流程是从第一个规则开始,一直到最后一个规则(从顶向下)。为了更好的说明这点,我们在client.cf文件中删除掉刚才的那个示例规则,然后换成以下规则:
Rx y
Rz x
Rx w
再次提醒你注意LHS与RHS之间必须用tab键隔开。
在测试这个新规则之前,你可以先猜测一下它们是如何工作的。第一个规则将工作区中的所有x重写成y。第二个规则则将工作区中的所有z重写成为x。最后一个规则将工作区中的所有x重写成w。
现在我们以规则测试的方法再次启动sendmail:
%./sendmail –Cclient.cf –bt
ADDRESS TEST MODE (ruleset 3 NOT automatically invoked)
Enter <ruleset> <address>
>
然后我们输入规则集0和字母x
> 0 x
rewrite:ruleset 0 input:x
rewrite:ruleset 0 returns:y
这个输入显示了所有在工作区中的x被第一个规则重写成了y,然后执行第二个规则,结果不区配,第三个规则也是不匹配:
x原来的输入
Rx y 第一个规则
匹配!重写成为y
y
Rz x 第二个规则
不匹配!跳过
Rx w 第三个规则
不匹配!跳过
------> 最后得到了y
这里有一个十分重要的一个问题,就是我们始终用的是当前工作区的内容与LHS比较。前面的规则会改变工作区。例如,本来第三个规则会将x改成w的,但由于第一个规则已经将其重写成为y了。
现在我们输入规则集0和字母z:
> 0 z
rewrite:ruleset 0 input:z
rewrite:ruleset 0 returns:w
为什么z被重写为w了呢?我们来看看发生了什么:
z原来的输入
Rx y 第一个规则
不匹配!跳过
Rz x 第二个规则
匹配!重写为x
x
Rx w 第三个规则
匹配!重写为w
------> 最后得到了w
在这个实例中,第一个规则没做任何事,第二个规则将z重写为了x,而第三个规则又将x重写成为了w。
现在,我们再输入一些不是x或z的字母,看看:
> 0 b
rewrite:ruleset 0 input:b
rewrite:ruleset 0 returns:b
我们发现,返回的地址与输入一样,由于三个规则都不匹配,所以工作区内的字符不有任何改变。
6.7 通配符
如果规则中的字符要与工作区内的完全相符的话,那么规则就没有存在的意义了。幸运的是,并非是这种情况。我们可以在规则中使用通配符以使得规则能够匹配一类的文本。例如,假设有一个规则:
R$+ RHS
LHS是以R开头的,也就是说在这个例子中,LHS是:$+。这是一个通配符。这个LHS能够匹配多个标记。地址在与LHS比较时,会被标记化,然后将分解的标记存放在工作区中,形如:
标记化
gw @ wash . gc .gov---- > 存放在工作区内
当比较工作区与LHS模式时,sendmail将从左到右检查工作区内的标记。每一个将与通配符($+)比较。如果标记与这个模式匹配,那么就将使LHS为true。
通配符$+能够匹配一个或多个标记。
工作区 模式
gw $+ 一个字符,匹配!
@ 多个字符,匹配!
wash
.
dc
.
gov
正如你所看到的一样,工作区中的所有标记都与$+匹配。也就是只要工作区不是空的,就与$+匹配。(有点象DOS命令行下的通配符*)。
6.7.1 LHS中的其它文本
一个$+是不足以处理所有可能的地址(一些特殊的错误地址),例如:
[email protected] 与$+匹配
@wash.dc.gov 也与$+匹配
为了使得LHS更加有效,sendmail允许其它文本字符出现在模式中。例如,我们想确保地址中有一个用户名,一个@符,和一个主机名组成,那我们就可以在LHS中使用模式:
$+@$+
这个模式被分解为三个标记:
$+ @ $+
现在一个正确的、有效的邮件地址就会与我们新的LHS模式相匹配了。
工作区 模式
gw $+ 一个字符,匹配!
@ @ 匹配!
wash$+ 区配多个字符
.
dc
.
gov
而一个无效的邮件地址,如@wash.dc.gov就会不匹配了:
工作区 模式
@$+ 匹配多个字符!
wash
.
dc
.
gov
@ 没有相匹配的项
$+
这样的结果就是不匹配了。
6.7.2 最小限度的匹配
到现在为止,还可能有一些产生混淆的小问题。当使用通配符$+时,sendmail一直采用的是最小限度匹配策略。
这个策略使用言语难以描述清楚,我们还是通过一个实例来说明,假设有规则:
R$+@$+
工作区内的地址是
a@b@c
那么匹配的结果是什么呢?
工作区 模式
a$+ 匹配多个字符!
@ @
b $+
@
c
6.7.3 LHS匹配实验
花一点时间将client.cf文件中前面的示例规则改成:
S0
R@ one
R@$+ two
R$+@$+ three
然后,我们再使sendmail运行在规则测试模式:
%./sendmail –Cclient.cf –bt
ADDRESS TEST MODE (ruleset 3 NOT automatically invoked)
Enter <ruleset> <address>
>
现在,我们查看一下已经设置的规则集
> =s 0
R@ one
R@ $+ two
R$+@$+ three
下面,我们就开始测试,首先我们输入单独的@符:
> 0 @
rewrite:ruleset 0 input:@
rewrite:ruleset 0 returns:one
可见,单独一个@将匹配规则一。
然后,我们输入:
> 0 @your.domain
rewrite:ruleset 0 input:@ your . domain
rewrite:ruleset 0 returns:two
而,我们输入:
> 0 [email protected]
rewrite:ruleset 0 input:you @ your . domain
rewrite:ruleset 0 returns:three
[目录]
规则集0
7. 规则集0
规则集0是用来为一个专门的接收地址选择邮件分发代理。
对于邮件消息的每一个接收者都将调用一次规则集0。对于每一个地址规则集0将做三次决定。这三次分别处理:邮件分发代理的符号名,邮件地址中的用户名,邮件地址中的主机名。我们在本章以及后的关于规则集的章节中都会展开说明。
7.1 引入规则集0
记得我们曾经在第6章“邮件中枢与分发代理”,设计过一个client.cf文件,使得sendmail将所有的邮件都转发给了邮件中枢来处理。当时我们在client.cf文件中做了以下声明:
# Delivery agent definition to forward mail to hub
Mhub, P=[IPC], S=0, R=0,F=mDFMuXa,T=DNS/RFC822/SMTP,A=IPC $h
在这种情况下,我们就无需设计任何重写接收者地址的规则,我们只需简单地将信发给邮件中枢就可以了。
以下就是一个规则集的实例:
S0
R$+ $#hub $@${REMOTE} $:$1 forward to hub
第一行(S0)说明规则集0定义开始。你已经在上一章中看到过这样的LHS,$+是一个通配符,它能够与任何一个地址匹配,当然如果是空地址就无法匹配了。
7.2 RHS的三个部分
规则集0的任务是负责将每一个地址解析成为三个部分:分发代理的符号名,主机的名字,在那台主机上的用户名。
大家应该还能记得“规则”就象是一个条件语句,如果LHS与工作区的内容匹配,那么sendmail就会执行RHS中的内容。
在规则“R$+ $#hub $@${REMOTE} $:$1”中,RHS将解析这三个部分:
$#hub 分发代理
$@${REMOTE} 主机名
$:$1 用户名
一些转换的通配符的含义如下表所示:
表9-1 规则集0转换符
转换符 说明
$# 邮件分发代理
$@ 主机名部分
$: 用户名部分
7.2.1 分发代理($#)
第一个部分是分发代理的名字,在$#后面跟着的就是分发代理名。在RHS中$#符号告诉sendmail后面跟着的就是分发代理的用户名。
我们在client.cf文件中定义了符号名hub:
# Delivery agent definition to forward mail to hub
Mhub, P=[IPC], S=0, R=0, F=mDFMuXa,T=DNS/RFC822/SMTP,A=IPC $h
当RHS拷贝到工作区时,首先将$#符号拷贝到工作区,然后拷贝分发代理的符号名,并且将它进行标记化。由于hub中并没有任何分隔符,所以最终拷贝到工作区的内容就是:
$# hub
7.2.2 主机名($@)
第二个部分就是主机名。转换符$@告诉Sendmail,后面的内容就是邮件将要发送到的主机。如果在RHS中遇到类似“${name}”的表示式,则意味着我们使用了宏。在这个例子中,我们使用了${REMOTE},这个宏在client.cf文件中有以下定义
D{REMOTE}mailhost
这样,最终
$@${REMOTE}
被拷贝到工作区后,就成为了:
$@ mailhost
7.2.3 用户名($:)
第三个部分就是用户名,RHS中的转换符$:意味后面跟着的是邮件接收者的用户名。它告诉sendmail后面跟的是用户名。在我们的例子中,使用的是$1。
那么这个$1是什么呢?请看下图:
$+ . $+
| |
$1 $2
也就是说在第一个“.”之前的部分就是$1,而之后的就是$2。
7.3 测试规则集0
为了使大家能够观察到实际的运行结果,我们花一些时间来做一些测试。首先我们在你的client.cf文件中加入规则集0:
S0
R$+ $#hub $@${REMOTE} $:$1 forward to hub
现在,我们以规则测试模式运行sendmail,然后输入地址boss@acme:
% ./sendmail –Cclient.cf –bt
ADDRESS TEST MODE (ruleset 3 NOT automatically invoked)
Enter <ruleset><address>
> 0 boss@acme
rewrite: ruleset 0 input: boss@acme
rewrite: ruleset 0 returns:$# hub $@ mailhost $: boss@acme
正如我们预料的一样,返回的是
$# hub $@ mailhost $: boss@acme
规则集0是与其它规则集不同的。我们将会从下一章开始介绍其它规则集。
1) 规则集0的规则可以使用$#,用于返回分发代理的符号名。如果其它规则集 使用$#,就会出现无法预测的错误。
2) 在规则集0的$#后面的$@和$:具有特别的用途,它如果出现在其它规则集中,或者不包含$#的其它规则集0中的规则时,用途不一样。这一点我们会在后面的章节中说明。
7.4 错误的分发代理
在邮件中枢模式下,尽管所有的邮件都将交由邮件中枢处理,但某些错误仍然会在本地处理。如果在本地已经发现了错误,将会立即对用户进行提示,而不是等邮件被退回。
在我们讨论邮件的分发代理的那一章中,我们曾经告诉您sendmail需要一个本地(local)的邮件转发代理。所以我们在client.cf文件中定义了一个:
Mlocal, P=/bin/mail, F=lsDFMAw5:/|@rmn, S=0, R=0, A=mail –d $u
在sendmail中,已经定了一个特殊的分发代理,它叫做error,这个分发代理你无法去定义它。这个分发代理是sendmail的一个处理机制。它对那些不合法的地址进行处理。
记得,我们曾经使用过以下演示规则:
S0
R@ one
R@$+ two
R$+@$+ three
当使用这个规则集时,如果处理地址“@host”的话,将会在工作区内重写为two。因此,我们使用一个sendmail内置的错误处理功能进行。
我们在client.cf文件的规则集0定义中加入 一行。
R@$+ $#error $: Missing user name
然后我们再次以规则测试模式启动sendmail,这个时候我们输入@acme会出现什么。
% ./sendmail –Cclient.cf –bt
ADDRESS TEST MODE (ruleset 3 NOT automatically invoked)
Enter <ruleset><address>
> 0 @acme
rewrite: ruleset 0 input: boss@acme
rewrite: ruleset 0 returns:$# error $:Missing user name
而如果我们输入合法的地址:boss@acme仍可得到正确的结果:
> 0 boss@acme
rewrite: ruleset 0 input: boss@acme
rewrite: ruleset 0 returns:$# hub $@ mailhost $: bos
[目录]
qmail
1 Qmail简介
1.1 什么是Qmail
Qmail是一种可以完全替代Sendmail-binmail体系的新一代Unix邮件系统.
1.2 为什么选择Qmail?
A. 安全----Qmail将Mail处理过程分为多个分过程,尽力避免用root用户运行.同时Qmail也禁止对特权用户(root,deamon等等)直接发信.
B. 可靠----Qmail的直接投递保证Email在投递过程中不会丢失.Qmail同时支持新的更可靠的信箱格式Maildir,保证系统在突然崩溃情况下不至破坏整个信箱.
C. 高效----在运行于奔腾的BSD/OS上,Qmail每天可以轻松的投递200000封信件.
D. 简单----Qmail要比其他的Internet Mail系统小得多.Qmail通过统一的向前机制完成forwarding,alias和maillist等功能,Qmail使用 简单高效队列来处理投递.Qmail-smtpd可以由inetd启动,节省了一定资源.
Qmail支持: host and user masquerading
full host hiding
virtual domains
null clients
list-owner rewriting
relay control
double-bounce recording
arbitrary RFC 822 address lists
cross-host mailing list loop detection
1.3 如何获得Qmail?
可以通过访问www.qmail.org或mirror站点下载qmail-1.03.tar.gz,以及获得
更多qmail的资料.
2 安装Qmail
获得qmail-1.03.tar.gz后,用tar命令解包
#tar xzvf qmail-1.03.tar.gz
进入qmail目录后,仔细阅读一下README和INSTALL文件.然后开始编译qmail.
2.1 建/var/qmail目录:
#mkdir /var/qmail
2.2 按照INSTALL.ids中方法建立qmail用户和组:
# groupadd nofiles
# useradd -g nofiles -d /var/qmail/alias alias
# useradd -g nofiles -d /var/qmail qmaild
# useradd -g nofiles -d /var/qmail qmaill
# useradd -g nofiles -d /var/qmail qmailp
# groupadd qmail
# useradd -g qmail -d /var/qmail qmailq
# useradd -g qmail -d /var/qmail qmailr
# useradd -g qmail -d /var/qmail qmails
2.3 make setup check
2.4 阅读INSTALL.ctl和FAQ,配置qmail,最简单的方法是
#./config
或者
#./config-fast your.full.home.name
2.5 建立系统别名
# (cd ~alias; touch .qmail-postmaster .qmail-mailer-daemon .qmail-root)
# chmod 644 ~alias/.qmail*
2.6 复制/var/qmail/boot/proc到/var/qmail/rc
# cp /var/qmail/boot/proc /var/qmail/rc
2.7 开始测试Qmail投递程序
启动qmail:
# csh -cf '/var/qmail/rc &'
先查看一下/var/log/maillog,搜索
qmail: status: local 0/10 remote 0/20
qmail-send通常是输出"status"或者"cannot start"如果不能正常启动.
用ps监视一下qmail守护进程,应该有五个相关进程:
qmail-send,以qmails用户运行
qmail-lspawn,以root用户运行
qmail-rspawn,以qmailr用户运行
qmail-clean,以qmailq用户运行
splogger,以qmaill用户运行
本地Mail测试:
% echo to: me | /var/qmail/bin/qmail-inject
注意:要用你的用户名代替me
Mail应该立即出现在你的信箱之中./var/log/maillog中应该有如下记录:
qmail: new msg 53
qmail: info msg 53: bytes 246 from <me@domain> qp 20345 uid 666
qmail: starting delivery 1: msg 53 to local me@domain
qmail: status: local 1/10 remote 0/20
qmail: delivery 1: success: did_1+0+0/
qmail: status: local 0/10 remote 0/20
qmail: end msg 53
53是inode号,20345是进程号,你的数字应该有所不同.
本地错误测试:
给一个不存在的本地用户发信:
% echo to: nonexistent | /var/qmail/bin/qmail-inject
qmail: new msg 53
qmail: info msg 53: bytes 246 from <me@domain> qp 20351 uid 666
qmail: starting delivery 2: msg 53 to local nonexistent@domain
qmail: status: local 1/10 remote 0/20
qmail: delivery 2: failure: No_such_address.__#5.1.1_/
qmail: status: local 0/10 remote 0/20
qmail: bounce msg 53 qp 20357
qmail: end msg 53
qmail: new msg 54
qmail: info msg 54: bytes 743 from <> qp 20357 uid 666
qmail: starting delivery 3: msg 54 to local me@domain
qmail: status: local 1/10 remote 0/20
qmail: delivery 3: success: did_1+0+0/
qmail: status: local 0/10 remote 0/20
qmail: end msg 54
你将立即收到弹回的信件.
远程投递测试: 向你在其他机器上的账户发信:
% echo to: me@wherever | /var/qmail/bin/qmail-inject
qmail: new msg 53
qmail: info msg 53: bytes 246 from <me@domain> qp 20372 uid 666
qmail: starting delivery 4: msg 53 to remote me@wherever
qmail: status: local 0/10 remote 1/20
qmail: delivery 4: success: 1.2.3.4_accepted_message./...
qmail: status: local 0/10 remote 0/20
qmail: end msg 53
投递过程将在starting delivery和success中有个间断,SMTP传送是相对较慢的然后去检查一下me@wherever是否正确收到email.
后面还有三个测试,分别测试本地postmaster,两次弹回mail和通过mail执行命令.具体方法可以在TEST.deliver中找到.
2.8 从Sendmail象Qmail转移.
2.8.1 首先找到sendmail的启动程序,一般是在/etc/rc.d目录中
看上去象这样的命令:
sendmail -bd -q15m
将其注释掉.
2.8.2 杀掉Sendmail进程.
如果sendmail有子进程,可以反复用-STOP和-CONT信号杀,直到没有子进程后用-TERM后加-CONT杀就可以了.
2.8.3 检查mail队列中是否还有信,如果认为有必要将他们发送出去,可以在以后一段时间不定期的运行sendmail.bak -q直到队列清空.
2.8.4 去掉sendmail等文件的setuid bit
# chmod 0 /usr/lib/sendmail
# chmod 0 /usr/sbin/sendmail
# chmod 0 /usr/lib/sendmail.mx
2.8.5
# mv /usr/lib/sendmail /usr/lib/sendmail.bak
# mv /usr/sbin/sendmail /usr/sbin/sendmail.bak
2.8.6 将
csh -cf '/var/qmail/rc &'
添加到启动文件中去.
2.8.7 安装qmail的sendmail外壳:
# ln -s /var/qmail/bin/sendmail /usr/lib/sendmail
# ln -s /var/qmail/bin/sendmail /usr/sbin/sendmail
2.8.8 在/etc/inetd.conf中设置qmail-smtpd:(是一行)
smtp stream tcp nowait qmaild /var/qmail/bin/tcp-env
tcp-env /var/qmail/bin/qmail-smtpd
2.8.9 重启动inetd,用-HUP信号杀inetd及可.
2.8.10 进行收信测试:仔细阅读TEST.receive
2.8.10.1 SMTP Server测试,me是你的用户,domain是你的机器名.
% telnet 127.0.0.1 25
Trying 127.0.0.1...
Connected to 127.0.0.1.
Escape character is '^]'.
220 domain ESMTP
helo dude
250 domain
mail <me@domain>
250 ok
rcpt <me@domain>
250 ok
data
354 go ahead
Subject: testing
This is a test.
.
250 ok 812345679 qp 12345
quit
221 domain
Connection closed by foreign host.
看看信箱中是否有新信.
2.8.10.2 从其他机器上发信给me@domain,看看是否能正确接收.
后面还有三个测试:远程错误测试,UA测试和远程PostMaster测试,具体方法可以参照TEST.receive
如果以上都没有问题,Qmail就算基本安装成功.
3 qmail的一些特殊配制
3.1 如何设置Smart Host:
# echo ":your.smart.host" > /var/qmail/control/smtproutes
3.2 如何设置允许转信:
将/etc/inetd.conf中smtp服务做如下修改:
smtp stream tcp nowait qmaild /usr/local/bin/tcpd
/var/qmail/bin/tcp-env /var/qmail/bin/qmail-smtpd
然后阅读tcpwarpper相关文档,在/etc/hosts.allow中加入
tcp-env: 1.2.3.4, 1.2.3.5: setenv = RELAYCLIENT
1.2.3.4和1.2.3.5是你的客户机ip地址,也可以使用tcpwarpper的所允许的其他地址形式.
3.3 如何设置虚拟主机
将虚拟主机名入/var/qmail/control/local和/var/qmail/control/rcpthosts然后在/var/qmail/control/virtualdomains中加入
your.virtual.domains:youracct
这样所有发往[email protected]都将以[email protected]的形式发给youracct,配合procmail和fetchmail,可以方便的分捡进行再投递.
[目录]
router
利用 Linux 当 router 其实很简单.首先你在 compile kernel 时要选IP forwarding. 用这个 kernel 开机,看看能不能抓到两块网路卡.如果不行的话在 /etc/lilo.conf 上加上
append="ether=irq,io-port,eth0 ether=irq,io-port,eth1"
再重跑 lilo. 或者如果你用 loadlin 的话,用
loadlin vmlinuz root=/dev/??? ro ether=irq,io,eth0 ether=irq,io,eth1
应该就可以了.注意看开机时有没有抓到? 或看 /proc/net/dev 有没有 eth0和 eth1? 有的话就对了(如果没有就要 recompile kernel 了.请参考 "为何我的网路连不出去" 一文).
接下来就是 config 这两个界面了.假设你的网路环境如下:
|
| +-----+
+---| DOS | 140.122.52.236
+---------+ eth0 +-----+ eth1 | +-----+
| gateway |------------------|Linux|----------+ 分出来的次子网
+---------+ +-----+ | +-----+
140.122.52.254 140.122.52.235 +---| Sun | 140.122.52.234
router | +-----+
|
可再接多台机器
eth0 界面是用来跟外面的网路相连的.你可以用一般的 netconfig 来设定它.(若不会的话还是请参考精华区中的文章).
我们是利用 Linux 的 Proxy Arp 功能,当外界有封包要送到你的次子网路时,Linux 会回应此封包.所以对外界来说你的次子网就是存在的了.因此很重要的一点是,你的次子网必须要是从你原来的子网路上割出来的!!
以上面的情形为例,如果预估你的次子网约有五、六台左右的机器,你可以决定要割出来的次子网的 IP 范围是从 140.122.52.232 -- 140.122.52.239 共8个位址(一定是 2^n 啦! 不懂吗? 自己想...:) 算一算
netmask 应是 255.255.255.248,
broadcast 是 140.122.52.239.
於是 eth1 应该这样 config:
# ifconfig eth1 140.122.52.235 netmask 255.255.255.248 broadcast 140.122.52.239
route add -net 140.122.52.232 netmask 255.255.255.248 gw
140.122.52.235 eth1
这样你所有从 140.122.52.232 到 140.122.52.239 的封包都会往 eth1 送.试试看能不能从你的次子网 telnet 到 Linux 上? 若可以的话就成功了一大半....^_^
不知你是否注意到,我将 eth0 和 eth1 的 IP address 设为一样! 常有人认为,有两块网路卡就要有两个 IP address. 但这是没有必要的! 你可以将两个设为相同!!
但是你若想试著直接 telnet 到外面去,就会发现不通! 这是因为尚未设定 arp table 的关系.以上面为例,如果你想要 DOS 那台机器可以跨出次子网,就要在Linux 上设 arp:
# arp -s 140.122.52.236 08:00:77:14:00:00 pub
^^^^^^^^^^^^^^^^^
↑
这是你 eth0 界面的 hardware address, 可用 ifconfig 查看
你可以把这些指令都放在/etc/rc.d/rc.inet1 中,这样开机时就可以设定好两个界面了!
[目录]
cvs
在/etc/services文件中有如下的两行,这是配置服务的:
cvspserver 2401/tcp # CVS client/server operations
cvspserver 2401/udp # CVS client/server operations
然后如果是使用inetd来启动,那么需要修改修改/etc/inetd.conf文件,如果使用xinetd,那么就需要在/etc/xinetd.d/目录下加入一个启动cvs的脚本我的系统使用的是red hat7.0,所以就加入如下的脚本到/etc/xinetd.d/下
[root@local /]# less /etc/xinetd.d/cvspserver
# default: off
# description: The SPOP3 service allows remote users to access their mail /
# using an POP3 client with SSL support such as Netscape /
# Communicator or fetchmail.
service cvspserver
{
socket_type = stream
protocol = tcp
wait = no
user = root
server = /usr/bin/cvs
server_args = --allow-root=/source pserver
log_on_success += USERID
log_on_failure += USERID
disable = no
}
如果使用inetd,也可以修改inetd.conf,很简单的。完成这些之后,就可以重新启动inetd/xinetd了。然后执行cvs的命令就可以正常使用cvs了。不过需要注意的是远程使用需要设置一下客户机:
CVSROOT=:pserver:user@host:/cvsrootdir:
export CVSROOT
cvs login
需要将前面两句加入到你的启动脚本中去。如果还有什么问题,请访问http://www.cvshome.org/docs/manual/
[目录]
bind
主域名服务器正常工作的必要条件:
1. 安装bind8软件
这在很多unix的发行版中都可以找到,在小红帽里有rpm包可以直接安装.或者去http://freesoft.online.sh.cn,
ftp://studio.sinet.net.cn都可以找到。
2. 几个必须的配置文件:
named.conf
named.ca
named.local
mater file(就是bind4里的zone file)
在这几个配置文件中,最主要的是named.conf,在/etc下,
它是named启动时缺省的启动文件.一个典型的named.conf文件至少包括
options,和zone.如:
options {
directory "/var/named";
};
zone "0.0.127.in-addr.arpa" in {
type master;
file "named.local";
};
zone "." in {
type hint;
file "named.ca";
};
zone "99.cn.net" in {
type master;
file "db.99.cn.net";
};
在这里面,options定义master file存放的路径,对应于某一个域,named就会到这里找数据文件,如有一个www.99.cn.net的require请求,named就会到/var/named下找db.99.cn.net这个文件,查出www.99.cn.net的ip. zone定义一个域,如99.cn.net这个域,type定义域名服务器的类型,master说明这是一个主域名服务器,第一个zone定义本地服务器为它自己的回送域的主服务器,将地址127.0.0.1映射为localhost,几乎在所有类型的域名服务器里都可以看见这个域。第二个zone用来定义一个高速缓存初始化文件,在named.ca里至少包含着根服务器的名字和地址(这些根服务器也会变化). 一般named.ca不需要修改,named.local里只需要修改SOA纪录里域和联系人。(有些书上说named.local里的NS纪录是个摆设,但我建议最好还是有这一句)
file定义99.cn.net这个域的master file.以下就是这个master file的内容:
@ IN SOA ns.your.domain. root.your.domain. (
1999110901 ; Serial Number
10800 ; Refresh after 3 hours
3600 ; Retry after 1 hour
3600000 ; Expire after 6 weeks
86400 ) ; Minimum TTL of 1 day
@ IN NS ns.your.domain.
localhost IN A 127.0.0.1
www IN A 202.98.106.12
这里面的@定义当前域,也就是your.domain, IN定义这是一个internet类型的纪录,SOA(start of authority)标志一个授权域的开始,ns.your.domain.为开创该域的服务器,可以用主域名服务器,root.your.domain.定义联系人,root后的.就是email里的@,括号里的几个数字定义和本域有关的几个参数,单位是秒,前四个参数用于辅域名服务器更新master file,其中;后面的是说明,serialnumber用于辅域名服务器判断主域名服务器的master file是否更新,所以如果你有辅域名服务器,在每次修改master file后就应该修改这个序列号,以便辅域名服务器更新这个域的 master file. refresh定义辅域名服务器刷新的时间,
retry定义如果主服务器未响应,辅服务器重试的时间间隔,expire定义这个域的过期时间,就是说如果辅服务器在连续42天里都没有从主服务器取到该域的信息,辅服务器就丢弃该域。第五个参数定义这个域在其他域名服务器的cache里的有效期,过了这个时间其他的域名服务器就会到这里来重新查询相关的信息。
NS表明这个域的域名服务器是ns.your.domain. NS纪录可以有多个。localhost和www这两条A纪录的含义是将localhost解析到127.0.0.1和将www.your.domain解析到202.98.106.12,现在一个基本的主域名服务器就建立好了,但需要特别说明的是,在master file里,ns.your.domain.后面的.这个.表示这是一个完整的纪录,否则服务器就会自动给你加上当前域,例如,www表示www.your.domain,而www.your.domain就会变成www.your.domain.your.domain,正确的表示方法可以是www或www.your.domain.如果不给这个.足够的重视,就很容易出错。
如果你需要维护很多域,可以在named.conf里加上相应的zone,然后在/var/named下建立该域的master file,最后用SIGHUP来reload域名服务器。如果你需要作反向解析,可以仿照上面named.conf里第一个zone进行,在它相应的master file里用PTR指针将ip转换为域名。
辅域名服务器的建立
建立辅域名服务器的方法和主域名服务器基本相同,主要的区别在于named.conf里的zone类型,它的type为slave,以下是一个辅域名服务器的named.conf:
options {
directory "/var/named";
};
zone "0.0.127.in-addr.arpa" in {
type master;
file "named.local";
};
zone "." in {
type hint;
file "named.ca";
};
zone "99.cn.net" in {
type slave;
file "db.99.cn.net";
masters { 202.98.21.141; };
可以看出,这个配置文件与主域名服务器的区别,type用salve定义该服务器为辅域名服务器,然后指出主域名服务器的ip。但是需注意第一个zone的type仍为master.另一个区别就是辅域名服务器的master file不用手动建立,它自己会从主域名服务器传过来,named-xfer缺省一次传10个域。辅域名服务器的master file和主域名服务器的master file基本没有什么区别。这样一个可以工作的辅域名服务器就建好了,如果这是一个新建的辅域名服务器,那么在你启动它后很快就会看见/var/named下多了很多master file. :)
还有一点我想说明的是,如果internet上的域名服务器乱了,那么整个internet就乱了,所以维护域名的人一定要谨慎从事。因为以前在internet上工作的都是bind4,如果你维护的域比较多,在从bind4转到bind8时可能会为建立格式完全不同的named.conf而头痛,一个好消息是在bind8的发行版中有一个named-bootconf的shell脚本和named-bootconf.pl的perl脚本可以帮你完成从bind4的named.boot到bind8的named.conf的转换,为了偷懒,我也写了一个perl脚本完成从主域名服务器的named.conf到辅域名服务器的named.conf的转换,见附录(仅供参考)。
附录:named-masterslave.pl
作用:将主域名服务器的named.conf转换为辅域名服务器的named.conf
用法:将主域名服务器的named.conf和这个脚本放在同一个目录下,将脚本中masters里的ip改为你自己的主域名服务器的ip,运行这个脚本,就会得到一个named.confd的文件,将文件中有关named.local和named.ca的zone按上面的例子修改,然后存成/etc/named.conf就可以了。
#!/usr/bin/perl
$filename="named.conf";
open(In,"$filename");
@lines=<In>;
close(In);
open(Out,">named.confd");
foreach $line(@lines) {
if ($line=~ /type/)
{ $line=~ s/master/slave/;}
if ($line=~ /file/) {
$line=~ s/;/;/n masters { 202.98.21.141; };/;}
print Out "$line";
}
close(Out);
=========================================================================================
建立安全的 DNS 服务器(Linux版本)
摘自:绿色兵团
和其它大型软件一样,BIND(DNS服务器)也因其体积庞大和功能繁杂而存在许多问题。因此针对BIND安全漏洞的系统入侵数量也在大幅度上升,最严重的甚至可获取目标主机的所有远程控制权。由于DNS服务器主机对网络系统有着很大的影响,如何避免这些系统
入侵也变得至关重要了。
这篇短文的主旨是讲述如何利用chroot()环境在RedHat Linux(或类似系统)中建立安全的BIND 8.x服务器。本文内容主要来自于Adam Shostack和他在这方面的文章(Solaris版本)。
步骤一:软件获取和安装
请到ISC FTP站点下载BIND的最新版本(本文内容在BIND 8.x版本中经过测试)。到Obtuse System FTP站点下载本文必需的免费软件:holelogd(及其它有用的工具)。该软件用于在chroot环境中建立/dev/log套接字(socket),从而使syslogd能够记录named进程的日志。OpenBSD系统的syslogd已内建了这一功能("syslogd -a /chroot/dev/log"),但Linux系统尚未实现这一功能。Holelogd软件就是用来模仿OpenBSD的这个功能。
按照软件文档安装holelogd(通常被安装到/usr/local/sbin)。
步骤二:构造静态(static)的named和named-xfer二进制文件
在编译和安装后,你需要构造可执行文件的静态链接版本。只要对%BIND%/src/port/linux目录下的Makefile.set文件稍加修改后即可。
修改文件内容:
'CDEBUG= -O2 -g'
替换为:
'CDEBUG= -O2 -static'
切换到BIND的源代码路径,执行"make clean"和"make"命令。在下面的步骤中将会把这些文件复制到chroot()目录下。
本步骤构造的静态链接执行文件在运行时无需装载动态链接库。在chroot()环境中,这种“独立”可执行文件可避免出现缺少链接库文件问题。它在chroot()环境中无需任何静态链接库,可使服务配置简单化。其它所有的网络守护进程也可以编译和使用这种静态链接版本。
步骤三:构造BIND目录
为chroot()环境构造BIND目录。这个目录将在chroot()环境中被BIND当作系统根目录。
/dev
/etc
/namedb
/usr
/sbin
/var
/run
需要复制以下文件到其下的相应子目录中,和进行一些必要的处理:
/
无
/etc
复制系统/etc目录下的named.conf文件
复制系统/etc目录下的localtime文件(为syslog提供正确的named日志记录时间)
创建仅包含named GID的/etc/group文件
/etc/namedb
复制系统/etc/namedb目录下的所有“区(zone)”数据库和文件
/dev
mknod ./null c 1 3; chmod 666 null(请参阅相应版本的mknod命令)
/usr/sbin
复制系统%BIND%/src/bin/named目录和系统%BIND%/src/bin/named-xfer目录下的named和named-xfer二进制文件(静态链接版本)
/var/run
无
另外还可根据需要指定日志记录目录(如/var/log)。
步骤四:添加named用户和组
在/etc/passwd和/etc/group文件中添加named用户和组。它们是DNS服务器运行时的UID/GID。
此时,你可以到chroot环境中执行"chown -R named.named /etc/namedb"命令。这样当你向系统发送中断信号(kill -INT )时,named进程能够保存服务器缓存和统计信息。如果该目录为root所有则named进程无法将输出写到目录中,但不会影响named服务器功能。另一个选择是仅改变目录权限(使named用户具有写权限),而属主仍然是root。这种方法也是可行的,但必须小心设置,确保其它用户不会修改named记录!
*** 重要警告***
不要用一个已存在的UID/GID(如"nobody")运行named。记住,以chroot环境中使用任何已存在的UID/GID都可能会影响到服务的安全性。必须养成在chroot环境中为每一个守护进程提供独立的UID/GID的习惯。
步骤五:编辑启动脚本
Linux使用SYS V风格的init文件,所以有几个地方都可以放置运行named的命令。(大多数情况下)最好将named初始化脚本放置到/etc/rc.d/init.d/named中。在其中你会找到有关named启动的那一节内容。我们需要添加和修改其中的某些行。
1、在运行named前插入一行以启动holelogd。需要向holelogd提供远程套接字位置的参数,它应该是在上面步骤中创建的chroot named dev目录。命令行内容如下:
# Start daemons.
echo -n "Staring holelogd: "
daemon /usr/local/sbin/holelogd /chroot/named/dev/log
echo
echo -n "Starting named: "
daemon named
echo
touch /var/lock/subsys/named
;;
2、另外还需要修改BIND的启动参数。BIND 8.x版本允许指定运行用户ID和组ID,它也应该是在上面步骤中特别创建的UID/GID:
# Start daemons.
echo -n "Staring holelogd: "
daemon /usr/local/sbin/holelogd /chroot/named/dev/log
echo
echo -n "Starting named: "
daemon /chroot/named/usr/sbin/named -u named -g named -t /chroot/named
echo
touch /var/lock/subsys/named
;;
3、named附带的"ndc"脚本可用于控制named的工作。需要编辑这个文件以将PID文件位置从/var/run/named.pid修改为/chroot/named/var/run/named.pid。
步骤六:服务器测试
输入如下命令启动holelogd进程
/usr/local/sbin/holelogd /chroot/named/dev/log
进入/chroot/named/dev/目录并输入ls -al。应该得到与下面类似的输出:
srw-rw-rw- 1 root wheel 0 Jan 01 12:00 log
设定的"s"位指示这是一个套接字(socket)文件。Chroot()环境中的named进程将通过该套接字与syslog通讯。
现在输入:
/chroot/named/usr/sbin/named -u named -g named -t /chroot/named
如果一切正常,named进程将启动,日志文件将记录named服务器"Ready to answer queries."。
进行适当的DNS测试以确保服务器能正确工作,然后重新启动系统并检验所有配置。BIND正常启动后会报告其chroot()目录和运行UID/GID。你可以使用lsof之类的程序列出主机中所有网络套接字进行检查。
当一切工作正常后,建议将/etc/namedb改名为/etc/namedb.orig或其它名字,同时chmod 000 /usr/sbin/named,这样可以确保这些旧版本的named不会因偶然的失误而使用。
2000-02-22
[目录]
openssh
openssh
ssh是Secure Shell的简称. 相当于一个在安全管道中的telnet, 所有数据都是加密传输的, 可以很好的防范网络窃听.openssh是Open BSD的一个部分, 是完全free的ssh实现, 同时支持ssh1, ssh2, 已经被移植到各种Unix平台上.openssh依赖于openssl, 因此安装openssh前, 必须先要安装openssl.
[目录]
简单使用
=================================
Server端的起动
=================================
单独运行, 直接运行
sshd
使用inetd起动
在/etc/inetd.conf中添加一行
ssh stream tcp nowait root /usr/sbin/tcpd ssh -i
然后重新起动inetd就可以了.
=================================
Client端
=================================
ssh -l user host
或者
ssh user@host
[目录]
配置文件
sshd的配置文件( /etc/ssh/sshd_config)
# This is ssh server systemwide configuration file.
Port 22
#Protocol 2,1
ListenAddress 0.0.0.0
#ListenAddress ::
HostKey /etc/ssh/ssh_host_key
ServerKeyBits 768
LoginGraceTime 600
KeyRegenerationInterval 3600
PermitRootLogin yes
#
# Don't read ~/.rhosts and ~/.shosts files
IgnoreRhosts yes
# Uncomment if you don't trust ~/.ssh/known_hosts for RhostsRSAAuthentication
#IgnoreUserKnownHosts yes
StrictModes yes
X11Forwarding no
X11DisplayOffset 10
PrintMotd yes
KeepAlive yes
# Logging
SyslogFacility AUTH
LogLevel INFO
#obsoletes QuietMode and FascistLogging
RhostsAuthentication no
#
# For this to work you will also need host keys in /etc/ssh_known_hosts
RhostsRSAAuthentication no
#
RSAAuthentication yes
# To disable tunneled clear text passwords, change to no here!
PasswordAuthentication yes
PermitEmptyPasswords no
# Uncomment to disable s/key passwords
#SkeyAuthentication no
# To change Kerberos options
#KerberosAuthentication no
#KerberosOrLocalPasswd yes
#AFSTokenPassing no
#KerberosTicketCleanup no
# Kerberos TGT Passing does only work with the AFS kaserver
#KerberosTgtPassing yes
CheckMail no
UseLogin no
#Subsystem sftp /usr/local/sbin/sftpd
[目录]
scp
scp 的使用方法
scp [-pqrvC46] [-P port] [-c cipher] [-i identity_file]
[[user@]host1:]file1 [...] [[user@]host2:]file2
使用ssh-keygen生成key
ssh-keygen [ -f key_file ] 生成RSA key, 用于ssh1协议
ssh-keygen -d [ -f key_file ] 生成DSA key, 用于ssh2协议
注意: 作为Host Key使用的key, passphrase必须为空
[目录]
使用key做验证
使用key做验证
缺省情况下, ssh仍然使用传统的口令验证. 可以通过简单的配置, 实现通过密匙的验证.
过程:
用ssh-keygen生成用户密匙, 存放在~/.ssh/下, 文件名为 identity.将~/.ssh/identity.pub复制到远程机器用户帐户上, 并添加到~/.ssh/authorized_keys中.必须确定远程机器上~/.ssh/authorized_keys属性为400.这样就可以通过key来验证用户了.可以通过一个命令来完成以上操作
ssh-keygen -f ~/.ssh/identity; cat ~/.ssh/identity.pub | /
ssh user@remote "mkdir ~/.ssh; > ~/.ssh/authorized_keys ; /
chmod 400 ~/.ssh/authorized_keys"
[目录]
rsync与ssh配合使用
rsync与ssh配合使用
rsync是个非常好的网络文件传输程序, 很适合大量文件的复制.
rsync可以与rsh, ssh配合使用, server/client都需要安装ssh/rsync
rsync -e /usr/bin/ssh ......
或者定义环境变量
export RSYNC_RSH=/usr/bin/ssh (bash)
就可以直接让rsync去使用ssh了.
[目录]
cvs
cvs中使用ssh
cvs可以使用rsh, ssh作为传输管道.
export CVS_RSH=/usr/bin/ssh
export CVSROOT=:ext:user@host:/path
[目录]
Khttpd
一、简介
从linux2.4.13开始,在Networking options出现了一个试验性的选项-"[ ] Kernel httpd acceleration (EXPERIMENTAL)",什么是kHTTPd呢?它是一个Linux环境下的web服务器。kHTTPd和其它web服务器的不同之处在于其是作为内核的一部分运行在Linux的内核中(可以看成是一个设备驱动)。
kHTTPd仅仅处理静态(基于静态文件的)的web页面,而将所有的对于非静态内容的请求传递给正常的运行于用户空间的web服务器来处理,如apache、Zeus等,而这些运行在用户空间的web服务器并不需要任何修改。
对于静态web页面的http请求的处理不是一个非常复杂过程,但是这却是web服务中非常重要的一个部分,因为至少网站中大多数图形都是静态的,而且还有很多html文件时静态的。一个常规的web服务器对于静态页面的http请求处理非常简单,仅仅是拷贝“文件到网络”的简单操作。如果这些操作在内核中完成将变得非常高效。例如也是完成类似的功能的NFS服务器也是运行在内核中的。
通过在内核中实现web请求处理加速,通常意义的web服务器-如apache等则可以专注于处理那些动态web请求。
注:这里Apache指代任何一个web服务器。
二、快速入门
1) 编译并加载模块
2) 如果需要,通过/proc/sys/net/khttpd来对模块进行配置
3) echo 1 /proc/sys/net/khttpd/start
卸载:
echo 1 /proc/sys/net/khttpd/stopecho 1 /proc/sys/net/khttpd/unload rmmod khttpd
三、配置
1、操作模式
这里有两种推荐操作模式:
1) "Apache" 是主web服务器,kHTTPd是辅助web服务器
clientport - 80
serverport - 8080 (or whatever)
2) kHTTPd是主web服务器, "Apache"是辅助web服务器
clientport - 8080 (or whatever)
serverport - 80
2、配置kHTTPd
在启动kHTTPd之前首先需要对它进行配置。这是通过/proc文件系统来进行的,因此可以在脚本中实现自动配置。大多数参数只能在kHTTPd没有启动以前才能设置。
一般可以配置以下参数:
1) kHTTPd监听的服务请求端口2) "Apache"监听的端口(在"localhost"接口中)3) web文档root目录(documentroot)4) 动态内容的请求所包含的字符串(可选的)[ 默认包括"cgi-bin"]
在这里指定的documentroot一定要保证和运行在用户空间的web服务器的documentroot相一致,因为kHTTPd可能会将任何请求重定向给用户空间的web服务器来处理。
一个典型的脚本(第一种操作模式)
#!/bin/shmodprobe khttpdecho 80 /proc/sys/net/khttpd/clientportecho 8080 /proc/sys/net/khttpd/serverportecho /var/www /proc/sys/net/khttpd/documentrootecho php3 /proc/sys/net/khttpd/dynamicecho shtml /proc/sys/net/khttpd/dynamicecho 1 /proc/sys/net/khttpd/start
对于第二种操作模式,其典型的脚本如下: