


PHP proxy implementation for unified mailbox login (swoole), _PHP tutorial
PHP 进行统一邮箱登陆的代理实现(swoole),
在工作的过程中,经常会有很多应用有发邮件的需求,这个时候需要在每个应用中配置smtp服务器。一旦公司调整了smtp服务器的配置,比如修改了密码等,这个时候对于维护的人员来说要逐一修改应用中smtp的配置。这样的情况虽然不多见,但遇上了还是很头痛的一件事情。
知道了问题,解决起来就有了方向。于是就有了自己开发一个简单的smtp代理的想法,这个代理主要的功能(参照问题)主要是:
1.接受指定IP应用的smtp请求;
2.应用不需要知道smtp的用户和密码;
3.转发应用的smtp请求。
开发的环境:Linux,php(swoole);
代码如下:
<?<span>php </span><span>/*</span><span>* * * SMTP Proxy Server * @author Terry Zhang, 2015-11-13 * * @version 1.0 * * 注意:本程序只能运行在cli模式,且需要扩展Swoole 1.7.20+的支持。 * * Swoole的源代码及安装请参考 https://github.com/swoole/swoole-src/ * * 本程序的使用场景: * * 在多个分散的系统中使用同一的邮件地址进行系统邮件发送时,一旦邮箱密码修改,则要修改每个系统的邮件配置参数。 * 同时,在每个系统中配置邮箱参数,使得邮箱的密码容易外泄。 * * 通过本代理进行邮件发送的客户端,可以随便指定用户名和密码。 * * </span><span>*/</span> <span>//</span><span>error_reporting(0);</span> <span>defined</span>('DEBUG_ON') or <span>define</span>('DEBUG_ON', <span>false</span><span>); </span><span>//</span><span>主目录</span> <span>defined</span>('BASE_PATH') or <span>define</span>('BASE_PATH',<span> __DIR__); </span><span>class</span><span> CSmtpProxy{ </span><span>//</span><span>软件版本</span> <span>const</span> VERSION = '1.0'<span>; </span><span>const</span> EOF = "\r\n"<span>; </span><span>public</span> <span>static</span> <span>$software</span> = "SMTP-Proxy-Server"<span>; </span><span>private</span> <span>static</span> <span>$server_mode</span> =<span> SWOOLE_PROCESS; </span><span>private</span> <span>static</span> <span>$pid_file</span><span>; </span><span>private</span> <span>static</span> <span>$log_file</span><span>; </span><span>private</span> <span>$smtp_host</span> = 'localhost'<span>; </span><span>private</span> <span>$smtp_port</span> = 25<span>; </span><span>private</span> <span>$smtp_user</span> = ''<span>; </span><span>private</span> <span>$smtp_pass</span> = ''<span>; </span><span>private</span> <span>$smtp_from</span> = ''<span>; </span><span>//</span><span>待写入文件的日志队列(缓冲区)</span> <span>private</span> <span>$queue</span> = <span>array</span><span>(); </span><span>public</span> <span>$host</span> = '0.0.0.0'<span>; </span><span>public</span> <span>$port</span> = 25<span>; </span><span>public</span> <span>$setting</span> = <span>array</span><span>(); </span><span>//</span><span>最大连接数</span> <span>public</span> <span>$max_connection</span> = 50<span>; </span><span>/*</span><span>* * @var swoole_server </span><span>*/</span> <span>protected</span> <span>$server</span><span>; </span><span>protected</span> <span>$connection</span> = <span>array</span><span>(); </span><span>public</span> <span>static</span> <span>function</span> setPidFile(<span>$pid_file</span><span>){ self</span>::<span>$pid_file</span> = <span>$pid_file</span><span>; } </span><span>public</span> <span>static</span> <span>function</span> start(<span>$startFunc</span><span>){ </span><span>if</span>(!<span>extension_loaded</span>('swoole'<span>)){ </span><span>exit</span>("Require extension `swoole`.\n"<span>); } </span><span>$pid_file</span> = self::<span>$pid_file</span><span>; </span><span>$server_pid</span> = 0<span>; </span><span>if</span>(<span>is_file</span>(<span>$pid_file</span><span>)){ </span><span>$server_pid</span> = <span>file_get_contents</span>(<span>$pid_file</span><span>); } </span><span>global</span> <span>$argv</span><span>; </span><span>if</span>(<span>empty</span>(<span>$argv</span>[1<span>])){ goto usage; }</span><span>elseif</span>(<span>$argv</span>[1] == 'reload'<span>){ </span><span>if</span> (<span>empty</span>(<span>$server_pid</span><span>)){ </span><span>exit</span>("SMTP Proxy Server is not running\n"<span>); } posix_kill(</span><span>$server_pid</span>,<span> SIGUSR1); </span><span>exit</span><span>; }</span><span>elseif</span> (<span>$argv</span>[1] == 'stop'<span>){ </span><span>if</span> (<span>empty</span>(<span>$server_pid</span><span>)){ </span><span>exit</span>("SMTP Proxy is not running\n"<span>); } posix_kill(</span><span>$server_pid</span>,<span> SIGTERM); </span><span>exit</span><span>; }</span><span>elseif</span> (<span>$argv</span>[1] == 'start'<span>){ </span><span>//</span><span>已存在ServerPID,并且进程存在</span> <span>if</span> (!<span>empty</span>(<span>$server_pid</span>) and posix_kill(<span>$server_pid</span>,(int) 0<span>)){ </span><span>exit</span>("SMTP Proxy is already running.\n"<span>); } </span><span>//</span><span>启动服务器</span> <span>$startFunc</span><span>(); }</span><span>else</span><span>{ usage</span>: <span>exit</span>("Usage: php {<span>$argv</span>[0]} start|stop|reload\n"<span>); } } </span><span>public</span> <span>function</span> __construct(<span>$host</span>,<span>$port</span><span>){ </span><span>$flag</span> =<span> SWOOLE_SOCK_TCP; </span><span>$this</span>->server = <span>new</span> swoole_server(<span>$host</span>,<span>$port</span>,self::<span>$server_mode</span>,<span>$flag</span><span>); </span><span>$this</span>->host = <span>$host</span><span>; </span><span>$this</span>->port = <span>$port</span><span>; </span><span>$this</span>->setting = <span>array</span><span>( </span>'backlog' => 128, 'dispatch_mode' => 2,<span> ); } </span><span>public</span> <span>function</span><span> daemonize(){ </span><span>$this</span>->setting['daemonize'] = 1<span>; } </span><span>public</span> <span>function</span> getConnectionInfo(<span>$fd</span><span>){ </span><span>return</span> <span>$this</span>->server->connection_info(<span>$fd</span><span>); } </span><span>/*</span><span>* * 启动服务进程 * @param array $setting * @throws Exception </span><span>*/</span> <span>public</span> <span>function</span> run(<span>$setting</span> = <span>array</span><span>()){ </span><span>$this</span>->setting = <span>array_merge</span>(<span>$this</span>->setting,<span>$setting</span><span>); </span><span>//</span><span>不使用swoole的默认日志</span> <span>if</span>(<span>isset</span>(<span>$this</span>->setting['log_file'<span>])){ self</span>::<span>$log_file</span> = <span>$this</span>->setting['log_file'<span>]; </span><span>unset</span>(<span>$this</span>->setting['log_file'<span>]); } </span><span>if</span>(<span>isset</span>(<span>$this</span>->setting['max_connection'<span>])){ </span><span>$this</span>->max_connection = <span>$this</span>->setting['max_connection'<span>]; </span><span>unset</span>(<span>$this</span>->setting['max_connection'<span>]); } </span><span>if</span>(<span>isset</span>(<span>$this</span>->setting['smtp_host'<span>])){ </span><span>$this</span>->smtp_host = <span>$this</span>->setting['smtp_host'<span>]; </span><span>unset</span>(<span>$this</span>->setting['smtp_host'<span>]); } </span><span>if</span>(<span>isset</span>(<span>$this</span>->setting['smtp_port'<span>])){ </span><span>$this</span>->smtp_port = <span>$this</span>->setting['smtp_port'<span>]; </span><span>unset</span>(<span>$this</span>->setting['smtp_port'<span>]); } </span><span>if</span>(<span>isset</span>(<span>$this</span>->setting['smtp_user'<span>])){ </span><span>$this</span>->smtp_user = <span>$this</span>->setting['smtp_user'<span>]; </span><span>unset</span>(<span>$this</span>->setting['smtp_user'<span>]); } </span><span>if</span>(<span>isset</span>(<span>$this</span>->setting['smtp_pass'<span>])){ </span><span>$this</span>->smtp_pass = <span>$this</span>->setting['smtp_pass'<span>]; </span><span>unset</span>(<span>$this</span>->setting['smtp_pass'<span>]); } </span><span>if</span>(<span>isset</span>(<span>$this</span>->setting['smtp_from'<span>])){ </span><span>$this</span>->smtp_from = <span>$this</span>->setting['smtp_from'<span>]; </span><span>unset</span>(<span>$this</span>->setting['smtp_from'<span>]); } </span><span>$this</span>->server->set(<span>$this</span>-><span>setting); </span><span>$version</span> = <span>explode</span>('.',<span> SWOOLE_VERSION); </span><span>if</span>(<span>$version</span>[0] == 1 && <span>$version</span>[1] < 7 && <span>$version</span>[2] <20<span>){ </span><span>throw</span> <span>new</span> <span>Exception</span>('Swoole version require 1.7.20 +.'<span>); } </span><span>//</span><span>事件绑定</span> <span>$this</span>->server->on('start',<span>array</span>(<span>$this</span>,'onMasterStart'<span>)); </span><span>$this</span>->server->on('shutdown',<span>array</span>(<span>$this</span>,'onMasterStop'<span>)); </span><span>$this</span>->server->on('ManagerStart',<span>array</span>(<span>$this</span>,'onManagerStart'<span>)); </span><span>$this</span>->server->on('ManagerStop',<span>array</span>(<span>$this</span>,'onManagerStop'<span>)); </span><span>$this</span>->server->on('WorkerStart',<span>array</span>(<span>$this</span>,'onWorkerStart'<span>)); </span><span>$this</span>->server->on('WorkerStop',<span>array</span>(<span>$this</span>,'onWorkerStop'<span>)); </span><span>$this</span>->server->on('WorkerError',<span>array</span>(<span>$this</span>,'onWorkerError'<span>)); </span><span>$this</span>->server->on('Connect',<span>array</span>(<span>$this</span>,'onConnect'<span>)); </span><span>$this</span>->server->on('Receive',<span>array</span>(<span>$this</span>,'onReceive'<span>)); </span><span>$this</span>->server->on('Close',<span>array</span>(<span>$this</span>,'onClose'<span>)); </span><span>$this</span>->server-><span>start(); } </span><span>public</span> <span>function</span> <span>log</span>(<span>$msg</span>,<span>$level</span> = 'debug',<span>$flush</span> = <span>false</span><span>){ </span><span>if</span><span>(DEBUG_ON){ </span><span>$log</span> = <span>date</span>('Y-m-d H:i:s').' ['.<span>$level</span>."]\t" .<span>$msg</span>."\n"<span>; </span><span>if</span>(!<span>empty</span>(self::<span>$log_file</span><span>)){ </span><span>$debug_file</span> = <span>dirname</span>(self::<span>$log_file</span>).'/debug.log'<span>; </span><span>file_put_contents</span>(<span>$debug_file</span>, <span>$log</span>,<span>FILE_APPEND); </span><span>if</span>(<span>filesize</span>(<span>$debug_file</span>) > 10485760){<span>//</span><span>10M</span> <span>unlink</span>(<span>$debug_file</span><span>); } } </span><span>echo</span> <span>$log</span><span>; } </span><span>if</span>(<span>$level</span> != 'debug'<span>){ </span><span>//</span><span>日志记录</span> <span>$this</span>->queue[] = <span>date</span>('Y-m-d H:i:s')."\t[".<span>$level</span>."]\t".<span>$msg</span><span>; } </span><span>if</span>(<span>count</span>(<span>$this</span>->queue)>10 && !<span>empty</span>(self::<span>$log_file</span>) || <span>$flush</span><span>){ </span><span>if</span> (<span>filesize</span>(self::<span>$log_file</span>) > 209715200){ <span>//</span><span>200M</span> <span>rename</span>(self::<span>$log_file</span>,self::<span>$log_file</span>.'.'.<span>date</span>('His'<span>)); } </span><span>$logs</span> = ''<span>; </span><span>foreach</span> (<span>$this</span>->queue <span>as</span> <span>$q</span><span>){ </span><span>$logs</span> .= <span>$q</span>."\n"<span>; } </span><span>file_put_contents</span>(self::<span>$log_file</span>, <span>$logs</span>,<span>FILE_APPEND); </span><span>$this</span>->queue = <span>array</span><span>(); } } </span><span>public</span> <span>function</span><span> shutdown(){ </span><span>return</span> <span>$this</span>->server-><span>shutdown(); } </span><span>public</span> <span>function</span> close(<span>$fd</span><span>){ </span><span>return</span> <span>$this</span>->server->close(<span>$fd</span><span>); } </span><span>public</span> <span>function</span> send(<span>$fd</span>,<span>$data</span><span>){ </span><span>$data</span> = <span>strtr</span>(<span>$data</span>,<span>array</span>("\n" => "", "\0" => "", "\r" => ""<span>)); </span><span>$this</span>-><span>log</span>("[P --> C]\t" . <span>$data</span><span>); </span><span>return</span> <span>$this</span>->server->send(<span>$fd</span>,<span>$data</span>.self::<span>EOF); } </span><span>/*</span><span>+++++++++++++++++++++++++++++++++++++++++++++++++++++++++ + 事件回调 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++</span><span>*/</span> <span>public</span> <span>function</span> onMasterStart(<span>$serv</span><span>){ </span><span>global</span> <span>$argv</span><span>; swoole_set_process_name(</span>'php '.<span>$argv</span>[0].': master -host='.<span>$this</span>->host.' -port='.<span>$this</span>-><span>port); </span><span>if</span>(!<span>empty</span>(<span>$this</span>->setting['pid_file'<span>])){ </span><span>file_put_contents</span>(self::<span>$pid_file</span>, <span>$serv</span>-><span>master_pid); } </span><span>$this</span>-><span>log</span>('Master started.'<span>); } </span><span>public</span> <span>function</span> onMasterStop(<span>$serv</span><span>){ </span><span>if</span> (!<span>empty</span>(<span>$this</span>->setting['pid_file'<span>])){ </span><span>unlink</span>(self::<span>$pid_file</span><span>); } </span><span>$this</span>->shm-><span>delete(); </span><span>$this</span>-><span>log</span>('Master stop.'<span>); } </span><span>public</span> <span>function</span> onManagerStart(<span>$serv</span><span>){ </span><span>global</span> <span>$argv</span><span>; swoole_set_process_name(</span>'php '.<span>$argv</span>[0].': manager'<span>); </span><span>$this</span>-><span>log</span>('Manager started.'<span>); } </span><span>public</span> <span>function</span> onManagerStop(<span>$serv</span><span>){ </span><span>$this</span>-><span>log</span>('Manager stop.'<span>); } </span><span>public</span> <span>function</span> onWorkerStart(<span>$serv</span>,<span>$worker_id</span><span>){ </span><span>global</span> <span>$argv</span><span>; </span><span>if</span>(<span>$worker_id</span> >= <span>$serv</span>->setting['worker_num'<span>]) { swoole_set_process_name(</span>"php {<span>$argv</span>[0]}: worker [task]"<span>); } </span><span>else</span><span> { swoole_set_process_name(</span>"php {<span>$argv</span>[0]}: worker [{<span>$worker_id</span>}]"<span>); } </span><span>$this</span>-><span>log</span>("Worker {<span>$worker_id</span>} started."<span>); } </span><span>public</span> <span>function</span> onWorkerStop(<span>$serv</span>,<span>$worker_id</span><span>){ </span><span>$this</span>-><span>log</span>("Worker {<span>$worker_id</span>} stop."<span>); } </span><span>public</span> <span>function</span> onWorkerError(<span>$serv</span>,<span>$worker_id</span>,<span>$worker_pid</span>,<span>$exit_code</span><span>){ </span><span>$this</span>-><span>log</span>("Worker {<span>$worker_id</span>} error:{<span>$exit_code</span>}."<span>); } </span><span>public</span> <span>function</span> onConnect(<span>$serv</span>,<span>$fd</span>,<span>$from_id</span><span>){ </span><span>if</span>(<span>count</span>(<span>$this</span>->server->connections) <= <span>$this</span>-><span>max_connection){ </span><span>$info</span> = <span>$this</span>->getConnectionInfo(<span>$fd</span><span>); </span><span>if</span>(<span>$this</span>->isIpAllow(<span>$info</span>['remote_ip'<span>])){ </span><span>//</span><span>建立服务器连接</span> <span>$cli</span> = <span>new</span> Client(SWOOLE_SOCK_TCP, SWOOLE_SOCK_ASYNC); <span>//</span><span>异步非阻塞</span> <span>$cli</span>->on('connect',<span>array</span>(<span>$this</span>,'onServerConnect'<span>)); </span><span>$cli</span>->on('receive',<span>array</span>(<span>$this</span>,'onServerReceive'<span>)); </span><span>$cli</span>->on('error',<span>array</span>(<span>$this</span>,'onServerError'<span>)); </span><span>$cli</span>->on('close',<span>array</span>(<span>$this</span>,'onServerClose'<span>)); </span><span>$cli</span>->fd = <span>$fd</span><span>; </span><span>$ip</span> = <span>gethostbyname</span>(<span>$this</span>-><span>smtp_host); </span><span>if</span>(<span>$cli</span>->connect(<span>$ip</span>,<span>$this</span>->smtp_port) !== <span>false</span><span>){ </span><span>$this</span>->connection[<span>$fd</span>] = <span>$cli</span><span>; }</span><span>else</span><span>{ </span><span>$this</span>->close(<span>$fd</span><span>); </span><span>$this</span>-><span>log</span>('Cannot connect to SMTP server. Connection #'.<span>$fd</span>.' close.'<span>); } }</span><span>else</span><span>{ </span><span>$this</span>-><span>log</span>('Blocked clinet connection, IP deny : '.<span>$info</span>['remote_ip'],'warn'<span>); </span><span>$this</span>->server->close(<span>$fd</span><span>); </span><span>$this</span>-><span>log</span>('Connection #'.<span>$fd</span>.' close.'<span>); } }</span><span>else</span><span>{ </span><span>$this</span>-><span>log</span>('Blocked clinet connection, too many connections.','warn'<span>); </span><span>$this</span>->server->close(<span>$fd</span><span>); } } </span><span>public</span> <span>function</span> onReceive(<span>$serv</span>,<span>$fd</span>,<span>$from_id</span>,<span>$recv_data</span><span>){ </span><span>$info</span> = <span>$this</span>->getConnectionInfo(<span>$fd</span><span>); </span><span>$this</span>-><span>log</span>("[P <-- C]\t".<span>trim</span>(<span>$recv_data</span><span>)); </span><span>//</span><span>禁止使用STARTTLS</span> <span>if</span>(<span>strtoupper</span>(<span>trim</span>(<span>$recv_data</span>)) == 'STARTTLS'<span>){ </span><span>$this</span>->server->send(<span>$fd</span>,"502 Not implemented".self::<span>EOF); </span><span>$this</span>-><span>log</span>("[P --> C]\t502 Not implemented"<span>); }</span><span>else</span><span>{ </span><span>//</span><span>重置登陆验证 </span> <span>if</span>(<span>preg_match</span>('/^AUTH\s+LOGIN(.*)/', <span>$recv_data</span>,<span>$m</span><span>)){ </span><span>$m</span>[1] = <span>trim</span>(<span>$m</span>[1<span>]); </span><span>if</span>(<span>empty</span>(<span>$m</span>[1<span>])){ </span><span>//</span><span>只发送AUTH LOGIN 接下来将发送用户名</span> <span>$this</span>->connection[<span>$fd</span>]->user = <span>$this</span>-><span>smtp_user; }</span><span>else</span><span>{ </span><span>$recv_data</span> = 'AUTH LOGIN '.<span>base64_encode</span>(<span>$this</span>->smtp_user).self::<span>EOF; </span><span>$this</span>->connection[<span>$fd</span>]->pass = <span>$this</span>-><span>smtp_pass; } }</span><span>else</span><span>{ </span><span>if</span>(<span>preg_match</span>('/^HELO.*|^EHLO.*/', <span>$recv_data</span><span>)){ </span><span>$recv_data</span> = 'HELO '.<span>$this</span>->smtp_host.self::<span>EOF; } </span><span>//</span><span>重置密码</span> <span>if</span>(!<span>empty</span>(<span>$this</span>->connection[<span>$fd</span>]-><span>pass)){ </span><span>$recv_data</span> = <span>base64_encode</span>(<span>$this</span>->connection[<span>$fd</span>]->pass).self::<span>EOF; </span><span>$this</span>->connection[<span>$fd</span>]->pass = ''<span>; } </span><span>//</span><span>重置用户名</span> <span>if</span>(!<span>empty</span>(<span>$this</span>->connection[<span>$fd</span>]-><span>user)){ </span><span>$recv_data</span> = <span>base64_encode</span>(<span>$this</span>->connection[<span>$fd</span>]->user).self::<span>EOF; </span><span>$this</span>->connection[<span>$fd</span>]->user = ''<span>; </span><span>$this</span>->connection[<span>$fd</span>]->pass = <span>$this</span>-><span>smtp_pass; } </span><span>//</span><span>重置mail from</span> <span>if</span>(<span>preg_match</span>('/^MAIL\s+FROM:.*/', <span>$recv_data</span><span>)){ </span><span>$recv_data</span> = 'MAIL FROM:<'.<span>$this</span>->smtp_from.'>'.self::<span>EOF; } } </span><span>if</span>(<span>$this</span>->connection[<span>$fd</span>]-><span>isConnected()){ </span><span>$this</span>->connection[<span>$fd</span>]->send(<span>$recv_data</span><span>); </span><span>$this</span>-><span>log</span>("[P --> S]\t".<span>trim</span>(<span>$recv_data</span><span>)); } } } </span><span>public</span> <span>function</span> onClose(<span>$serv</span>,<span>$fd</span>,<span>$from_id</span><span>){ </span><span>if</span>(<span>isset</span>(<span>$this</span>->connection[<span>$fd</span><span>])){ </span><span>if</span>(<span>$this</span>->connection[<span>$fd</span>]-><span>isConnected()){ </span><span>$this</span>->connection[<span>$fd</span>]-><span>close(); </span><span>$this</span>-><span>log</span>('Connection on SMTP server close.'<span>); } } </span><span>$this</span>-><span>log</span>('Connection #'.<span>$fd</span>.' close. Flush the logs.','debug',<span>true</span><span>); } </span><span>/*</span><span>--------------------------------------------- * * 服务器连接事件回调 * ----------------------------------------------</span><span>*/</span> <span>public</span> <span>function</span> onServerConnect(<span>$cli</span><span>){ </span><span>$this</span>-><span>log</span>('Connected to SMTP server.'<span>); } </span><span>public</span> <span>function</span> onServerReceive(<span>$cli</span>,<span>$data</span><span>){ </span><span>$this</span>-><span>log</span>("[P <-- S]\t".<span>trim</span>(<span>$data</span><span>)); </span><span>if</span>(<span>$this</span>->server->send(<span>$cli</span>->fd,<span>$data</span><span>)){ </span><span>$this</span>-><span>log</span>("[P --> C]\t".<span>trim</span>(<span>$data</span><span>)); } } </span><span>public</span> <span>function</span> onServerError(<span>$cli</span><span>){ </span><span>$this</span>->server->close(<span>$cli</span>-><span>fd); </span><span>$this</span>-><span>log</span>('Connection on SMTP server error: '.<span>$cli</span>->errCode.' '.socket_strerror(<span>$cli</span>->errCode),'warn'<span>); } </span><span>public</span> <span>function</span> onServerClose(<span>$cli</span><span>){ </span><span>$this</span>-><span>log</span>('Connection on SMTP server close.'<span>); </span><span>$this</span>->server->close(<span>$cli</span>-><span>fd); } </span><span>/*</span><span>* * IP地址过滤 * @param unknown $ip * @return boolean </span><span>*/</span> <span>public</span> <span>function</span> isIpAllow(<span>$ip</span><span>){ </span><span>$pass</span> = <span>false</span><span>; </span><span>if</span>(<span>isset</span>(<span>$this</span>->setting['ip']['allow'<span>])){ </span><span>foreach</span> (<span>$this</span>->setting['ip']['allow'] <span>as</span> <span>$addr</span><span>){ </span><span>$pattern</span> = '/'.<span>str_replace</span>('*','\d+',<span>str_replace</span>('.', '\.', <span>$addr</span>)).'/'<span>; </span><span>if</span>(<span>preg_match</span>(<span>$pattern</span>, <span>$ip</span>) && !<span>empty</span>(<span>$addr</span><span>)){ </span><span>$pass</span> = <span>true</span><span>; </span><span>break</span><span>; } } } </span><span>if</span>(<span>$pass</span><span>){ </span><span>if</span>(<span>isset</span>(<span>$this</span>->setting['ip']['deny'<span>])){ </span><span>foreach</span> (<span>$this</span>->setting['ip']['deny'] <span>as</span> <span>$addr</span><span>){ </span><span>$pattern</span> = '/'.<span>str_replace</span>('*','\d+',<span>str_replace</span>('.', '\.', <span>$addr</span>)).'/'<span>; </span><span>if</span>(<span>preg_match</span>(<span>$pattern</span>, <span>$ip</span>) && !<span>empty</span>(<span>$addr</span><span>)){ </span><span>$pass</span> = <span>false</span><span>; </span><span>break</span><span>; } } } } </span><span>return</span> <span>$pass</span><span>; } } </span><span>class</span> Client <span>extends</span><span> swoole_client{ </span><span>/*</span><span>* * 记录当前连接 * @var unknown </span><span>*/</span> <span>public</span> <span>$fd</span><span> ; </span><span>public</span> <span>$user</span> = ''<span>; </span><span>/*</span><span>* * smtp登陆密码 * @var unknown </span><span>*/</span> <span>public</span> <span>$pass</span> = ''<span>; }</span>
配置文件例子:
<span>/*</span><span>* * 运行配置 </span><span>*/</span> <span>return</span> <span>array</span><span>( </span>'worker_num' => 12, 'log_file' => BASE_PATH.'/logs/proxyserver.log', 'pid_file' => BASE_PATH.'/logs/proxyserver.pid', 'heartbeat_idle_time' => 300, 'heartbeat_check_interval' => 60, 'max_connection' => 50, <br /> //配置真实的smtp信息 'smtp_host' => '', 'smtp_port' => 25, 'smtp_user' => '', 'smtp_pass' => '', 'smtp_from' => '', 'ip' => <span>array</span><span>( </span>'allow' => <span>array</span>('192.168.0.*'), 'deny' => <span>array</span>('192.168.10.*','192.168.100.*'),<span> ) );</span>
运行例子:
defined('BASE_PATH') or define('BASE_PATH', __DIR__); defined('DEBUG_ON') or define('DEBUG_ON', true); //服务器配置 require BASE_PATH.'/CSmtpProxy.php'; $settings = require BASE_PATH.'/conf/config.php'; CSmtpProxy::setPidFile($settings['pid_file']); CSmtpProxy::start(function(){ global $settings; $serv = new CSmtpProxy('0.0.0.0', 25); $serv->daemonize(); $serv->run($settings); });
应用配置:
smtp host: 192.168.0.* //指定smtpproxy 运行的服务器IP。
port: 25
user: xxxx //随意填写
pass: xxxx //随意填写
from: xxxx@xxxx.com // 根据情况填写
——————————————————————————————————————————————————————
存在的问题:
1、不支持ssl模式;
2、应用的from还是要填写正确,否则发出的邮件发件人会显示错误。

Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics

OKX Ouyi is a leading cryptocurrency trading platform. This article will provide detailed steps to guide you on how to register an OKX Ouyi official website account. You will learn how to access the official website, choose the registration method, fill in the necessary information, and complete the registration process. The article also contains information about precautions, such as the importance of using real personal information and setting a strong password.

This guide provides detailed download and installation steps for the official Bitget Exchange app, suitable for Android and iOS systems. The guide integrates information from multiple authoritative sources, including the official website, the App Store, and Google Play, and emphasizes considerations during download and account management. Users can download the app from official channels, including app store, official website APK download and official website jump, and complete registration, identity verification and security settings. In addition, the guide covers frequently asked questions and considerations, such as

A detailed introduction to the login operation of the Sesame Open Exchange web version, including login steps and password recovery process. It also provides solutions to common problems such as login failure, unable to open the page, and unable to receive verification codes to help you log in to the platform smoothly.

Gateio Exchange app download channels for old versions, covering official, third-party application markets, forum communities and other channels. It also provides download precautions to help you easily obtain old versions and solve the problems of discomfort in using new versions or device compatibility.

Gate.io (Sesame Open Door) is the world's leading cryptocurrency trading platform. This article provides a complete tutorial on spot trading of Gate.io. The tutorial covers steps such as account registration and login, KYC certification, fiat currency and digital currency recharge, trading pair selection, limit/market transaction orders, and orders and transaction records viewing, helping you quickly get started on the Gate.io platform for cryptocurrency trading. Whether a beginner or a veteran, you can benefit from this tutorial and easily master the Gate.io trading skills.

As the world's leading digital asset trading platform, Ouyi OKX attracts many investors with its rich trading products, strong security guarantees and convenient user experience. However, the risks of network security are becoming increasingly severe, and how to safely register the official Ouyi OKX account is crucial. This article will provide the latest registration portal for Ouyi OKX official website, and explain in detail the steps and precautions for safe registration, including how to identify the official website, set a strong password, enable two-factor verification, etc., to help you start your digital asset investment journey safely and conveniently. Please note that there are risks in digital asset investment, please make cautious decisions.

The login method of the Sesame Open Door Trading Platform is convenient. Users only need to visit their official website (please search for the domain name yourself), enter the registered email/mobile phone number and password to log in. The platform may enable security verification mechanisms such as 2FA to ensure account security.

This article introduces in detail two methods for downloading Binance APP on Apple iOS system and Android system mobile phones. For iOS systems, since the App Store in China cannot be downloaded directly, users need to use the Apple ID in the outer zone, and they can choose to borrow or register the Apple ID in the outer zone to download it. Android users can directly search and install it in the app store, or visit Binance's official website to scan the QR code to download the installation package. It should be noted that when downloading applications from unofficial channels, you may need to enable the application installation permissions of unknown sources on your phone. No matter which system you are, you can use the Binance APP after downloading.
