mirror of
https://github.com/phpv8/v8js.git
synced 2024-12-22 03:01:51 +00:00
Merge branch 'php7' into php8
This commit is contained in:
commit
7887ec6475
16
README.md
16
README.md
@ -104,6 +104,15 @@ class V8Js
|
|||||||
public function setModuleNormaliser(callable $normaliser)
|
public function setModuleNormaliser(callable $normaliser)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provate a function or method to be used to convert/proxy PHP exceptions to JS.
|
||||||
|
* This can be any valid PHP callable.
|
||||||
|
* The converter function will receive the PHP Exception instance that has not been caught and
|
||||||
|
* is due to be forwarded to JS. Pass NULL as $filter to uninstall an existing filter.
|
||||||
|
*/
|
||||||
|
public function setExceptionFilter(callable $filter)
|
||||||
|
{}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compiles and executes script in object's context with optional identifier string.
|
* Compiles and executes script in object's context with optional identifier string.
|
||||||
* A time limit (milliseconds) and/or memory limit (bytes) can be provided to restrict execution. These options will throw a V8JsTimeLimitException or V8JsMemoryLimitException.
|
* A time limit (milliseconds) and/or memory limit (bytes) can be provided to restrict execution. These options will throw a V8JsTimeLimitException or V8JsMemoryLimitException.
|
||||||
@ -369,3 +378,10 @@ objects obeying the above rules and re-thrown in JavaScript context. If they
|
|||||||
are not caught by JavaScript code the execution stops and a
|
are not caught by JavaScript code the execution stops and a
|
||||||
`V8JsScriptException` is thrown, which has the original PHP exception accessible
|
`V8JsScriptException` is thrown, which has the original PHP exception accessible
|
||||||
via `getPrevious` method.
|
via `getPrevious` method.
|
||||||
|
|
||||||
|
Consider that the JS code has access to methods like `getTrace` on the exception
|
||||||
|
object. This might be unwanted behaviour, if you execute untrusted code.
|
||||||
|
Using `setExceptionFilter` method a callable can be provided, that may convert
|
||||||
|
the PHP exception to some other value that is safe to expose. The filter may
|
||||||
|
also decide not to propagate the exception to JS at all by either re-throwing
|
||||||
|
the passed exception or throwing another exception.
|
||||||
|
26
config.m4
26
config.m4
@ -39,11 +39,15 @@ if test "$PHP_V8JS" != "no"; then
|
|||||||
|
|
||||||
|
|
||||||
AC_CACHE_CHECK(for C standard version, ac_cv_v8_cstd, [
|
AC_CACHE_CHECK(for C standard version, ac_cv_v8_cstd, [
|
||||||
ac_cv_v8_cstd="c++14"
|
ac_cv_v8_cstd="c++17"
|
||||||
old_CPPFLAGS=$CPPFLAGS
|
old_CPPFLAGS=$CPPFLAGS
|
||||||
AC_LANG_PUSH([C++])
|
AC_LANG_PUSH([C++])
|
||||||
CPPFLAGS="-std="$ac_cv_v8_cstd
|
CPPFLAGS="-std="$ac_cv_v8_cstd
|
||||||
AC_RUN_IFELSE([AC_LANG_SOURCE([[int main() { return 0; }]])],[],[ac_cv_v8_cstd="c++1y"],[])
|
AC_RUN_IFELSE([AC_LANG_SOURCE([[int main() { return 0; }]])],[],[
|
||||||
|
ac_cv_v8_cstd="c++14"
|
||||||
|
CPPFLAGS="-std="$ac_cv_v8_cstd
|
||||||
|
AC_RUN_IFELSE([AC_LANG_SOURCE([[int main() { return 0; }]])],[],[ ac_cv_v8_cstd="c++1y" ],[])
|
||||||
|
],[])
|
||||||
AC_LANG_POP([C++])
|
AC_LANG_POP([C++])
|
||||||
CPPFLAGS=$old_CPPFLAGS
|
CPPFLAGS=$old_CPPFLAGS
|
||||||
]);
|
]);
|
||||||
@ -173,6 +177,24 @@ int main ()
|
|||||||
V8_SEARCH_BLOB([snapshot_blob.bin], [PHP_V8_SNAPSHOT_BLOB_PATH])
|
V8_SEARCH_BLOB([snapshot_blob.bin], [PHP_V8_SNAPSHOT_BLOB_PATH])
|
||||||
|
|
||||||
|
|
||||||
|
dnl
|
||||||
|
dnl Check for v8::V8::InitializeSandbox
|
||||||
|
dnl
|
||||||
|
AC_CACHE_CHECK([for v8::V8::InitializeSandbox], ac_cv_has_initialize_sandbox, [
|
||||||
|
AC_LINK_IFELSE([AC_LANG_PROGRAM([
|
||||||
|
#define V8_ENABLE_SANDBOX 1
|
||||||
|
#include <v8.h>
|
||||||
|
], [ v8::V8::InitializeSandbox(); ])], [
|
||||||
|
ac_cv_has_initialize_sandbox=yes
|
||||||
|
], [
|
||||||
|
ac_cv_has_initialize_sandbox=no
|
||||||
|
])
|
||||||
|
])
|
||||||
|
if test "x$ac_cv_has_initialize_sandbox" = "xyes"; then
|
||||||
|
AC_DEFINE([V8_HAS_INITIALIZE_SANDBOX], [1],
|
||||||
|
[Define if V8::InitializeSandbox must be called.])
|
||||||
|
fi
|
||||||
|
|
||||||
dnl
|
dnl
|
||||||
dnl Check for v8::ArrayBuffer::Allocator::NewDefaultAllocator
|
dnl Check for v8::ArrayBuffer::Allocator::NewDefaultAllocator
|
||||||
dnl
|
dnl
|
||||||
|
@ -52,6 +52,10 @@ extern "C" {
|
|||||||
#undef COMPILER
|
#undef COMPILER
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef V8_HAS_INITIALIZE_SANDBOX
|
||||||
|
#define V8_ENABLE_SANDBOX 1
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <v8.h>
|
#include <v8.h>
|
||||||
#include <v8-platform.h>
|
#include <v8-platform.h>
|
||||||
|
|
||||||
|
35
tests/exception_filter_001.phpt
Normal file
35
tests/exception_filter_001.phpt
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
--TEST--
|
||||||
|
Test V8::setExceptionFilter() : String conversion
|
||||||
|
--SKIPIF--
|
||||||
|
<?php require_once(dirname(__FILE__) . '/skipif.inc'); ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
class myv8 extends V8Js
|
||||||
|
{
|
||||||
|
public function throwException(string $message) {
|
||||||
|
throw new Exception($message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$v8 = new myv8();
|
||||||
|
$v8->setExceptionFilter(function (Throwable $ex) {
|
||||||
|
echo "exception filter called.\n";
|
||||||
|
return $ex->getMessage();
|
||||||
|
});
|
||||||
|
|
||||||
|
$v8->executeString('
|
||||||
|
try {
|
||||||
|
PHP.throwException("Oops");
|
||||||
|
}
|
||||||
|
catch (e) {
|
||||||
|
var_dump(typeof e); // string
|
||||||
|
var_dump(e);
|
||||||
|
}
|
||||||
|
', null, V8Js::FLAG_PROPAGATE_PHP_EXCEPTIONS);
|
||||||
|
?>
|
||||||
|
===EOF===
|
||||||
|
--EXPECT--
|
||||||
|
exception filter called.
|
||||||
|
string(6) "string"
|
||||||
|
string(4) "Oops"
|
||||||
|
===EOF===
|
32
tests/exception_filter_002.phpt
Normal file
32
tests/exception_filter_002.phpt
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
--TEST--
|
||||||
|
Test V8::setExceptionFilter() : Filter handling on exception in setModuleLoader
|
||||||
|
--SKIPIF--
|
||||||
|
<?php require_once(dirname(__FILE__) . '/skipif.inc'); ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
|
||||||
|
$v8 = new V8Js();
|
||||||
|
$v8->setModuleLoader(function ($path) {
|
||||||
|
throw new Error('moep');
|
||||||
|
});
|
||||||
|
|
||||||
|
$v8->setExceptionFilter(function (Throwable $ex) {
|
||||||
|
echo "exception filter called.\n";
|
||||||
|
return $ex->getMessage();
|
||||||
|
});
|
||||||
|
|
||||||
|
$v8->executeString('
|
||||||
|
try {
|
||||||
|
require("file");
|
||||||
|
} catch(e) {
|
||||||
|
var_dump(e);
|
||||||
|
}
|
||||||
|
', null, V8Js::FLAG_PROPAGATE_PHP_EXCEPTIONS);
|
||||||
|
|
||||||
|
?>
|
||||||
|
===EOF===
|
||||||
|
--EXPECT--
|
||||||
|
exception filter called.
|
||||||
|
string(4) "moep"
|
||||||
|
===EOF===
|
||||||
|
|
34
tests/exception_filter_003.phpt
Normal file
34
tests/exception_filter_003.phpt
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
--TEST--
|
||||||
|
Test V8::setExceptionFilter() : Filter handling on exception in setModuleNormaliser
|
||||||
|
--SKIPIF--
|
||||||
|
<?php require_once(dirname(__FILE__) . '/skipif.inc'); ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
|
||||||
|
$v8 = new V8Js();
|
||||||
|
$v8->setModuleNormaliser(function ($path) {
|
||||||
|
throw new Error('blarg');
|
||||||
|
});
|
||||||
|
$v8->setModuleLoader(function ($path) {
|
||||||
|
throw new Error('moep');
|
||||||
|
});
|
||||||
|
|
||||||
|
$v8->setExceptionFilter(function (Throwable $ex) {
|
||||||
|
echo "exception filter called.\n";
|
||||||
|
return $ex->getMessage();
|
||||||
|
});
|
||||||
|
|
||||||
|
$v8->executeString('
|
||||||
|
try {
|
||||||
|
require("file");
|
||||||
|
} catch(e) {
|
||||||
|
var_dump(e);
|
||||||
|
}
|
||||||
|
', null, V8Js::FLAG_PROPAGATE_PHP_EXCEPTIONS);
|
||||||
|
|
||||||
|
?>
|
||||||
|
===EOF===
|
||||||
|
--EXPECT--
|
||||||
|
exception filter called.
|
||||||
|
string(5) "blarg"
|
||||||
|
===EOF===
|
37
tests/exception_filter_004.phpt
Normal file
37
tests/exception_filter_004.phpt
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
--TEST--
|
||||||
|
Test V8::setExceptionFilter() : Filter handling on exception in converter
|
||||||
|
--SKIPIF--
|
||||||
|
<?php require_once(dirname(__FILE__) . '/skipif.inc'); ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
class myv8 extends V8Js
|
||||||
|
{
|
||||||
|
public function throwException(string $message) {
|
||||||
|
throw new Exception($message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$v8 = new myv8();
|
||||||
|
$v8->setExceptionFilter(function (Throwable $ex) {
|
||||||
|
throw new Exception('moep');
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
$v8->executeString('
|
||||||
|
try {
|
||||||
|
PHP.throwException("Oops");
|
||||||
|
print("done\\n");
|
||||||
|
}
|
||||||
|
catch (e) {
|
||||||
|
print("caught\\n");
|
||||||
|
var_dump(e);
|
||||||
|
}
|
||||||
|
', null, V8Js::FLAG_PROPAGATE_PHP_EXCEPTIONS);
|
||||||
|
} catch (Exception $ex) {
|
||||||
|
echo "caught in php: " . $ex->getMessage() . PHP_EOL;
|
||||||
|
}
|
||||||
|
?>
|
||||||
|
===EOF===
|
||||||
|
--EXPECT--
|
||||||
|
caught in php: moep
|
||||||
|
===EOF===
|
53
tests/exception_filter_005.phpt
Normal file
53
tests/exception_filter_005.phpt
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
--TEST--
|
||||||
|
Test V8::setExceptionFilter() : Uninstall filter on NULL
|
||||||
|
--SKIPIF--
|
||||||
|
<?php require_once(dirname(__FILE__) . '/skipif.inc'); ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
class myv8 extends V8Js
|
||||||
|
{
|
||||||
|
public function throwException(string $message) {
|
||||||
|
throw new Exception($message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$v8 = new myv8();
|
||||||
|
$v8->setExceptionFilter(function (Throwable $ex) {
|
||||||
|
echo "exception filter called.\n";
|
||||||
|
return "moep";
|
||||||
|
});
|
||||||
|
|
||||||
|
$v8->executeString('
|
||||||
|
try {
|
||||||
|
PHP.throwException("Oops");
|
||||||
|
}
|
||||||
|
catch (e) {
|
||||||
|
var_dump(e);
|
||||||
|
}
|
||||||
|
', null, V8Js::FLAG_PROPAGATE_PHP_EXCEPTIONS);
|
||||||
|
|
||||||
|
$v8->setExceptionFilter(null);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$v8->executeString('
|
||||||
|
try {
|
||||||
|
PHP.throwException("Oops");
|
||||||
|
print("done\\n");
|
||||||
|
}
|
||||||
|
catch (e) {
|
||||||
|
print("caught\\n");
|
||||||
|
var_dump(e.getMessage());
|
||||||
|
}
|
||||||
|
', null, V8Js::FLAG_PROPAGATE_PHP_EXCEPTIONS);
|
||||||
|
} catch (Exception $ex) {
|
||||||
|
echo "caught in php: " . $ex->getMessage() . PHP_EOL;
|
||||||
|
}
|
||||||
|
|
||||||
|
?>
|
||||||
|
===EOF===
|
||||||
|
--EXPECT--
|
||||||
|
exception filter called.
|
||||||
|
string(4) "moep"
|
||||||
|
caught
|
||||||
|
string(4) "Oops"
|
||||||
|
===EOF===
|
38
tests/exception_filter_006.phpt
Normal file
38
tests/exception_filter_006.phpt
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
--TEST--
|
||||||
|
Test V8::setExceptionFilter() : re-throw exception in exception filter
|
||||||
|
--SKIPIF--
|
||||||
|
<?php require_once(dirname(__FILE__) . '/skipif.inc'); ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
class myv8 extends V8Js
|
||||||
|
{
|
||||||
|
public function throwException(string $message) {
|
||||||
|
throw new Exception($message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$v8 = new myv8();
|
||||||
|
$v8->setExceptionFilter(function (Throwable $ex) {
|
||||||
|
// re-throw exception so it is not forwarded
|
||||||
|
throw $ex;
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
$v8->executeString('
|
||||||
|
try {
|
||||||
|
PHP.throwException("Oops");
|
||||||
|
print("done\\n");
|
||||||
|
}
|
||||||
|
catch (e) {
|
||||||
|
print("caught\\n");
|
||||||
|
var_dump(e);
|
||||||
|
}
|
||||||
|
', null, V8Js::FLAG_PROPAGATE_PHP_EXCEPTIONS);
|
||||||
|
} catch (Exception $ex) {
|
||||||
|
echo "caught in php: " . $ex->getMessage() . PHP_EOL;
|
||||||
|
}
|
||||||
|
?>
|
||||||
|
===EOF===
|
||||||
|
--EXPECT--
|
||||||
|
caught in php: Oops
|
||||||
|
===EOF===
|
55
tests/exception_filter_basic.phpt
Normal file
55
tests/exception_filter_basic.phpt
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
--TEST--
|
||||||
|
Test V8::setExceptionFilter() : Simple test
|
||||||
|
--SKIPIF--
|
||||||
|
<?php require_once(dirname(__FILE__) . '/skipif.inc'); ?>
|
||||||
|
--FILE--
|
||||||
|
<?php
|
||||||
|
class myv8 extends V8Js
|
||||||
|
{
|
||||||
|
public function throwException(string $message) {
|
||||||
|
throw new Exception($message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class ExceptionFilter {
|
||||||
|
private $ex;
|
||||||
|
|
||||||
|
public function __construct(Throwable $ex) {
|
||||||
|
echo "ExceptionFilter::__construct called!\n";
|
||||||
|
var_dump($ex->getMessage());
|
||||||
|
|
||||||
|
$this->ex = $ex;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getMessage() {
|
||||||
|
echo "getMessage called\n";
|
||||||
|
return $this->ex->getMessage();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$v8 = new myv8();
|
||||||
|
$v8->setExceptionFilter(function (Throwable $ex) {
|
||||||
|
echo "exception filter called.\n";
|
||||||
|
return new ExceptionFilter($ex);
|
||||||
|
});
|
||||||
|
|
||||||
|
$v8->executeString('
|
||||||
|
try {
|
||||||
|
PHP.throwException("Oops");
|
||||||
|
}
|
||||||
|
catch (e) {
|
||||||
|
var_dump(e.getMessage()); // calls ExceptionFilter::getMessage
|
||||||
|
var_dump(typeof e.getTrace);
|
||||||
|
}
|
||||||
|
', null, V8Js::FLAG_PROPAGATE_PHP_EXCEPTIONS);
|
||||||
|
?>
|
||||||
|
===EOF===
|
||||||
|
--EXPECT--
|
||||||
|
exception filter called.
|
||||||
|
ExceptionFilter::__construct called!
|
||||||
|
string(4) "Oops"
|
||||||
|
getMessage called
|
||||||
|
string(4) "Oops"
|
||||||
|
string(9) "undefined"
|
||||||
|
===EOF===
|
||||||
|
|
@ -80,6 +80,7 @@ static void v8js_free_storage(zend_object *object) /* {{{ */
|
|||||||
|
|
||||||
zval_ptr_dtor(&c->module_normaliser);
|
zval_ptr_dtor(&c->module_normaliser);
|
||||||
zval_ptr_dtor(&c->module_loader);
|
zval_ptr_dtor(&c->module_loader);
|
||||||
|
zval_ptr_dtor(&c->exception_filter);
|
||||||
|
|
||||||
/* Delete PHP global object from JavaScript */
|
/* Delete PHP global object from JavaScript */
|
||||||
if (!c->context.IsEmpty()) {
|
if (!c->context.IsEmpty()) {
|
||||||
@ -312,6 +313,7 @@ static PHP_METHOD(V8Js, __construct)
|
|||||||
|
|
||||||
ZVAL_NULL(&c->module_normaliser);
|
ZVAL_NULL(&c->module_normaliser);
|
||||||
ZVAL_NULL(&c->module_loader);
|
ZVAL_NULL(&c->module_loader);
|
||||||
|
ZVAL_NULL(&c->exception_filter);
|
||||||
|
|
||||||
// Isolate execution
|
// Isolate execution
|
||||||
v8::Isolate *isolate = c->isolate;
|
v8::Isolate *isolate = c->isolate;
|
||||||
@ -701,6 +703,21 @@ static PHP_METHOD(V8Js, setModuleLoader)
|
|||||||
}
|
}
|
||||||
/* }}} */
|
/* }}} */
|
||||||
|
|
||||||
|
/* {{{ proto void V8Js::setExceptionFilter(callable factory)
|
||||||
|
*/
|
||||||
|
static PHP_METHOD(V8Js, setExceptionFilter)
|
||||||
|
{
|
||||||
|
zval *callable;
|
||||||
|
|
||||||
|
if (zend_parse_parameters(ZEND_NUM_ARGS(), "z", &callable) == FAILURE) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
v8js_ctx *c = Z_V8JS_CTX_OBJ_P(getThis());
|
||||||
|
ZVAL_COPY(&c->exception_filter, callable);
|
||||||
|
}
|
||||||
|
/* }}} */
|
||||||
|
|
||||||
/* {{{ proto void V8Js::setTimeLimit(int time_limit)
|
/* {{{ proto void V8Js::setTimeLimit(int time_limit)
|
||||||
*/
|
*/
|
||||||
static PHP_METHOD(V8Js, setTimeLimit)
|
static PHP_METHOD(V8Js, setTimeLimit)
|
||||||
@ -937,6 +954,10 @@ ZEND_BEGIN_ARG_INFO_EX(arginfo_v8js_setmoduleloader, 0, 0, 1)
|
|||||||
ZEND_ARG_INFO(0, callable)
|
ZEND_ARG_INFO(0, callable)
|
||||||
ZEND_END_ARG_INFO()
|
ZEND_END_ARG_INFO()
|
||||||
|
|
||||||
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_v8js_setexceptionfilter, 0, 0, 1)
|
||||||
|
ZEND_ARG_INFO(0, callable)
|
||||||
|
ZEND_END_ARG_INFO()
|
||||||
|
|
||||||
ZEND_BEGIN_ARG_INFO_EX(arginfo_v8js_setaverageobjectsize, 0, 0, 1)
|
ZEND_BEGIN_ARG_INFO_EX(arginfo_v8js_setaverageobjectsize, 0, 0, 1)
|
||||||
ZEND_ARG_INFO(0, average_object_size)
|
ZEND_ARG_INFO(0, average_object_size)
|
||||||
ZEND_END_ARG_INFO()
|
ZEND_END_ARG_INFO()
|
||||||
@ -963,6 +984,7 @@ const zend_function_entry v8js_methods[] = { /* {{{ */
|
|||||||
PHP_ME(V8Js, executeScript, arginfo_v8js_executescript, ZEND_ACC_PUBLIC)
|
PHP_ME(V8Js, executeScript, arginfo_v8js_executescript, ZEND_ACC_PUBLIC)
|
||||||
PHP_ME(V8Js, setModuleNormaliser, arginfo_v8js_setmodulenormaliser, ZEND_ACC_PUBLIC)
|
PHP_ME(V8Js, setModuleNormaliser, arginfo_v8js_setmodulenormaliser, ZEND_ACC_PUBLIC)
|
||||||
PHP_ME(V8Js, setModuleLoader, arginfo_v8js_setmoduleloader, ZEND_ACC_PUBLIC)
|
PHP_ME(V8Js, setModuleLoader, arginfo_v8js_setmoduleloader, ZEND_ACC_PUBLIC)
|
||||||
|
PHP_ME(V8Js, setExceptionFilter, arginfo_v8js_setexceptionfilter, ZEND_ACC_PUBLIC)
|
||||||
PHP_ME(V8Js, setTimeLimit, arginfo_v8js_settimelimit, ZEND_ACC_PUBLIC)
|
PHP_ME(V8Js, setTimeLimit, arginfo_v8js_settimelimit, ZEND_ACC_PUBLIC)
|
||||||
PHP_ME(V8Js, setMemoryLimit, arginfo_v8js_setmemorylimit, ZEND_ACC_PUBLIC)
|
PHP_ME(V8Js, setMemoryLimit, arginfo_v8js_setmemorylimit, ZEND_ACC_PUBLIC)
|
||||||
PHP_ME(V8Js, setAverageObjectSize, arginfo_v8js_setaverageobjectsize, ZEND_ACC_PUBLIC)
|
PHP_ME(V8Js, setAverageObjectSize, arginfo_v8js_setaverageobjectsize, ZEND_ACC_PUBLIC)
|
||||||
|
@ -53,6 +53,7 @@ struct v8js_ctx {
|
|||||||
|
|
||||||
zval module_normaliser;
|
zval module_normaliser;
|
||||||
zval module_loader;
|
zval module_loader;
|
||||||
|
zval exception_filter;
|
||||||
|
|
||||||
std::vector<char *> modules_stack;
|
std::vector<char *> modules_stack;
|
||||||
std::map<char *, v8js_persistent_value_t, cmp_str> modules_loaded;
|
std::map<char *, v8js_persistent_value_t, cmp_str> modules_loaded;
|
||||||
|
@ -162,7 +162,11 @@ static PHP_MSHUTDOWN_FUNCTION(v8js)
|
|||||||
|
|
||||||
if(v8_initialized) {
|
if(v8_initialized) {
|
||||||
v8::V8::Dispose();
|
v8::V8::Dispose();
|
||||||
|
#if PHP_V8_API_VERSION >= 10000000
|
||||||
|
v8::V8::DisposePlatform();
|
||||||
|
#else
|
||||||
v8::V8::ShutdownPlatform();
|
v8::V8::ShutdownPlatform();
|
||||||
|
#endif
|
||||||
// @fixme call virtual destructor somehow
|
// @fixme call virtual destructor somehow
|
||||||
//delete v8js_process_globals.v8_platform;
|
//delete v8js_process_globals.v8_platform;
|
||||||
}
|
}
|
||||||
|
@ -19,6 +19,7 @@
|
|||||||
#include "php_v8js_macros.h"
|
#include "php_v8js_macros.h"
|
||||||
#include "v8js_commonjs.h"
|
#include "v8js_commonjs.h"
|
||||||
#include "v8js_exceptions.h"
|
#include "v8js_exceptions.h"
|
||||||
|
#include "v8js_object_export.h"
|
||||||
|
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#include "zend_exceptions.h"
|
#include "zend_exceptions.h"
|
||||||
@ -337,14 +338,7 @@ V8JS_METHOD(require)
|
|||||||
|
|
||||||
// Check if an exception was thrown
|
// Check if an exception was thrown
|
||||||
if (EG(exception)) {
|
if (EG(exception)) {
|
||||||
if (c->flags & V8JS_FLAG_PROPAGATE_PHP_EXCEPTIONS) {
|
info.GetReturnValue().Set(v8js_propagate_exception(c));
|
||||||
zval tmp_zv;
|
|
||||||
ZVAL_OBJ(&tmp_zv, EG(exception));
|
|
||||||
info.GetReturnValue().Set(isolate->ThrowException(zval_to_v8js(&tmp_zv, isolate)));
|
|
||||||
zend_clear_exception();
|
|
||||||
} else {
|
|
||||||
v8js_terminate_execution(isolate);
|
|
||||||
}
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -466,15 +460,7 @@ V8JS_METHOD(require)
|
|||||||
efree(normalised_module_id);
|
efree(normalised_module_id);
|
||||||
efree(normalised_path);
|
efree(normalised_path);
|
||||||
|
|
||||||
if (c->flags & V8JS_FLAG_PROPAGATE_PHP_EXCEPTIONS) {
|
info.GetReturnValue().Set(v8js_propagate_exception(c));
|
||||||
zval tmp_zv;
|
|
||||||
ZVAL_OBJ(&tmp_zv, EG(exception));
|
|
||||||
info.GetReturnValue().Set(isolate->ThrowException(zval_to_v8js(&tmp_zv, isolate)));
|
|
||||||
zend_clear_exception();
|
|
||||||
} else {
|
|
||||||
v8js_terminate_execution(isolate);
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -485,7 +471,7 @@ V8JS_METHOD(require)
|
|||||||
|
|
||||||
efree(normalised_path);
|
efree(normalised_path);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(Z_TYPE(module_code) == IS_OBJECT) {
|
if(Z_TYPE(module_code) == IS_OBJECT) {
|
||||||
v8::Local<v8::Object> newobj = zval_to_v8js(&module_code, isolate)->ToObject(isolate->GetEnteredOrMicrotaskContext()).ToLocalChecked();
|
v8::Local<v8::Object> newobj = zval_to_v8js(&module_code, isolate)->ToObject(isolate->GetEnteredOrMicrotaskContext()).ToLocalChecked();
|
||||||
|
@ -34,6 +34,41 @@ extern "C" {
|
|||||||
|
|
||||||
static void v8js_weak_object_callback(const v8::WeakCallbackInfo<zend_object> &data);
|
static void v8js_weak_object_callback(const v8::WeakCallbackInfo<zend_object> &data);
|
||||||
|
|
||||||
|
v8::Local<v8::Value> v8js_propagate_exception(v8js_ctx *ctx) /* {{{ */
|
||||||
|
{
|
||||||
|
v8::Local<v8::Value> return_value = v8::Null(ctx->isolate);
|
||||||
|
|
||||||
|
if (!(ctx->flags & V8JS_FLAG_PROPAGATE_PHP_EXCEPTIONS)) {
|
||||||
|
v8js_terminate_execution(ctx->isolate);
|
||||||
|
return return_value;
|
||||||
|
}
|
||||||
|
|
||||||
|
zval tmp_zv;
|
||||||
|
|
||||||
|
if (Z_TYPE(ctx->exception_filter) != IS_NULL) {
|
||||||
|
zval params[1];
|
||||||
|
ZVAL_OBJ(¶ms[0], EG(exception));
|
||||||
|
Z_ADDREF_P(¶ms[0]);
|
||||||
|
zend_clear_exception();
|
||||||
|
call_user_function(EG(function_table), NULL, &ctx->exception_filter, &tmp_zv, 1, params);
|
||||||
|
zval_ptr_dtor(¶ms[0]);
|
||||||
|
|
||||||
|
if(EG(exception)) {
|
||||||
|
// exception proxy threw exception itself, don't forward, just stop execution.
|
||||||
|
v8js_terminate_execution(ctx->isolate);
|
||||||
|
} else {
|
||||||
|
return_value = ctx->isolate->ThrowException(zval_to_v8js(&tmp_zv, ctx->isolate));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ZVAL_OBJ(&tmp_zv, EG(exception));
|
||||||
|
return_value = ctx->isolate->ThrowException(zval_to_v8js(&tmp_zv, ctx->isolate));
|
||||||
|
zend_clear_exception();
|
||||||
|
}
|
||||||
|
|
||||||
|
return return_value;
|
||||||
|
}
|
||||||
|
/* }}} */
|
||||||
|
|
||||||
/* Callback for PHP methods and functions */
|
/* Callback for PHP methods and functions */
|
||||||
static void v8js_call_php_func(zend_object *object, zend_function *method_ptr, const v8::FunctionCallbackInfo<v8::Value>& info) /* {{{ */
|
static void v8js_call_php_func(zend_object *object, zend_function *method_ptr, const v8::FunctionCallbackInfo<v8::Value>& info) /* {{{ */
|
||||||
{
|
{
|
||||||
@ -167,14 +202,7 @@ failure:
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(EG(exception)) {
|
if(EG(exception)) {
|
||||||
if(ctx->flags & V8JS_FLAG_PROPAGATE_PHP_EXCEPTIONS) {
|
return_value = v8js_propagate_exception(ctx);
|
||||||
zval tmp_zv;
|
|
||||||
ZVAL_OBJ(&tmp_zv, EG(exception));
|
|
||||||
return_value = isolate->ThrowException(zval_to_v8js(&tmp_zv, isolate));
|
|
||||||
zend_clear_exception();
|
|
||||||
} else {
|
|
||||||
v8js_terminate_execution(isolate);
|
|
||||||
}
|
|
||||||
} else if (Z_TYPE(retval) == IS_OBJECT && Z_OBJ(retval) == object) {
|
} else if (Z_TYPE(retval) == IS_OBJECT && Z_OBJ(retval) == object) {
|
||||||
// special case: "return $this"
|
// special case: "return $this"
|
||||||
return_value = info.Holder();
|
return_value = info.Holder();
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
#define V8JS_OBJECT_EXPORT_H
|
#define V8JS_OBJECT_EXPORT_H
|
||||||
|
|
||||||
v8::Local<v8::Value> v8js_hash_to_jsobj(zval *value, v8::Isolate *isolate);
|
v8::Local<v8::Value> v8js_hash_to_jsobj(zval *value, v8::Isolate *isolate);
|
||||||
|
v8::Local<v8::Value> v8js_propagate_exception(v8js_ctx *ctx);
|
||||||
|
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
|
@ -71,6 +71,10 @@ void v8js_v8_init() /* {{{ */
|
|||||||
v8js_process_globals.v8_platform = v8::platform::NewDefaultPlatform();
|
v8js_process_globals.v8_platform = v8::platform::NewDefaultPlatform();
|
||||||
v8::V8::InitializePlatform(v8js_process_globals.v8_platform.get());
|
v8::V8::InitializePlatform(v8js_process_globals.v8_platform.get());
|
||||||
|
|
||||||
|
#ifdef V8_HAS_INITIALIZE_SANDBOX
|
||||||
|
v8::V8::InitializeSandbox();
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Set V8 command line flags (must be done before V8::Initialize()!) */
|
/* Set V8 command line flags (must be done before V8::Initialize()!) */
|
||||||
if (v8js_process_globals.v8_flags) {
|
if (v8js_process_globals.v8_flags) {
|
||||||
size_t flags_len = strlen(v8js_process_globals.v8_flags);
|
size_t flags_len = strlen(v8js_process_globals.v8_flags);
|
||||||
|
@ -80,7 +80,7 @@ void v8js_register_accessors(std::vector<v8js_accessor_ctx*> *accessor_list, v8:
|
|||||||
ctx->isolate = isolate;
|
ctx->isolate = isolate;
|
||||||
|
|
||||||
/* Set the variable fetch callback for given symbol on named property */
|
/* Set the variable fetch callback for given symbol on named property */
|
||||||
php_obj->SetAccessor(V8JS_STRL(ZSTR_VAL(property_name), static_cast<int>(ZSTR_LEN(property_name))), v8js_fetch_php_variable, NULL, v8::External::New(isolate, ctx), v8::PROHIBITS_OVERWRITING, v8::ReadOnly, v8::AccessorSignature::New(isolate, php_obj_t));
|
php_obj->SetAccessor(V8JS_STRL(ZSTR_VAL(property_name), static_cast<int>(ZSTR_LEN(property_name))), v8js_fetch_php_variable, NULL, v8::External::New(isolate, ctx), v8::PROHIBITS_OVERWRITING, v8::ReadOnly);
|
||||||
|
|
||||||
/* record the context so we can free it later */
|
/* record the context so we can free it later */
|
||||||
accessor_list->push_back(ctx);
|
accessor_list->push_back(ctx);
|
||||||
|
Loading…
Reference in New Issue
Block a user