123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982 |
- /*
- +----------------------------------------------------------------------+
- | Twig Extension |
- +----------------------------------------------------------------------+
- | Copyright (c) 2011 Derick Rethans |
- +----------------------------------------------------------------------+
- | Redistribution and use in source and binary forms, with or without |
- | modification, are permitted provided that the conditions mentioned |
- | in the accompanying LICENSE file are met (BSD, revised). |
- +----------------------------------------------------------------------+
- | Author: Derick Rethans <derick@derickrethans.nl> |
- +----------------------------------------------------------------------+
- */
-
- #ifdef HAVE_CONFIG_H
- #include "config.h"
- #endif
-
- #include "php.h"
- #include "php_twig.h"
- #include "ext/standard/php_string.h"
- #include "ext/standard/php_smart_str.h"
-
- #include "Zend/zend_object_handlers.h"
- #include "Zend/zend_interfaces.h"
- #include "Zend/zend_exceptions.h"
-
- #ifndef Z_ADDREF_P
- #define Z_ADDREF_P(pz) (pz)->refcount++
- #endif
-
- #if PHP_VERSION_ID >= 50300
- #define APPLY_TSRMLS_DC TSRMLS_DC
- #define APPLY_TSRMLS_CC TSRMLS_CC
- #define APPLY_TSRMLS_FETCH()
- #else
- #define APPLY_TSRMLS_DC
- #define APPLY_TSRMLS_CC
- #define APPLY_TSRMLS_FETCH() TSRMLS_FETCH()
- #endif
-
- ZEND_BEGIN_ARG_INFO_EX(twig_template_get_attribute_args, ZEND_SEND_BY_VAL, ZEND_RETURN_VALUE, 6)
- ZEND_ARG_INFO(0, template)
- ZEND_ARG_INFO(0, object)
- ZEND_ARG_INFO(0, item)
- ZEND_ARG_INFO(0, arguments)
- ZEND_ARG_INFO(0, type)
- ZEND_ARG_INFO(0, isDefinedTest)
- ZEND_END_ARG_INFO()
-
- zend_function_entry twig_functions[] = {
- PHP_FE(twig_template_get_attributes, twig_template_get_attribute_args)
- {NULL, NULL, NULL}
- };
-
-
- zend_module_entry twig_module_entry = {
- STANDARD_MODULE_HEADER,
- "twig",
- twig_functions,
- NULL,
- NULL,
- NULL,
- NULL,
- NULL,
- PHP_TWIG_VERSION,
- STANDARD_MODULE_PROPERTIES
- };
-
-
- #ifdef COMPILE_DL_TWIG
- ZEND_GET_MODULE(twig)
- #endif
-
- int TWIG_ARRAY_KEY_EXISTS(zval *array, char* key, int key_len)
- {
- if (Z_TYPE_P(array) != IS_ARRAY) {
- return 0;
- }
- return zend_symtable_exists(Z_ARRVAL_P(array), key, key_len + 1);
- }
-
- int TWIG_INSTANCE_OF(zval *object, zend_class_entry *interface TSRMLS_DC)
- {
- if (Z_TYPE_P(object) != IS_OBJECT) {
- return 0;
- }
- return instanceof_function(Z_OBJCE_P(object), interface TSRMLS_CC);
- }
-
- int TWIG_INSTANCE_OF_USERLAND(zval *object, char *interface TSRMLS_DC)
- {
- zend_class_entry **pce;
- if (Z_TYPE_P(object) != IS_OBJECT) {
- return 0;
- }
- if (zend_lookup_class(interface, strlen(interface), &pce TSRMLS_CC) == FAILURE) {
- return 0;
- }
- return instanceof_function(Z_OBJCE_P(object), *pce TSRMLS_CC);
- }
-
- zval *TWIG_GET_ARRAYOBJECT_ELEMENT(zval *object, zval *offset TSRMLS_DC)
- {
- zend_class_entry *ce = Z_OBJCE_P(object);
- zval *retval;
-
- if (Z_TYPE_P(object) == IS_OBJECT) {
- SEPARATE_ARG_IF_REF(offset);
- zend_call_method_with_1_params(&object, ce, NULL, "offsetget", &retval, offset);
-
- zval_ptr_dtor(&offset);
-
- if (!retval) {
- if (!EG(exception)) {
- zend_error(E_ERROR, "Undefined offset for object of type %s used as array", ce->name);
- }
- return NULL;
- }
-
- return retval;
- }
- return NULL;
- }
-
- int TWIG_ISSET_ARRAYOBJECT_ELEMENT(zval *object, zval *offset TSRMLS_DC)
- {
- zend_class_entry *ce = Z_OBJCE_P(object);
- zval *retval;
-
- if (Z_TYPE_P(object) == IS_OBJECT) {
- SEPARATE_ARG_IF_REF(offset);
- zend_call_method_with_1_params(&object, ce, NULL, "offsetexists", &retval, offset);
-
- zval_ptr_dtor(&offset);
-
- if (!retval) {
- if (!EG(exception)) {
- zend_error(E_ERROR, "Undefined offset for object of type %s used as array", ce->name);
- }
- return 0;
- }
-
- return (retval && Z_TYPE_P(retval) == IS_BOOL && Z_LVAL_P(retval));
- }
- return 0;
- }
-
- char *TWIG_STRTOLOWER(const char *str, int str_len)
- {
- char *item_dup;
-
- item_dup = estrndup(str, str_len);
- php_strtolower(item_dup, str_len);
- return item_dup;
- }
-
- zval *TWIG_CALL_USER_FUNC_ARRAY(zval *object, char *function, zval *arguments TSRMLS_DC)
- {
- zend_fcall_info fci;
- zval ***args = NULL;
- int arg_count = 0;
- HashTable *table;
- HashPosition pos;
- int i = 0;
- zval *retval_ptr;
- zval *zfunction;
-
- if (arguments) {
- table = HASH_OF(arguments);
- args = safe_emalloc(sizeof(zval **), table->nNumOfElements, 0);
-
- zend_hash_internal_pointer_reset_ex(table, &pos);
-
- while (zend_hash_get_current_data_ex(table, (void **)&args[i], &pos) == SUCCESS) {
- i++;
- zend_hash_move_forward_ex(table, &pos);
- }
- arg_count = table->nNumOfElements;
- }
-
- MAKE_STD_ZVAL(zfunction);
- ZVAL_STRING(zfunction, function, 1);
- fci.size = sizeof(fci);
- fci.function_table = EG(function_table);
- fci.function_name = zfunction;
- fci.symbol_table = NULL;
- #if PHP_VERSION_ID >= 50300
- fci.object_ptr = object;
- #else
- fci.object_pp = &object;
- #endif
- fci.retval_ptr_ptr = &retval_ptr;
- fci.param_count = arg_count;
- fci.params = args;
- fci.no_separation = 0;
-
- if (zend_call_function(&fci, NULL TSRMLS_CC) == FAILURE) {
- zval_dtor(zfunction);
- efree(zfunction);
- zend_throw_exception_ex(zend_exception_get_default(TSRMLS_C), 0 TSRMLS_CC, "Could not execute %s::%s()", zend_get_class_entry(object TSRMLS_CC)->name, function TSRMLS_CC);
- }
-
- if (args) {
- efree(fci.params);
- }
- zval_dtor(zfunction);
- efree(zfunction);
- return retval_ptr;
- }
-
- int TWIG_CALL_BOOLEAN(zval *object, char *functionName TSRMLS_DC)
- {
- zval *ret;
- int res;
-
- ret = TWIG_CALL_USER_FUNC_ARRAY(object, functionName, NULL TSRMLS_CC);
- res = Z_LVAL_P(ret);
- zval_ptr_dtor(&ret);
- return res;
- }
-
- zval *TWIG_GET_STATIC_PROPERTY(zval *class, char *prop_name TSRMLS_DC)
- {
- zval **tmp_zval;
- zend_class_entry *ce;
-
- if (class == NULL || Z_TYPE_P(class) != IS_OBJECT) {
- return NULL;
- }
-
- ce = zend_get_class_entry(class TSRMLS_CC);
- #if PHP_VERSION_ID >= 50400
- tmp_zval = zend_std_get_static_property(ce, prop_name, strlen(prop_name), 0, NULL TSRMLS_CC);
- #else
- tmp_zval = zend_std_get_static_property(ce, prop_name, strlen(prop_name), 0 TSRMLS_CC);
- #endif
- return *tmp_zval;
- }
-
- zval *TWIG_GET_ARRAY_ELEMENT_ZVAL(zval *class, zval *prop_name TSRMLS_DC)
- {
- zval **tmp_zval;
- char *tmp_name;
-
- if (class == NULL || Z_TYPE_P(class) != IS_ARRAY || Z_TYPE_P(prop_name) != IS_STRING) {
- if (class != NULL && Z_TYPE_P(class) == IS_OBJECT && TWIG_INSTANCE_OF(class, zend_ce_arrayaccess TSRMLS_CC)) {
- // array access object
- return TWIG_GET_ARRAYOBJECT_ELEMENT(class, prop_name TSRMLS_CC);
- }
- return NULL;
- }
-
- convert_to_string(prop_name);
- tmp_name = Z_STRVAL_P(prop_name);
- if (zend_symtable_find(HASH_OF(class), tmp_name, strlen(tmp_name)+1, (void**) &tmp_zval) == SUCCESS) {
- return *tmp_zval;
- }
- return NULL;
- }
-
- zval *TWIG_GET_ARRAY_ELEMENT(zval *class, char *prop_name, int prop_name_length TSRMLS_DC)
- {
- zval **tmp_zval;
-
- if (class == NULL/* || Z_TYPE_P(class) != IS_ARRAY*/) {
- return NULL;
- }
-
- if (class != NULL && Z_TYPE_P(class) == IS_OBJECT && TWIG_INSTANCE_OF(class, zend_ce_arrayaccess TSRMLS_CC)) {
- // array access object
- zval *tmp_name_zval;
- zval *tmp_ret_zval;
-
- ALLOC_INIT_ZVAL(tmp_name_zval);
- ZVAL_STRING(tmp_name_zval, prop_name, 1);
- tmp_ret_zval = TWIG_GET_ARRAYOBJECT_ELEMENT(class, tmp_name_zval TSRMLS_CC);
- zval_dtor(tmp_name_zval);
- efree(tmp_name_zval);
- return tmp_ret_zval;
- }
-
- if (zend_symtable_find(HASH_OF(class), prop_name, prop_name_length+1, (void**)&tmp_zval) == SUCCESS) {
- return *tmp_zval;
- }
- return NULL;
- }
-
- zval *TWIG_PROPERTY(zval *object, zval *propname TSRMLS_DC)
- {
- zval *tmp = NULL;
-
- if (Z_OBJ_HT_P(object)->read_property) {
- #if PHP_VERSION_ID >= 50400
- tmp = Z_OBJ_HT_P(object)->read_property(object, propname, BP_VAR_IS, NULL TSRMLS_CC);
- #else
- tmp = Z_OBJ_HT_P(object)->read_property(object, propname, BP_VAR_IS TSRMLS_CC);
- #endif
- if (tmp != EG(uninitialized_zval_ptr)) {
- return tmp;
- } else {
- return NULL;
- }
- }
- return tmp;
- }
-
- int TWIG_HAS_PROPERTY(zval *object, zval *propname TSRMLS_DC)
- {
- if (Z_OBJ_HT_P(object)->has_property) {
- #if PHP_VERSION_ID >= 50400
- return Z_OBJ_HT_P(object)->has_property(object, propname, 0, NULL TSRMLS_CC);
- #else
- return Z_OBJ_HT_P(object)->has_property(object, propname, 0 TSRMLS_CC);
- #endif
- }
- return 0;
- }
-
- int TWIG_HAS_DYNAMIC_PROPERTY(zval *object, char *prop, int prop_len TSRMLS_DC)
- {
- if (Z_OBJ_HT_P(object)->get_properties) {
- return zend_hash_quick_exists(
- Z_OBJ_HT_P(object)->get_properties(object TSRMLS_CC), // the properties hash
- prop, // property name
- prop_len + 1, // property length
- zend_get_hash_value(prop, prop_len + 1) // hash value
- );
- }
- return 0;
- }
-
- zval *TWIG_PROPERTY_CHAR(zval *object, char *propname TSRMLS_DC)
- {
- zval *tmp_name_zval, *tmp;
-
- ALLOC_INIT_ZVAL(tmp_name_zval);
- ZVAL_STRING(tmp_name_zval, propname, 1);
- tmp = TWIG_PROPERTY(object, tmp_name_zval TSRMLS_CC);
- zval_dtor(tmp_name_zval);
- efree(tmp_name_zval);
- return tmp;
- }
-
- int TWIG_CALL_B_0(zval *object, char *method)
- {
- return 0;
- }
-
- zval *TWIG_CALL_S(zval *object, char *method, char *arg0 TSRMLS_DC)
- {
- zend_fcall_info fci;
- zval **args[1];
- zval *argument;
- zval *zfunction;
- zval *retval_ptr;
-
- MAKE_STD_ZVAL(argument);
- ZVAL_STRING(argument, arg0, 1);
- args[0] = &argument;
-
- MAKE_STD_ZVAL(zfunction);
- ZVAL_STRING(zfunction, method, 1);
- fci.size = sizeof(fci);
- fci.function_table = EG(function_table);
- fci.function_name = zfunction;
- fci.symbol_table = NULL;
- #if PHP_VERSION_ID >= 50300
- fci.object_ptr = object;
- #else
- fci.object_pp = &object;
- #endif
- fci.retval_ptr_ptr = &retval_ptr;
- fci.param_count = 1;
- fci.params = args;
- fci.no_separation = 0;
-
- if (zend_call_function(&fci, NULL TSRMLS_CC) == FAILURE) {
- zval_dtor(argument);
- return 0;
- }
- zval_dtor(zfunction);
- efree(zfunction);
- zval_dtor(argument);
- efree(argument);
- return retval_ptr;
- }
-
- int TWIG_CALL_SB(zval *object, char *method, char *arg0 TSRMLS_DC)
- {
- zval *retval_ptr;
- int success;
-
- retval_ptr = TWIG_CALL_S(object, method, arg0 TSRMLS_CC);
- success = (retval_ptr && (Z_TYPE_P(retval_ptr) == IS_BOOL) && Z_LVAL_P(retval_ptr));
-
- if (retval_ptr) {
- zval_ptr_dtor(&retval_ptr);
- }
-
- return success;
- }
-
- int TWIG_CALL_Z(zval *object, char *method, zval *arg1 TSRMLS_DC)
- {
- zend_fcall_info fci;
- zval **args[1];
- zval *zfunction;
- zval *retval_ptr;
- int success;
-
- args[0] = &arg1;
-
- MAKE_STD_ZVAL(zfunction);
- ZVAL_STRING(zfunction, method, 1);
- fci.size = sizeof(fci);
- fci.function_table = EG(function_table);
- fci.function_name = zfunction;
- fci.symbol_table = NULL;
- #if PHP_VERSION_ID >= 50300
- fci.object_ptr = object;
- #else
- fci.object_pp = &object;
- #endif
- fci.retval_ptr_ptr = &retval_ptr;
- fci.param_count = 1;
- fci.params = args;
- fci.no_separation = 0;
-
- if (zend_call_function(&fci, NULL TSRMLS_CC) == FAILURE) {
- zval_dtor(zfunction);
- efree(zfunction);
- if (retval_ptr) {
- zval_ptr_dtor(&retval_ptr);
- }
- return 0;
- }
-
- zval_dtor(zfunction);
- efree(zfunction);
-
- success = (retval_ptr && (Z_TYPE_P(retval_ptr) == IS_BOOL) && Z_LVAL_P(retval_ptr));
- if (retval_ptr) {
- zval_ptr_dtor(&retval_ptr);
- }
-
- return success;
- }
-
- int TWIG_CALL_ZZ(zval *object, char *method, zval *arg1, zval *arg2 TSRMLS_DC)
- {
- zend_fcall_info fci;
- zval **args[2];
- zval *zfunction;
- zval *retval_ptr;
- int success;
-
- args[0] = &arg1;
- args[1] = &arg2;
-
- MAKE_STD_ZVAL(zfunction);
- ZVAL_STRING(zfunction, method, 1);
- fci.size = sizeof(fci);
- fci.function_table = EG(function_table);
- fci.function_name = zfunction;
- fci.symbol_table = NULL;
- #if PHP_VERSION_ID >= 50300
- fci.object_ptr = object;
- #else
- fci.object_pp = &object;
- #endif
- fci.retval_ptr_ptr = &retval_ptr;
- fci.param_count = 2;
- fci.params = args;
- fci.no_separation = 0;
-
- if (zend_call_function(&fci, NULL TSRMLS_CC) == FAILURE) {
- zval_dtor(zfunction);
- return 0;
- }
-
- zval_dtor(zfunction);
-
- success = (retval_ptr && (Z_TYPE_P(retval_ptr) == IS_BOOL) && Z_LVAL_P(retval_ptr));
- if (retval_ptr) {
- zval_ptr_dtor(&retval_ptr);
- }
-
- return success;
- }
-
- #ifndef Z_SET_REFCOUNT_P
- # define Z_SET_REFCOUNT_P(pz, rc) pz->refcount = rc
- # define Z_UNSET_ISREF_P(pz) pz->is_ref = 0
- #endif
-
- void TWIG_NEW(zval *object, char *class, zval *arg0, zval *arg1 TSRMLS_DC)
- {
- zend_class_entry **pce;
-
- if (zend_lookup_class(class, strlen(class), &pce TSRMLS_CC) == FAILURE) {
- return;
- }
-
- Z_TYPE_P(object) = IS_OBJECT;
- object_init_ex(object, *pce);
- Z_SET_REFCOUNT_P(object, 1);
- Z_UNSET_ISREF_P(object);
-
- TWIG_CALL_ZZ(object, "__construct", arg0, arg1 TSRMLS_CC);
- }
-
- static int twig_add_array_key_to_string(void *pDest APPLY_TSRMLS_DC, int num_args, va_list args, zend_hash_key *hash_key)
- {
- smart_str *buf;
- char *joiner;
-
- buf = va_arg(args, smart_str*);
- joiner = va_arg(args, char*);
-
- if (buf->len != 0) {
- smart_str_appends(buf, joiner);
- }
-
- if (hash_key->nKeyLength == 0) {
- smart_str_append_long(buf, (long) hash_key->h);
- } else {
- char *key, *tmp_str;
- int key_len, tmp_len;
- key = php_addcslashes(hash_key->arKey, hash_key->nKeyLength - 1, &key_len, 0, "'\\", 2 TSRMLS_CC);
- tmp_str = php_str_to_str_ex(key, key_len, "\0", 1, "' . \"\\0\" . '", 12, &tmp_len, 0, NULL);
-
- smart_str_appendl(buf, tmp_str, tmp_len);
- efree(key);
- efree(tmp_str);
- }
-
- return 0;
- }
-
- char *TWIG_IMPLODE_ARRAY_KEYS(char *joiner, zval *array TSRMLS_DC)
- {
- smart_str collector = { 0, 0, 0 };
-
- smart_str_appendl(&collector, "", 0);
- zend_hash_apply_with_arguments(HASH_OF(array) APPLY_TSRMLS_CC, twig_add_array_key_to_string, 2, &collector, joiner);
- smart_str_0(&collector);
-
- return collector.c;
- }
-
- static void TWIG_THROW_EXCEPTION(char *exception_name TSRMLS_DC, char *message, ...)
- {
- char *buffer;
- va_list args;
- zend_class_entry **pce;
-
- if (zend_lookup_class(exception_name, strlen(exception_name), &pce TSRMLS_CC) == FAILURE)
- {
- return;
- }
-
- va_start(args, message);
- vspprintf(&buffer, 0, message, args);
- va_end(args);
-
- zend_throw_exception_ex(*pce, 0 TSRMLS_CC, buffer);
- }
-
- char *TWIG_GET_CLASS_NAME(zval *object TSRMLS_DC)
- {
- char *class_name;
- zend_uint class_name_len;
-
- if (Z_TYPE_P(object) != IS_OBJECT) {
- return "";
- }
- zend_get_object_classname(object, (const char **) &class_name, &class_name_len TSRMLS_CC);
- return class_name;
- }
-
- static int twig_add_method_to_class(void *pDest APPLY_TSRMLS_DC, int num_args, va_list args, zend_hash_key *hash_key)
- {
- zval *retval;
- char *item;
- size_t item_len;
- zend_function *mptr = (zend_function *) pDest;
-
- if (!(mptr->common.fn_flags & ZEND_ACC_PUBLIC)) {
- return 0;
- }
-
- retval = va_arg(args, zval*);
-
- item_len = strlen(mptr->common.function_name);
- item = estrndup(mptr->common.function_name, item_len);
- php_strtolower(item, item_len);
-
- add_assoc_stringl_ex(retval, item, item_len+1, item, item_len, 0);
-
- return 0;
- }
-
- static int twig_add_property_to_class(void *pDest APPLY_TSRMLS_DC, int num_args, va_list args, zend_hash_key *hash_key)
- {
- zend_class_entry *ce;
- zval *retval;
- char *class_name, *prop_name;
- zend_property_info *pptr = (zend_property_info *) pDest;
-
- if (!(pptr->flags & ZEND_ACC_PUBLIC)) {
- return 0;
- }
-
- ce = *va_arg(args, zend_class_entry**);
- retval = va_arg(args, zval*);
-
- zend_unmangle_property_name(pptr->name, pptr->name_length, (const char **) &class_name, (const char **) &prop_name);
-
- add_assoc_string(retval, prop_name, prop_name, 1);
-
- return 0;
- }
-
- static void twig_add_class_to_cache(zval *cache, zval *object, char *class_name TSRMLS_DC)
- {
- zval *class_info, *class_methods, *class_properties;
- zend_class_entry *class_ce;
-
- class_ce = zend_get_class_entry(object TSRMLS_CC);
-
- ALLOC_INIT_ZVAL(class_info);
- ALLOC_INIT_ZVAL(class_methods);
- ALLOC_INIT_ZVAL(class_properties);
- array_init(class_info);
- array_init(class_methods);
- array_init(class_properties);
- // add all methods to self::cache[$class]['methods']
- zend_hash_apply_with_arguments(&class_ce->function_table APPLY_TSRMLS_CC, twig_add_method_to_class, 1, class_methods);
- zend_hash_apply_with_arguments(&class_ce->properties_info APPLY_TSRMLS_CC, twig_add_property_to_class, 2, &class_ce, class_properties);
-
- add_assoc_zval(class_info, "methods", class_methods);
- add_assoc_zval(class_info, "properties", class_properties);
- add_assoc_zval(cache, class_name, class_info);
- }
-
- /* {{{ proto mixed twig_template_get_attributes(TwigTemplate template, mixed object, mixed item, array arguments, string type, boolean isDefinedTest, boolean ignoreStrictCheck)
- A C implementation of TwigTemplate::getAttribute() */
- PHP_FUNCTION(twig_template_get_attributes)
- {
- zval *template;
- zval *object;
- char *item;
- int item_len;
- zval zitem;
- zval *arguments = NULL;
- zval *ret = NULL;
- char *type = NULL;
- int type_len = 0;
- zend_bool isDefinedTest = 0;
- zend_bool ignoreStrictCheck = 0;
- int free_ret = 0;
- zval *tmp_self_cache;
-
-
- if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ozs|asbb", &template, &object, &item, &item_len, &arguments, &type, &type_len, &isDefinedTest, &ignoreStrictCheck) == FAILURE) {
- return;
- }
-
- INIT_PZVAL(&zitem);
- ZVAL_STRINGL(&zitem, item, item_len, 0);
-
- switch (is_numeric_string(item, item_len, &Z_LVAL(zitem), &Z_DVAL(zitem), 0)) {
- case IS_LONG:
- Z_TYPE(zitem) = IS_LONG;
- break;
- case IS_DOUBLE:
- Z_TYPE(zitem) = IS_DOUBLE;
- convert_to_long(&zitem);
- break;
- }
-
- if (!type) {
- type = "any";
- }
-
- /*
- // array
- if (Twig_TemplateInterface::METHOD_CALL !== $type) {
- if ((is_array($object) && array_key_exists($item, $object))
- || ($object instanceof ArrayAccess && isset($object[$item]))
- ) {
- if ($isDefinedTest) {
- return true;
- }
-
- return $object[$item];
- }
- */
- if (strcmp("method", type) != 0) {
- // printf("XXXmethod: %s\n", type);
- if ((TWIG_ARRAY_KEY_EXISTS(object, item, item_len))
- || (TWIG_INSTANCE_OF(object, zend_ce_arrayaccess TSRMLS_CC) && TWIG_ISSET_ARRAYOBJECT_ELEMENT(object, &zitem TSRMLS_CC))
- ) {
- zval *ret;
-
- if (isDefinedTest) {
- RETURN_TRUE;
- }
-
- ret = TWIG_GET_ARRAY_ELEMENT(object, item, item_len TSRMLS_CC);
- if (!ret) {
- ret = &EG(uninitialized_zval);
- }
- RETVAL_ZVAL(ret, 1, 0);
- if (free_ret) {
- zval_ptr_dtor(&ret);
- }
- return;
- }
- /*
- if (Twig_TemplateInterface::ARRAY_CALL === $type) {
- if ($isDefinedTest) {
- return false;
- }
- if ($ignoreStrictCheck || !$this->env->isStrictVariables()) {
- return null;
- }
- */
- if (strcmp("array", type) == 0) {
- if (isDefinedTest) {
- RETURN_FALSE;
- }
- if (ignoreStrictCheck || !TWIG_CALL_BOOLEAN(TWIG_PROPERTY_CHAR(template, "env" TSRMLS_CC), "isStrictVariables" TSRMLS_CC)) {
- return;
- }
- /*
- if (is_object($object)) {
- throw new Twig_Error_Runtime(sprintf('Key "%s" in object (with ArrayAccess) of type "%s" does not exist', $item, get_class($object)));
- // array
- } else {
- throw new Twig_Error_Runtime(sprintf('Key "%s" for array with keys "%s" does not exist', $item, implode(', ', array_keys($object))));
- }
- }
- }
- */
- if (Z_TYPE_P(object) == IS_OBJECT) {
- TWIG_THROW_EXCEPTION("Twig_Error_Runtime" TSRMLS_CC, "Key \"%s\" in object (with ArrayAccess) of type \"%s\" does not exist", item, TWIG_GET_CLASS_NAME(object TSRMLS_CC));
- } else {
- TWIG_THROW_EXCEPTION("Twig_Error_Runtime" TSRMLS_CC, "Key \"%s\" for array with keys \"%s\" does not exist", item, TWIG_IMPLODE_ARRAY_KEYS(", ", object TSRMLS_CC));
- }
- return;
- }
- }
-
- /*
- if (!is_object($object)) {
- if ($isDefinedTest) {
- return false;
- }
- */
-
- if (Z_TYPE_P(object) != IS_OBJECT) {
- if (isDefinedTest) {
- RETURN_FALSE;
- }
- /*
- if ($ignoreStrictCheck || !$this->env->isStrictVariables()) {
- return null;
- }
- throw new Twig_Error_Runtime(sprintf('Item "%s" for "%s" does not exist', $item, implode(', ', array_keys($object))));
- }
- */
- if (ignoreStrictCheck || !TWIG_CALL_BOOLEAN(TWIG_PROPERTY_CHAR(template, "env" TSRMLS_CC), "isStrictVariables" TSRMLS_CC)) {
- RETURN_FALSE;
- }
- if (Z_TYPE_P(object) == IS_ARRAY) {
- TWIG_THROW_EXCEPTION("Twig_Error_Runtime" TSRMLS_CC, "Item \"%s\" for \"Array\" does not exist", item);
- } else {
- Z_ADDREF_P(object);
- convert_to_string_ex(&object);
- TWIG_THROW_EXCEPTION("Twig_Error_Runtime" TSRMLS_CC, "Item \"%s\" for \"%s\" does not exist", item, Z_STRVAL_P(object));
- zval_ptr_dtor(&object);
- }
- return;
- }
- /*
- // get some information about the object
- $class = get_class($object);
- if (!isset(self::$cache[$class])) {
- $r = new ReflectionClass($class);
- self::$cache[$class] = array('methods' => array(), 'properties' => array());
- foreach ($r->getMethods(ReflectionMethod::IS_PUBLIC) as $method) {
- self::$cache[$class]['methods'][strtolower($method->getName())] = true;
- }
-
- foreach ($r->getProperties(ReflectionProperty::IS_PUBLIC) as $property) {
- self::$cache[$class]['properties'][$property->getName()] = true;
- }
- }
- */
- if (Z_TYPE_P(object) == IS_OBJECT) {
- char *class_name = NULL;
-
- class_name = TWIG_GET_CLASS_NAME(object TSRMLS_CC);
- tmp_self_cache = TWIG_GET_STATIC_PROPERTY(template, "cache" TSRMLS_CC);
-
- if (!TWIG_GET_ARRAY_ELEMENT(tmp_self_cache, class_name, strlen(class_name) TSRMLS_CC)) {
- twig_add_class_to_cache(tmp_self_cache, object, class_name TSRMLS_CC);
- }
- efree(class_name);
- }
-
- /*
- // object property
- if (Twig_TemplateInterface::METHOD_CALL !== $type) {
- if (isset(self::$cache[$class]['properties'][$item])
- || isset($object->$item) || array_key_exists($item, $object)
- ) {
- if ($isDefinedTest) {
- return true;
- }
- if ($this->env->hasExtension('sandbox')) {
- $this->env->getExtension('sandbox')->checkPropertyAllowed($object, $item);
- }
-
- return $object->$item;
- }
- }
- */
- if (strcmp("method", type) != 0) {
- zval *tmp_class, *tmp_properties, *tmp_item;
- char *class_name = NULL;
-
- class_name = TWIG_GET_CLASS_NAME(object TSRMLS_CC);
- tmp_class = TWIG_GET_ARRAY_ELEMENT(tmp_self_cache, class_name, strlen(class_name) TSRMLS_CC);
- tmp_properties = TWIG_GET_ARRAY_ELEMENT(tmp_class, "properties", strlen("properties") TSRMLS_CC);
- tmp_item = TWIG_GET_ARRAY_ELEMENT(tmp_properties, item, item_len TSRMLS_CC);
-
- efree(class_name);
-
- if (tmp_item || TWIG_HAS_PROPERTY(object, &zitem TSRMLS_CC) || TWIG_HAS_DYNAMIC_PROPERTY(object, item, item_len TSRMLS_CC)) {
- if (isDefinedTest) {
- RETURN_TRUE;
- }
- if (TWIG_CALL_SB(TWIG_PROPERTY_CHAR(template, "env" TSRMLS_CC), "hasExtension", "sandbox" TSRMLS_CC)) {
- TWIG_CALL_ZZ(TWIG_CALL_S(TWIG_PROPERTY_CHAR(template, "env" TSRMLS_CC), "getExtension", "sandbox" TSRMLS_CC), "checkPropertyAllowed", object, &zitem TSRMLS_CC);
- }
- if (EG(exception)) {
- return;
- }
-
- ret = TWIG_PROPERTY(object, &zitem TSRMLS_CC);
- RETURN_ZVAL(ret, 1, 0);
- }
- }
- /*
- // object method
- $lcItem = strtolower($item);
- if (isset(self::$cache[$class]['methods'][$lcItem])) {
- $method = $item;
- } elseif (isset(self::$cache[$class]['methods']['get'.$lcItem])) {
- $method = 'get'.$item;
- } elseif (isset(self::$cache[$class]['methods']['is'.$lcItem])) {
- $method = 'is'.$item;
- } elseif (isset(self::$cache[$class]['methods']['__call'])) {
- $method = $item;
- */
- {
- char *lcItem = TWIG_STRTOLOWER(item, item_len);
- int lcItem_length;
- char *method = NULL;
- char *tmp_method_name_get;
- char *tmp_method_name_is;
- zval *tmp_class, *tmp_methods;
- char *class_name = NULL;
-
- class_name = TWIG_GET_CLASS_NAME(object TSRMLS_CC);
- lcItem_length = strlen(lcItem);
- tmp_method_name_get = emalloc(4 + lcItem_length);
- tmp_method_name_is = emalloc(3 + lcItem_length);
-
- sprintf(tmp_method_name_get, "get%s", lcItem);
- sprintf(tmp_method_name_is, "is%s", lcItem);
-
- tmp_class = TWIG_GET_ARRAY_ELEMENT(tmp_self_cache, class_name, strlen(class_name) TSRMLS_CC);
- tmp_methods = TWIG_GET_ARRAY_ELEMENT(tmp_class, "methods", strlen("methods") TSRMLS_CC);
- efree(class_name);
-
- if (TWIG_GET_ARRAY_ELEMENT(tmp_methods, lcItem, lcItem_length TSRMLS_CC)) {
- method = item;
- } else if (TWIG_GET_ARRAY_ELEMENT(tmp_methods, tmp_method_name_get, lcItem_length + 3 TSRMLS_CC)) {
- method = tmp_method_name_get;
- } else if (TWIG_GET_ARRAY_ELEMENT(tmp_methods, tmp_method_name_is, lcItem_length + 2 TSRMLS_CC)) {
- method = tmp_method_name_is;
- } else if (TWIG_GET_ARRAY_ELEMENT(tmp_methods, "__call", 6 TSRMLS_CC)) {
- method = item;
- /*
- } else {
- if ($isDefinedTest) {
- return false;
- }
- if ($ignoreStrictCheck || !$this->env->isStrictVariables()) {
- return null;
- }
- throw new Twig_Error_Runtime(sprintf('Method "%s" for object "%s" does not exist', $item, get_class($object)));
- }
- if ($isDefinedTest) {
- return true;
- }
- */
- } else {
- efree(tmp_method_name_get);
- efree(tmp_method_name_is);
- efree(lcItem);
-
- if (isDefinedTest) {
- RETURN_FALSE;
- }
- if (ignoreStrictCheck || !TWIG_CALL_BOOLEAN(TWIG_PROPERTY_CHAR(template, "env" TSRMLS_CC), "isStrictVariables" TSRMLS_CC)) {
- return;
- }
- TWIG_THROW_EXCEPTION("Twig_Error_Runtime" TSRMLS_CC, "Method \"%s\" for object \"%s\" does not exist", item, TWIG_GET_CLASS_NAME(object TSRMLS_CC));
- return;
- }
-
- if (isDefinedTest) {
- efree(tmp_method_name_get);
- efree(tmp_method_name_is);
- efree(lcItem);
- RETURN_TRUE;
- }
- /*
- if ($this->env->hasExtension('sandbox')) {
- $this->env->getExtension('sandbox')->checkMethodAllowed($object, $method);
- }
- */
- if (TWIG_CALL_SB(TWIG_PROPERTY_CHAR(template, "env" TSRMLS_CC), "hasExtension", "sandbox" TSRMLS_CC)) {
- TWIG_CALL_ZZ(TWIG_CALL_S(TWIG_PROPERTY_CHAR(template, "env" TSRMLS_CC), "getExtension", "sandbox" TSRMLS_CC), "checkMethodAllowed", object, &zitem TSRMLS_CC);
- }
- if (EG(exception)) {
- efree(tmp_method_name_get);
- efree(tmp_method_name_is);
- efree(lcItem);
- return;
- }
- /*
- $ret = call_user_func_array(array($object, $method), $arguments);
- */
- if (Z_TYPE_P(object) == IS_OBJECT) {
- ret = TWIG_CALL_USER_FUNC_ARRAY(object, method, arguments TSRMLS_CC);
- free_ret = 1;
- }
- efree(tmp_method_name_get);
- efree(tmp_method_name_is);
- efree(lcItem);
- }
- /*
- if ($object instanceof Twig_TemplateInterface) {
- return new Twig_Markup($ret, $this->env->getCharset());
- }
- */
- if (TWIG_INSTANCE_OF_USERLAND(object, "Twig_TemplateInterface" TSRMLS_CC)) {
- zval *charset = TWIG_CALL_USER_FUNC_ARRAY(TWIG_PROPERTY_CHAR(template, "env" TSRMLS_CC), "getCharset", NULL TSRMLS_CC);
- TWIG_NEW(return_value, "Twig_Markup", ret, charset TSRMLS_CC);
- zval_ptr_dtor(&charset);
- if (ret) {
- zval_ptr_dtor(&ret);
- }
- return;
- }
- /*
- return $ret;
- */
- if (ret) {
- RETVAL_ZVAL(ret, 1, 0);
- if (free_ret) {
- zval_ptr_dtor(&ret);
- }
- }
- }
|