/* +----------------------------------------------------------------------+ | 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 | | Rob Richards | +----------------------------------------------------------------------+ */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "php.h" #if defined(HAVE_LIBXML) && defined(HAVE_DOM) #include "php_dom.h" /* * class DOMNode * * URL: https://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1950641247 * Since: */ zend_string *dom_node_concatenated_name_helper(size_t name_len, const char *name, size_t prefix_len, const char *prefix) { if (UNEXPECTED(prefix_len > ZSTR_MAX_LEN / 2 - 1 || name_len > ZSTR_MAX_LEN / 2 - 1)) { return zend_empty_string; } zend_string *str = zend_string_alloc(prefix_len + 1 + name_len, false); memcpy(ZSTR_VAL(str), prefix, prefix_len); ZSTR_VAL(str)[prefix_len] = ':'; memcpy(ZSTR_VAL(str) + prefix_len + 1, name, name_len + 1 /* include \0 */); return str; } zend_string *dom_node_get_node_name_attribute_or_element(const xmlNode *nodep) { size_t name_len = strlen((const char *) nodep->name); if (nodep->ns != NULL && nodep->ns->prefix != NULL) { return dom_node_concatenated_name_helper(name_len, (const char *) nodep->name, strlen((const char *) nodep->ns->prefix), (const char *) nodep->ns->prefix); } else { return zend_string_init((const char *) nodep->name, name_len, false); } } bool php_dom_is_node_connected(const xmlNode *node) { ZEND_ASSERT(node != NULL); do { if (node->type == XML_DOCUMENT_NODE || node->type == XML_HTML_DOCUMENT_NODE) { return true; } node = node->parent; } while (node != NULL); return false; } /* {{{ nodeName string readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-F68D095 Since: */ int dom_node_node_name_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } switch (nodep->type) { case XML_ATTRIBUTE_NODE: case XML_ELEMENT_NODE: ZVAL_STR(retval, dom_node_get_node_name_attribute_or_element(nodep)); break; case XML_NAMESPACE_DECL: { xmlNsPtr ns = nodep->ns; if (ns != NULL && ns->prefix) { xmlChar *qname = xmlStrdup((xmlChar *) "xmlns"); qname = xmlStrcat(qname, (xmlChar *) ":"); qname = xmlStrcat(qname, nodep->name); ZVAL_STRING(retval, (const char *) qname); xmlFree(qname); } else { ZVAL_STRING(retval, (const char *) nodep->name); } break; } case XML_DOCUMENT_TYPE_NODE: case XML_DTD_NODE: case XML_PI_NODE: case XML_ENTITY_DECL: case XML_ENTITY_REF_NODE: case XML_NOTATION_NODE: ZVAL_STRING(retval, (char *) nodep->name); break; case XML_CDATA_SECTION_NODE: ZVAL_STRING(retval, "#cdata-section"); break; case XML_COMMENT_NODE: ZVAL_STRING(retval, "#comment"); break; case XML_HTML_DOCUMENT_NODE: case XML_DOCUMENT_NODE: ZVAL_STRING(retval, "#document"); break; case XML_DOCUMENT_FRAG_NODE: ZVAL_STRING(retval, "#document-fragment"); break; case XML_TEXT_NODE: ZVAL_STRING(retval, "#text"); break; EMPTY_SWITCH_DEFAULT_CASE(); } return SUCCESS; } /* }}} */ /* {{{ nodeValue string readonly=no URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-F68D080 Since: */ int dom_node_node_value_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } /* Access to Element node is implemented as a convenience method */ switch (nodep->type) { case XML_ATTRIBUTE_NODE: case XML_TEXT_NODE: case XML_ELEMENT_NODE: case XML_COMMENT_NODE: case XML_CDATA_SECTION_NODE: case XML_PI_NODE: php_dom_get_content_into_zval(nodep, retval, true); break; case XML_NAMESPACE_DECL: { char *str = (char *) xmlNodeGetContent(nodep->children); if (str != NULL) { ZVAL_STRING(retval, str); xmlFree(str); } else { ZVAL_NULL(retval); } break; } default: ZVAL_NULL(retval); break; } return SUCCESS; } int dom_node_node_value_write(dom_object *obj, zval *newval) { xmlNode *nodep = dom_object_get_node(obj); zend_string *str; if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } str = zval_try_get_string(newval); if (UNEXPECTED(!str)) { return FAILURE; } /* Access to Element node is implemented as a convenience method */ switch (nodep->type) { case XML_ATTRIBUTE_NODE: case XML_ELEMENT_NODE: dom_remove_all_children(nodep); ZEND_FALLTHROUGH; case XML_TEXT_NODE: case XML_COMMENT_NODE: case XML_CDATA_SECTION_NODE: case XML_PI_NODE: xmlNodeSetContentLen(nodep, (xmlChar *) ZSTR_VAL(str), ZSTR_LEN(str)); break; default: break; } php_libxml_invalidate_node_list_cache(obj->document); zend_string_release_ex(str, 0); return SUCCESS; } /* }}} */ /* {{{ nodeType int readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-111237558 Since: */ int dom_node_node_type_read(dom_object *obj, zval *retval) { xmlNode *nodep; nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } /* Specs dictate that they are both type XML_DOCUMENT_TYPE_NODE */ if (nodep->type == XML_DTD_NODE) { ZVAL_LONG(retval, XML_DOCUMENT_TYPE_NODE); } else { ZVAL_LONG(retval, nodep->type); } return SUCCESS; } /* }}} */ static zend_result dom_node_parent_get(dom_object *obj, zval *retval, bool only_element) { xmlNodePtr nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } xmlNodePtr nodeparent = nodep->parent; if (!nodeparent || (only_element && nodeparent->type != XML_ELEMENT_NODE)) { ZVAL_NULL(retval); return SUCCESS; } php_dom_create_object(nodeparent, retval, obj); return SUCCESS; } /* {{{ parentNode ?DomNode readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1060184317 Since: */ int dom_node_parent_node_read(dom_object *obj, zval *retval) { return dom_node_parent_get(obj, retval, false); } /* }}} */ /* {{{ parentElement ?DomElement readonly=yes URL: https://dom.spec.whatwg.org/#parent-element Since: */ int dom_node_parent_element_read(dom_object *obj, zval *retval) { return dom_node_parent_get(obj, retval, true); } /* }}} */ /* {{{ childNodes DomNodeList readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1451460987 Since: */ int dom_node_child_nodes_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); dom_object *intern; if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } php_dom_create_iterator(retval, DOM_NODELIST); intern = Z_DOMOBJ_P(retval); dom_namednode_iter(obj, XML_ELEMENT_NODE, intern, NULL, NULL, 0, NULL, 0); return SUCCESS; } /* }}} */ /* {{{ firstChild DomNode readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-169727388 Since: */ int dom_node_first_child_read(dom_object *obj, zval *retval) { xmlNode *nodep, *first = NULL; nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } if (dom_node_children_valid(nodep) == SUCCESS) { first = nodep->children; } if (!first) { ZVAL_NULL(retval); return SUCCESS; } php_dom_create_object(first, retval, obj); return SUCCESS; } /* }}} */ /* {{{ lastChild DomNode readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-61AD09FB Since: */ int dom_node_last_child_read(dom_object *obj, zval *retval) { xmlNode *nodep, *last = NULL; nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } if (dom_node_children_valid(nodep) == SUCCESS) { last = nodep->last; } if (!last) { ZVAL_NULL(retval); return SUCCESS; } php_dom_create_object(last, retval, obj); return SUCCESS; } /* }}} */ /* {{{ previousSibling DomNode readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-640FB3C8 Since: */ int dom_node_previous_sibling_read(dom_object *obj, zval *retval) { xmlNode *nodep, *prevsib; nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } prevsib = nodep->prev; if (!prevsib) { ZVAL_NULL(retval); return SUCCESS; } php_dom_create_object(prevsib, retval, obj); return SUCCESS; } /* }}} */ /* {{{ nextSibling DomNode readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-6AC54C2F Since: */ int dom_node_next_sibling_read(dom_object *obj, zval *retval) { xmlNode *nodep, *nextsib; nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } nextsib = nodep->next; if (!nextsib) { ZVAL_NULL(retval); return SUCCESS; } php_dom_create_object(nextsib, retval, obj); return SUCCESS; } /* }}} */ /* {{{ previousElementSibling DomNode readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-640FB3C8 Since: */ int dom_node_previous_element_sibling_read(dom_object *obj, zval *retval) { xmlNode *nodep, *prevsib; nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } prevsib = nodep->prev; while (prevsib && prevsib->type != XML_ELEMENT_NODE) { prevsib = prevsib->prev; } if (!prevsib) { ZVAL_NULL(retval); return SUCCESS; } php_dom_create_object(prevsib, retval, obj); return SUCCESS; } /* }}} */ /* {{{ nextElementSibling DomNode readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-6AC54C2F Since: */ int dom_node_next_element_sibling_read(dom_object *obj, zval *retval) { xmlNode *nodep, *nextsib; nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } nextsib = nodep->next; while (nextsib != NULL && nextsib->type != XML_ELEMENT_NODE) { nextsib = nextsib->next; } if (!nextsib) { ZVAL_NULL(retval); return SUCCESS; } php_dom_create_object(nextsib, retval, obj); return SUCCESS; } /* }}} */ /* {{{ attributes DomNamedNodeMap readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-84CF096 Since: */ int dom_node_attributes_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); dom_object *intern; if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } if (nodep->type == XML_ELEMENT_NODE) { php_dom_create_iterator(retval, DOM_NAMEDNODEMAP); intern = Z_DOMOBJ_P(retval); dom_namednode_iter(obj, XML_ATTRIBUTE_NODE, intern, NULL, NULL, 0, NULL, 0); } else { ZVAL_NULL(retval); } return SUCCESS; } /* }}} */ /* {{{ isConnected boolean readonly=yes URL: https://dom.spec.whatwg.org/#dom-node-isconnected Since: */ int dom_node_is_connected_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } ZVAL_BOOL(retval, php_dom_is_node_connected(nodep)); return SUCCESS; } /* }}} */ /* {{{ ownerDocument DomDocument readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-node-ownerDoc Since: */ int dom_node_owner_document_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); xmlDocPtr docp; if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } if (nodep->type == XML_DOCUMENT_NODE || nodep->type == XML_HTML_DOCUMENT_NODE) { ZVAL_NULL(retval); return SUCCESS; } docp = nodep->doc; if (!docp) { return FAILURE; } php_dom_create_object((xmlNodePtr) docp, retval, obj); return SUCCESS; } /* }}} */ /* {{{ namespaceUri string readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-NodeNSname Since: DOM Level 2 */ int dom_node_namespace_uri_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); char *str = NULL; if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } switch (nodep->type) { case XML_ELEMENT_NODE: case XML_ATTRIBUTE_NODE: case XML_NAMESPACE_DECL: if (nodep->ns != NULL) { str = (char *) nodep->ns->href; } break; default: str = NULL; break; } if (str != NULL) { ZVAL_STRING(retval, str); } else { ZVAL_NULL(retval); } return SUCCESS; } /* }}} */ /* {{{ prefix string readonly=no URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-NodeNSPrefix Since: DOM Level 2 */ int dom_node_prefix_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); xmlNsPtr ns; char *str = NULL; if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } switch (nodep->type) { case XML_ELEMENT_NODE: case XML_ATTRIBUTE_NODE: case XML_NAMESPACE_DECL: ns = nodep->ns; if (ns != NULL && ns->prefix) { str = (char *) ns->prefix; } break; default: str = NULL; break; } if (str == NULL) { ZVAL_EMPTY_STRING(retval); } else { ZVAL_STRING(retval, str); } return SUCCESS; } int dom_node_prefix_write(dom_object *obj, zval *newval) { zend_string *prefix_str; xmlNode *nodep, *nsnode = NULL; xmlNsPtr ns = NULL, curns; char *strURI; char *prefix; nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } switch (nodep->type) { case XML_ELEMENT_NODE: nsnode = nodep; ZEND_FALLTHROUGH; case XML_ATTRIBUTE_NODE: if (nsnode == NULL) { nsnode = nodep->parent; if (nsnode == NULL) { nsnode = xmlDocGetRootElement(nodep->doc); } } /* Typed property, this is already a string */ ZEND_ASSERT(Z_TYPE_P(newval) == IS_STRING); prefix_str = Z_STR_P(newval); prefix = ZSTR_VAL(prefix_str); if (nsnode && nodep->ns != NULL && !xmlStrEqual(nodep->ns->prefix, (xmlChar *)prefix)) { strURI = (char *) nodep->ns->href; if (strURI == NULL || (zend_string_equals_literal(prefix_str, "xml") && strcmp(strURI, (char *) XML_XML_NAMESPACE)) || (nodep->type == XML_ATTRIBUTE_NODE && zend_string_equals_literal(prefix_str, "xmlns") && strcmp(strURI, (char *) DOM_XMLNS_NAMESPACE)) || (nodep->type == XML_ATTRIBUTE_NODE && !strcmp((char *) nodep->name, "xmlns"))) { ns = NULL; } else { curns = nsnode->nsDef; while (curns != NULL) { if (xmlStrEqual((xmlChar *)prefix, curns->prefix) && xmlStrEqual(nodep->ns->href, curns->href)) { ns = curns; break; } curns = curns->next; } if (ns == NULL) { ns = xmlNewNs(nsnode, nodep->ns->href, (xmlChar *)prefix); } } if (ns == NULL) { php_dom_throw_error(NAMESPACE_ERR, dom_get_strict_error(obj->document)); return FAILURE; } xmlSetNs(nodep, ns); } break; default: break; } return SUCCESS; } /* }}} */ /* {{{ localName string readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-NodeNSLocalN Since: DOM Level 2 */ int dom_node_local_name_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } if (nodep->type == XML_ELEMENT_NODE || nodep->type == XML_ATTRIBUTE_NODE || nodep->type == XML_NAMESPACE_DECL) { ZVAL_STRING(retval, (char *) (nodep->name)); } else { ZVAL_NULL(retval); } return SUCCESS; } /* }}} */ /* {{{ baseURI string readonly=yes URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-baseURI Since: DOM Level 3 */ int dom_node_base_uri_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); xmlChar *baseuri; if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } baseuri = xmlNodeGetBase(nodep->doc, nodep); if (baseuri) { ZVAL_STRING(retval, (char *) (baseuri)); xmlFree(baseuri); } else { ZVAL_NULL(retval); } return SUCCESS; } /* }}} */ /* {{{ textContent string readonly=no URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-textContent Since: DOM Level 3 */ int dom_node_text_content_read(dom_object *obj, zval *retval) { xmlNode *nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } php_dom_get_content_into_zval(nodep, retval, false); return SUCCESS; } int dom_node_text_content_write(dom_object *obj, zval *newval) { xmlNode *nodep = dom_object_get_node(obj); if (nodep == NULL) { php_dom_throw_error(INVALID_STATE_ERR, 1); return FAILURE; } php_libxml_invalidate_node_list_cache(obj->document); /* Typed property, this is already a string */ ZEND_ASSERT(Z_TYPE_P(newval) == IS_STRING); const xmlChar *xmlChars = (const xmlChar *) Z_STRVAL_P(newval); int type = nodep->type; /* We can't directly call xmlNodeSetContent, because it might encode the string through * xmlStringLenGetNodeList for types XML_DOCUMENT_FRAG_NODE, XML_ELEMENT_NODE, XML_ATTRIBUTE_NODE. * See tree.c:xmlNodeSetContent in libxml. * In these cases we need to use a text node to avoid the encoding. * For the other cases, we *can* rely on xmlNodeSetContent because it is either a no-op, or handles * the content without encoding. */ if (type == XML_DOCUMENT_FRAG_NODE || type == XML_ELEMENT_NODE || type == XML_ATTRIBUTE_NODE) { dom_remove_all_children(nodep); xmlNode *textNode = xmlNewText(xmlChars); xmlAddChild(nodep, textNode); } else { xmlNodeSetContent(nodep, xmlChars); } return SUCCESS; } /* }}} */ /* Returns true if the node had the same document reference, false otherwise. */ static bool dom_set_document_ref_obj_single(xmlNodePtr node, php_libxml_ref_obj *document) { dom_object *childobj = php_dom_object_get_data(node); if (!childobj) { return true; } if (!childobj->document) { childobj->document = document; document->refcount++; return true; } return false; } void dom_set_document_ref_pointers_attr(xmlAttrPtr attr, php_libxml_ref_obj *document) { ZEND_ASSERT(document != NULL); dom_set_document_ref_obj_single((xmlNodePtr) attr, document); for (xmlNodePtr attr_child = attr->children; attr_child; attr_child = attr_child->next) { dom_set_document_ref_obj_single(attr_child, document); } } static bool dom_set_document_ref_pointers_node(xmlNodePtr node, php_libxml_ref_obj *document) { ZEND_ASSERT(document != NULL); if (!dom_set_document_ref_obj_single(node, document)) { return false; } if (node->type == XML_ELEMENT_NODE) { for (xmlAttrPtr attr = node->properties; attr; attr = attr->next) { dom_set_document_ref_pointers_attr(attr, document); } } return true; } /* TODO: on 8.4 replace the loop with the tree walk helper function. */ void dom_set_document_ref_pointers(xmlNodePtr node, php_libxml_ref_obj *document) { if (!document) { return; } if (!dom_set_document_ref_pointers_node(node, document)) { return; } xmlNodePtr base = node; node = node->children; while (node != NULL) { ZEND_ASSERT(node != base); if (!dom_set_document_ref_pointers_node(node, document)) { break; } if (node->type == XML_ELEMENT_NODE) { if (node->children) { node = node->children; continue; } } if (node->next) { node = node->next; } else { /* Go upwards, until we find a parent node with a next sibling, or until we hit the base. */ do { node = node->parent; if (node == base) { return; } } while (node->next == NULL); node = node->next; } } } static xmlNodePtr _php_dom_insert_fragment(xmlNodePtr nodep, xmlNodePtr prevsib, xmlNodePtr nextsib, xmlNodePtr fragment, dom_object *intern, dom_object *childobj) /* {{{ */ { xmlNodePtr newchild = fragment->children; if (newchild) { if (prevsib == NULL) { nodep->children = newchild; } else { prevsib->next = newchild; } newchild->prev = prevsib; if (nextsib == NULL) { nodep->last = fragment->last; } else { fragment->last->next = nextsib; nextsib->prev = fragment->last; } /* Assign parent node pointer */ xmlNodePtr node = newchild; while (node != NULL) { node->parent = nodep; if (node == fragment->last) { break; } node = node->next; } fragment->children = NULL; fragment->last = NULL; } return newchild; } /* }}} */ static bool dom_node_check_legacy_insertion_validity(xmlNodePtr parentp, xmlNodePtr child, bool stricterror, bool warn_empty_fragment) { if (dom_node_is_read_only(parentp) == SUCCESS || (child->parent != NULL && dom_node_is_read_only(child->parent) == SUCCESS)) { php_dom_throw_error(NO_MODIFICATION_ALLOWED_ERR, stricterror); return false; } if (dom_hierarchy(parentp, child) == FAILURE) { php_dom_throw_error(HIERARCHY_REQUEST_ERR, stricterror); return false; } if (child->doc != parentp->doc && child->doc != NULL) { php_dom_throw_error(WRONG_DOCUMENT_ERR, stricterror); return false; } if (warn_empty_fragment && child->type == XML_DOCUMENT_FRAG_NODE && child->children == NULL) { /* TODO Drop Warning? */ php_error_docref(NULL, E_WARNING, "Document Fragment is empty"); return false; } /* In old DOM only text nodes and entity nodes can be added as children to attributes. */ if (parentp->type == XML_ATTRIBUTE_NODE && child->type != XML_TEXT_NODE && child->type != XML_ENTITY_REF_NODE) { php_dom_throw_error(HIERARCHY_REQUEST_ERR, stricterror); return false; } /* Attributes must be in elements. */ if (child->type == XML_ATTRIBUTE_NODE && parentp->type != XML_ELEMENT_NODE) { php_dom_throw_error(HIERARCHY_REQUEST_ERR, stricterror); return false; } /* Documents can never be a child. */ if (child->type == XML_DOCUMENT_NODE || child->type == XML_HTML_DOCUMENT_NODE) { php_dom_throw_error(HIERARCHY_REQUEST_ERR, stricterror); return false; } return true; } /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-952280727 Since: */ PHP_METHOD(DOMNode, insertBefore) { zval *id, *node, *ref = NULL; xmlNodePtr child, new_child, parentp, refp = NULL; dom_object *intern, *childobj, *refpobj; int ret, stricterror; id = ZEND_THIS; if (zend_parse_parameters(ZEND_NUM_ARGS(), "O|O!", &node, dom_node_class_entry, &ref, dom_node_class_entry) == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(parentp, id, xmlNodePtr, intern); if (dom_node_children_valid(parentp) == FAILURE) { RETURN_FALSE; } DOM_GET_OBJ(child, node, xmlNodePtr, childobj); new_child = NULL; stricterror = dom_get_strict_error(intern->document); if (!dom_node_check_legacy_insertion_validity(parentp, child, stricterror, true)) { RETURN_FALSE; } if (ref != NULL) { DOM_GET_OBJ(refp, ref, xmlNodePtr, refpobj); if (refp->parent != parentp) { php_dom_throw_error(NOT_FOUND_ERR, stricterror); RETURN_FALSE; } } if (child->doc == NULL && parentp->doc != NULL) { dom_set_document_ref_pointers(child, intern->document); } php_libxml_invalidate_node_list_cache(intern->document); if (ref != NULL) { if (child->parent != NULL) { xmlUnlinkNode(child); } if (child->type == XML_TEXT_NODE && (refp->type == XML_TEXT_NODE || (refp->prev != NULL && refp->prev->type == XML_TEXT_NODE))) { new_child = child; new_child->parent = refp->parent; new_child->next = refp; new_child->prev = refp->prev; refp->prev = new_child; if (new_child->prev != NULL) { new_child->prev->next = new_child; } if (new_child->parent != NULL) { if (new_child->parent->children == refp) { new_child->parent->children = new_child; } } } else if (child->type == XML_ATTRIBUTE_NODE) { xmlAttrPtr lastattr; if (child->ns == NULL) lastattr = xmlHasProp(refp->parent, child->name); else lastattr = xmlHasNsProp(refp->parent, child->name, child->ns->href); if (lastattr != NULL && lastattr->type != XML_ATTRIBUTE_DECL) { if (lastattr != (xmlAttrPtr) child) { xmlUnlinkNode((xmlNodePtr) lastattr); php_libxml_node_free_resource((xmlNodePtr) lastattr); } else { DOM_RET_OBJ(child, &ret, intern); return; } } new_child = xmlAddPrevSibling(refp, child); if (UNEXPECTED(NULL == new_child)) { goto cannot_add; } } else if (child->type == XML_DOCUMENT_FRAG_NODE) { xmlNodePtr last = child->last; new_child = _php_dom_insert_fragment(parentp, refp->prev, refp, child, intern, childobj); dom_reconcile_ns_list(parentp->doc, new_child, last); } else { new_child = xmlAddPrevSibling(refp, child); if (UNEXPECTED(NULL == new_child)) { goto cannot_add; } dom_reconcile_ns(parentp->doc, new_child); } } else { if (child->parent != NULL){ xmlUnlinkNode(child); } if (child->type == XML_TEXT_NODE && parentp->last != NULL && parentp->last->type == XML_TEXT_NODE) { child->parent = parentp; new_child = child; if (parentp->children == NULL) { parentp->children = child; parentp->last = child; } else { child = parentp->last; child->next = new_child; new_child->prev = child; parentp->last = new_child; } } else if (child->type == XML_ATTRIBUTE_NODE) { xmlAttrPtr lastattr; if (child->ns == NULL) lastattr = xmlHasProp(parentp, child->name); else lastattr = xmlHasNsProp(parentp, child->name, child->ns->href); if (lastattr != NULL && lastattr->type != XML_ATTRIBUTE_DECL) { if (lastattr != (xmlAttrPtr) child) { xmlUnlinkNode((xmlNodePtr) lastattr); php_libxml_node_free_resource((xmlNodePtr) lastattr); } else { DOM_RET_OBJ(child, &ret, intern); return; } } new_child = xmlAddChild(parentp, child); if (UNEXPECTED(NULL == new_child)) { goto cannot_add; } } else if (child->type == XML_DOCUMENT_FRAG_NODE) { xmlNodePtr last = child->last; new_child = _php_dom_insert_fragment(parentp, parentp->last, NULL, child, intern, childobj); dom_reconcile_ns_list(parentp->doc, new_child, last); } else { new_child = xmlAddChild(parentp, child); if (UNEXPECTED(NULL == new_child)) { goto cannot_add; } dom_reconcile_ns(parentp->doc, new_child); } } DOM_RET_OBJ(new_child, &ret, intern); return; cannot_add: zend_throw_error(NULL, "Cannot add newnode as the previous sibling of refnode"); RETURN_THROWS(); } /* }}} end dom_node_insert_before */ /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-785887307 Since: */ PHP_METHOD(DOMNode, replaceChild) { zval *id, *newnode, *oldnode; xmlNodePtr newchild, oldchild, nodep; dom_object *intern, *newchildobj, *oldchildobj; int stricterror; bool replacedoctype = false; int ret; id = ZEND_THIS; if (zend_parse_parameters(ZEND_NUM_ARGS(), "OO", &newnode, dom_node_class_entry, &oldnode, dom_node_class_entry) == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); if (dom_node_children_valid(nodep) == FAILURE) { RETURN_FALSE; } DOM_GET_OBJ(newchild, newnode, xmlNodePtr, newchildobj); DOM_GET_OBJ(oldchild, oldnode, xmlNodePtr, oldchildobj); if (!nodep->children) { RETURN_FALSE; } stricterror = dom_get_strict_error(intern->document); if (!dom_node_check_legacy_insertion_validity(nodep, newchild, stricterror, false)) { RETURN_FALSE; } /* This is already disallowed by libxml, but we should check it here to avoid * breaking assumptions and assertions. */ if ((oldchild->type == XML_ATTRIBUTE_NODE) != (newchild->type == XML_ATTRIBUTE_NODE)) { php_dom_throw_error(HIERARCHY_REQUEST_ERR, stricterror); RETURN_FALSE; } if (oldchild->parent != nodep) { php_dom_throw_error(NOT_FOUND_ERR, stricterror); RETURN_FALSE; } if (newchild->doc == NULL && nodep->doc != NULL) { dom_set_document_ref_pointers(newchild, intern->document); } if (newchild->type == XML_DOCUMENT_FRAG_NODE) { xmlNodePtr prevsib, nextsib; prevsib = oldchild->prev; nextsib = oldchild->next; xmlUnlinkNode(oldchild); xmlNodePtr last = newchild->last; newchild = _php_dom_insert_fragment(nodep, prevsib, nextsib, newchild, intern, newchildobj); if (newchild) { dom_reconcile_ns_list(nodep->doc, newchild, last); } } else if (oldchild != newchild) { xmlDtdPtr intSubset = xmlGetIntSubset(nodep->doc); replacedoctype = (intSubset == (xmlDtd *) oldchild); xmlReplaceNode(oldchild, newchild); dom_reconcile_ns(nodep->doc, newchild); if (replacedoctype) { nodep->doc->intSubset = (xmlDtd *) newchild; } } php_libxml_invalidate_node_list_cache(intern->document); DOM_RET_OBJ(oldchild, &ret, intern); } /* }}} end dom_node_replace_child */ /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1734834066 Since: */ PHP_METHOD(DOMNode, removeChild) { zval *id, *node; xmlNodePtr child, nodep; dom_object *intern, *childobj; int ret, stricterror; id = ZEND_THIS; if (zend_parse_parameters(ZEND_NUM_ARGS(), "O", &node, dom_node_class_entry) == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); if (dom_node_children_valid(nodep) == FAILURE) { RETURN_FALSE; } DOM_GET_OBJ(child, node, xmlNodePtr, childobj); stricterror = dom_get_strict_error(intern->document); if (dom_node_is_read_only(nodep) == SUCCESS || (child->parent != NULL && dom_node_is_read_only(child->parent) == SUCCESS)) { php_dom_throw_error(NO_MODIFICATION_ALLOWED_ERR, stricterror); RETURN_FALSE; } if (!nodep->children || child->parent != nodep) { php_dom_throw_error(NOT_FOUND_ERR, stricterror); RETURN_FALSE; } xmlUnlinkNode(child); php_libxml_invalidate_node_list_cache(intern->document); DOM_RET_OBJ(child, &ret, intern); } /* }}} end dom_node_remove_child */ /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-184E7107 Since: */ PHP_METHOD(DOMNode, appendChild) { zval *id, *node; xmlNodePtr child, nodep, new_child = NULL; dom_object *intern, *childobj; int ret, stricterror; id = ZEND_THIS; if (zend_parse_parameters(ZEND_NUM_ARGS(), "O", &node, dom_node_class_entry) == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); if (dom_node_children_valid(nodep) == FAILURE) { RETURN_FALSE; } DOM_GET_OBJ(child, node, xmlNodePtr, childobj); stricterror = dom_get_strict_error(intern->document); if (!dom_node_check_legacy_insertion_validity(nodep, child, stricterror, true)) { RETURN_FALSE; } if (child->doc == NULL && nodep->doc != NULL) { dom_set_document_ref_pointers(child, intern->document); } if (child->parent != NULL){ xmlUnlinkNode(child); } if (child->type == XML_TEXT_NODE && nodep->last != NULL && nodep->last->type == XML_TEXT_NODE) { child->parent = nodep; new_child = child; if (nodep->children == NULL) { nodep->children = child; nodep->last = child; } else { child = nodep->last; child->next = new_child; new_child->prev = child; nodep->last = new_child; } } else if (child->type == XML_ATTRIBUTE_NODE) { xmlAttrPtr lastattr; if (child->ns == NULL) lastattr = xmlHasProp(nodep, child->name); else lastattr = xmlHasNsProp(nodep, child->name, child->ns->href); if (lastattr != NULL && lastattr->type != XML_ATTRIBUTE_DECL) { if (lastattr != (xmlAttrPtr) child) { xmlUnlinkNode((xmlNodePtr) lastattr); php_libxml_node_free_resource((xmlNodePtr) lastattr); } } new_child = xmlAddChild(nodep, child); if (UNEXPECTED(new_child == NULL)) { goto cannot_add; } } else if (child->type == XML_DOCUMENT_FRAG_NODE) { xmlNodePtr last = child->last; new_child = _php_dom_insert_fragment(nodep, nodep->last, NULL, child, intern, childobj); dom_reconcile_ns_list(nodep->doc, new_child, last); } else { new_child = xmlAddChild(nodep, child); if (UNEXPECTED(new_child == NULL)) { goto cannot_add; } dom_reconcile_ns(nodep->doc, new_child); } php_libxml_invalidate_node_list_cache(intern->document); DOM_RET_OBJ(new_child, &ret, intern); return; cannot_add: // TODO Convert to Error? php_error_docref(NULL, E_WARNING, "Couldn't append node"); RETURN_FALSE; } /* }}} end dom_node_append_child */ /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-810594187 Since: */ PHP_METHOD(DOMNode, hasChildNodes) { zval *id; xmlNode *nodep; dom_object *intern; id = ZEND_THIS; if (zend_parse_parameters_none() == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); if (dom_node_children_valid(nodep) == FAILURE) { RETURN_FALSE; } if (nodep->children) { RETURN_TRUE; } else { RETURN_FALSE; } } /* }}} end dom_node_has_child_nodes */ /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-3A0ED0A4 Since: */ PHP_METHOD(DOMNode, cloneNode) { zval *id; xmlNode *n, *node; int ret; dom_object *intern; bool recursive = 0; id = ZEND_THIS; if (zend_parse_parameters(ZEND_NUM_ARGS(), "|b", &recursive) == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(n, id, xmlNodePtr, intern); node = xmlDocCopyNode(n, n->doc, recursive); if (!node) { RETURN_FALSE; } /* When deep is false Element nodes still require the attributes Following taken from libxml as xmlDocCopyNode doesn't do this */ if (n->type == XML_ELEMENT_NODE && recursive == 0) { if (n->nsDef != NULL) { node->nsDef = xmlCopyNamespaceList(n->nsDef); } if (n->ns != NULL) { xmlNsPtr ns; ns = xmlSearchNs(n->doc, node, n->ns->prefix); if (ns == NULL) { ns = xmlSearchNs(n->doc, n, n->ns->prefix); if (ns != NULL) { xmlNodePtr root = node; while (root->parent != NULL) { root = root->parent; } node->ns = xmlNewNs(root, ns->href, ns->prefix); } } else { node->ns = ns; } } if (n->properties != NULL) { node->properties = xmlCopyPropList(node, n->properties); } } /* If document cloned we want a new document proxy */ if (node->doc != n->doc) { intern = NULL; } DOM_RET_OBJ(node, &ret, intern); } /* }}} end dom_node_clone_node */ /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-normalize Since: */ PHP_METHOD(DOMNode, normalize) { zval *id; xmlNode *nodep; dom_object *intern; id = ZEND_THIS; if (zend_parse_parameters_none() == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); php_libxml_invalidate_node_list_cache(intern->document); dom_normalize(nodep); } /* }}} end dom_node_normalize */ /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Level-2-Core-Node-supports Since: DOM Level 2 */ PHP_METHOD(DOMNode, isSupported) { zend_string *feature, *version; if (zend_parse_parameters(ZEND_NUM_ARGS(), "SS", &feature, &version) == FAILURE) { RETURN_THROWS(); } RETURN_BOOL(dom_has_feature(feature, version)); } /* }}} end dom_node_is_supported */ /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-NodeHasAttrs Since: DOM Level 2 */ PHP_METHOD(DOMNode, hasAttributes) { zval *id; xmlNode *nodep; dom_object *intern; id = ZEND_THIS; if (zend_parse_parameters_none() == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); if (nodep->type != XML_ELEMENT_NODE) RETURN_FALSE; if (nodep->properties) { RETURN_TRUE; } else { RETURN_FALSE; } } /* }}} end dom_node_has_attributes */ /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-isSameNode Since: DOM Level 3 */ PHP_METHOD(DOMNode, isSameNode) { zval *id, *node; xmlNodePtr nodeotherp, nodep; dom_object *intern, *nodeotherobj; id = ZEND_THIS; if (zend_parse_parameters(ZEND_NUM_ARGS(), "O", &node, dom_node_class_entry) == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); DOM_GET_OBJ(nodeotherp, node, xmlNodePtr, nodeotherobj); if (nodep == nodeotherp) { RETURN_TRUE; } else { RETURN_FALSE; } } /* }}} end dom_node_is_same_node */ static bool php_dom_node_is_content_equal(const xmlNode *this, const xmlNode *other) { xmlChar *this_content = xmlNodeGetContent(this); xmlChar *other_content = xmlNodeGetContent(other); bool result = xmlStrEqual(this_content, other_content); xmlFree(this_content); xmlFree(other_content); return result; } static bool php_dom_node_is_ns_uri_equal(const xmlNode *this, const xmlNode *other) { const xmlChar *this_ns = this->ns ? this->ns->href : NULL; const xmlChar *other_ns = other->ns ? other->ns->href : NULL; return xmlStrEqual(this_ns, other_ns); } static bool php_dom_node_is_ns_prefix_equal(const xmlNode *this, const xmlNode *other) { const xmlChar *this_ns = this->ns ? this->ns->prefix : NULL; const xmlChar *other_ns = other->ns ? other->ns->prefix : NULL; return xmlStrEqual(this_ns, other_ns); } static bool php_dom_node_is_equal_node(const xmlNode *this, const xmlNode *other); #define PHP_DOM_FUNC_CAT(prefix, suffix) prefix##_##suffix /* xmlNode and xmlNs have incompatible struct layouts, i.e. the next field is in a different offset */ #define PHP_DOM_DEFINE_LIST_COUNTER_HELPER(type) \ static size_t PHP_DOM_FUNC_CAT(php_dom_node_count_list_size, type)(const type *node) \ { \ size_t counter = 0; \ while (node) { \ counter++; \ node = node->next; \ } \ return counter; \ } #define PHP_DOM_DEFINE_LIST_EQUALITY_ORDERED_HELPER(type) \ static bool PHP_DOM_FUNC_CAT(php_dom_node_list_equality_check_ordered, type)(const type *list1, const type *list2) \ { \ size_t count = PHP_DOM_FUNC_CAT(php_dom_node_count_list_size, type)(list1); \ if (count != PHP_DOM_FUNC_CAT(php_dom_node_count_list_size, type)(list2)) { \ return false; \ } \ for (size_t i = 0; i < count; i++) { \ if (!php_dom_node_is_equal_node((const xmlNode *) list1, (const xmlNode *) list2)) { \ return false; \ } \ list1 = list1->next; \ list2 = list2->next; \ } \ return true; \ } #define PHP_DOM_DEFINE_LIST_EQUALITY_UNORDERED_HELPER(type) \ static bool PHP_DOM_FUNC_CAT(php_dom_node_list_equality_check_unordered, type)(const type *list1, const type *list2)\ { \ size_t count = PHP_DOM_FUNC_CAT(php_dom_node_count_list_size, type)(list1); \ if (count != PHP_DOM_FUNC_CAT(php_dom_node_count_list_size, type)(list2)) { \ return false; \ } \ for (const type *n1 = list1; n1 != NULL; n1 = n1->next) { \ bool found = false; \ for (const type *n2 = list2; n2 != NULL && !found; n2 = n2->next) { \ if (php_dom_node_is_equal_node((const xmlNode *) n1, (const xmlNode *) n2)) { \ found = true; \ } \ } \ if (!found) { \ return false; \ } \ } \ return true; \ } PHP_DOM_DEFINE_LIST_COUNTER_HELPER(xmlNode) PHP_DOM_DEFINE_LIST_COUNTER_HELPER(xmlNs) PHP_DOM_DEFINE_LIST_EQUALITY_ORDERED_HELPER(xmlNode) PHP_DOM_DEFINE_LIST_EQUALITY_UNORDERED_HELPER(xmlNode) PHP_DOM_DEFINE_LIST_EQUALITY_UNORDERED_HELPER(xmlNs) static bool php_dom_node_is_equal_node(const xmlNode *this, const xmlNode *other) { ZEND_ASSERT(this != NULL); ZEND_ASSERT(other != NULL); if (this->type != other->type) { return false; } /* Notes: * - XML_DOCUMENT_TYPE_NODE is no longer created by libxml2, we only have to support XML_DTD_NODE. * - element and attribute declarations are not exposed as nodes in DOM, so no comparison is needed for those. */ if (this->type == XML_ELEMENT_NODE) { return xmlStrEqual(this->name, other->name) && php_dom_node_is_ns_prefix_equal(this, other) && php_dom_node_is_ns_uri_equal(this, other) /* Check attributes first, then namespace declarations, then children */ && php_dom_node_list_equality_check_unordered_xmlNode((const xmlNode *) this->properties, (const xmlNode *) other->properties) && php_dom_node_list_equality_check_unordered_xmlNs(this->nsDef, other->nsDef) && php_dom_node_list_equality_check_ordered_xmlNode(this->children, other->children); } else if (this->type == XML_DTD_NODE) { /* Note: in the living spec entity declarations and notations are no longer compared because they're considered obsolete. */ const xmlDtd *this_dtd = (const xmlDtd *) this; const xmlDtd *other_dtd = (const xmlDtd *) other; return xmlStrEqual(this_dtd->name, other_dtd->name) && xmlStrEqual(this_dtd->ExternalID, other_dtd->ExternalID) && xmlStrEqual(this_dtd->SystemID, other_dtd->SystemID); } else if (this->type == XML_PI_NODE) { return xmlStrEqual(this->name, other->name) && xmlStrEqual(this->content, other->content); } else if (this->type == XML_TEXT_NODE || this->type == XML_COMMENT_NODE || this->type == XML_CDATA_SECTION_NODE) { return xmlStrEqual(this->content, other->content); } else if (this->type == XML_ATTRIBUTE_NODE) { const xmlAttr *this_attr = (const xmlAttr *) this; const xmlAttr *other_attr = (const xmlAttr *) other; return xmlStrEqual(this_attr->name, other_attr->name) && php_dom_node_is_ns_uri_equal(this, other) && php_dom_node_is_content_equal(this, other); } else if (this->type == XML_ENTITY_REF_NODE) { return xmlStrEqual(this->name, other->name); } else if (this->type == XML_ENTITY_DECL || this->type == XML_NOTATION_NODE || this->type == XML_ENTITY_NODE) { const xmlEntity *this_entity = (const xmlEntity *) this; const xmlEntity *other_entity = (const xmlEntity *) other; return this_entity->etype == other_entity->etype && xmlStrEqual(this_entity->name, other_entity->name) && xmlStrEqual(this_entity->ExternalID, other_entity->ExternalID) && xmlStrEqual(this_entity->SystemID, other_entity->SystemID) && php_dom_node_is_content_equal(this, other); } else if (this->type == XML_NAMESPACE_DECL) { const xmlNs *this_ns = (const xmlNs *) this; const xmlNs *other_ns = (const xmlNs *) other; return xmlStrEqual(this_ns->prefix, other_ns->prefix) && xmlStrEqual(this_ns->href, other_ns->href); } else if (this->type == XML_DOCUMENT_FRAG_NODE || this->type == XML_HTML_DOCUMENT_NODE || this->type == XML_DOCUMENT_NODE) { return php_dom_node_list_equality_check_ordered_xmlNode(this->children, other->children); } return false; } /* {{{ URL: https://dom.spec.whatwg.org/#dom-node-isequalnode (for everything still in the living spec) * URL: https://www.w3.org/TR/2004/REC-DOM-Level-3-Core-20040407/DOM3-Core.html#core-Node3-isEqualNode (for old nodes removed from the living spec) Since: DOM Level 3 */ PHP_METHOD(DOMNode, isEqualNode) { zval *id, *node; xmlNodePtr otherp, nodep; dom_object *unused_intern; id = ZEND_THIS; if (zend_parse_parameters(ZEND_NUM_ARGS(), "O!", &node, dom_node_class_entry) == FAILURE) { RETURN_THROWS(); } if (node == NULL) { RETURN_FALSE; } DOM_GET_THIS_OBJ(nodep, id, xmlNodePtr, unused_intern); DOM_GET_OBJ(otherp, node, xmlNodePtr, unused_intern); if (nodep == otherp) { RETURN_TRUE; } /* Empty fragments/documents only match if they're both empty */ if (UNEXPECTED(nodep == NULL || otherp == NULL)) { RETURN_BOOL(nodep == NULL && otherp == NULL); } RETURN_BOOL(php_dom_node_is_equal_node(nodep, otherp)); } /* }}} end DOMNode::isEqualNode */ /* {{{ URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-lookupNamespacePrefix Since: DOM Level 3 */ PHP_METHOD(DOMNode, lookupPrefix) { zval *id; xmlNodePtr nodep, lookupp = NULL; dom_object *intern; xmlNsPtr nsptr; size_t uri_len = 0; char *uri; id = ZEND_THIS; if (zend_parse_parameters(ZEND_NUM_ARGS(), "s", &uri, &uri_len) == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); if (uri_len > 0) { switch (nodep->type) { case XML_ELEMENT_NODE: lookupp = nodep; break; case XML_DOCUMENT_NODE: case XML_HTML_DOCUMENT_NODE: lookupp = xmlDocGetRootElement((xmlDocPtr) nodep); break; case XML_ENTITY_NODE : case XML_NOTATION_NODE: case XML_DOCUMENT_FRAG_NODE: case XML_DOCUMENT_TYPE_NODE: case XML_DTD_NODE: RETURN_NULL(); break; default: lookupp = nodep->parent; } if (lookupp != NULL) { nsptr = xmlSearchNsByHref(lookupp->doc, lookupp, (xmlChar *) uri); if (nsptr && nsptr->prefix != NULL) { RETURN_STRING((char *) nsptr->prefix); } } } RETURN_NULL(); } /* }}} end dom_node_lookup_prefix */ /* {{{ URL: http://www.w3.org/TR/DOM-Level-3-Core/core.html#Node3-isDefaultNamespace Since: DOM Level 3 */ PHP_METHOD(DOMNode, isDefaultNamespace) { zval *id; xmlNodePtr nodep; dom_object *intern; xmlNsPtr nsptr; size_t uri_len = 0; char *uri; id = ZEND_THIS; if (zend_parse_parameters(ZEND_NUM_ARGS(), "s", &uri, &uri_len) == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); if (nodep->type == XML_DOCUMENT_NODE || nodep->type == XML_HTML_DOCUMENT_NODE) { nodep = xmlDocGetRootElement((xmlDocPtr) nodep); } if (nodep && uri_len > 0) { nsptr = xmlSearchNs(nodep->doc, nodep, NULL); if (nsptr && xmlStrEqual(nsptr->href, (xmlChar *) uri)) { RETURN_TRUE; } } RETURN_FALSE; } /* }}} end dom_node_is_default_namespace */ /* {{{ URL: http://www.w3.org/TR/DOM-Level-3-Core/core.html#Node3-lookupNamespaceURI Since: DOM Level 3 */ PHP_METHOD(DOMNode, lookupNamespaceURI) { zval *id; xmlNodePtr nodep; dom_object *intern; xmlNsPtr nsptr; size_t prefix_len; char *prefix; id = ZEND_THIS; if (zend_parse_parameters(ZEND_NUM_ARGS(), "s!", &prefix, &prefix_len) == FAILURE) { RETURN_THROWS(); } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); if (nodep->type == XML_DOCUMENT_NODE || nodep->type == XML_HTML_DOCUMENT_NODE) { nodep = xmlDocGetRootElement((xmlDocPtr) nodep); if (nodep == NULL) { RETURN_NULL(); } } nsptr = xmlSearchNs(nodep->doc, nodep, (xmlChar *) prefix); if (nsptr && nsptr->href != NULL) { RETURN_STRING((char *) nsptr->href); } RETURN_NULL(); } /* }}} end dom_node_lookup_namespace_uri */ static void dom_canonicalization(INTERNAL_FUNCTION_PARAMETERS, int mode) /* {{{ */ { zval *id; zval *xpath_array=NULL, *ns_prefixes=NULL; xmlNodePtr nodep; xmlDocPtr docp; xmlNodeSetPtr nodeset = NULL; dom_object *intern; bool exclusive=0, with_comments=0; xmlChar **inclusive_ns_prefixes = NULL; char *file = NULL; int ret = -1; size_t file_len = 0; xmlOutputBufferPtr buf; xmlXPathContextPtr ctxp=NULL; xmlXPathObjectPtr xpathobjp=NULL; id = ZEND_THIS; if (mode == 0) { if (zend_parse_parameters(ZEND_NUM_ARGS(), "|bba!a!", &exclusive, &with_comments, &xpath_array, &ns_prefixes) == FAILURE) { RETURN_THROWS(); } } else { if (zend_parse_parameters(ZEND_NUM_ARGS(), "s|bba!a!", &file, &file_len, &exclusive, &with_comments, &xpath_array, &ns_prefixes) == FAILURE) { RETURN_THROWS(); } } DOM_GET_OBJ(nodep, id, xmlNodePtr, intern); docp = nodep->doc; if (! docp) { zend_throw_error(NULL, "Node must be associated with a document"); RETURN_THROWS(); } php_libxml_invalidate_node_list_cache_from_doc(docp); if (xpath_array == NULL) { if (nodep->type != XML_DOCUMENT_NODE) { ctxp = xmlXPathNewContext(docp); ctxp->node = nodep; xpathobjp = xmlXPathEvalExpression((xmlChar *) "(.//. | .//@* | .//namespace::*)", ctxp); ctxp->node = NULL; if (xpathobjp && xpathobjp->type == XPATH_NODESET) { nodeset = xpathobjp->nodesetval; } else { if (xpathobjp) { xmlXPathFreeObject(xpathobjp); } xmlXPathFreeContext(ctxp); zend_throw_error(NULL, "XPath query did not return a nodeset"); RETURN_THROWS(); } } } else { /*xpath query from xpath_array */ HashTable *ht = Z_ARRVAL_P(xpath_array); zval *tmp; char *xquery; /* Find "query" key */ tmp = zend_hash_find_deref(ht, ZSTR_KNOWN(ZEND_STR_QUERY)); if (!tmp) { /* if mode == 0 then $xpath arg is 3, if mode == 1 then $xpath is 4 */ zend_argument_value_error(3 + mode, "must have a \"query\" key"); RETURN_THROWS(); } if (Z_TYPE_P(tmp) != IS_STRING) { /* if mode == 0 then $xpath arg is 3, if mode == 1 then $xpath is 4 */ zend_argument_type_error(3 + mode, "\"query\" option must be a string, %s given", zend_zval_value_name(tmp)); RETURN_THROWS(); } xquery = Z_STRVAL_P(tmp); ctxp = xmlXPathNewContext(docp); ctxp->node = nodep; tmp = zend_hash_str_find_deref(ht, "namespaces", sizeof("namespaces")-1); if (tmp && Z_TYPE_P(tmp) == IS_ARRAY && !HT_IS_PACKED(Z_ARRVAL_P(tmp))) { zval *tmpns; zend_string *prefix; ZEND_HASH_MAP_FOREACH_STR_KEY_VAL(Z_ARRVAL_P(tmp), prefix, tmpns) { ZVAL_DEREF(tmpns); if (Z_TYPE_P(tmpns) == IS_STRING) { if (prefix) { xmlXPathRegisterNs(ctxp, (xmlChar *) ZSTR_VAL(prefix), (xmlChar *) Z_STRVAL_P(tmpns)); } } } ZEND_HASH_FOREACH_END(); } xpathobjp = xmlXPathEvalExpression((xmlChar *) xquery, ctxp); ctxp->node = NULL; if (xpathobjp && xpathobjp->type == XPATH_NODESET) { nodeset = xpathobjp->nodesetval; } else { if (xpathobjp) { xmlXPathFreeObject(xpathobjp); } xmlXPathFreeContext(ctxp); zend_throw_error(NULL, "XPath query did not return a nodeset"); RETURN_THROWS(); } } if (ns_prefixes != NULL) { if (exclusive) { zval *tmpns; int nscount = 0; inclusive_ns_prefixes = safe_emalloc(zend_hash_num_elements(Z_ARRVAL_P(ns_prefixes)) + 1, sizeof(xmlChar *), 0); ZEND_HASH_FOREACH_VAL(Z_ARRVAL_P(ns_prefixes), tmpns) { ZVAL_DEREF(tmpns); if (Z_TYPE_P(tmpns) == IS_STRING) { inclusive_ns_prefixes[nscount++] = (xmlChar *) Z_STRVAL_P(tmpns); } } ZEND_HASH_FOREACH_END(); inclusive_ns_prefixes[nscount] = NULL; } else { php_error_docref(NULL, E_NOTICE, "Inclusive namespace prefixes only allowed in exclusive mode."); } } if (mode == 1) { buf = xmlOutputBufferCreateFilename(file, NULL, 0); } else { buf = xmlAllocOutputBuffer(NULL); } if (buf != NULL) { ret = xmlC14NDocSaveTo(docp, nodeset, exclusive, inclusive_ns_prefixes, with_comments, buf); } if (inclusive_ns_prefixes != NULL) { efree(inclusive_ns_prefixes); } if (xpathobjp != NULL) { xmlXPathFreeObject(xpathobjp); } if (ctxp != NULL) { xmlXPathFreeContext(ctxp); } if (buf == NULL || ret < 0) { RETVAL_FALSE; } else { if (mode == 0) { #ifdef LIBXML2_NEW_BUFFER ret = xmlOutputBufferGetSize(buf); #else ret = buf->buffer->use; #endif if (ret > 0) { #ifdef LIBXML2_NEW_BUFFER RETVAL_STRINGL((char *) xmlOutputBufferGetContent(buf), ret); #else RETVAL_STRINGL((char *) buf->buffer->content, ret); #endif } else { RETVAL_EMPTY_STRING(); } } } if (buf) { int bytes; bytes = xmlOutputBufferClose(buf); if (mode == 1 && (ret >= 0)) { RETURN_LONG(bytes); } } } /* }}} */ /* {{{ Canonicalize nodes to a string */ PHP_METHOD(DOMNode, C14N) { dom_canonicalization(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0); } /* }}} */ /* {{{ Canonicalize nodes to a file */ PHP_METHOD(DOMNode, C14NFile) { dom_canonicalization(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1); } /* }}} */ /* {{{ Gets an xpath for a node */ PHP_METHOD(DOMNode, getNodePath) { zval *id; xmlNode *nodep; dom_object *intern; char *value; if (zend_parse_parameters_none() == FAILURE) { RETURN_THROWS(); } DOM_GET_THIS_OBJ(nodep, id, xmlNodePtr, intern); value = (char *) xmlGetNodePath(nodep); if (value == NULL) { /* TODO Research if can return empty string */ RETURN_NULL(); } else { RETVAL_STRING(value); xmlFree(value); } } /* }}} */ /* {{{ Gets line number for a node */ PHP_METHOD(DOMNode, getLineNo) { zval *id; xmlNode *nodep; dom_object *intern; if (zend_parse_parameters_none() == FAILURE) { RETURN_THROWS(); } DOM_GET_THIS_OBJ(nodep, id, xmlNodePtr, intern); RETURN_LONG(xmlGetLineNo(nodep)); } /* }}} */ /* {{{ URL: https://dom.spec.whatwg.org/#dom-node-contains Since: */ PHP_METHOD(DOMNode, contains) { zval *other, *id; xmlNodePtr otherp, thisp; dom_object *unused_intern; ZEND_PARSE_PARAMETERS_START(1, 1) Z_PARAM_OBJECT_OR_NULL(other) ZEND_PARSE_PARAMETERS_END(); if (other == NULL) { RETURN_FALSE; } if (UNEXPECTED(!instanceof_function(Z_OBJCE_P(other), dom_node_class_entry) && !instanceof_function(Z_OBJCE_P(other), dom_namespace_node_class_entry))) { zend_argument_type_error(1, "must be of type DOMNode|DOMNameSpaceNode|null, %s given", zend_zval_value_name(other)); RETURN_THROWS(); } DOM_GET_OBJ(otherp, other, xmlNodePtr, unused_intern); DOM_GET_THIS_OBJ(thisp, id, xmlNodePtr, unused_intern); do { if (otherp == thisp) { RETURN_TRUE; } otherp = otherp->parent; } while (otherp); RETURN_FALSE; } /* }}} */ /* {{{ URL: https://dom.spec.whatwg.org/#dom-node-getrootnode Since: */ PHP_METHOD(DOMNode, getRootNode) { zval *id; xmlNodePtr thisp; dom_object *intern; /* Unused now because we don't support the shadow DOM nodes. Options only influence shadow DOM nodes. */ zval *options = NULL; if (zend_parse_parameters(ZEND_NUM_ARGS(), "|a!", &options) == FAILURE) { RETURN_THROWS(); } DOM_GET_THIS_OBJ(thisp, id, xmlNodePtr, intern); while (thisp->parent) { thisp = thisp->parent; } int ret; DOM_RET_OBJ(thisp, &ret, intern); } /* }}} */ /** * We want to block the serialization and unserialization of DOM classes. * However, using @not-serializable makes the child classes also not serializable, even if the user implements the methods. * So instead, we implement the methods wherein we throw exceptions. * The reason we choose these methods is because: * - If the user implements __serialize / __unserialize, the respective throwing methods are not called. * - If the user implements __sleep / __wakeup, then it's also not a problem because they will not enter the throwing methods. */ PHP_METHOD(DOMNode, __sleep) { if (zend_parse_parameters_none() != SUCCESS) { RETURN_THROWS(); } zend_throw_exception_ex(NULL, 0, "Serialization of '%s' is not allowed, unless serialization methods are implemented in a subclass", ZSTR_VAL(Z_OBJCE_P(ZEND_THIS)->name)); RETURN_THROWS(); } PHP_METHOD(DOMNode, __wakeup) { if (zend_parse_parameters_none() != SUCCESS) { RETURN_THROWS(); } zend_throw_exception_ex(NULL, 0, "Unserialization of '%s' is not allowed, unless unserialization methods are implemented in a subclass", ZSTR_VAL(Z_OBJCE_P(ZEND_THIS)->name)); RETURN_THROWS(); } #endif