你在这里

搭建一个PHP版本passmail服务器给AI 2服务器发送改密码链接邮件

主标签

一.网络要求:

   a.   学生机(客户机):              可以访问App Inventor 2服务器;可以登录自己邮箱查看邮件。

   b.   App Inventor 2服务器:         可以访问存放 passmail的服务器,可以被学生机访问,无需对外网(internet)提供服务。

   c.  passmail的服务器:                 能被App Inventor 2服务器访问(或者就是安装在App Inventor 2服务器上),可以访问外网,无需对外网(internet)提供服务。

二.基本组件:

      nginx + php 或者 apache+php。

三.搭建passmail的服务器过程:

   1.常规安装nginx或apahce,并支持PHP。

   2.新建一个子目录passmail,并将下列内容保存为  index.php

<?php
 error_reporting(0);
 $debug=0;                                  //调试
 $smtp_host="smtp.qq.com";                  //发送邮件服务器地址(SMTP)
 $smtp_port="465";                          //服务器端口
 $smtp_ssl = "1";                           //邮件服务器是否要求加密连接(SSL),1,是;0,否
 $smtp_user="fs**t@qq.com";                 //邮件登陆帐号
 $smtp_pass ="x****c";                      //帐号密码
 $smtp_mail="fs**t@qq.com";                 //发件人邮箱  与邮件登陆账号不同可能会被认为发垃圾邮件
 $password="cha***me";                       //与WEB-INF\appengine-web.xml中<property name="localauth.mailserver.password" value="changeme" />value值一致

 $mail_subject="设置您的登陆密码";            //邮件主题

 $pass=filterStr($_POST["pass"]);
 $url=filterStr($_POST["url"]);
 $email=filterStr($_POST["email"]);

header('Cache-Control:no-cache,must-revalidate');  
header('Pragma:no-cache');  

if($pass=="" and $debug==1){
echo'<!doctype html>
<html>
 <head>
  <meta charset="UTF-8">
   <title>passmail</title>
 </head>
 <body>
    <form method="post" >
    <input type="hidden" name="pass" value="'.$password.'">
    <input type="hidden" name="email" value="'.$smtp_mail.'">
    <input type="hidden" name="url" value="http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'">
    <input type="submit"  style="font-size:20px;width: 250px;height:100px;" value="给自己发测试邮件">
  </from>
 </body>
</html>
';
  exit();
}

if($pass!==$password ){
   if($debug==1){ echo "Password error.";}
   exit();
}

if (!validEmail($email)) {
   if($debug==1){ echo "Email error.";}
   exit();
}
$content="<a  href='".$url."' target='_blank'>".$url."</a>";
$content="请复制链接在浏览器地址栏打开,设置您的登陆密码:".$content;
$content.="<br>请注意使用chrome浏览器。";
$mail = new MySendMail();
$mail->setServer($smtp_host,  $smtp_user,$smtp_pass,  $smtp_port, $smtp_ssl);

$mail->setFrom($smtp_mail);

$mail->setReceiver($email);

$mail->setMail( $mail_subject, $content);

$sendinfo=$mail->sendMail();

if(!$sendinfo and $debug==1){

   echo $mail->error();

}
function filterStr($str){
       $str = trim($str);
       $str = strip_tags($str);
       $str = htmlspecialchars($str,ENT_QUOTES);
       $str = addslashes($str);
       return $str;
}

function validEmail($email)
{
   $isValid = true;
   $atIndex = strrpos($email, "@");
   if (is_bool($atIndex) && !$atIndex)
   {
      $isValid = false;
   }
   else
   {
      $domain = substr($email, $atIndex+1);
      $local = substr($email, 0, $atIndex);
      $localLen = strlen($local);
      $domainLen = strlen($domain);
      if ($localLen < 1 || $localLen > 64)
      {
         // local part length exceeded
         $isValid = false;
      }
      else if ($domainLen < 1 || $domainLen > 255)
      {
         // domain part length exceeded
         $isValid = false;
      }
      else if ($local[0] == '.' || $local[$localLen-1] == '.')
      {
         // local part starts or ends with '.'
         $isValid = false;
      }
      else if (preg_match('/\\.\\./', $local))
      {
         // local part has two consecutive dots
         $isValid = false;
      }
      else if (!preg_match('/^[A-Za-z0-9\\-\\.]+$/', $domain))
      {
         // character not valid in domain part
         $isValid = false;
      }
      else if (preg_match('/\\.\\./', $domain))
      {
         // domain part has two consecutive dots
         $isValid = false;
      }
      else if(!preg_match('/^(\\\\.|[A-Za-z0-9!#%&`_=\\/$\'*+?^{}|~.-])+$/', str_replace("\\\\","",$local)))
      {
         // character not valid in local part unless
         // local part is quoted
         if (!preg_match('/^"(\\\\"|[^"])+"$/',
             str_replace("\\\\","",$local)))
         {
            $isValid = false;
         }
      }
      if ($isValid && !(checkdnsrr($domain,"MX") || checkdnsrr($domain,"A")))
      {
         // domain not found in DNS
         $isValid = false;
      }
   }
   return $isValid;
}

class MySendMail {
     /**
     * @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) {
         //echo 'Send command:' . $command . ',expected code:' . $code . '<br />';
         //发送命令给服务器
        try{
             if(socket_write($this->_socket, $command, strlen($command))){
                 //当邮件内容分多次发送时,没有$code,服务器没有返回
                if(empty($code))  {
                     return true;
                 }
                 //读取服务器返回
                $data = trim(socket_read($this->_socket, 1024));
                 //echo 'response:' . $data . '<br /><br />';
                 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) {
         //echo 'Send command:' . $command . ',expected code:' . $code . '<br />';
         try {
             if(fwrite($this->_socket, $command)){
                 //当邮件内容分多次发送时,没有$code,服务器没有返回
                if(empty($code))  {
                     return true;
                 }
                 //读取服务器返回
                $data = trim(fread($this->_socket, 1024));
                 //echo 'response:' . $data . '<br /><br />';
                 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;
     }
 }
 ?>

    3.修改3行$debug=1;  第4-8行,根据自己qq邮箱或其他支持smtp功能的邮箱设置参数。

    4.浏览器打开http://passmail服务器的IP/passmail ,点击页面中的”给自己发测试邮件“ 按钮,自己设置的邮箱中应该能收到一份测试的邮件。若没收到,请检查自己php环境和第4-8行邮箱设置信息。

四、配置war\WEB-INF\appengine-web.xml文件,修改2行:

 <!-- Flags associated with localauth code -->

 <property name="localauth.mailserver" value="https://osiris.mit.edu/passmail/" />

 <property name="localauth.mailserver.password" value="changeme" />

         其中”https://osiris.mit.edu/passmail/“修改为 http://passmail服务器的IP/passmail;

        ”changeme“ 修改为index。php第10行中相关值。

            最好改回:$debug=0;

五、上述功能已经整合到个人修改版:   https://gte.fsyz.net/node/1878   

邮件发送类来自: http://m.jb51.net/article/61771.htm