summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAnatol Belski <ab@php.net>2016-03-17 14:28:32 +0100
committerAnatol Belski <ab@php.net>2016-03-17 14:28:58 +0100
commit1b632cfe834bfd87d182566d7d960df7d10ded72 (patch)
tree37e88bfd9a592f617cce7c9e242d69379addee23
parenteb405623ec01d61ea34f5fb1d4c6f39c4eb187aa (diff)
downloadphp-git-1b632cfe834bfd87d182566d7d960df7d10ded72.tar.gz
Backport patch for bug #71820
Rev b4eedd128ba9f61be08a50c94afd72837d7cf70b
-rw-r--r--ext/mysqli/mysqli.c15
-rw-r--r--ext/mysqli/tests/bug71820.phpt104
-rw-r--r--ext/pgsql/pgsql.c15
-rw-r--r--ext/pgsql/tests/bug71820.phpt93
4 files changed, 225 insertions, 2 deletions
diff --git a/ext/mysqli/mysqli.c b/ext/mysqli/mysqli.c
index f19f6a8504..211f8ea319 100644
--- a/ext/mysqli/mysqli.c
+++ b/ext/mysqli/mysqli.c
@@ -1295,9 +1295,13 @@ void php_mysqli_fetch_into_hash(INTERNAL_FUNCTION_PARAMETERS, int override_flags
zend_fcall_info fci;
zend_fcall_info_cache fcc;
zval *retval_ptr;
+ zend_bool props_merged = 0;
object_and_properties_init(return_value, ce, NULL);
- zend_merge_properties(return_value, Z_ARRVAL(dataset), 1 TSRMLS_CC);
+ if (!ce->__set) {
+ props_merged = 1;
+ zend_merge_properties(return_value, Z_ARRVAL(dataset), 1 TSRMLS_CC);
+ }
if (ce->constructor) {
fci.size = sizeof(fci);
@@ -1331,6 +1335,10 @@ void php_mysqli_fetch_into_hash(INTERNAL_FUNCTION_PARAMETERS, int override_flags
if (zend_call_function(&fci, &fcc TSRMLS_CC) == FAILURE) {
zend_throw_exception_ex(zend_exception_get_default(TSRMLS_C), 0 TSRMLS_CC, "Could not execute %s::%s()", ce->name, ce->constructor->common.function_name);
+ if (fci.params) {
+ efree(fci.params);
+ }
+ return;
} else {
if (retval_ptr) {
zval_ptr_dtor(&retval_ptr);
@@ -1341,6 +1349,11 @@ void php_mysqli_fetch_into_hash(INTERNAL_FUNCTION_PARAMETERS, int override_flags
}
} else if (ctor_params) {
zend_throw_exception_ex(zend_exception_get_default(TSRMLS_C), 0 TSRMLS_CC, "Class %s does not have a constructor hence you cannot use ctor_params", ce->name);
+ return;
+ }
+
+ if (!props_merged) {
+ zend_merge_properties(return_value, Z_ARRVAL(dataset), 1 TSRMLS_CC);
}
}
}
diff --git a/ext/mysqli/tests/bug71820.phpt b/ext/mysqli/tests/bug71820.phpt
new file mode 100644
index 0000000000..02a25939e2
--- /dev/null
+++ b/ext/mysqli/tests/bug71820.phpt
@@ -0,0 +1,104 @@
+--TEST--
+Bug #71820 __set has to be called after constructor, mysqli part
+--SKIPIF--
+<?php
+require_once('skipif.inc');
+require_once('skipifconnectfailure.inc');
+require_once("connect.inc");
+?>
+--FILE--
+<?php
+
+include "connect.inc";
+
+$tableName = 'test_mysqli_fetch_object';
+
+class TestRow
+{
+
+ private $set_from_constructor;
+ private $data;
+ private $hello = "world";
+
+ public function __construct($set_from_constructor)
+ {
+ $this->set_from_constructor = $set_from_constructor;
+ }
+
+ public function __set($name, $value)
+ {
+ if (!isset($this->data[$name])) {
+ /* $this->set_from_constructor has an expected value */
+ $this->data[$name] = 42 == $this->set_from_constructor ? $value : -1;
+ return;
+ }
+ throw new \Exception('Duplicity column name.');
+ }
+
+}
+
+
+if (!($connection = my_mysqli_connect($host, $user, $passwd, $db, $port, $socket))) {
+ printf("[001] Cannot connect to the server");
+}
+
+$rc = mysqli_query($connection, "DROP TABLE IF EXISTS $tableName");
+if (!$rc)
+ printf("[002] [%d] %s\n", mysqli_errno($connection), mysqli_error($connection));
+
+$table = <<<SQL
+CREATE TABLE $tableName (
+ id int NOT NULL auto_increment primary key,
+ name varchar(255) NOT NULL
+);
+SQL;
+
+$rc = mysqli_query($connection, $table);
+if (!$rc)
+ printf("[003] [%d] %s\n", mysqli_errno($connection), mysqli_error($connection));
+
+$rc = mysqli_query($connection, "INSERT INTO " . $tableName . " (name) VALUES ('Doe'), ('Joe')");
+if (!$rc)
+ printf("[004] [%d] %s\n", mysqli_errno($connection), mysqli_error($connection));
+
+$result = mysqli_query($connection, 'SELECT * FROM ' . $tableName . ' LIMIT 10');
+if (!$result)
+ printf("[005] [%d] %s\n", mysqli_errno($result), mysqli_error($result));
+
+
+while ($row = mysqli_fetch_object($result, 'TestRow', [42])) {
+ var_dump($row);
+}
+
+mysqli_close($connection);
+
+?>
+==DONE==
+--EXPECTF--
+object(TestRow)#%d (3) {
+ ["set_from_constructor":"TestRow":private]=>
+ int(42)
+ ["data":"TestRow":private]=>
+ array(2) {
+ ["id"]=>
+ string(1) "1"
+ ["name"]=>
+ string(3) "Doe"
+ }
+ ["hello":"TestRow":private]=>
+ string(5) "world"
+}
+object(TestRow)#%d (3) {
+ ["set_from_constructor":"TestRow":private]=>
+ int(42)
+ ["data":"TestRow":private]=>
+ array(2) {
+ ["id"]=>
+ string(1) "2"
+ ["name"]=>
+ string(3) "Joe"
+ }
+ ["hello":"TestRow":private]=>
+ string(5) "world"
+}
+==DONE==
diff --git a/ext/pgsql/pgsql.c b/ext/pgsql/pgsql.c
index 8f3518d7c2..84195f7142 100644
--- a/ext/pgsql/pgsql.c
+++ b/ext/pgsql/pgsql.c
@@ -2784,9 +2784,13 @@ static void php_pgsql_fetch_hash(INTERNAL_FUNCTION_PARAMETERS, long result_type,
zend_fcall_info fci;
zend_fcall_info_cache fcc;
zval *retval_ptr;
+ zend_bool props_merged = 0;
object_and_properties_init(return_value, ce, NULL);
- zend_merge_properties(return_value, Z_ARRVAL(dataset), 1 TSRMLS_CC);
+ if (!ce->__set) {
+ props_merged = 1;
+ zend_merge_properties(return_value, Z_ARRVAL(dataset), 1 TSRMLS_CC);
+ }
if (ce->constructor) {
fci.size = sizeof(fci);
@@ -2820,6 +2824,10 @@ static void php_pgsql_fetch_hash(INTERNAL_FUNCTION_PARAMETERS, long result_type,
if (zend_call_function(&fci, &fcc TSRMLS_CC) == FAILURE) {
zend_throw_exception_ex(zend_exception_get_default(TSRMLS_C), 0 TSRMLS_CC, "Could not execute %s::%s()", ce->name, ce->constructor->common.function_name);
+ if (fci.params) {
+ efree(fci.params);
+ }
+ return;
} else {
if (retval_ptr) {
zval_ptr_dtor(&retval_ptr);
@@ -2830,6 +2838,11 @@ static void php_pgsql_fetch_hash(INTERNAL_FUNCTION_PARAMETERS, long result_type,
}
} else if (ctor_params) {
zend_throw_exception_ex(zend_exception_get_default(TSRMLS_C), 0 TSRMLS_CC, "Class %s does not have a constructor hence you cannot use ctor_params", ce->name);
+ return;
+ }
+
+ if (!props_merged) {
+ zend_merge_properties(return_value, Z_ARRVAL(dataset), 1 TSRMLS_CC);
}
}
}
diff --git a/ext/pgsql/tests/bug71820.phpt b/ext/pgsql/tests/bug71820.phpt
new file mode 100644
index 0000000000..4d99e3468e
--- /dev/null
+++ b/ext/pgsql/tests/bug71820.phpt
@@ -0,0 +1,93 @@
+--TEST--
+Bug #71820 pg_fetch_object bind parameters before call constructor
+--SKIPIF--
+<?php
+require_once('skipif.inc');
+?>
+--FILE--
+<?php
+
+require_once('config.inc');
+
+$tableName = 'test_pg_fetch_object';
+
+class TestRow
+{
+
+ private $set_from_constructor;
+ private $data;
+ private $hello = 42;
+
+ public function __construct($set_from_constructor)
+ {
+ $this->set_from_constructor = $set_from_constructor;
+ }
+
+ public function __set($name, $value)
+ {
+ if (!isset($this->data[$name])) {
+ /* $this->set_from_constructor has an expected value */
+ $this->data[$name] = 42 == $this->set_from_constructor ? $value : -1;
+ return;
+ }
+ throw new \Exception('Duplicity column name.');
+ }
+
+}
+
+$connection = pg_connect($conn_str);
+
+if (!$connection) {
+ die('Connection faild.');
+}
+
+$table = <<<SQL
+CREATE TABLE IF NOT EXISTS $tableName (
+ id serial NOT NULL,
+ name character varying NOT NULL
+);
+SQL;
+pg_query($connection, $table);
+
+pg_query_params('INSERT INTO ' . $tableName . ' (name) VALUES ($1), ($2);', ['$1' => 'Doe', '$2' => 'Joe']);
+
+$result = pg_query('SELECT * FROM ' . $tableName . ' LIMIT 10;');
+
+while ($row = pg_fetch_object($result, NULL, 'TestRow', [42])) {
+ var_dump($row);
+}
+
+pg_query($connection, "DROP TABLE $tableName");
+
+pg_close($connection);
+
+?>
+==DONE==
+--EXPECTF--
+object(TestRow)#%d (3) {
+ ["set_from_constructor":"TestRow":private]=>
+ int(42)
+ ["data":"TestRow":private]=>
+ array(2) {
+ ["id"]=>
+ string(1) "1"
+ ["name"]=>
+ string(3) "Doe"
+ }
+ ["hello":"TestRow":private]=>
+ int(42)
+}
+object(TestRow)#%d (3) {
+ ["set_from_constructor":"TestRow":private]=>
+ int(42)
+ ["data":"TestRow":private]=>
+ array(2) {
+ ["id"]=>
+ string(1) "2"
+ ["name"]=>
+ string(3) "Joe"
+ }
+ ["hello":"TestRow":private]=>
+ int(42)
+}
+==DONE==