从PHP存储MySQL中的IP

时间:2017-03-04 06:15:10

标签: php mysql pdo

经过几个小时的搜索,我只能找到“过时”和/或“不完整”的答案。 (显然它们早于PDO。)我正在编写PHP代码(版本7.0.15),并使用MySQL版本5.7.17-0(两者都在KUbuntu“虚拟机”上)。虽然我已经使用计算机超过45年,但我对PHP和MySQL还不熟悉。

我可以用PHP获取访问者的IP地址。然后我想检查“try_ur_table”以查看它是否已经有一个条目,如果没有,则插入一个条目然后查找,这样我就可以在程序的其他部分使用“ur_index”了。 ur_index是int(11),ur_ip是binary(16)。

问题在于,每次运行代码时,第一个选择都会失败,因此会生成一个新条目,然后第二个选择也无法找到匹配项!

以下是相关的代码段:

  try
  {
    $pdc = new PDO('mysql:host=localhost;dbname=toy_database', 'xxxxx', 'xxxxx' );
    $pdc->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
    $pdc->exec('SET NAMES "utf8"');
  }
  catch (PDOException $e)
  {
    $output = 'Unable to connect tothe databaseserver. ' . $e->getMessage();
    include 'errout.html.php';
    exit();
  }
  // Find the caller data...
  if ( isset( $_SERVER[ 'REMOTE_ADDR' ] ) )
  {
    $cd = inet_pton( $_SERVER[ 'REMOTE_ADDR' ] );
    if ( $cd )
    {
      // inet_pton thinks it succeeded...
      try
      {
        $sql = "SELECT * FROM try_ur_table WHERE ur_ip = ? ";
        $rt = $pdc->prepare($sql);
        $rt->execute(array($cd));
        $u_list = $rt->fetchAll();
      }
      catch (PDOException $e)
      {
        $output = 'Problem looking for ur_ip.  ' .  $e->getMessage();
        include 'errout.html.php';
        exit();
      }
      if ( $u_list == NULL )
      {
        // New user!
        try
        {
          $sqm = "INSERT INTO try_ur_table SET ur_ip=?";
          $rs = $pdc->prepare($sqm);
          $rs->execute(array($cd));
        }
        catch (PDOException $e)
        {
          $output = 'Problem inserting new ur_ip. ' . $e->getMessage();
          include 'errout.html.php';
          exit();
        }
        // Now go find the new entry...
        try
        {
          $sql = "SELECT * FROM try_ur_table WHERE ur_ip = ? ";
          $rt = $pdc->prepare($sql);
          $rt->execute(array($cd));
          $u_list = $rt->fetchAll();
        }
        catch (PDOException $e)
        {
          $output = 'Problem looking for new ur_ip.  ' . $e->getMessage();
          include 'errout.html.php';
          exit();
        }
      } // $u_list == NULL
      // At this point there should be exactly one row in $u_list...
    } // $cd != false
  }
  else
  {
    // ! isset( $_SERVER[ 'REMOTE_ADDR' ]
    $cd = false;
  }

其他测试显示$ _SERVER ['REMOTE_ADDR']返回127.0.0.1(这是有意义的,因为这是“本地主机”)。但是,每次我运行上面的代码,phpMyAdmin说ur_ip是0x7f00000001000000000000000000000(希望我正确地计算了零 - 似乎无法从phpMyAdmin复制和粘贴,但这是次要的)。此外,由于我有ur_index,我已经尝试了基于它的选择,当我尝试通过inet_ntop()运行ur_ip时,我得到垃圾,既不是有效的IPv4也不是IPv6地址。

1 个答案:

答案 0 :(得分:0)

您的问题源于二进制类型。要匹配存储的值,您应该填充要比较的数据,如the documentation

中所示

或者你可以简单地使用VARBINARY来避免麻烦。

此外,您的代码非常复制,因此难以阅读和维护。异常的想法是你将它们全部捕获一次。还有一些条件没用,你应该使用fetchAll()而不是fetch method that is appropriate for your task

当然,尝试解码而不是实际的二进制值,但是phpmyadmin显示的十六进制表示确实不会给你任何明智的结果。

以下是您被要求提供的最小,完整和可验证的示例。只要提供了正确的凭据,它就代表了实际问题并适用于任何关心运行它的人。

try {
    $pdc = new PDO('mysql:host=localhost;dbname=toy_database;charset=utf8', 'xxxxx', 'xxxxx' );
    $pdc->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);

    // let's create a temporary table for the example purpose
    // change varbinary for binary and it goes astray
    $pdc->query("CREATE temporary TABLE try_ur_table 
                 (ur_index int primary key auto_increment,ur_ip varbinary(16))");

    $cd = inet_pton( $_SERVER["REMOTE_ADDR"] );

    $sql = "SELECT ur_index FROM try_ur_table WHERE ur_ip = ? ";
    $rt = $pdc->prepare($sql);
    $rt->execute(array($cd));
    $ur_index = $rt->fetchColumn();

    if ( !$ur_index )
    {
        $sqm = "INSERT INTO try_ur_table SET ur_ip=?";
        $pdc->prepare($sqm)->execute(array($cd));
        // here: that's all you need to get the index
        $ur_index = $pdc->lastInsertId();
    }

    // OK, let's verify
    $sql = "SELECT ur_ip FROM try_ur_table WHERE ur_ip = ? ";
    $rt = $pdc->prepare($sql);
    $rt->execute(array($cd));
    $ur_ip = $rt->fetchColumn();
    var_dump($ur_ip === $cd, bin2hex($ur_ip));

} catch (PDOException $e) {
    error_log($e);
    $output = ini_get('display_errors') ? $e : "There is a temporary problem. Try again later";
    include 'errout.html.php';
    exit();
}