php守候進程-發送隊列郵件 (一)

在linux系統下運行php

./demo.php

注:只能在liunx系統下運行html

demo.conflinux

# demo.conf

# daemon mode
daemon yes

# child number
child_num 5

# child user and group
user nobody
group nogroup

# retry times
retry_times 3

# pid file
pid_file /tmp/demo.pid

# log file
log_file /tmp/demo.log

demo.phpredis

#!/usr/bin/env php
<?php

error_reporting(0);

include "SendMail.php";

define("EXIT_SUCCESS", 		0);
define("EXIT_FAILURE", 		1);

define("DEMO_LOG_DEBUG", 	0);
define("DEMO_LOG_INFO",  	1);
define("DEMO_LOG_WARN",  	2);
define("DEMO_LOG_ERR",   	3);

// default config
$config = array(
	"daemon"		=> FALSE,
	"child_num"		=> 1,
	"user"			=> NULL,
	"group"			=> NULL,
	"retry_times"	=> 0,
	"pid_file"		=> NULL,
	"log"			=> STDERR,
	);

// parse config file
parse_config($config);

if ($argc > 1) {
	switch ($argv[1]) {

	case "help":
		exit("Usage: ./demo.php {start|restart|reload|quit}\n\n");

	case "start":
		break;

	case "restart":
	case "reload":
	case "quit":
		$fp = fopen($config["pid_file"], "r");
		if (!$fp) {
			exit("Can't open pid file '{$config["pid_file"]}'.\n");
		}

		$pid = fgets($fp, 6);
		if ($pid) {
			$pid = intval($pid);
		}
		fclose($fp);

		if ($argv[1] == "restart" || $argv[1] == "quit") {
			posix_kill($pid, SIGINT);
			if ($argv[1] == "restart") {
				unlink($config["pid_file"]);
			} else {
				exit(EXIT_SUCCESS);
			}

		} else {
			posix_kill($pid, SIGHUP);
			exit(EXIT_SUCCESS);
		}
		break;
	}
}

if (file_exists($config["pid_file"])) {
	exit("pid file '{$config["pid_file"]}' already exits.\n");
}

if ($config["daemon"]) {
	if (pcntl_fork() > 0) {
		// parent exit
		exit(EXIT_SUCCESS);
	}

	// set session id
	posix_setsid();

	// write pid file
	if ($config["pid_file"]) {
		$fp = fopen($config["pid_file"], "w");
		if (!$fp) {
			exit("Can't open pid file '{$config["pid_file"]}'.\n");
		}

		if (!fputs($fp, posix_getpid())) {
			exit("Can't write pid to file '{$config["pid_file"]}'.\n");
		}
		fclose($fp);
	}
}

write_log(DEMO_LOG_INFO, "main process started");

declare(ticks = 1);

pcntl_signal(SIGTERM, SIG_IGN);

pcntl_signal(SIGHUP,  "reload");
pcntl_signal(SIGINT,  "quit");
pcntl_signal(SIGCHLD, "handle_child");

$childs = array();

$reparse_config = FALSE;

$run = TRUE;
while ($run) {

	if ($reparse_config) {
		write_log(DEMO_LOG_INFO, "reparse config file");
		parse_config($config);
		posix_kill(0, SIGTERM);
		$reparse_config = FALSE;
	}

	for ($i = count($childs); $i < $config["child_num"]; ++$i) {
		if (($child_pid = pcntl_fork()) == 0) {
			do_child($config);
			exit(EXIT_SUCCESS);
		}

		write_log(DEMO_LOG_INFO, "start child process: %d", $child_pid);
		$childs[$child_pid] = 1;
	}

	sleep(1);
}

// kill child
posix_kill(0, SIGTERM);

// remove pid file
@unlink($config["pid_file"]);

write_log(DEMO_LOG_INFO, "main process exit");
exit(EXIT_SUCCESS);

function reload() {
	$GLOBALS["reparse_config"] = TRUE;
}

function quit() {
	$GLOBALS["run"] = FALSE;
}

function handle_child() {
	while (($child_pid = pcntl_waitpid(-1, $status, WNOHANG)) > 0) {
		write_log(DEMO_LOG_INFO, "exit child process: %d", $child_pid);
		unset($GLOBALS["childs"][$child_pid]);
	}
}

function parse_config(Array &$config) {
	$fp = fopen("demo.conf", "r");
	if (!$fp) {
		exit("Can't open config file.\n");
	}

	$lineno = 0;
	while (($line = fgets($fp, 1024)) !== FALSE) {
		++$lineno;

		$line = trim($line);
		if (!$line || $line[0] == "#") {
			continue;
		}

		$params = preg_split("/\s+/", $line);
		switch (strtolower($params[0])) {

		case "daemon":
			if ($params[1] == "yes") {
				$config["daemon"] = TRUE;
			} else if ($params[1] == "no") {
				$config["daemon"] = FALSE;
			} else {
				$err = "daemon value must be 'yes' or 'no'";
				goto parse_failed;
			}
			break;

		case "child_num":
			$child_num = intval($params[1]);
			if ($child_num < 0 || $child_num > 1024) {
				$err = "invalid child_num value '{$params[1]}'";
				goto parse_failed;
			}
			$config["child_num"] = $child_num;
			break;

		case "retry_times":
			$retry_times = intval($params[1]);
			if ($retry_times < 1 || $retry_times > 100) {
				$err = "invalid retry_times value '{$params[1]}'";
				goto parse_failed;
			}
			$config["retry_times"] = $retry_times;
			break;

		case "user":
			$user = posix_getpwnam($params[1]);
			if (!$user) {
				$err = "invalid user value '{$params[1]}'";
				goto parse_failed;
			}
			$config["user"] = $user["uid"];
			break;

		case "group":
			$group = posix_getgrnam($params[1]);
			if (!$group) {
				$err = "invalid group value '{$params[1]}'";
				goto parse_failed;
			}
			$config["group"] = $group["gid"];
			break;

		case "pid_file":
			$config["pid_file"] = $params[1];
			break;

		case "log_file":
			$log = fopen($params[1], "a");
			if (!$log) {
				$err = "Can't open log file '{$params[1]}'";
				goto parse_failed;
			}
			$config["log"] = $log;
			break;
		}

		continue;

	parse_failed:
		fprintf(STDERR, "\n*** FATAL CONFIG FILE ERROR ***\n");
		fprintf(STDERR, "Reading the configuration file, at line %d\n", $lineno);
		fprintf(STDERR, ">>> '%s'\n", $line);
		fprintf(STDERR, "%s\n", $err);
		exit(EXIT_FAILURE);
	}
	fclose($fp);
}

function write_log($level, $fmt) {
	$chars = ".-*#";

	if (func_num_args() > 2) {
		$args = func_get_args();
		$err  = vsprintf($args[1], array_slice($args, 2));

	} else {
		$err = $fmt;
	}

	fprintf($GLOBALS["config"]["log"], "[%d] [%s] %s %s\n", 
			posix_getpid(), date("Y-m-d H:i:s"), $chars[$level], $err);
}

function do_child() {
	global $run;

	$user 		 = $GLOBALS["config"]["user"];
	$group 		 = $GLOBALS["config"]["group"];
	$retry_times = $GLOBALS["config"]["retry_times"];

	if ($user) {
		posix_setuid($user);
	}

	if ($group) {
		posix_setuid($group);
	}

	pcntl_signal(SIGTERM, "quit");

	$redis = new Redis;
	$redis->pconnect("127.0.0.1", 6379);

	while ($run) {
		try {
			$email = $redis->lpop("email_queue");
			if ($email) {
				for ($i = 0; $i < $retry_times; ++$i) {
					if (do_sendmail($email)) {
						write_log(DEMO_LOG_INFO, "send mail to '%s' success", $email);
						break;
					}

					write_log(DEMO_LOG_ERR, "send mail to '%s' failed, try again", $email);
				}

				if ($i == $retry_times) {
					write_log(DEMO_LOG_ERR, "send mail to '%s' failed", $email);
				}
			}

		} catch (RedisException $e) {
			write_log(LOG_ERR, "receive message failed: %s", $e->getMessage());
			exit(EXIT_FAILURE);
		}

		sleep(1);
	}

	exit(EXIT_SUCCESS);
}

function do_sendmail($email) {
	$mail = new SendMail();
	$mail->setServer("smtp.exmail.qq.com", "dingpeilong@xywy.com", "PLDing1989.com", 465, 1);
	$mail->setFrom("dingpeilong@xywy.com");
	$mail->setReceiver("77676182@qq.com");
	$mail->setMail("test", "**hello world!**");
	return $mail->sendMail();
}

SendMail.php安全

<?php
/**
* 郵件發送類
* 支持發送純文本郵件和HTML格式的郵件,能夠多收件人,多抄送,多祕密抄送,帶附件(單個或多個附件),支持到服務器的ssl鏈接
* 須要的php擴展:sockets、Fileinfo和openssl。
* 編碼格式是UTF-8,傳輸編碼格式是base64
* @example
* $mail = new SendMail();
* $mail->setServer("smtp@126.com", "XXXXX@126.com", "XXXXX"); //設置smtp服務器,普通鏈接方式
* $mail->setServer("smtp.gmail.com", "XXXXX@gmail.com", "XXXXX", 465, true); //設置smtp服務器,到服務器的SSL鏈接
* $mail->setFrom("XXXXX"); //設置發件人
* $mail->setReceiver("XXXXX"); //設置收件人,多個收件人,調用屢次
* $mail->setCc("XXXX"); //設置抄送,多個抄送,調用屢次
* $mail->setBcc("XXXXX"); //設置祕密抄送,多個祕密抄送,調用屢次
* $mail->addAttachment("XXXX"); //添加附件,多個附件,調用屢次
* $mail->setMail("test", "**test**"); //設置郵件主題、內容
* $mail->sendMail(); //發送
*/
class SendMail {
    /**
    * @var string 郵件傳輸代理用戶名
    * @access protected
    */
    protected $_userName;

    /**
    * @var string 郵件傳輸代理密碼
    * @access protected
    */
    protected $_password;

    /**
    * @var string 郵件傳輸代理服務器地址
    * @access protected
    */
    protected $_sendServer;

    /**
    * @var int 郵件傳輸代理服務器端口
    * @access protected
    */
    protected $_port;

    /**
    * @var string 發件人
    * @access protected
    */
    protected $_from;

    /**
    * @var array 收件人
    * @access protected
    */
    protected $_to = array();

    /**
    * @var array 抄送
    * @access protected
    */
    protected $_cc = array();

    /**
    * @var array 祕密抄送
    * @access protected
    */
    protected $_bcc = array();

    /**
    * @var string 主題
    * @access protected
    */
    protected $_subject;

    /**
    * @var string 郵件正文
    * @access protected
    */
    protected $_body;

    /**
    * @var array 附件
    * @access protected
    */
    protected $_attachment = array();

    /**
    * @var reource socket資源
    * @access protected
    */
    protected $_socket;

    /**
    * @var reource 是不是安全鏈接
    * @access protected
    */
    protected $_isSecurity;

    /**
    * @var string 錯誤信息
    * @access protected
    */
    protected $_errorMessage;

    /**
    * 設置郵件傳輸代理,若是是能夠匿名發送有郵件的服務器,只需傳遞代理服務器地址就行
    * @access public
    * @param string $server 代理服務器的ip或者域名
    * @param string $username 認證帳號
    * @param string $password 認證密碼
    * @param int $port 代理服務器的端口,smtp默認25號端口
    * @param boolean $isSecurity 到服務器的鏈接是否爲安全鏈接,默認false
    * @return boolean
    */
    public function setServer($server, $username="", $password="", $port=25, $isSecurity=false) {
        $this->_sendServer = $server;
        $this->_port = $port;
        $this->_isSecurity = $isSecurity;
        $this->_userName = empty($username) ? "" : base64_encode($username);
        $this->_password = empty($password) ? "" : base64_encode($password);
        return true;
    }

    /**
    * 設置發件人
    * @access public
    * @param string $from 發件人地址
    * @return boolean
    */
    public function setFrom($from) {
        $this->_from = $from;
        return true;
    }

    /**
    * 設置收件人,多個收件人,調用屢次.
    * @access public
    * @param string $to 收件人地址
    * @return boolean
    */
    public function setReceiver($to) {
        $this->_to[] = $to;
        return true;
    }

    /**
    * 設置抄送,多個抄送,調用屢次.
    * @access public
    * @param string $cc 抄送地址
    * @return boolean
    */
    public function setCc($cc) {
        $this->_cc[] = $cc;
        return true;
    }

    /**
    * 設置祕密抄送,多個祕密抄送,調用屢次
    * @access public
    * @param string $bcc 祕密抄送地址
    * @return boolean
    */
    public function setBcc($bcc) {
        $this->_bcc[] = $bcc;
        return true;
    }

    /**
    * 設置郵件附件,多個附件,調用屢次
    * @access public
    * @param string $file 文件地址
    * @return boolean
    */
    public function addAttachment($file) {
        if(!file_exists($file)) {
            $this->_errorMessage = "file " . $file . " does not exist.";
            return false;
        }
        $this->_attachment[] = $file;
        return true;
    }

    /**
    * 設置郵件信息
    * @access public
    * @param string $body 郵件主題
    * @param string $subject 郵件主體內容,能夠是純文本,也但是是HTML文本
    * @return boolean
    */
    public function setMail($subject, $body) {
        $this->_subject = base64_encode($subject);
        $this->_body = base64_encode($body);
        return true;
    }

    /**
    * 發送郵件
    * @access public
    * @return boolean
    */
    public function sendMail() {
        $command = $this->getCommand();

        $this->_isSecurity ? $this->socketSecurity() : $this->socket();

        foreach ($command as $value) {
            $result = $this->_isSecurity ? $this->sendCommandSecurity($value[0], $value[1]) : $this->sendCommand($value[0], $value[1]);
            if($result) {
                continue;
            }
            else{
                return false;
            }
        }

        //其實這裏也不必關閉,smtp命令:QUIT發出以後,服務器就關閉了鏈接,本地的socket資源會自動釋放
        $this->_isSecurity ? $this->closeSecutity() : $this->close();
        return true;
    }

    /**
    * 返回錯誤信息
    * @return string
    */
    public function error(){
        if(!isset($this->_errorMessage)) {
            $this->_errorMessage = "";
        }
        return $this->_errorMessage;
    }

    /**
    * 返回mail命令
    * @access protected
    * @return array
    */
    protected function getCommand() {
        $separator = "----=_Part_" . md5($this->_from . time()) . uniqid(); //分隔符

        $command = array(
                array("HELO sendmail\r\n", 250)
            );
        if(!empty($this->_userName)){
            $command[] = array("AUTH LOGIN\r\n", 334);
            $command[] = array($this->_userName . "\r\n", 334);
            $command[] = array($this->_password . "\r\n", 235);
        }

        //設置發件人
        $command[] = array("MAIL FROM: <" . $this->_from . ">\r\n", 250);
        $header = "FROM: <" . $this->_from . ">\r\n";

        //設置收件人
        if(!empty($this->_to)) {
            $count = count($this->_to);
            if($count == 1){
                $command[] = array("RCPT TO: <" . $this->_to[0] . ">\r\n", 250);
                $header .= "TO: <" . $this->_to[0] .">\r\n";
            }
            else{
                for($i=0; $i<$count; $i++){
                    $command[] = array("RCPT TO: <" . $this->_to[$i] . ">\r\n", 250);
                    if($i == 0){
                        $header .= "TO: <" . $this->_to[$i] .">";
                    }
                    elseif($i + 1 == $count){
                        $header .= ",<" . $this->_to[$i] .">\r\n";
                    }
                    else{
                        $header .= ",<" . $this->_to[$i] .">";
                    }
                }
            }
        }

        //設置抄送
        if(!empty($this->_cc)) {
            $count = count($this->_cc);
            if($count == 1){
                $command[] = array("RCPT TO: <" . $this->_cc[0] . ">\r\n", 250);
                $header .= "CC: <" . $this->_cc[0] .">\r\n";
            }
            else{
                for($i=0; $i<$count; $i++){
                    $command[] = array("RCPT TO: <" . $this->_cc[$i] . ">\r\n", 250);
                    if($i == 0){
                    $header .= "CC: <" . $this->_cc[$i] .">";
                    }
                    elseif($i + 1 == $count){
                        $header .= ",<" . $this->_cc[$i] .">\r\n";
                    }
                    else{
                        $header .= ",<" . $this->_cc[$i] .">";
                    }
                }
            }
        }

        //設置祕密抄送
        if(!empty($this->_bcc)) {
            $count = count($this->_bcc);
            if($count == 1) {
                $command[] = array("RCPT TO: <" . $this->_bcc[0] . ">\r\n", 250);
                $header .= "BCC: <" . $this->_bcc[0] .">\r\n";
            }
            else{
                for($i=0; $i<$count; $i++){
                    $command[] = array("RCPT TO: <" . $this->_bcc[$i] . ">\r\n", 250);
                    if($i == 0){
                    $header .= "BCC: <" . $this->_bcc[$i] .">";
                    }
                    elseif($i + 1 == $count){
                        $header .= ",<" . $this->_bcc[$i] .">\r\n";
                    }
                    else{
                        $header .= ",<" . $this->_bcc[$i] .">";
                    }
                }
            }
        }

        //主題
        $header .= "Subject: =?UTF-8?B?" . $this->_subject ."?=\r\n";
        if(isset($this->_attachment)) {
            //含有附件的郵件頭須要聲明成這個
            $header .= "Content-Type: multipart/mixed;\r\n";
        }
        elseif(false){
            //郵件體含有圖片資源的,且包含的圖片在郵件內部時聲明成這個,若是是引用的遠程圖片,就不須要了
            $header .= "Content-Type: multipart/related;\r\n";
        }
        else{
            //html或者純文本的郵件聲明成這個
            $header .= "Content-Type: multipart/alternative;\r\n";
        }

        //郵件頭分隔符
        $header .= "\t" . 'boundary="' . $separator . '"';

        $header .= "\r\nMIME-Version: 1.0\r\n";

        //這裏開始是郵件的body部分,body部分分紅幾段發送
        $header .= "\r\n--" . $separator . "\r\n";
        $header .= "Content-Type:text/html; charset=utf-8\r\n";
        $header .= "Content-Transfer-Encoding: base64\r\n\r\n";
        $header .= $this->_body . "\r\n";
        $header .= "--" . $separator . "\r\n";

        //加入附件
        if(!empty($this->_attachment)){
            $count = count($this->_attachment);
            for($i=0; $i<$count; $i++){
                $header .= "\r\n--" . $separator . "\r\n";
                $header .= "Content-Type: " . $this->getMIMEType($this->_attachment[$i]) . '; name="=?UTF-8?B?' . base64_encode( basename($this->_attachment[$i]) ) . '?="' . "\r\n";
                $header .= "Content-Transfer-Encoding: base64\r\n";
                $header .= 'Content-Disposition: attachment; filename="=?UTF-8?B?' . base64_encode( basename($this->_attachment[$i]) ) . '?="' . "\r\n";
                $header .= "\r\n";
                $header .= $this->readFile($this->_attachment[$i]);
                $header .= "\r\n--" . $separator . "\r\n";
            }
        }

        //結束郵件數據發送
        $header .= "\r\n.\r\n";

        $command[] = array("DATA\r\n", 354);
        $command[] = array($header, 250);
        $command[] = array("QUIT\r\n", 221);

        return $command;
    }

    /**
    * 發送命令
    * @access protected
    * @param string $command 發送到服務器的smtp命令
    * @param int $code 指望服務器返回的響應嗎
    * @return boolean
    */
    protected function sendCommand($command, $code) {
        //發送命令給服務器
        try{
            if(@socket_write($this->_socket, $command, strlen($command))){

                //當郵件內容分屢次發送時,沒有$code,服務器沒有返回
                if(empty($code))  {
                    return true;
                }

                //讀取服務器返回
                $data = trim(socket_read($this->_socket, 1024));

                if($data) {
                    $pattern = "/^".$code."+?/";
                    if(preg_match($pattern, $data)) {
                        return true;
                    }
                    else{
                        $this->_errorMessage = "Error:" . $data . "|**| command:";
                        return false;
                    }
                }
                else{
                    $this->_errorMessage = "Error:" . socket_strerror(socket_last_error());
                    return false;
                }
            }
            else{
                $this->_errorMessage = "Error:" . socket_strerror(socket_last_error());
                return false;
            }
        }catch(Exception $e) {
            $this->_errorMessage = "Error:" . $e->getMessage();
        }
    }

    /**
    * 安全鏈接發送命令
    * @access protected
    * @param string $command 發送到服務器的smtp命令
    * @param int $code 指望服務器返回的響應嗎
    * @return boolean
    */
    protected function sendCommandSecurity($command, $code) {
        try {
            if(fwrite($this->_socket, $command)){
                //當郵件內容分屢次發送時,沒有$code,服務器沒有返回
                if(empty($code))  {
                    return true;
                }
                //讀取服務器返回
                $data = trim(fread($this->_socket, 1024));

                if($data) {
                    $pattern = "/^".$code."+?/";
                    if(preg_match($pattern, $data)) {
                        return true;
                    }
                    else{
                        $this->_errorMessage = "Error:" . $data . "|**| command:";
                        return false;
                    }
                }
                else{
                    return false;
                }
            }
            else{
                $this->_errorMessage = "Error: " . $command . " send failed";
                return false;
            }
        }catch(Exception $e) {
            $this->_errorMessage = "Error:" . $e->getMessage();
        }
    }

    /**
    * 讀取附件文件內容,返回base64編碼後的文件內容
    * @access protected
    * @param string $file 文件
    * @return mixed
    */
    protected function readFile($file) {
        if(file_exists($file)) {
            $file_obj = file_get_contents($file);
            return base64_encode($file_obj);
        }
        else {
            $this->_errorMessage = "file " . $file . " dose not exist";
            return false;
        }
    }

    /**
    * 獲取附件MIME類型
    * @access protected
    * @param string $file 文件
    * @return mixed
    */
    protected function getMIMEType($file) {
        if(file_exists($file)) {
            $mime = mime_content_type($file);
            /*if(! preg_match("/gif|jpg|png|jpeg/", $mime)){
                $mime = "application/octet-stream";
            }*/
            return $mime;
        }
        else {
            return false;
        }
    }

    /**
    * 創建到服務器的網絡鏈接
    * @access protected
    * @return boolean
    */
    protected function socket() {
        //建立socket資源
        $this->_socket = socket_create(AF_INET, SOCK_STREAM, getprotobyname('tcp'));

        if(!$this->_socket) {
            $this->_errorMessage = socket_strerror(socket_last_error());
            return false;
        }

        socket_set_block($this->_socket);//設置阻塞模式

        //鏈接服務器
        if(!@socket_connect($this->_socket, $this->_sendServer, $this->_port)) {
            $this->_errorMessage = socket_strerror(socket_last_error());
            return false;
        }
        $str = socket_read($this->_socket, 1024);
        if(!preg_match("/220+?/", $str)){
            $this->_errorMessage = $str;
            return false;
        }

        return true;
    }

    /**
    * 創建到服務器的SSL網絡鏈接
    * @access protected
    * @return boolean
    */
    protected function socketSecurity() {
        $remoteAddr = "tcp://" . $this->_sendServer . ":" . $this->_port;
        $this->_socket = stream_socket_client($remoteAddr, $errno, $errstr, 30);
        if(!$this->_socket){
            $this->_errorMessage = $errstr;
            return false;
        }

        //設置加密鏈接,默認是ssl,若是須要tls鏈接,能夠查看php手冊stream_socket_enable_crypto函數的解釋
        @stream_socket_enable_crypto($this->_socket, true, STREAM_CRYPTO_METHOD_SSLv23_CLIENT);

        stream_set_blocking($this->_socket, 1); //設置阻塞模式
        $str = fread($this->_socket, 1024);
        if(!preg_match("/220+?/", $str)){
            $this->_errorMessage = $str;
            return false;
        }

        return true;
    }

    /**
    * 關閉socket
    * @access protected
    * @return boolean
    */
    protected function close() {
        if(isset($this->_socket) && is_object($this->_socket)) {
            $this->_socket->close();
            return true;
        }
        $this->_errorMessage = "No resource can to be close";
        return false;
    }

    /**
    * 關閉安全socket
    * @access protected
    * @return boolean
    */
    protected function closeSecutity() {
        if(isset($this->_socket) && is_object($this->_socket)) {
            stream_socket_shutdown($this->_socket, STREAM_SHUT_WR);
            return true;
        }
        $this->_errorMessage = "No resource can to be close";
        return false;
    }
}
相關文章
相關標籤/搜索