Database: Add support for connecting to IPv6 hosts

IPv4 addresses are scarce, overworked, and underpaid. They're ready to retire, but we just won't let them go. If you care about their wellbeing, switch to IPv6 today.

Props schlessera, birgire.
Fixes #41722.


Built from https://develop.svn.wordpress.org/trunk@41629


git-svn-id: http://core.svn.wordpress.org/trunk@41464 1a063a9b-81f0-0310-95a4-ce76da25c4cd
This commit is contained in:
Gary Pendergast 2017-09-28 05:37:44 +00:00
parent 0028a9555e
commit dbd93ddbb6
2 changed files with 66 additions and 20 deletions

View File

@ -4,7 +4,7 @@
* *
* @global string $wp_version * @global string $wp_version
*/ */
$wp_version = '4.9-alpha-41628'; $wp_version = '4.9-alpha-41629';
/** /**
* Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema. * Holds the WordPress DB revision, increments when changes are made to the WordPress DB schema.

View File

@ -1460,24 +1460,23 @@ class wpdb {
if ( $this->use_mysqli ) { if ( $this->use_mysqli ) {
$this->dbh = mysqli_init(); $this->dbh = mysqli_init();
// mysqli_real_connect doesn't support the host param including a port or socket $host = $this->dbhost;
// like mysql_connect does. This duplicates how mysql_connect detects a port and/or socket file. $port = null;
$port = null; $socket = null;
$socket = null; $is_ipv6 = false;
$host = $this->dbhost;
$port_or_socket = strstr( $host, ':' ); if ( $host_data = $this->parse_db_host( $this->dbhost ) ) {
if ( ! empty( $port_or_socket ) ) { list( $host, $port, $socket, $is_ipv6 ) = $host_data;
$host = substr( $host, 0, strpos( $host, ':' ) ); }
$port_or_socket = substr( $port_or_socket, 1 );
if ( 0 !== strpos( $port_or_socket, '/' ) ) { /*
$port = intval( $port_or_socket ); * If using the `mysqlnd` library, the IPv6 address needs to be
$maybe_socket = strstr( $port_or_socket, ':' ); * enclosed in square brackets, whereas it doesn't while using the
if ( ! empty( $maybe_socket ) ) { * `libmysqlclient` library.
$socket = substr( $maybe_socket, 1 ); * @see https://bugs.php.net/bug.php?id=67563
} */
} else { if ( $is_ipv6 && extension_loaded( 'mysqlnd' ) ) {
$socket = $port_or_socket; $host = "[$host]";
}
} }
if ( WP_DEBUG ) { if ( WP_DEBUG ) {
@ -1489,7 +1488,8 @@ class wpdb {
if ( $this->dbh->connect_errno ) { if ( $this->dbh->connect_errno ) {
$this->dbh = null; $this->dbh = null;
/* It's possible ext/mysqli is misconfigured. Fall back to ext/mysql if: /*
* It's possible ext/mysqli is misconfigured. Fall back to ext/mysql if:
* - We haven't previously connected, and * - We haven't previously connected, and
* - WP_USE_EXT_MYSQL isn't set to false, and * - WP_USE_EXT_MYSQL isn't set to false, and
* - ext/mysql is loaded. * - ext/mysql is loaded.
@ -1569,6 +1569,52 @@ class wpdb {
return false; return false;
} }
/**
* Parse the DB_HOST setting to interpret it for mysqli_real_connect.
*
* mysqli_real_connect doesn't support the host param including a port or
* socket like mysql_connect does. This duplicates how mysql_connect detects
* a port and/or socket file.
*
* @since 4.9.0
*
* @param string $host The DB_HOST setting to parse.
* @return array|bool Array containing the host, the port, the socket and whether
* it is an IPv6 address, in that order. If $host couldn't be parsed,
* returns false.
*/
public function parse_db_host( $host ) {
$port = null;
$socket = null;
$is_ipv6 = false;
// We need to check for an IPv6 address first.
// An IPv6 address will always contain at least two colons.
if ( substr_count( $host, ':' ) > 1 ) {
$pattern = '#^(?:\[)?(?<host>[0-9a-fA-F:]+)(?:\]:(?<port>[\d]+))?(?:/(?<socket>.+))?#';
$is_ipv6 = true;
} else {
// We seem to be dealing with an IPv4 address.
$pattern = '#^(?<host>[^:/]*)(?::(?<port>[\d]+))?(?::(?<socket>.+))?#';
}
$matches = array();
$result = preg_match( $pattern, $host, $matches );
if ( 1 !== $result ) {
// Couldn't parse the address, bail.
return false;
}
foreach ( array( 'host', 'port', 'socket' ) as $component ) {
if ( array_key_exists( $component, $matches ) ) {
$$component = $matches[$component];
}
}
return array( $host, $port, $socket, $is_ipv6 );
}
/** /**
* Checks that the connection to the database is still up. If not, try to reconnect. * Checks that the connection to the database is still up. If not, try to reconnect.
* *