Unverified Commit c79ba5c1 authored by Paul Yang's avatar Paul Yang Committed by GitHub

Merge pull request #4034 from TeBoring/php-timestamp-bug

Avoid calling method from php extension directly
parents 269884a1 8d6f13e8
...@@ -29,7 +29,6 @@ ...@@ -29,7 +29,6 @@
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#include <php.h> #include <php.h>
#include <ext/date/php_date.h>
#include <stdlib.h> #include <stdlib.h>
#include "protobuf.h" #include "protobuf.h"
...@@ -1124,17 +1123,41 @@ PHP_METHOD(Timestamp, fromDateTime) { ...@@ -1124,17 +1123,41 @@ PHP_METHOD(Timestamp, fromDateTime) {
zval* datetime; zval* datetime;
zval member; zval member;
PHP_PROTO_CE_DECLARE date_interface_ce;
if (php_proto_zend_lookup_class("\\DatetimeInterface", 18,
&date_interface_ce) == FAILURE) {
zend_error(E_ERROR, "Make sure date extension is enabled.");
return;
}
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "O", &datetime, if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "O", &datetime,
php_date_get_date_ce()) == FAILURE) { PHP_PROTO_CE_UNREF(date_interface_ce)) == FAILURE) {
zend_error(E_USER_ERROR, "Expect DatetimeInterface.");
return; return;
} }
php_date_obj* dateobj = UNBOX(php_date_obj, datetime); // Get timestamp from Datetime object.
if (!dateobj->time->sse_uptodate) { zval retval;
timelib_update_ts(dateobj->time, NULL); zval function_name;
int64_t timestamp;
#if PHP_MAJOR_VERSION < 7
INIT_ZVAL(retval);
INIT_ZVAL(function_name);
#endif
PHP_PROTO_ZVAL_STRING(&function_name, "date_timestamp_get", 1);
if (call_user_function(EG(function_table), NULL, &function_name, &retval, 1,
ZVAL_PTR_TO_CACHED_PTR(datetime) TSRMLS_CC) == FAILURE) {
zend_error(E_ERROR, "Cannot get timestamp from DateTime.");
return;
} }
int64_t timestamp = dateobj->time->sse; protobuf_convert_to_int64(&retval, &timestamp);
zval_dtor(&retval);
zval_dtor(&function_name);
// Set seconds // Set seconds
MessageHeader* self = UNBOX(MessageHeader, getThis()); MessageHeader* self = UNBOX(MessageHeader, getThis());
...@@ -1142,20 +1165,18 @@ PHP_METHOD(Timestamp, fromDateTime) { ...@@ -1142,20 +1165,18 @@ PHP_METHOD(Timestamp, fromDateTime) {
upb_msgdef_ntofz(self->descriptor->msgdef, "seconds"); upb_msgdef_ntofz(self->descriptor->msgdef, "seconds");
void* storage = message_data(self); void* storage = message_data(self);
void* memory = slot_memory(self->descriptor->layout, storage, field); void* memory = slot_memory(self->descriptor->layout, storage, field);
*(int64_t*)memory = dateobj->time->sse; *(int64_t*)memory = timestamp;
// Set nanos // Set nanos
field = upb_msgdef_ntofz(self->descriptor->msgdef, "nanos"); field = upb_msgdef_ntofz(self->descriptor->msgdef, "nanos");
storage = message_data(self); storage = message_data(self);
memory = slot_memory(self->descriptor->layout, storage, field); memory = slot_memory(self->descriptor->layout, storage, field);
*(int32_t*)memory = 0; *(int32_t*)memory = 0;
RETURN_NULL();
} }
PHP_METHOD(Timestamp, toDateTime) { PHP_METHOD(Timestamp, toDateTime) {
zval datetime;
php_date_instantiate(php_date_get_date_ce(), &datetime TSRMLS_CC);
php_date_obj* dateobj = UNBOX(php_date_obj, &datetime);
// Get seconds // Get seconds
MessageHeader* self = UNBOX(MessageHeader, getThis()); MessageHeader* self = UNBOX(MessageHeader, getThis());
const upb_fielddef* field = const upb_fielddef* field =
...@@ -1176,14 +1197,38 @@ PHP_METHOD(Timestamp, toDateTime) { ...@@ -1176,14 +1197,38 @@ PHP_METHOD(Timestamp, toDateTime) {
strftime(formated_time, sizeof(formated_time), "%Y-%m-%dT%H:%M:%SUTC", strftime(formated_time, sizeof(formated_time), "%Y-%m-%dT%H:%M:%SUTC",
utc_time); utc_time);
if (!php_date_initialize(dateobj, formated_time, strlen(formated_time), NULL, // Create Datetime object.
NULL, 0 TSRMLS_CC)) { zval datetime;
zval_dtor(&datetime); zval formated_time_php;
RETURN_NULL(); zval function_name;
int64_t timestamp = 0;
#if PHP_MAJOR_VERSION < 7
INIT_ZVAL(function_name);
INIT_ZVAL(formated_time_php);
#endif
PHP_PROTO_ZVAL_STRING(&function_name, "date_create", 1);
PHP_PROTO_ZVAL_STRING(&formated_time_php, formated_time, 1);
CACHED_VALUE params[1] = {ZVAL_TO_CACHED_VALUE(formated_time_php)};
if (call_user_function(EG(function_table), NULL,
&function_name, &datetime, 1,
params TSRMLS_CC) == FAILURE) {
zend_error(E_ERROR, "Cannot create DateTime.");
return;
} }
zval_dtor(&formated_time_php);
zval_dtor(&function_name);
#if PHP_MAJOR_VERSION < 7
zval* datetime_ptr = &datetime; zval* datetime_ptr = &datetime;
PHP_PROTO_RETVAL_ZVAL(datetime_ptr); PHP_PROTO_RETVAL_ZVAL(datetime_ptr);
#else
ZVAL_OBJ(return_value, Z_OBJ(datetime));
#endif
} }
// ----------------------------------------------------------------------------- // -----------------------------------------------------------------------------
......
...@@ -182,8 +182,15 @@ zend_function_entry protobuf_functions[] = { ...@@ -182,8 +182,15 @@ zend_function_entry protobuf_functions[] = {
ZEND_FE_END ZEND_FE_END
}; };
static const zend_module_dep protobuf_deps[] = {
ZEND_MOD_OPTIONAL("date")
ZEND_MOD_END
};
zend_module_entry protobuf_module_entry = { zend_module_entry protobuf_module_entry = {
STANDARD_MODULE_HEADER, STANDARD_MODULE_HEADER_EX,
NULL,
protobuf_deps,
PHP_PROTOBUF_EXTNAME, // extension name PHP_PROTOBUF_EXTNAME, // extension name
protobuf_functions, // function list protobuf_functions, // function list
PHP_MINIT(protobuf), // process startup PHP_MINIT(protobuf), // process startup
......
...@@ -182,6 +182,8 @@ ...@@ -182,6 +182,8 @@
#define CACHED_TO_ZVAL_PTR(VALUE) (VALUE) #define CACHED_TO_ZVAL_PTR(VALUE) (VALUE)
#define CACHED_PTR_TO_ZVAL_PTR(VALUE) (*VALUE) #define CACHED_PTR_TO_ZVAL_PTR(VALUE) (*VALUE)
#define ZVAL_PTR_TO_CACHED_PTR(VALUE) (&VALUE) #define ZVAL_PTR_TO_CACHED_PTR(VALUE) (&VALUE)
#define ZVAL_PTR_TO_CACHED_VALUE(VALUE) (VALUE)
#define ZVAL_TO_CACHED_VALUE(VALUE) (&VALUE)
#define CREATE_OBJ_ON_ALLOCATED_ZVAL_PTR(zval_ptr, class_type) \ #define CREATE_OBJ_ON_ALLOCATED_ZVAL_PTR(zval_ptr, class_type) \
ZVAL_OBJ(zval_ptr, class_type->create_object(class_type TSRMLS_CC)); ZVAL_OBJ(zval_ptr, class_type->create_object(class_type TSRMLS_CC));
...@@ -452,6 +454,8 @@ static inline int php_proto_zend_hash_get_current_data_ex(HashTable* ht, ...@@ -452,6 +454,8 @@ static inline int php_proto_zend_hash_get_current_data_ex(HashTable* ht,
#define CACHED_TO_ZVAL_PTR(VALUE) (&VALUE) #define CACHED_TO_ZVAL_PTR(VALUE) (&VALUE)
#define CACHED_PTR_TO_ZVAL_PTR(VALUE) (VALUE) #define CACHED_PTR_TO_ZVAL_PTR(VALUE) (VALUE)
#define ZVAL_PTR_TO_CACHED_PTR(VALUE) (VALUE) #define ZVAL_PTR_TO_CACHED_PTR(VALUE) (VALUE)
#define ZVAL_PTR_TO_CACHED_VALUE(VALUE) (*VALUE)
#define ZVAL_TO_CACHED_VALUE(VALUE) (VALUE)
#define CREATE_OBJ_ON_ALLOCATED_ZVAL_PTR(zval_ptr, class_type) \ #define CREATE_OBJ_ON_ALLOCATED_ZVAL_PTR(zval_ptr, class_type) \
ZVAL_OBJ(zval_ptr, class_type->create_object(class_type)); ZVAL_OBJ(zval_ptr, class_type->create_object(class_type));
......
...@@ -126,6 +126,18 @@ $from = new \Google\Protobuf\Timestamp(); ...@@ -126,6 +126,18 @@ $from = new \Google\Protobuf\Timestamp();
$from->setSeconds(1); $from->setSeconds(1);
assert(1, $from->getSeconds()); assert(1, $from->getSeconds());
$timestamp = new \Google\Protobuf\Timestamp();
date_default_timezone_set('UTC');
$from = new DateTime('2011-01-01T15:03:01.012345UTC');
$timestamp->fromDateTime($from);
assert($from->format('U') == $timestamp->getSeconds());
assert(0 == $timestamp->getNanos());
$to = $timestamp->toDateTime();
assert(\DateTime::class == get_class($to));
assert($from->format('U') == $to->format('U'));
$from = new \Google\Protobuf\Value(); $from = new \Google\Protobuf\Value();
$from->setNumberValue(1); $from->setNumberValue(1);
assert(1, $from->getNumberValue()); assert(1, $from->getNumberValue());
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment