Current File : /home/pacjaorg/www/nsa/administrator/components/com_akeebabackup/engine/FixMySQLHostname.php |
<?php
/**
* Akeeba Engine
*
* @package akeebaengine
* @copyright Copyright (c)2006-2022 Nicholas K. Dionysopoulos / Akeeba Ltd
* @license GNU General Public License version 3, or later
*/
namespace Akeeba\Engine;
defined('AKEEBAENGINE') || die();
trait FixMySQLHostname
{
/**
* Tries to parse all the weird hostname definitions and normalize them into something that the MySQLi connector
* will understand. Please note that there are some differences to the old MySQL driver:
*
* * Port and socket MUST be provided separately from the hostname. Hostnames in the form of 127.0.0.1:8336 are no
* longer acceptable.
*
* * The hostname "localhost" has special meaning. It means "use named pipes / sockets". Anything else uses TCP/IP.
* This is the ONLY way to specify a. TCP/IP or b. named pipes / sockets connection.
*
* * You SHOULD NOT use a numeric TCP/IP port with hostname localhost. For some strange reason it's still allowed
* but the manual is self-contradicting over what this really does...
*
* * Likewise you CANNOT use a socket / named pipe path with hostname other than localhost. Named pipes and sockets
* can only be used with the local machine, therefore the hostname MUST be localhost.
*
* * You cannot give a TCP/IP port number in the socket parameter or a named pipe / socket path to the port
* parameter. This leads to an error.
*
* * You cannot use an empty string, 0 or any other non-null value when you want to omit either of the port or
* socket parameters.
*
* * Persistent connections must be prefixed with the string literal 'p:'. Therefore you cannot have a hostname
* called 'p' (not to mention that'd be daft). You can also not specify something like 'p:1234' to make a
* persistent connection to a port. This wasn't even supported by the old MySQL driver. As a result we don't even
* try to catch that degenerate case.
*
* This method will try to apply all of the aforementioned rules with one additional disambiguation rule:
*
* A port / socket set in the hostname overrides a port specified separately. A port specified separately overrides
* a socket specified separately.
*
* @param string $host The hostname. Can contain legacy hostname:port or hostname:sc=ocket definitions.
* @param int $port The port. Alternatively it can contain the path to the socket.
* @param string $socket The path to the socket. You could abuse it to enter the port number. DON'T!
*
* @return void All parameters are passed by reference.
*
* @since 9.2.3
*/
protected function fixHostnamePortSocket(&$host, &$port, &$socket)
{
// Is this a persistent connection? Persistent connections are indicated by the literal "p:" in front of the hostname
$isPersistent = (substr($host, 0, 2) == 'p:');
$host = $isPersistent ? substr($host, 2) : $host;
// If the hostname looks like a *NIX filename we need to treat it as a socket.
if (preg_match('#^/([^/]*/)?[^/]#', $host))
{
$socket = $host;
$host = null;
}
// Special case: Windows named pipe (\\.\something\or\another), with or without parentheses.
$isNamedPipe = false;
if (preg_match("#^\(?\\\\\\\\\.\\\\#", $host))
{
$isNamedPipe = true;
$socket = $host;
$host = '.';
}
/*
* Unlike mysql_connect(), mysqli_connect() takes the port and socket as separate arguments. Therefore, we
* have to extract them from the host string.
*/
$port = !empty($port) ? $port : 3306;
if ($host === 'localhost')
{
$port = null;
}
// UNIX socket URI, e.g. 'unix:/path/to/unix/socket.sock'
elseif (preg_match('/^unix:(?P<socket>[^:]+)$/', $host, $matches))
{
$host = null;
$socket = $matches['socket'];
$port = null;
}
// It's an IPv4 address with or without port
elseif (preg_match('/^(?P<host>((25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?))(:(?P<port>.+))?$/', $host, $matches))
{
$host = $matches['host'];
if (!empty($matches['port']))
{
$port = $matches['port'];
}
}
// Square-bracketed IPv6 address with or without port, e.g. [fe80:102::2%eth1]:3306
elseif (preg_match('/^(?P<host>\[.*\])(:(?P<port>.+))?$/', $host, $matches))
{
$host = $matches['host'];
if (!empty($matches['port']))
{
$port = $matches['port'];
}
}
// Named host (e.g example.com or localhost) with or without port
elseif (preg_match('/^(?P<host>(\w+:\/{2,3})?[a-z0-9\.\-]+)(:(?P<port>[^:]+))?$/i', $host, $matches))
{
$host = $matches['host'];
if (!empty($matches['port']))
{
$port = $matches['port'];
}
}
// Empty host, just port, e.g. ':3306'
elseif (preg_match('/^:(?P<port>[^:]+)$/', $host, $matches))
{
$host = '127.0.0.1';
$port = $matches['port'];
}
// ... else we assume normal (naked) IPv6 address, so host and port stay as they are or default
// If there is both a valid port and a valid socket we will choose the socket instead
if (is_numeric($port) && !empty($socket))
{
$port = null;
}
// Get the port number or socket name
if (is_numeric($port))
{
$port = (int) $port;
$socket = '';
}
elseif (is_string($port) && empty($socket))
{
$socket = $port;
$port = null;
}
// If there is a socket the hostname must be null
if (!empty($socket))
{
$host = null;
}
// If there is a socket the port must be null
if (!empty($socket))
{
$port = null;
}
// If there is a numeric port and the hostname is 'localhost' convert to 127.0.0.1
if (is_numeric($port) && ($host === 'localhost'))
{
$host = '127.0.0.1';
}
/**
* Special case: MySQL sockets on Windows need to be enclosed with parentheses and have \\.\ in front.
*
* @see https://dev.mysql.com/doc/mysql-shell/8.0/en/mysql-shell-connection-socket.html
* @see https://www.php.net/manual/en/mysqli.quickstart.connections.php
*/
if (!empty($socket) && $isNamedPipe)
{
$host = '.';
/**
* Remove any existing parentheses, otherwise URL-decode the socket (in case it was given in the correct
* percent encoded format).
*/
if (substr($socket, 0, 1) === '(' && substr($socket, -1) === ')')
{
$socket = substr($socket, 1, -1);
}
else
{
$socket = rawurldecode($socket);
}
// If the socket doesn't already start with \\.\ add it
if (substr($socket, 0, 4) !== '\\\\.\\')
{
$socket = '\\\\.\\' . $socket;
}
$socket = '(' . $socket . ')';
}
// Finally, if it's a persistent connection we have to prefix the hostname with 'p:'
$host = ($isPersistent && $host !== null) ? "p:$host" : $host;
}
}