mirror of
https://github.com/php/php-src.git
synced 2025-08-15 21:48:51 +02:00
291 lines
8.6 KiB
C
291 lines
8.6 KiB
C
/*
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) The PHP Group |
|
|
+----------------------------------------------------------------------+
|
|
| This source file is subject to version 3.01 of the PHP license, |
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
| available through the world-wide-web at the following url: |
|
|
| https://www.php.net/license/3_01.txt |
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
+----------------------------------------------------------------------+
|
|
| Authors: Christian Stocker <chregu@php.net> |
|
|
| Rob Richards <rrichards@php.net> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "php.h"
|
|
#if defined(HAVE_LIBXML) && defined(HAVE_DOM)
|
|
#include "php_dom.h"
|
|
#include "dom_ce.h"
|
|
|
|
typedef struct _nodeIterator nodeIterator;
|
|
struct _nodeIterator {
|
|
int cur;
|
|
int index;
|
|
xmlNode *node;
|
|
};
|
|
|
|
/* Function pointer typedef changed in 2.9.8, see https://github.com/GNOME/libxml2/commit/e03f0a199a67017b2f8052354cf732b2b4cae787 */
|
|
#if LIBXML_VERSION >= 20908
|
|
static void itemHashScanner (void *payload, void *data, const xmlChar *name) /* {{{ */
|
|
#else
|
|
static void itemHashScanner (void *payload, void *data, xmlChar *name)
|
|
#endif
|
|
{
|
|
nodeIterator *priv = data;
|
|
|
|
if (priv->cur < priv->index) {
|
|
priv->cur++;
|
|
} else {
|
|
if (priv->node == NULL) {
|
|
priv->node = payload;
|
|
}
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
xmlNodePtr create_notation(const xmlChar *name, const xmlChar *ExternalID, const xmlChar *SystemID) /* {{{ */
|
|
{
|
|
xmlEntityPtr ret = xmlMalloc(sizeof(xmlEntity));
|
|
memset(ret, 0, sizeof(xmlEntity));
|
|
ret->type = XML_NOTATION_NODE;
|
|
ret->name = xmlStrdup(name);
|
|
ret->ExternalID = xmlStrdup(ExternalID);
|
|
ret->SystemID = xmlStrdup(SystemID);
|
|
return (xmlNodePtr) ret;
|
|
}
|
|
/* }}} */
|
|
|
|
static xmlNode *php_dom_libxml_hash_iter_ex(xmlHashTable *ht, int index)
|
|
{
|
|
int htsize;
|
|
|
|
if ((htsize = xmlHashSize(ht)) > 0 && index < htsize) {
|
|
nodeIterator iter;
|
|
iter.cur = 0;
|
|
iter.index = index;
|
|
iter.node = NULL;
|
|
xmlHashScan(ht, itemHashScanner, &iter);
|
|
return iter.node;
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
xmlNode *php_dom_libxml_hash_iter(dom_nnodemap_object *objmap, int index)
|
|
{
|
|
xmlNode *curnode = php_dom_libxml_hash_iter_ex(objmap->ht, index);
|
|
|
|
if (curnode != NULL && objmap->nodetype != XML_ENTITY_NODE) {
|
|
xmlNotation *notation = (xmlNotation *) curnode;
|
|
curnode = create_notation(notation->name, notation->PublicID, notation->SystemID);
|
|
}
|
|
|
|
return curnode;
|
|
}
|
|
|
|
static void php_dom_iterator_dtor(zend_object_iterator *iter) /* {{{ */
|
|
{
|
|
php_dom_iterator *iterator = (php_dom_iterator *)iter;
|
|
|
|
zval_ptr_dtor(&iterator->intern.data);
|
|
zval_ptr_dtor(&iterator->curobj);
|
|
}
|
|
/* }}} */
|
|
|
|
static zend_result php_dom_iterator_valid(zend_object_iterator *iter) /* {{{ */
|
|
{
|
|
|
|
php_dom_iterator *iterator = (php_dom_iterator *)iter;
|
|
|
|
if (Z_TYPE(iterator->curobj) != IS_UNDEF) {
|
|
return SUCCESS;
|
|
} else {
|
|
return FAILURE;
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
zval *php_dom_iterator_current_data(zend_object_iterator *iter) /* {{{ */
|
|
{
|
|
php_dom_iterator *iterator = (php_dom_iterator *)iter;
|
|
|
|
return &iterator->curobj;
|
|
}
|
|
/* }}} */
|
|
|
|
static void php_dom_iterator_current_key(zend_object_iterator *iter, zval *key) /* {{{ */
|
|
{
|
|
php_dom_iterator *iterator = (php_dom_iterator *)iter;
|
|
zval *object = &iterator->intern.data;
|
|
|
|
if (instanceof_function(Z_OBJCE_P(object), dom_nodelist_class_entry)) {
|
|
ZVAL_LONG(key, iter->index);
|
|
} else {
|
|
dom_object *intern = Z_DOMOBJ_P(&iterator->curobj);
|
|
|
|
if (intern != NULL && intern->ptr != NULL) {
|
|
xmlNodePtr curnode = (xmlNodePtr)((php_libxml_node_ptr *)intern->ptr)->node;
|
|
ZVAL_STRINGL(key, (char *) curnode->name, xmlStrlen(curnode->name));
|
|
} else {
|
|
ZVAL_NULL(key);
|
|
}
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
static void php_dom_iterator_move_forward(zend_object_iterator *iter) /* {{{ */
|
|
{
|
|
xmlNodePtr curnode = NULL;
|
|
|
|
php_dom_iterator *iterator = (php_dom_iterator *)iter;
|
|
|
|
zval *object = &iterator->intern.data;
|
|
dom_object *nnmap = Z_DOMOBJ_P(object);
|
|
dom_nnodemap_object *objmap = nnmap->ptr;
|
|
|
|
dom_object *intern = Z_DOMOBJ_P(&iterator->curobj);
|
|
|
|
if (intern != NULL && intern->ptr != NULL) {
|
|
if (objmap->nodetype != XML_ENTITY_NODE &&
|
|
objmap->nodetype != XML_NOTATION_NODE) {
|
|
if (objmap->nodetype == DOM_NODESET) {
|
|
HashTable *nodeht = HASH_OF(&objmap->baseobj_zv);
|
|
zval *entry;
|
|
zend_hash_move_forward_ex(nodeht, &iterator->pos);
|
|
if ((entry = zend_hash_get_current_data_ex(nodeht, &iterator->pos))) {
|
|
zval_ptr_dtor(&iterator->curobj);
|
|
ZVAL_COPY(&iterator->curobj, entry);
|
|
return;
|
|
}
|
|
} else {
|
|
if (objmap->nodetype == XML_ATTRIBUTE_NODE ||
|
|
objmap->nodetype == XML_ELEMENT_NODE) {
|
|
curnode = (xmlNodePtr)((php_libxml_node_ptr *)intern->ptr)->node;
|
|
curnode = curnode->next;
|
|
} else {
|
|
/* The collection is live, we nav the tree from the base object if we cannot
|
|
* use the cache to restart from the last point. */
|
|
xmlNodePtr basenode = dom_object_get_node(objmap->baseobj);
|
|
|
|
/* We have a strong reference to the base node via baseobj_zv, this cannot become NULL */
|
|
ZEND_ASSERT(basenode != NULL);
|
|
|
|
int previndex;
|
|
if (php_dom_is_cache_tag_stale_from_node(&iterator->cache_tag, basenode)) {
|
|
php_dom_mark_cache_tag_up_to_date_from_node(&iterator->cache_tag, basenode);
|
|
previndex = 0;
|
|
if (basenode->type == XML_DOCUMENT_NODE || basenode->type == XML_HTML_DOCUMENT_NODE) {
|
|
curnode = xmlDocGetRootElement((xmlDoc *) basenode);
|
|
} else {
|
|
curnode = basenode->children;
|
|
}
|
|
} else {
|
|
previndex = iter->index - 1;
|
|
curnode = (xmlNodePtr)((php_libxml_node_ptr *)intern->ptr)->node;
|
|
}
|
|
curnode = dom_get_elements_by_tag_name_ns_raw(
|
|
basenode, curnode, objmap->ns, objmap->local, objmap->local_lower, &previndex, iter->index);
|
|
}
|
|
}
|
|
} else {
|
|
curnode = php_dom_libxml_hash_iter(objmap, iter->index);
|
|
}
|
|
}
|
|
|
|
zval_ptr_dtor(&iterator->curobj);
|
|
ZVAL_UNDEF(&iterator->curobj);
|
|
|
|
if (curnode) {
|
|
php_dom_create_object(curnode, &iterator->curobj, objmap->baseobj);
|
|
}
|
|
}
|
|
/* }}} */
|
|
|
|
static const zend_object_iterator_funcs php_dom_iterator_funcs = {
|
|
php_dom_iterator_dtor,
|
|
php_dom_iterator_valid,
|
|
php_dom_iterator_current_data,
|
|
php_dom_iterator_current_key,
|
|
php_dom_iterator_move_forward,
|
|
NULL,
|
|
NULL,
|
|
NULL, /* get_gc */
|
|
};
|
|
|
|
zend_object_iterator *php_dom_get_iterator(zend_class_entry *ce, zval *object, int by_ref) /* {{{ */
|
|
{
|
|
dom_object *intern;
|
|
dom_nnodemap_object *objmap;
|
|
xmlNodePtr curnode=NULL;
|
|
int curindex = 0;
|
|
HashTable *nodeht;
|
|
zval *entry;
|
|
php_dom_iterator *iterator;
|
|
|
|
if (by_ref) {
|
|
zend_throw_error(NULL, "An iterator cannot be used with foreach by reference");
|
|
return NULL;
|
|
}
|
|
iterator = emalloc(sizeof(php_dom_iterator));
|
|
zend_iterator_init(&iterator->intern);
|
|
iterator->cache_tag.modification_nr = 0;
|
|
|
|
ZVAL_OBJ_COPY(&iterator->intern.data, Z_OBJ_P(object));
|
|
iterator->intern.funcs = &php_dom_iterator_funcs;
|
|
|
|
ZVAL_UNDEF(&iterator->curobj);
|
|
|
|
intern = Z_DOMOBJ_P(object);
|
|
objmap = (dom_nnodemap_object *)intern->ptr;
|
|
if (objmap != NULL) {
|
|
if (objmap->nodetype != XML_ENTITY_NODE &&
|
|
objmap->nodetype != XML_NOTATION_NODE) {
|
|
if (objmap->nodetype == DOM_NODESET) {
|
|
nodeht = HASH_OF(&objmap->baseobj_zv);
|
|
zend_hash_internal_pointer_reset_ex(nodeht, &iterator->pos);
|
|
if ((entry = zend_hash_get_current_data_ex(nodeht, &iterator->pos))) {
|
|
ZVAL_COPY(&iterator->curobj, entry);
|
|
}
|
|
} else {
|
|
xmlNodePtr basep = (xmlNode *)dom_object_get_node(objmap->baseobj);
|
|
if (!basep) {
|
|
goto err;
|
|
}
|
|
if (objmap->nodetype == XML_ATTRIBUTE_NODE || objmap->nodetype == XML_ELEMENT_NODE) {
|
|
if (objmap->nodetype == XML_ATTRIBUTE_NODE) {
|
|
curnode = (xmlNodePtr) basep->properties;
|
|
} else {
|
|
curnode = (xmlNodePtr) basep->children;
|
|
}
|
|
} else {
|
|
xmlNodePtr nodep = basep;
|
|
if (nodep->type == XML_DOCUMENT_NODE || nodep->type == XML_HTML_DOCUMENT_NODE) {
|
|
nodep = xmlDocGetRootElement((xmlDoc *) nodep);
|
|
} else {
|
|
nodep = nodep->children;
|
|
}
|
|
curnode = dom_get_elements_by_tag_name_ns_raw(
|
|
basep, nodep, objmap->ns, objmap->local, objmap->local_lower, &curindex, 0);
|
|
}
|
|
}
|
|
} else {
|
|
curnode = php_dom_libxml_hash_iter(objmap, 0);
|
|
}
|
|
}
|
|
err:
|
|
if (curnode) {
|
|
php_dom_create_object(curnode, &iterator->curobj, objmap->baseobj);
|
|
}
|
|
|
|
return &iterator->intern;
|
|
}
|
|
/* }}} */
|
|
|
|
#endif
|