tags:

views:

1489

answers:

2

Hello,

I am trying to store IPv6 addresses in MySQL 5.0 in an efficient way. I have read the other questions related to this, such as this one. The author of that question eventually chose for two BIGINT fields. My searches have also turned up another often used mechanism: Using a DECIMAL(39,0) to store the IPv6 address. I have two questions about that.

  1. What are the advantages and disadvantages of using DECIMAL(39,0) over the other methods such as 2*BIGINT?
  2. How do I convert (in PHP) from the binary format as returned by inet_pton() to a decimal string format usable by MySQL, and how do I convert back so I can pretty-print with inet_ntop()?
+6  A: 

Here are the functions I now use to convert IP addresses from and to DECIMAL(39,0) format. They are named inet_ptod and inet_dtop for "presentation-to-decimal" and "decimal-to-presentation". It needs IPv6 and bcmath support in PHP.

/**
 * Convert an IP address from presentation to decimal(39,0) format suitable for storage in MySQL
 *
 * @param string $ip_address An IP address in IPv4, IPv6 or decimal notation
 * @return string The IP address in decimal notation
 */
function inet_ptod($ip_address)
{
    // IPv4 address
    if (strpos($ip_address, ':') === false && strpos($ip_address, '.') !== false) {
     $ip_address = '::' . $ip_address;
    }

    // IPv6 address
    if (strpos($ip_address, ':') !== false) {
     $network = inet_pton($ip_address);
     $parts = unpack('N*', $network);

     foreach ($parts as &$part) {
      if ($part < 0) {
       $part = bcadd((string) $part, '4294967296');
      }

      if (!is_string($part)) {
       $part = (string) $part;
      }
     }

     $decimal = $parts[4];
     $decimal = bcadd($decimal, bcmul($parts[3], '4294967296'));
     $decimal = bcadd($decimal, bcmul($parts[2], '18446744073709551616'));
     $decimal = bcadd($decimal, bcmul($parts[1], '79228162514264337593543950336'));

     return $decimal;
    }

    // Decimal address
    return $ip_address;
}

/**
 * Convert an IP address from decimal format to presentation format
 *
 * @param string $decimal An IP address in IPv4, IPv6 or decimal notation
 * @return string The IP address in presentation format
 */
function inet_dtop($decimal)
{
    // IPv4 or IPv6 format
    if (strpos($decimal, ':') !== false || strpos($decimal, '.') !== false) {
     return $decimal;
    }

    // Decimal format
    $parts = array();
    $parts[1] = bcdiv($decimal, '79228162514264337593543950336', 0);
    $decimal = bcsub($decimal, bcmul($parts[1], '79228162514264337593543950336'));
    $parts[2] = bcdiv($decimal, '18446744073709551616', 0);
    $decimal = bcsub($decimal, bcmul($parts[2], '18446744073709551616'));
    $parts[3] = bcdiv($decimal, '4294967296', 0);
    $decimal = bcsub($decimal, bcmul($parts[3], '4294967296'));
    $parts[4] = $decimal;

    foreach ($parts as &$part) {
     if (bccomp($part, '2147483647') == 1) {
      $part = bcsub($part, '4294967296');
     }

     $part = (int) $part;
    }

    $network = pack('N4', $parts[1], $parts[2], $parts[3], $parts[4]);
    $ip_address = inet_ntop($network);

    // Turn IPv6 to IPv4 if it's IPv4
    if (preg_match('/^::\d+.\d+.\d+.\d+$/', $ip_address)) {
     return substr($ip_address, 2);
    }

    return $ip_address;
}
Sander Marechal
Boy. Good thing there aren't any arbitrary magic numbers in that code.
Frank Farmer
They're not arbitrary or magic if you know your powers of 2 ;-)
Sander Marechal
+1 I wish could up vote you 500 times for posting those functions! Thanks.
Tim Santeford
79228162514264337593543950336 = 2^96 --- 18446744073709551616 = 2^64 --- 4294967296 = 2^32 --- Do note that you sadly can't use this shorthand notation as-is, because they're to big for PHP.
alexanderpas
+3  A: 

We went for a VARBINARY(16) column instead and use inet_pton and inet_ntop to do the conversions:

http://labs.watchmouse.com/2009/10/extending-mysql-5-with-ipv6-functions/

The functions can be loaded into a running MySQL server and will give you INET6_NTOP and INET6_PTON in SQL, just as the familiar INET_NTOA and INET_ATON functions for IPv4.

Nice. Not usable on standard shared hosting, but definitely an option if you run your own server.
Sander Marechal
I will vouch for this approach as well. Obviously, it doesn't work in many shared hosting environments, but storing the address this way also allows MUCH more efficient matching (for example, based on netmask) than in native PHP.
David Eads