2012-03-22 161 views
0

我已经开始使用Zend_Validate_EmailAddress并将mxdeep选项设置为true。对于IP地址在保留IP范围内的MX记录,我认为我得到了一些错误的否定结论。Zend验证电子邮件地址和深度MX检查

一个很好的例子是MX records for harn.ufl.edu。由于IP地址在128.0.0.0/16范围内,它似乎失败了。但它有一个使用8.6.245.30的记录,它不在保留范围内。另一个例子是MX record for martinhealth.org。它的MX记录域使用198.136.38.2。

这是一个技术上不正确的事情,但实际上工作?

+0

你的Windows使用它? _在Windows环境中MX检查仅在使用PHP 5.3或更高版本时可用。在PHP 5.3以下,即使在选项中激活了MX检查,也不会使用._ – 2012-03-22 19:24:48

+0

看起来您在'Zend_Validate_EmailAddress'中遇到了一个错误。 128.'范围中唯一保留的IP地址来自'128.0.0.0' - '128.0.255.255'。 harn的邮件服务器不在此范围内,但似乎Zend_Validate_Email错误地计算了子网掩码导致误报的情况。我会看看我能否找出更多。 – drew010 2012-03-22 20:26:00

+0

如果您即将到来,我设置了这个[聊天](http://chat.stackoverflow.com/rooms/9213/zend-validate-emailaddress)来讨论这个问题,我相信我看到ZF的问题在哪里。 – drew010 2012-03-22 20:48:27

回答

0

作为我发表的评论上的评论,在Zend_Validate_EmailAddress::_isReserved有一个错误。它不仅有问题,而且很难理解逻辑流程。这是一个private函数,所以我将它改为protected,这样我就可以在我的子类中覆盖它。 $_invalidIp阵列中也有一些不正确的范围。

对于我的逻辑检查,我决定最简单的(最清晰的)方式来比较IP地址是将它们转换为它们的十进制整数等值。

这是我的子类:

class My_Validate_EmailAddressDeep extends Zend_Validate_EmailAddress 
{ 
    /** 
    * @var array 
    */ 
    protected $_messageTemplates = array(
     self::INVALID   => "Invalid type given. String expected", 
     self::INVALID_FORMAT  => "'%value%' is not a valid email address in the basic [user]@[hostname] format", 
     self::INVALID_HOSTNAME => "The '%hostname%' part of '%value%' is not a valid hostname", 
     self::INVALID_MX_RECORD => "'%hostname%' does not appear to be configured to accept email", 
     self::INVALID_SEGMENT => "'%hostname%' does not appear to be configured to accept external email", 
     self::DOT_ATOM   => null, 
     self::QUOTED_STRING  => null, 
     self::INVALID_LOCAL_PART => "The '%localPart%' part of '%value%' is not valid", 
     self::LENGTH_EXCEEDED => "'%value%' is longer than the allowed length for an email address", 
    ); 

    /** 
    * Internal options array 
    * @var array 
    */ 
    protected $_options = array(
     'allow' => Zend_Validate_Hostname::ALLOW_DNS, 
     'deep' => true, 
     'domain' => true, 
     'hostname' => null, 
     'mx' => true, 
    ); 

    /** 
    * @see http://en.wikipedia.org/wiki/Reserved_IP_addresses#Reserved_IPv4_addresses 
    * @var array [first octet] => [[CIDR] => [[range start], [range end]]] 
    */ 
    protected $_reservedIps = array(
     '0' => array('0.0.0.0/8' => array('0.0.0.0', '0.255.255.255',),), 
     '10' => array('10.0.0.0/8' => array('10.0.0.0', '10.255.255.255',),), 
     '127' => array('127.0.0.0/8' => array('127.0.0.0', '127.255.255.255',),), 
     '169' => array('169.254.0.0/16' => array('169.254.0.0', '169.254.255.255',),), 
     '172' => array('172.16.0.0/12' => array('172.16.0.0', '172.31.255.255',),), 
     '192' => array(
      '192.0.2.0/24' => array('192.0.2.0', '192.0.2.255',), 
      '192.88.99.0/24' => array('192.88.99.0', '192.88.99.255',), 
      '192.168.0.0/16' => array('192.168.0.0', '192.168.255.255',), 
     ), 
     '198' => array(
      '198.18.0.0/15' => array('198.18.0.0', '198.19.255.255',), 
      '198.51.100.0/24' => array('198.51.100.0', '198.51.100.255',), 
     ), 
     '203' => array('203.0.113.0/24' => array('203.0.113.0', '203.0.113.255',),), 
     '224' => array('224.0.0.0/4' => array('224.0.0.0', '239.255.255.255',),), 
     '240' => array('240.0.0.0/4' => array('240.0.0.0', '255.255.255.255',),), 
    ); 

    /** 
    * Returns if the given host is reserved 
    * 
    * @param string $host 
    * @return boolean 
    */ 
    protected function _isReserved($host) 
    { 
     if (!preg_match('/^([0-9]{1,3}\.){3}[0-9]{1,3}$/', $host)) { 
      $host = gethostbyname($host); 
     } 

     $octets = explode('.', $host); 
     if (224 <= (int) $octets[0]) { 
      // IP Addresses beginning with 224 or greater are all reserved, short-circuit range checks 
      return true; 
     } elseif (array_key_exists($octets[0], $this->_reservedIps)) { 
      // for integer comparisons 
      $intIp = $this->_ipToInt($host); 

      // loop over reserved IP addresses 
      foreach ($this->_reservedIps as $ranges) { 
       foreach ($ranges as $range) { 
        if (($this->_ipToInt($range[0]) <= $intIp) 
          && ($this->_ipToInt($range[1]) >= $intIp)) { 
         // the IP address falls in a reserved range 
         return true; 
        } 
       } 
      } 

      // the IP address did not fall in a reserved range 
      return false; 
     } else { 
      return false; 
     } 
    } 

    /** 
    * Convert a dot-decimal IP address to it's decimal integer equivalent 
    * 
    * @param string $ip 
    * @return integer 
    */ 
    protected function _ipToInt($ip) 
    { 
     $octets = explode('.', $ip); 
     foreach ($octets as $key => $octet) { 
      $octets[$key] = str_pad(decbin($octet), 8, '0', STR_PAD_LEFT); 
     } 
     $bin = implode('', $octets); 
     return bindec($bin); 
    } 
} 
+0

我终于在Zend Framework问题跟踪器中提交了一个错误:http://framework.zend.com/issues/browse/ZF-12160 – Sonny 2012-04-18 13:36:29

相关问题