Add charset= support to PDO mysql DSN
Adds support for charset= in the PDO mysql DSN. Closes #1309 Closes #1489 Reviewed By: @ptarjan Differential Revision: D1137883
Esse commit está contido em:
@@ -8,6 +8,11 @@ before_script:
|
||||
# for some tests
|
||||
- time sudo locale-gen de_DE && sudo locale-gen zh_CN.utf8 && sudo locale-gen fr_FR
|
||||
- time HPHP_HOME=`pwd` make -j 6
|
||||
# mysql configuration for unit-tests
|
||||
- mysql -e 'CREATE DATABASE IF NOT EXISTS hhvm;'
|
||||
- export PDO_MYSQL_TEST_DSN="mysql:host=127.0.0.1;dbname=hhvm"
|
||||
- export PDO_MYSQL_TEST_USER="travis"
|
||||
- export PDO_MYSQL_TEST_PASS=""
|
||||
|
||||
# The larger test suites (slow, zend) take longer than 50 miuntes to run. Since
|
||||
# we have no way to change the timeout, we subdivide the jit/interp repo/normal
|
||||
|
||||
@@ -244,8 +244,9 @@ bool PDOMySqlConnection::create(CArrRef options) {
|
||||
char *host = NULL, *unix_socket = NULL;
|
||||
unsigned int port = 3306;
|
||||
char *dbname;
|
||||
char *charset = nullptr;
|
||||
struct pdo_data_src_parser vars[] = {
|
||||
{ "charset", NULL, 0 },
|
||||
{ "charset", nullptr, 0 },
|
||||
{ "dbname", "", 0 },
|
||||
{ "host", "localhost", 0 },
|
||||
{ "port", "3306", 0 },
|
||||
@@ -270,6 +271,7 @@ bool PDOMySqlConnection::create(CArrRef options) {
|
||||
|
||||
m_max_buffer_size = 1024*1024;
|
||||
m_buffered = m_emulate_prepare = 1;
|
||||
charset = vars[0].optval;
|
||||
|
||||
/* handle MySQL options */
|
||||
if (!options.empty()) {
|
||||
@@ -350,6 +352,13 @@ bool PDOMySqlConnection::create(CArrRef options) {
|
||||
}
|
||||
}
|
||||
|
||||
if (charset) {
|
||||
if (mysql_options(m_server, MYSQL_SET_CHARSET_NAME, charset)) {
|
||||
handleError(__FILE__, __LINE__);
|
||||
goto cleanup;
|
||||
}
|
||||
}
|
||||
|
||||
dbname = vars[1].optval;
|
||||
host = vars[2].optval;
|
||||
if (vars[3].optval) {
|
||||
|
||||
@@ -0,0 +1,177 @@
|
||||
<?php
|
||||
require_once(dirname(__FILE__) . DIRECTORY_SEPARATOR . 'config.inc');
|
||||
require_once(dirname(__FILE__) . '/../../pdo/tests/pdo_test.inc');
|
||||
|
||||
class MySQLPDOTest extends PDOTest {
|
||||
|
||||
static function factory($classname = 'PDO', $drop_test_tables = false, $myattr = null, $mydsn = null) {
|
||||
|
||||
$dsn = self::getDSN($mydsn);
|
||||
$user = PDO_MYSQL_TEST_USER;
|
||||
$pass = PDO_MYSQL_TEST_PASS;
|
||||
$attr = getenv('PDOTEST_ATTR');
|
||||
|
||||
if (is_string($attr) && strlen($attr)) {
|
||||
$attr = unserialize($attr);
|
||||
} else {
|
||||
$attr = null;
|
||||
}
|
||||
if ($user === false)
|
||||
$user = NULL;
|
||||
if ($pass === false)
|
||||
$pass = NULL;
|
||||
|
||||
$db = new $classname($dsn, $user, $pass, $attr);
|
||||
if (!$db) {
|
||||
die("Could not create PDO object (DSN=$dsn, user=$user)\n");
|
||||
}
|
||||
|
||||
$db->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_WARNING);
|
||||
$db->setAttribute(PDO::ATTR_CASE, PDO::CASE_LOWER);
|
||||
|
||||
return $db;
|
||||
}
|
||||
|
||||
static function createTestTable($db, $engine = null) {
|
||||
if (!$engine)
|
||||
$engine = PDO_MYSQL_TEST_ENGINE;
|
||||
|
||||
$db->exec('DROP TABLE IF EXISTS test');
|
||||
$db->exec('CREATE TABLE test(id INT, label CHAR(1), PRIMARY KEY(id)) ENGINE=' . $engine);
|
||||
$db->exec("INSERT INTO test(id, label) VALUES (1, 'a'), (2, 'b'), (3, 'c'), (4, 'd'), (5, 'e'), (6, 'f')");
|
||||
}
|
||||
|
||||
static function getTableEngine() {
|
||||
return PDO_MYSQL_TEST_ENGINE;
|
||||
}
|
||||
|
||||
|
||||
static function getDSN($new_options = null, $addition = '') {
|
||||
if (!$new_options)
|
||||
return PDO_MYSQL_TEST_DSN . $addition;
|
||||
|
||||
$old_options = array();
|
||||
$dsn = substr(PDO_MYSQL_TEST_DSN,
|
||||
strpos(PDO_MYSQL_TEST_DSN, ':') + 1,
|
||||
strlen(PDO_MYSQL_TEST_DSN));
|
||||
|
||||
// no real parser - any excotic setting can fool us
|
||||
$parts = explode(';', $dsn);
|
||||
foreach ($parts as $k => $v) {
|
||||
$tmp = explode('=', $v);
|
||||
if (count($tmp) == 2)
|
||||
$old_options[$tmp[0]] = $tmp[1];
|
||||
}
|
||||
|
||||
$options = $old_options;
|
||||
foreach ($new_options as $k => $v)
|
||||
$options[$k] = $v;
|
||||
|
||||
$dsn = 'mysql:';
|
||||
foreach ($options as $k => $v)
|
||||
$dsn .= sprintf('%s=%s;', $k, $v);
|
||||
|
||||
if ($addition)
|
||||
$dsn .= $addition;
|
||||
else
|
||||
$dsn = substr($dsn, 0, strlen($dsn) -1);
|
||||
|
||||
return $dsn;
|
||||
}
|
||||
|
||||
static function getClientVersion($db) {
|
||||
return self::extractVersion($db->getAttribute(PDO::ATTR_CLIENT_VERSION));
|
||||
}
|
||||
|
||||
static function getServerVersion($db) {
|
||||
return self::extractVersion($db->getAttribute(PDO::ATTR_SERVER_VERSION));
|
||||
}
|
||||
|
||||
static function extractVersion($version_string) {
|
||||
/*
|
||||
TODO:
|
||||
We're a bit in trouble: PDO_MYSQL returns version strings.
|
||||
That's wrong according to the manual. According to the manual
|
||||
integers should be returned. However, this code needs to work
|
||||
with stinky PDO_MYSQL and hopefully better PDO_MYSQLND.
|
||||
*/
|
||||
|
||||
// already an int value?
|
||||
if (is_int($version_string))
|
||||
return $version_string;
|
||||
|
||||
// string but int value?
|
||||
$tmp = (int)$version_string;
|
||||
if (((string)$tmp) === $version_string)
|
||||
return $tmp;
|
||||
|
||||
// stinky string which we need to parse
|
||||
$parts = explode('.', $version_string);
|
||||
if (count($parts) != 3)
|
||||
return -1;
|
||||
|
||||
$version = (int)$parts[0] * 10000;
|
||||
$version+= (int)$parts[1] * 100;
|
||||
$version+= (int)$parts[2];
|
||||
|
||||
return $version;
|
||||
}
|
||||
|
||||
static function getTempDir() {
|
||||
|
||||
if (!function_exists('sys_get_temp_dir')) {
|
||||
|
||||
if (!empty($_ENV['TMP']))
|
||||
return realpath( $_ENV['TMP'] );
|
||||
if (!empty($_ENV['TMPDIR']))
|
||||
return realpath( $_ENV['TMPDIR'] );
|
||||
if (!empty($_ENV['TEMP']))
|
||||
return realpath( $_ENV['TEMP'] );
|
||||
|
||||
$temp_file = tempnam(md5(uniqid(rand(), TRUE)), '');
|
||||
if ($temp_file) {
|
||||
$temp_dir = realpath(dirname($temp_file));
|
||||
unlink($temp_file);
|
||||
return $temp_dir;
|
||||
}
|
||||
return FALSE;
|
||||
} else {
|
||||
return sys_get_temp_dir();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
static function detect_transactional_mysql_engine($db) {
|
||||
foreach ($db->query("show variables like 'have%'") as $row) {
|
||||
if (!empty($row) && $row[1] == 'YES' && ($row[0] == 'have_innodb' || $row[0] == 'have_bdb')) {
|
||||
return str_replace("have_", "", $row[0]);
|
||||
}
|
||||
}
|
||||
/* MySQL 5.6.1+ */
|
||||
foreach ($db->query("SHOW ENGINES") as $row) {
|
||||
if (isset($row['engine']) && isset($row['support'])) {
|
||||
if ('InnoDB' == $row['engine'] && ('YES' == $row['support'] || 'DEFAULT' == $row['support']))
|
||||
return 'innodb';
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
static function isPDOMySQLnd() {
|
||||
ob_start();
|
||||
phpinfo();
|
||||
$tmp = ob_get_contents();
|
||||
ob_end_clean();
|
||||
return (preg_match('/PDO Driver for MySQL.*enabled/', $tmp) &&
|
||||
preg_match('/Client API version.*mysqlnd/', $tmp));
|
||||
}
|
||||
|
||||
static function dropTestTable($db = NULL) {
|
||||
if (is_null($db))
|
||||
$db = self::factory();
|
||||
|
||||
$db->exec('DROP TABLE IF EXISTS test');
|
||||
}
|
||||
|
||||
}
|
||||
?>
|
||||
@@ -0,0 +1,177 @@
|
||||
<?php
|
||||
require_once(dirname(__FILE__) . DIRECTORY_SEPARATOR . 'config.inc');
|
||||
require_once(dirname(__FILE__) . '/../../pdo/tests/pdo_test.inc');
|
||||
|
||||
class MySQLPDOTest extends PDOTest {
|
||||
|
||||
static function factory($classname = 'PDO', $drop_test_tables = false, $myattr = null, $mydsn = null) {
|
||||
|
||||
$dsn = self::getDSN($mydsn);
|
||||
$user = PDO_MYSQL_TEST_USER;
|
||||
$pass = PDO_MYSQL_TEST_PASS;
|
||||
$attr = getenv('PDOTEST_ATTR');
|
||||
|
||||
if (is_string($attr) && strlen($attr)) {
|
||||
$attr = unserialize($attr);
|
||||
} else {
|
||||
$attr = null;
|
||||
}
|
||||
if ($user === false)
|
||||
$user = NULL;
|
||||
if ($pass === false)
|
||||
$pass = NULL;
|
||||
|
||||
$db = new $classname($dsn, $user, $pass, $attr);
|
||||
if (!$db) {
|
||||
die("Could not create PDO object (DSN=$dsn, user=$user)\n");
|
||||
}
|
||||
|
||||
$db->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_WARNING);
|
||||
$db->setAttribute(PDO::ATTR_CASE, PDO::CASE_LOWER);
|
||||
|
||||
return $db;
|
||||
}
|
||||
|
||||
static function createTestTable($db, $engine = null) {
|
||||
if (!$engine)
|
||||
$engine = PDO_MYSQL_TEST_ENGINE;
|
||||
|
||||
$db->exec('DROP TABLE IF EXISTS test');
|
||||
$db->exec('CREATE TABLE test(id INT, label CHAR(1), PRIMARY KEY(id)) ENGINE=' . $engine);
|
||||
$db->exec("INSERT INTO test(id, label) VALUES (1, 'a'), (2, 'b'), (3, 'c'), (4, 'd'), (5, 'e'), (6, 'f')");
|
||||
}
|
||||
|
||||
static function getTableEngine() {
|
||||
return PDO_MYSQL_TEST_ENGINE;
|
||||
}
|
||||
|
||||
|
||||
static function getDSN($new_options = null, $addition = '') {
|
||||
if (!$new_options)
|
||||
return PDO_MYSQL_TEST_DSN . $addition;
|
||||
|
||||
$old_options = array();
|
||||
$dsn = substr(PDO_MYSQL_TEST_DSN,
|
||||
strpos(PDO_MYSQL_TEST_DSN, ':') + 1,
|
||||
strlen(PDO_MYSQL_TEST_DSN));
|
||||
|
||||
// no real parser - any excotic setting can fool us
|
||||
$parts = explode(';', $dsn);
|
||||
foreach ($parts as $k => $v) {
|
||||
$tmp = explode('=', $v);
|
||||
if (count($tmp) == 2)
|
||||
$old_options[$tmp[0]] = $tmp[1];
|
||||
}
|
||||
|
||||
$options = $old_options;
|
||||
foreach ($new_options as $k => $v)
|
||||
$options[$k] = $v;
|
||||
|
||||
$dsn = 'mysql:';
|
||||
foreach ($options as $k => $v)
|
||||
$dsn .= sprintf('%s=%s;', $k, $v);
|
||||
|
||||
if ($addition)
|
||||
$dsn .= $addition;
|
||||
else
|
||||
$dsn = substr($dsn, 0, strlen($dsn) -1);
|
||||
|
||||
return $dsn;
|
||||
}
|
||||
|
||||
static function getClientVersion($db) {
|
||||
return self::extractVersion($db->getAttribute(PDO::ATTR_CLIENT_VERSION));
|
||||
}
|
||||
|
||||
static function getServerVersion($db) {
|
||||
return self::extractVersion($db->getAttribute(PDO::ATTR_SERVER_VERSION));
|
||||
}
|
||||
|
||||
static function extractVersion($version_string) {
|
||||
/*
|
||||
TODO:
|
||||
We're a bit in trouble: PDO_MYSQL returns version strings.
|
||||
That's wrong according to the manual. According to the manual
|
||||
integers should be returned. However, this code needs to work
|
||||
with stinky PDO_MYSQL and hopefully better PDO_MYSQLND.
|
||||
*/
|
||||
|
||||
// already an int value?
|
||||
if (is_int($version_string))
|
||||
return $version_string;
|
||||
|
||||
// string but int value?
|
||||
$tmp = (int)$version_string;
|
||||
if (((string)$tmp) === $version_string)
|
||||
return $tmp;
|
||||
|
||||
// stinky string which we need to parse
|
||||
$parts = explode('.', $version_string);
|
||||
if (count($parts) != 3)
|
||||
return -1;
|
||||
|
||||
$version = (int)$parts[0] * 10000;
|
||||
$version+= (int)$parts[1] * 100;
|
||||
$version+= (int)$parts[2];
|
||||
|
||||
return $version;
|
||||
}
|
||||
|
||||
static function getTempDir() {
|
||||
|
||||
if (!function_exists('sys_get_temp_dir')) {
|
||||
|
||||
if (!empty($_ENV['TMP']))
|
||||
return realpath( $_ENV['TMP'] );
|
||||
if (!empty($_ENV['TMPDIR']))
|
||||
return realpath( $_ENV['TMPDIR'] );
|
||||
if (!empty($_ENV['TEMP']))
|
||||
return realpath( $_ENV['TEMP'] );
|
||||
|
||||
$temp_file = tempnam(md5(uniqid(rand(), TRUE)), '');
|
||||
if ($temp_file) {
|
||||
$temp_dir = realpath(dirname($temp_file));
|
||||
unlink($temp_file);
|
||||
return $temp_dir;
|
||||
}
|
||||
return FALSE;
|
||||
} else {
|
||||
return sys_get_temp_dir();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
static function detect_transactional_mysql_engine($db) {
|
||||
foreach ($db->query("show variables like 'have%'") as $row) {
|
||||
if (!empty($row) && $row[1] == 'YES' && ($row[0] == 'have_innodb' || $row[0] == 'have_bdb')) {
|
||||
return str_replace("have_", "", $row[0]);
|
||||
}
|
||||
}
|
||||
/* MySQL 5.6.1+ */
|
||||
foreach ($db->query("SHOW ENGINES") as $row) {
|
||||
if (isset($row['engine']) && isset($row['support'])) {
|
||||
if ('InnoDB' == $row['engine'] && ('YES' == $row['support'] || 'DEFAULT' == $row['support']))
|
||||
return 'innodb';
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
static function isPDOMySQLnd() {
|
||||
ob_start();
|
||||
phpinfo();
|
||||
$tmp = ob_get_contents();
|
||||
ob_end_clean();
|
||||
return (preg_match('/PDO Driver for MySQL.*enabled/', $tmp) &&
|
||||
preg_match('/Client API version.*mysqlnd/', $tmp));
|
||||
}
|
||||
|
||||
static function dropTestTable($db = NULL) {
|
||||
if (is_null($db))
|
||||
$db = self::factory();
|
||||
|
||||
$db->exec('DROP TABLE IF EXISTS test');
|
||||
}
|
||||
|
||||
}
|
||||
?>
|
||||
+22
-22
@@ -1,22 +1,22 @@
|
||||
<?php
|
||||
require_once(dirname(__FILE__) . DIRECTORY_SEPARATOR . 'mysql_pdo_test.inc');
|
||||
|
||||
/* Connect to mysql to determine the current charset so we can diffinate it */
|
||||
$link = MySQLPDOTest::factory();
|
||||
$charset = $link->query("SHOW VARIABLES LIKE 'character_set_connection'")->fetchObject()->value;
|
||||
|
||||
/* Make sure that we don't attempt to set the current character set to make this case useful */
|
||||
$new_charset = ($charset == 'latin1' ? 'ascii' : 'latin1');
|
||||
|
||||
/* Done with the original connection, create a second link to test the character set being defined */
|
||||
unset($link);
|
||||
|
||||
$link = MySQLPDOTest::factory('PDO', false, null, Array('charset' => $new_charset));
|
||||
$conn_charset = $link->query("SHOW VARIABLES LIKE 'character_set_connection'")->fetchObject()->value;
|
||||
|
||||
if ($charset !== $conn_charset) {
|
||||
echo "done!\n";
|
||||
} else {
|
||||
echo "failed!\n";
|
||||
}
|
||||
?>
|
||||
<?php
|
||||
require_once(dirname(__FILE__) . DIRECTORY_SEPARATOR . 'mysql_pdo_test.inc');
|
||||
|
||||
/* Connect to mysql to determine the current charset so we can diffinate it */
|
||||
$link = MySQLPDOTest::factory();
|
||||
$charset = $link->query("SHOW VARIABLES LIKE 'character_set_connection'")->fetchObject()->value;
|
||||
|
||||
/* Make sure that we don't attempt to set the current character set to make this case useful */
|
||||
$new_charset = ($charset == 'latin1' ? 'ascii' : 'latin1');
|
||||
|
||||
/* Done with the original connection, create a second link to test the character set being defined */
|
||||
unset($link);
|
||||
|
||||
$link = MySQLPDOTest::factory('PDO', false, null, Array('charset' => $new_charset));
|
||||
$conn_charset = $link->query("SHOW VARIABLES LIKE 'character_set_connection'")->fetchObject()->value;
|
||||
|
||||
if ($charset !== $conn_charset) {
|
||||
echo "done!\n";
|
||||
} else {
|
||||
echo "failed!\n";
|
||||
}
|
||||
?>
|
||||
@@ -0,0 +1,7 @@
|
||||
<?php
|
||||
|
||||
if (!extension_loaded("pdo_mysql")) exit("skip pdo_mysql extension not loaded");
|
||||
|
||||
if (false === getenv('PDO_MYSQL_TEST_DSN')) exit("skip PDO_MYSQL_TEST_DSN env variable is not defined");
|
||||
if (false === getenv('PDO_MYSQL_TEST_USER')) exit("skip PDO_MYSQL_TEST_USER env variable is not defined");
|
||||
if (false === getenv('PDO_MYSQL_TEST_PASS')) exit("skip PDO_MYSQL_TEST_PASS env variable is not defined");
|
||||
Referência em uma Nova Issue
Bloquear um usuário