mirror of
https://github.com/phpv8/v8js.git
synced 2024-12-22 08:11:52 +00:00
Handle property visibility and __get, refs #79
Protected and private properties should not be available to JS context. Instead call __get function, if the property is not accessible.
This commit is contained in:
parent
c54b49d2a6
commit
170b1ff94c
33
tests/property_visibility.phpt
Normal file
33
tests/property_visibility.phpt
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
--TEST--
|
||||||
|
Test V8::executeString() : Property visibility
|
||||||
|
--SKIPIF--
|
||||||
|
<?php require_once(dirname(__FILE__) . '/skipif.inc'); ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
|
||||||
|
class Foo {
|
||||||
|
private $privBar = "privBar";
|
||||||
|
protected $protBar = "protBar";
|
||||||
|
public $pubBar = "pubBar";
|
||||||
|
}
|
||||||
|
|
||||||
|
$js = new V8Js();
|
||||||
|
|
||||||
|
$js->foo = new Foo();
|
||||||
|
|
||||||
|
$script = <<<END
|
||||||
|
|
||||||
|
var_dump(PHP.foo.privBr);
|
||||||
|
var_dump(PHP.foo.protBar);
|
||||||
|
var_dump(PHP.foo.pubBar);
|
||||||
|
|
||||||
|
END;
|
||||||
|
|
||||||
|
$js->executeString($script);
|
||||||
|
?>
|
||||||
|
===EOF===
|
||||||
|
--EXPECT--
|
||||||
|
NULL
|
||||||
|
NULL
|
||||||
|
string(6) "pubBar"
|
||||||
|
===EOF===
|
44
tests/property_visibility__get.phpt
Normal file
44
tests/property_visibility__get.phpt
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
--TEST--
|
||||||
|
Test V8::executeString() : Property visibility __get
|
||||||
|
--SKIPIF--
|
||||||
|
<?php require_once(dirname(__FILE__) . '/skipif.inc'); ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
|
||||||
|
class Foo {
|
||||||
|
private $privBar = "privBar";
|
||||||
|
protected $protBar = "protBar";
|
||||||
|
public $pubBar = "pubBar";
|
||||||
|
|
||||||
|
public function __get($key)
|
||||||
|
{
|
||||||
|
var_dump($key);
|
||||||
|
return 42;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$js = new V8Js();
|
||||||
|
|
||||||
|
$js->foo = new Foo();
|
||||||
|
|
||||||
|
$script = <<<END
|
||||||
|
|
||||||
|
var_dump(PHP.foo.unknownBar);
|
||||||
|
var_dump(PHP.foo.privBar);
|
||||||
|
var_dump(PHP.foo.protBar);
|
||||||
|
var_dump(PHP.foo.pubBar);
|
||||||
|
|
||||||
|
END;
|
||||||
|
|
||||||
|
$js->executeString($script);
|
||||||
|
?>
|
||||||
|
===EOF===
|
||||||
|
--EXPECT--
|
||||||
|
string(10) "unknownBar"
|
||||||
|
int(42)
|
||||||
|
string(7) "privBar"
|
||||||
|
int(42)
|
||||||
|
string(7) "protBar"
|
||||||
|
int(42)
|
||||||
|
string(6) "pubBar"
|
||||||
|
===EOF===
|
@ -604,15 +604,52 @@ static inline v8::Local<v8::Value> php_v8js_named_property_callback(v8::Local<v8
|
|||||||
}
|
}
|
||||||
if (callback_type == V8JS_PROP_GETTER) {
|
if (callback_type == V8JS_PROP_GETTER) {
|
||||||
/* Nope, not a method -- must be a (case-sensitive) property */
|
/* Nope, not a method -- must be a (case-sensitive) property */
|
||||||
php_value = zend_read_property(scope, object, V8JS_CONST name, name_len, true TSRMLS_CC);
|
zval zname;
|
||||||
// special case uninitialized_zval_ptr and return an empty value
|
ZVAL_STRINGL(&zname, name, name_len, 0);
|
||||||
// (indicating that we don't intercept this property) if the
|
zend_property_info *property_info = zend_get_property_info(ce, &zname, 1 TSRMLS_CC);
|
||||||
// property doesn't exist.
|
|
||||||
if (php_value == EG(uninitialized_zval_ptr)) {
|
if(property_info && property_info->flags & ZEND_ACC_PUBLIC) {
|
||||||
ret_value = v8::Handle<v8::Value>();
|
php_value = zend_read_property(NULL, object, V8JS_CONST name, name_len, true TSRMLS_CC);
|
||||||
} else {
|
// special case uninitialized_zval_ptr and return an empty value
|
||||||
// wrap it
|
// (indicating that we don't intercept this property) if the
|
||||||
|
// property doesn't exist.
|
||||||
|
if (php_value == EG(uninitialized_zval_ptr)) {
|
||||||
|
ret_value = v8::Handle<v8::Value>();
|
||||||
|
} else {
|
||||||
|
// wrap it
|
||||||
|
ret_value = zval_to_v8js(php_value, isolate TSRMLS_CC);
|
||||||
|
/* We don't own the reference to php_value... unless the
|
||||||
|
* returned refcount was 0, in which case the below code
|
||||||
|
* will free it. */
|
||||||
|
zval_add_ref(&php_value);
|
||||||
|
zval_ptr_dtor(&php_value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (zend_hash_find(&ce->function_table, "__get", 6, (void**)&method_ptr) == SUCCESS
|
||||||
|
/* Allow only public methods */
|
||||||
|
&& ((method_ptr->common.fn_flags & ZEND_ACC_PUBLIC) != 0)) {
|
||||||
|
/* Okay, let's call __get. */
|
||||||
|
zend_fcall_info fci;
|
||||||
|
|
||||||
|
zval fmember;
|
||||||
|
ZVAL_STRING(&fmember, "__get", 0);
|
||||||
|
|
||||||
|
fci.size = sizeof(fci);
|
||||||
|
fci.function_table = &ce->function_table;
|
||||||
|
fci.function_name = &fmember;
|
||||||
|
fci.symbol_table = NULL;
|
||||||
|
fci.object_ptr = object;
|
||||||
|
fci.retval_ptr_ptr = &php_value;
|
||||||
|
fci.param_count = 1;
|
||||||
|
|
||||||
|
zval *zname_ptr = &zname;
|
||||||
|
zval **zname_ptr_ptr = &zname_ptr;
|
||||||
|
fci.params = &zname_ptr_ptr;
|
||||||
|
|
||||||
|
zend_call_function(&fci, NULL TSRMLS_CC);
|
||||||
|
|
||||||
ret_value = zval_to_v8js(php_value, isolate TSRMLS_CC);
|
ret_value = zval_to_v8js(php_value, isolate TSRMLS_CC);
|
||||||
|
|
||||||
/* We don't own the reference to php_value... unless the
|
/* We don't own the reference to php_value... unless the
|
||||||
* returned refcount was 0, in which case the below code
|
* returned refcount was 0, in which case the below code
|
||||||
* will free it. */
|
* will free it. */
|
||||||
|
Loading…
Reference in New Issue
Block a user