mirror of
https://github.com/php/php-src.git
synced 2025-01-10 04:54:47 +08:00
19523f9a39
add test
2328 lines
58 KiB
C
2328 lines
58 KiB
C
/*
|
|
+----------------------------------------------------------------------+
|
|
| PHP Version 5 |
|
|
+----------------------------------------------------------------------+
|
|
| Copyright (c) 1997-2006 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: |
|
|
| http://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> |
|
|
+----------------------------------------------------------------------+
|
|
*/
|
|
|
|
/* $Id$ */
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "php.h"
|
|
#if HAVE_LIBXML && HAVE_DOM
|
|
#include "php_dom.h"
|
|
#include <libxml/SAX.h>
|
|
#ifdef LIBXML_SCHEMAS_ENABLED
|
|
#include <libxml/relaxng.h>
|
|
#include <libxml/xmlschemas.h>
|
|
#endif
|
|
|
|
typedef struct _idsIterator idsIterator;
|
|
struct _idsIterator {
|
|
xmlChar *elementId;
|
|
xmlNode *element;
|
|
};
|
|
|
|
#define DOM_LOAD_STRING 0
|
|
#define DOM_LOAD_FILE 1
|
|
|
|
/*
|
|
* class DOMDocument extends DOMNode
|
|
*
|
|
* URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-i-Document
|
|
* Since:
|
|
*/
|
|
|
|
zend_function_entry php_dom_document_class_functions[] = {
|
|
PHP_FALIAS(createElement, dom_document_create_element, NULL)
|
|
PHP_FALIAS(createDocumentFragment, dom_document_create_document_fragment, NULL)
|
|
PHP_FALIAS(createTextNode, dom_document_create_text_node, NULL)
|
|
PHP_FALIAS(createComment, dom_document_create_comment, NULL)
|
|
PHP_FALIAS(createCDATASection, dom_document_create_cdatasection, NULL)
|
|
PHP_FALIAS(createProcessingInstruction, dom_document_create_processing_instruction, NULL)
|
|
PHP_FALIAS(createAttribute, dom_document_create_attribute, NULL)
|
|
PHP_FALIAS(createEntityReference, dom_document_create_entity_reference, NULL)
|
|
PHP_FALIAS(getElementsByTagName, dom_document_get_elements_by_tag_name, NULL)
|
|
PHP_FALIAS(importNode, dom_document_import_node, NULL)
|
|
PHP_FALIAS(createElementNS, dom_document_create_element_ns, NULL)
|
|
PHP_FALIAS(createAttributeNS, dom_document_create_attribute_ns, NULL)
|
|
PHP_FALIAS(getElementsByTagNameNS, dom_document_get_elements_by_tag_name_ns, NULL)
|
|
PHP_FALIAS(getElementById, dom_document_get_element_by_id, NULL)
|
|
PHP_FALIAS(adoptNode, dom_document_adopt_node, NULL)
|
|
PHP_FALIAS(normalizeDocument, dom_document_normalize_document, NULL)
|
|
PHP_FALIAS(renameNode, dom_document_rename_node, NULL)
|
|
PHP_ME(domdocument, load, NULL, ZEND_ACC_PUBLIC|ZEND_ACC_ALLOW_STATIC)
|
|
PHP_FALIAS(save, dom_document_save, NULL)
|
|
PHP_ME(domdocument, loadXML, NULL, ZEND_ACC_PUBLIC|ZEND_ACC_ALLOW_STATIC)
|
|
PHP_FALIAS(saveXML, dom_document_savexml, NULL)
|
|
PHP_ME(domdocument, __construct, NULL, ZEND_ACC_PUBLIC)
|
|
PHP_FALIAS(validate, dom_document_validate, NULL)
|
|
PHP_FALIAS(xinclude, dom_document_xinclude, NULL)
|
|
#if defined(LIBXML_HTML_ENABLED)
|
|
PHP_ME(domdocument, loadHTML, NULL, ZEND_ACC_PUBLIC|ZEND_ACC_ALLOW_STATIC)
|
|
PHP_ME(domdocument, loadHTMLFile, NULL, ZEND_ACC_PUBLIC|ZEND_ACC_ALLOW_STATIC)
|
|
PHP_FALIAS(saveHTML, dom_document_save_html, NULL)
|
|
PHP_FALIAS(saveHTMLFile, dom_document_save_html_file, NULL)
|
|
#endif /* defined(LIBXML_HTML_ENABLED) */
|
|
#if defined(LIBXML_SCHEMAS_ENABLED)
|
|
PHP_FALIAS(schemaValidate, dom_document_schema_validate_file, NULL)
|
|
PHP_FALIAS(schemaValidateSource, dom_document_schema_validate_xml, NULL)
|
|
PHP_FALIAS(relaxNGValidate, dom_document_relaxNG_validate_file, NULL)
|
|
PHP_FALIAS(relaxNGValidateSource, dom_document_relaxNG_validate_xml, NULL)
|
|
#endif
|
|
PHP_ME(domdocument, registerNodeClass, NULL, ZEND_ACC_PUBLIC)
|
|
{NULL, NULL, NULL}
|
|
};
|
|
|
|
/* {{{ docType DOMDocumentType
|
|
readonly=yes
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-B63ED1A31
|
|
Since:
|
|
*/
|
|
int dom_document_doctype_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
xmlDoc *docp;
|
|
xmlDtdPtr dtdptr;
|
|
int ret;
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(obj);
|
|
|
|
if (docp == NULL) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
|
|
return FAILURE;
|
|
}
|
|
|
|
dtdptr = xmlGetIntSubset(docp);
|
|
if (!dtdptr) {
|
|
return FAILURE;
|
|
}
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
if (NULL == (*retval = php_dom_create_object((xmlNodePtr) dtdptr, &ret, NULL, *retval, obj TSRMLS_CC))) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot create required DOM object");
|
|
return FAILURE;
|
|
}
|
|
return SUCCESS;
|
|
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
/* {{{ implementation DOMImplementation
|
|
readonly=yes
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1B793EBA
|
|
Since:
|
|
*/
|
|
int dom_document_implementation_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
ALLOC_ZVAL(*retval);
|
|
php_dom_create_implementation(retval TSRMLS_CC);
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
/* {{{ documentElement DOMElement
|
|
readonly=yes
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-87CD092
|
|
Since:
|
|
*/
|
|
int dom_document_document_element_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
xmlDoc *docp;
|
|
xmlNode *root;
|
|
int ret;
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(obj);
|
|
|
|
if (docp == NULL) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
|
|
return FAILURE;
|
|
}
|
|
|
|
root = xmlDocGetRootElement(docp);
|
|
if (!root) {
|
|
return FAILURE;
|
|
}
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
if (NULL == (*retval = php_dom_create_object(root, &ret, NULL, *retval, obj TSRMLS_CC))) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot create required DOM object");
|
|
return FAILURE;
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
/* {{{ encoding string
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Document3-encoding
|
|
Since: DOM Level 3
|
|
*/
|
|
int dom_document_encoding_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
xmlDoc *docp;
|
|
char *encoding;
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(obj);
|
|
|
|
if (docp == NULL) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
|
|
return FAILURE;
|
|
}
|
|
|
|
encoding = (char *) docp->encoding;
|
|
ALLOC_ZVAL(*retval);
|
|
|
|
if (encoding != NULL) {
|
|
ZVAL_XML_STRING(*retval, encoding, ZSTR_DUPLICATE);
|
|
} else {
|
|
ZVAL_NULL(*retval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_encoding_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
xmlDoc *docp;
|
|
xmlCharEncodingHandlerPtr handler;
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(obj);
|
|
|
|
if (docp == NULL) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
|
|
return FAILURE;
|
|
}
|
|
|
|
if (newval->type != IS_STRING) {
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_string_with_converter(newval, UG(utf8_conv));
|
|
}
|
|
|
|
handler = xmlFindCharEncodingHandler(Z_STRVAL_P(newval));
|
|
|
|
if (handler != NULL) {
|
|
xmlCharEncCloseFunc(handler);
|
|
if (docp->encoding != NULL) {
|
|
xmlFree((xmlChar *)docp->encoding);
|
|
}
|
|
docp->encoding = xmlStrdup((const xmlChar *) Z_STRVAL_P(newval));
|
|
} else {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid Document Encoding");
|
|
}
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
/* {{{ standalone boolean
|
|
readonly=no
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Document3-standalone
|
|
Since: DOM Level 3
|
|
*/
|
|
int dom_document_standalone_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
xmlDoc *docp;
|
|
int standalone;
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(obj);
|
|
|
|
if (docp == NULL) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
|
|
return FAILURE;
|
|
}
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
standalone = docp->standalone;
|
|
ZVAL_BOOL(*retval, standalone);
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_standalone_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
xmlDoc *docp;
|
|
int standalone;
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(obj);
|
|
|
|
if (docp == NULL) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
|
|
return FAILURE;
|
|
}
|
|
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_long(newval);
|
|
|
|
standalone = Z_LVAL_P(newval);
|
|
if (standalone > 0) {
|
|
docp->standalone = 1;
|
|
}
|
|
else if (standalone < 0) {
|
|
docp->standalone = -1;
|
|
}
|
|
else {
|
|
docp->standalone = 0;
|
|
}
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
/* {{{ version string
|
|
readonly=no
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Document3-version
|
|
Since: DOM Level 3
|
|
*/
|
|
int dom_document_version_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
xmlDoc *docp;
|
|
char *version;
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(obj);
|
|
|
|
if (docp == NULL) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
|
|
return FAILURE;
|
|
}
|
|
|
|
version = (char *) docp->version;
|
|
ALLOC_ZVAL(*retval);
|
|
|
|
if (version != NULL) {
|
|
ZVAL_XML_STRING(*retval, version, ZSTR_DUPLICATE);
|
|
} else {
|
|
ZVAL_NULL(*retval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_version_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
xmlDoc *docp;
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(obj);
|
|
|
|
if (docp == NULL) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
|
|
return FAILURE;
|
|
}
|
|
|
|
if (docp->version != NULL) {
|
|
xmlFree((xmlChar *) docp->version );
|
|
}
|
|
|
|
if (newval->type != IS_STRING) {
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_string_with_converter(newval, UG(utf8_conv));
|
|
}
|
|
|
|
docp->version = xmlStrdup((const xmlChar *) Z_STRVAL_P(newval));
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
/* {{{ strictErrorChecking boolean
|
|
readonly=no
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Document3-strictErrorChecking
|
|
Since: DOM Level 3
|
|
*/
|
|
int dom_document_strict_error_checking_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
ZVAL_BOOL(*retval, doc_prop->stricterror);
|
|
} else {
|
|
ZVAL_FALSE(*retval);
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_strict_error_checking_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_boolean(newval);
|
|
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
doc_prop->stricterror = Z_LVAL_P(newval);
|
|
}
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
/* {{{ formatOutput boolean
|
|
readonly=no
|
|
*/
|
|
int dom_document_format_output_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
ZVAL_BOOL(*retval, doc_prop->formatoutput);
|
|
} else {
|
|
ZVAL_FALSE(*retval);
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_format_output_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_boolean(newval);
|
|
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
doc_prop->formatoutput = Z_LVAL_P(newval);
|
|
}
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ validateOnParse boolean
|
|
readonly=no
|
|
*/
|
|
int dom_document_validate_on_parse_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
ZVAL_BOOL(*retval, doc_prop->validateonparse);
|
|
} else {
|
|
ZVAL_FALSE(*retval);
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_validate_on_parse_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_boolean(newval);
|
|
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
doc_prop->validateonparse = Z_LVAL_P(newval);
|
|
}
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ resolveExternals boolean
|
|
readonly=no
|
|
*/
|
|
int dom_document_resolve_externals_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
ZVAL_BOOL(*retval, doc_prop->resolveexternals);
|
|
} else {
|
|
ZVAL_FALSE(*retval);
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_resolve_externals_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_boolean(newval);
|
|
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
doc_prop->resolveexternals = Z_LVAL_P(newval);
|
|
}
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ preserveWhiteSpace boolean
|
|
readonly=no
|
|
*/
|
|
int dom_document_preserve_whitespace_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
ZVAL_BOOL(*retval, doc_prop->preservewhitespace);
|
|
} else {
|
|
ZVAL_FALSE(*retval);
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_preserve_whitespace_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_boolean(newval);
|
|
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
doc_prop->preservewhitespace = Z_LVAL_P(newval);
|
|
}
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ recover boolean
|
|
readonly=no
|
|
*/
|
|
int dom_document_recover_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
ZVAL_BOOL(*retval, doc_prop->recover);
|
|
} else {
|
|
ZVAL_FALSE(*retval);
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_recover_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_boolean(newval);
|
|
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
doc_prop->recover = Z_LVAL_P(newval);
|
|
}
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ substituteEntities boolean
|
|
readonly=no
|
|
*/
|
|
int dom_document_substitue_entities_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
ZVAL_BOOL(*retval, doc_prop->substituteentities);
|
|
} else {
|
|
ZVAL_FALSE(*retval);
|
|
}
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_substitue_entities_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
dom_doc_propsptr doc_prop;
|
|
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_boolean(newval);
|
|
|
|
if (obj->document) {
|
|
doc_prop = dom_get_doc_props(obj->document);
|
|
doc_prop->substituteentities = Z_LVAL_P(newval);
|
|
}
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
/* }}} */
|
|
|
|
|
|
/* {{{ documentURI string
|
|
readonly=no
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Document3-documentURI
|
|
Since: DOM Level 3
|
|
*/
|
|
int dom_document_document_uri_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
xmlDoc *docp;
|
|
char *url;
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(obj);
|
|
|
|
if (docp == NULL) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
|
|
return FAILURE;
|
|
}
|
|
|
|
ALLOC_ZVAL(*retval);
|
|
url = (char *) docp->URL;
|
|
if (url != NULL) {
|
|
ZVAL_XML_STRING(*retval, url, ZSTR_DUPLICATE);
|
|
} else {
|
|
ZVAL_NULL(*retval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
int dom_document_document_uri_write(dom_object *obj, zval *newval TSRMLS_DC)
|
|
{
|
|
zval value_copy;
|
|
xmlDoc *docp;
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(obj);
|
|
|
|
if (docp == NULL) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
|
|
return FAILURE;
|
|
}
|
|
|
|
if (docp->URL != NULL) {
|
|
xmlFree((xmlChar *) docp->URL);
|
|
}
|
|
|
|
if (newval->type != IS_STRING) {
|
|
if(newval->refcount > 1) {
|
|
value_copy = *newval;
|
|
zval_copy_ctor(&value_copy);
|
|
newval = &value_copy;
|
|
}
|
|
convert_to_string_with_converter(newval, UG(utf8_conv));
|
|
}
|
|
|
|
docp->URL = xmlStrdup((const xmlChar *) Z_STRVAL_P(newval));
|
|
|
|
if (newval == &value_copy) {
|
|
zval_dtor(newval);
|
|
}
|
|
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
/* {{{ config DOMConfiguration
|
|
readonly=yes
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Document3-config
|
|
Since: DOM Level 3
|
|
*/
|
|
int dom_document_config_read(dom_object *obj, zval **retval TSRMLS_DC)
|
|
{
|
|
ALLOC_ZVAL(*retval);
|
|
ZVAL_NULL(*retval);
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* }}} */
|
|
|
|
|
|
|
|
/* {{{ proto DOMElement dom_document_create_element(string tagName [, string value]) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-2141741547
|
|
Since:
|
|
*/
|
|
PHP_FUNCTION(dom_document_create_element)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlNode *node;
|
|
xmlDocPtr docp;
|
|
dom_object *intern;
|
|
int ret, name_len, value_len;
|
|
char *name, *value = NULL;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os&|s&", &id, dom_document_class_entry, &name, &name_len, UG(utf8_conv), &value, &value_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
if (xmlValidateName((xmlChar *) name, 0) != 0) {
|
|
php_dom_throw_error(INVALID_CHARACTER_ERR, dom_get_strict_error(intern->document) TSRMLS_CC);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
node = xmlNewDocNode(docp, NULL, name, value);
|
|
if (!node) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_RET_OBJ(rv, node, &ret, intern);
|
|
}
|
|
/* }}} end dom_document_create_element */
|
|
|
|
|
|
/* {{{ proto DOMDocumentFragment dom_document_create_document_fragment() U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-35CB04B5
|
|
Since:
|
|
*/
|
|
PHP_FUNCTION(dom_document_create_document_fragment)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlNode *node;
|
|
xmlDocPtr docp;
|
|
dom_object *intern;
|
|
int ret;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O", &id, dom_document_class_entry) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
node = xmlNewDocFragment(docp);
|
|
if (!node) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_RET_OBJ(rv, node, &ret, intern);
|
|
}
|
|
/* }}} end dom_document_create_document_fragment */
|
|
|
|
|
|
/* {{{ proto DOMText dom_document_create_text_node(string data) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1975348127
|
|
Since:
|
|
*/
|
|
PHP_FUNCTION(dom_document_create_text_node)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlNode *node;
|
|
xmlDocPtr docp;
|
|
int ret, value_len;
|
|
dom_object *intern;
|
|
char *value;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os&", &id, dom_document_class_entry, &value, &value_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
node = xmlNewDocText(docp, (xmlChar *) value);
|
|
if (!node) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_RET_OBJ(rv, node, &ret, intern);
|
|
}
|
|
/* }}} end dom_document_create_text_node */
|
|
|
|
|
|
/* {{{ proto DOMComment dom_document_create_comment(string data) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1334481328
|
|
Since:
|
|
*/
|
|
PHP_FUNCTION(dom_document_create_comment)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlNode *node;
|
|
xmlDocPtr docp;
|
|
int ret, value_len;
|
|
dom_object *intern;
|
|
char *value;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os&", &id, dom_document_class_entry, &value, &value_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
node = xmlNewDocComment(docp, (xmlChar *) value);
|
|
if (!node) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_RET_OBJ(rv, node, &ret, intern);
|
|
}
|
|
/* }}} end dom_document_create_comment */
|
|
|
|
|
|
/* {{{ proto DOMCdataSection dom_document_create_cdatasection(string data) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-D26C0AF8
|
|
Since:
|
|
*/
|
|
PHP_FUNCTION(dom_document_create_cdatasection)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlNode *node;
|
|
xmlDocPtr docp;
|
|
int ret, value_len;
|
|
dom_object *intern;
|
|
char *value;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os&", &id, dom_document_class_entry, &value, &value_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
node = xmlNewCDataBlock(docp, (xmlChar *) value, value_len);
|
|
if (!node) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_RET_OBJ(rv, node, &ret, intern);
|
|
}
|
|
/* }}} end dom_document_create_cdatasection */
|
|
|
|
|
|
/* {{{ proto DOMProcessingInstruction dom_document_create_processing_instruction(string target, string data) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-135944439
|
|
Since:
|
|
*/
|
|
PHP_FUNCTION(dom_document_create_processing_instruction)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlNode *node;
|
|
xmlDocPtr docp;
|
|
int ret, value_len, name_len = 0;
|
|
dom_object *intern;
|
|
char *name, *value = NULL;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os&|s&", &id, dom_document_class_entry, &name, &name_len, UG(utf8_conv), &value, &value_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
if (xmlValidateName((xmlChar *) name, 0) != 0) {
|
|
php_dom_throw_error(INVALID_CHARACTER_ERR, dom_get_strict_error(intern->document) TSRMLS_CC);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
node = xmlNewPI((xmlChar *) name, (xmlChar *) value);
|
|
if (!node) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
node->doc = docp;
|
|
|
|
DOM_RET_OBJ(rv, node, &ret, intern);
|
|
}
|
|
/* }}} end dom_document_create_processing_instruction */
|
|
|
|
|
|
/* {{{ proto DOMAttr dom_document_create_attribute(string name) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1084891198
|
|
Since:
|
|
*/
|
|
PHP_FUNCTION(dom_document_create_attribute)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlAttrPtr node;
|
|
xmlDocPtr docp;
|
|
int ret, name_len;
|
|
dom_object *intern;
|
|
char *name;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os&", &id, dom_document_class_entry, &name, &name_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
if (xmlValidateName((xmlChar *) name, 0) != 0) {
|
|
php_dom_throw_error(INVALID_CHARACTER_ERR, dom_get_strict_error(intern->document) TSRMLS_CC);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
node = xmlNewDocProp(docp, (xmlChar *) name, NULL);
|
|
if (!node) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_RET_OBJ(rv, (xmlNodePtr) node, &ret, intern);
|
|
|
|
}
|
|
/* }}} end dom_document_create_attribute */
|
|
|
|
|
|
/* {{{ proto DOMEntityReference dom_document_create_entity_reference(string name) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-392B75AE
|
|
Since:
|
|
*/
|
|
PHP_FUNCTION(dom_document_create_entity_reference)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlNode *node;
|
|
xmlDocPtr docp = NULL;
|
|
dom_object *intern;
|
|
int ret, name_len;
|
|
char *name;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os&", &id, dom_document_class_entry, &name, &name_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
if (xmlValidateName((xmlChar *) name, 0) != 0) {
|
|
php_dom_throw_error(INVALID_CHARACTER_ERR, dom_get_strict_error(intern->document) TSRMLS_CC);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
node = xmlNewReference(docp, name);
|
|
if (!node) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_RET_OBJ(rv, (xmlNodePtr) node, &ret, intern);
|
|
}
|
|
/* }}} end dom_document_create_entity_reference */
|
|
|
|
|
|
/* {{{ proto DOMNodeList dom_document_get_elements_by_tag_name(string tagname) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-A6C9094
|
|
Since:
|
|
*/
|
|
PHP_FUNCTION(dom_document_get_elements_by_tag_name)
|
|
{
|
|
zval *id;
|
|
xmlDocPtr docp;
|
|
int name_len;
|
|
dom_object *intern, *namednode;
|
|
char *name;
|
|
xmlChar *local;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os&", &id, dom_document_class_entry, &name, &name_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
php_dom_create_interator(return_value, DOM_NODELIST TSRMLS_CC);
|
|
namednode = (dom_object *)zend_objects_get_address(return_value TSRMLS_CC);
|
|
local = xmlCharStrndup(name, name_len);
|
|
dom_namednode_iter(intern, 0, namednode, NULL, local, NULL TSRMLS_CC);
|
|
}
|
|
/* }}} end dom_document_get_elements_by_tag_name */
|
|
|
|
|
|
/* {{{ proto DOMNode dom_document_import_node(DOMNode importedNode, boolean deep) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Core-Document-importNode
|
|
Since: DOM Level 2
|
|
*/
|
|
PHP_FUNCTION(dom_document_import_node)
|
|
{
|
|
zval *rv = NULL;
|
|
zval *id, *node;
|
|
xmlDocPtr docp;
|
|
xmlNodePtr nodep, retnodep;
|
|
dom_object *intern, *nodeobj;
|
|
int ret;
|
|
long recursive = 0;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "OO|l", &id, dom_document_class_entry, &node, dom_node_class_entry, &recursive) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
DOM_GET_OBJ(nodep, node, xmlNodePtr, nodeobj);
|
|
|
|
if (nodep->type == XML_HTML_DOCUMENT_NODE || nodep->type == XML_DOCUMENT_NODE
|
|
|| nodep->type == XML_DOCUMENT_TYPE_NODE) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot import: Node Type Not Supported");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (nodep->doc == docp) {
|
|
retnodep = nodep;
|
|
} else {
|
|
retnodep = xmlDocCopyNode(nodep, docp, recursive);
|
|
if (!retnodep) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
}
|
|
|
|
DOM_RET_OBJ(rv, (xmlNodePtr) retnodep, &ret, intern);
|
|
}
|
|
/* }}} end dom_document_import_node */
|
|
|
|
|
|
/* {{{ proto DOMElement dom_document_create_element_ns(string namespaceURI, string qualifiedName [,string value]) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-DocCrElNS
|
|
Since: DOM Level 2
|
|
*/
|
|
PHP_FUNCTION(dom_document_create_element_ns)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlDocPtr docp;
|
|
xmlNodePtr nodep = NULL;
|
|
xmlNsPtr nsptr = NULL;
|
|
int ret, uri_len = 0, name_len = 0, value_len = 0;
|
|
char *uri, *name, *value = NULL;
|
|
char *localname = NULL, *prefix = NULL;
|
|
int errorcode;
|
|
dom_object *intern;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os!&s&|s&", &id, dom_document_class_entry, &uri, &uri_len, UG(utf8_conv), &name, &name_len, UG(utf8_conv), &value, &value_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
errorcode = dom_check_qname(name, &localname, &prefix, uri_len, name_len);
|
|
|
|
if (errorcode == 0) {
|
|
if (xmlValidateName((xmlChar *) localname, 0) == 0) {
|
|
nodep = xmlNewDocNode (docp, NULL, localname, value);
|
|
if (nodep != NULL && uri != NULL) {
|
|
nsptr = xmlSearchNsByHref (nodep->doc, nodep, uri);
|
|
if (nsptr == NULL) {
|
|
nsptr = dom_get_ns(nodep, uri, &errorcode, prefix);
|
|
}
|
|
xmlSetNs(nodep, nsptr);
|
|
}
|
|
} else {
|
|
errorcode = INVALID_CHARACTER_ERR;
|
|
}
|
|
}
|
|
|
|
xmlFree(localname);
|
|
if (prefix != NULL) {
|
|
xmlFree(prefix);
|
|
}
|
|
|
|
if (errorcode != 0) {
|
|
if (nodep != NULL) {
|
|
xmlFreeNode(nodep);
|
|
}
|
|
php_dom_throw_error(errorcode, dom_get_strict_error(intern->document) TSRMLS_CC);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (nodep == NULL) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
|
|
nodep->ns = nsptr;
|
|
|
|
DOM_RET_OBJ(rv, nodep, &ret, intern);
|
|
}
|
|
/* }}} end dom_document_create_element_ns */
|
|
|
|
|
|
/* {{{ proto DOMAttr dom_document_create_attribute_ns(string namespaceURI, string qualifiedName) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-DocCrAttrNS
|
|
Since: DOM Level 2
|
|
*/
|
|
PHP_FUNCTION(dom_document_create_attribute_ns)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlDocPtr docp;
|
|
xmlNodePtr nodep = NULL, root;
|
|
xmlNsPtr nsptr;
|
|
int ret, uri_len = 0, name_len = 0;
|
|
char *uri, *name;
|
|
char *localname = NULL, *prefix = NULL;
|
|
dom_object *intern;
|
|
int errorcode;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os!&s&", &id, dom_document_class_entry, &uri, &uri_len, UG(utf8_conv), &name, &name_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
root = xmlDocGetRootElement(docp);
|
|
if (root != NULL) {
|
|
errorcode = dom_check_qname(name, &localname, &prefix, uri_len, name_len);
|
|
if (errorcode == 0) {
|
|
if (xmlValidateName((xmlChar *) localname, 0) == 0) {
|
|
nodep = (xmlNodePtr) xmlNewDocProp(docp, localname, NULL);
|
|
if (nodep != NULL && uri_len > 0) {
|
|
nsptr = xmlSearchNsByHref (nodep->doc, root, uri);
|
|
if (nsptr == NULL) {
|
|
nsptr = dom_get_ns(root, uri, &errorcode, prefix);
|
|
}
|
|
xmlSetNs(nodep, nsptr);
|
|
}
|
|
} else {
|
|
errorcode = INVALID_CHARACTER_ERR;
|
|
}
|
|
}
|
|
} else {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Document Missing Root Element");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
xmlFree(localname);
|
|
if (prefix != NULL) {
|
|
xmlFree(prefix);
|
|
}
|
|
|
|
if (errorcode != 0) {
|
|
if (nodep != NULL) {
|
|
xmlFreeProp((xmlAttrPtr) nodep);
|
|
}
|
|
php_dom_throw_error(errorcode, dom_get_strict_error(intern->document) TSRMLS_CC);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (nodep == NULL) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_RET_OBJ(rv, nodep, &ret, intern);
|
|
}
|
|
/* }}} end dom_document_create_attribute_ns */
|
|
|
|
|
|
/* {{{ proto DOMNodeList dom_document_get_elements_by_tag_name_ns(string namespaceURI, string localName) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-getElBTNNS
|
|
Since: DOM Level 2
|
|
*/
|
|
PHP_FUNCTION(dom_document_get_elements_by_tag_name_ns)
|
|
{
|
|
zval *id;
|
|
xmlDocPtr docp;
|
|
int uri_len, name_len;
|
|
dom_object *intern, *namednode;
|
|
char *uri, *name;
|
|
xmlChar *local, *nsuri;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os&s&", &id, dom_document_class_entry, &uri, &uri_len, UG(utf8_conv), &name, &name_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
php_dom_create_interator(return_value, DOM_NODELIST TSRMLS_CC);
|
|
namednode = (dom_object *)zend_objects_get_address(return_value TSRMLS_CC);
|
|
local = xmlCharStrndup(name, name_len);
|
|
nsuri = xmlCharStrndup(uri, uri_len);
|
|
dom_namednode_iter(intern, 0, namednode, NULL, local, nsuri TSRMLS_CC);
|
|
}
|
|
/* }}} end dom_document_get_elements_by_tag_name_ns */
|
|
|
|
|
|
/* {{{ proto DOMElement dom_document_get_element_by_id(string elementId) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-getElBId
|
|
Since: DOM Level 2
|
|
*/
|
|
PHP_FUNCTION(dom_document_get_element_by_id)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlDocPtr docp;
|
|
xmlAttrPtr attrp;
|
|
int ret, idname_len;
|
|
dom_object *intern;
|
|
char *idname;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os&", &id, dom_document_class_entry, &idname, &idname_len, UG(utf8_conv)) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
attrp = xmlGetID(docp, (xmlChar *) idname);
|
|
|
|
if (attrp && attrp->parent) {
|
|
DOM_RET_OBJ(rv, (xmlNodePtr) attrp->parent, &ret, intern);
|
|
} else {
|
|
RETVAL_NULL();
|
|
}
|
|
|
|
}
|
|
/* }}} end dom_document_get_element_by_id */
|
|
|
|
|
|
/* {{{ proto DOMNode dom_document_adopt_node(DOMNode source) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Document3-adoptNode
|
|
Since: DOM Level 3
|
|
*/
|
|
PHP_FUNCTION(dom_document_adopt_node)
|
|
{
|
|
DOM_NOT_IMPLEMENTED();
|
|
}
|
|
/* }}} end dom_document_adopt_node */
|
|
|
|
|
|
/* {{{ proto void dom_document_normalize_document() U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Document3-normalizeDocument
|
|
Since: DOM Level 3
|
|
*/
|
|
PHP_FUNCTION(dom_document_normalize_document)
|
|
{
|
|
zval *id;
|
|
xmlDocPtr docp;
|
|
dom_object *intern;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O", &id, dom_document_class_entry) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
dom_normalize((xmlNodePtr) docp TSRMLS_CC);
|
|
}
|
|
/* }}} end dom_document_normalize_document */
|
|
|
|
|
|
/* {{{ proto DOMNode dom_document_rename_node(node n, string namespaceURI, string qualifiedName) U
|
|
URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Document3-renameNode
|
|
Since: DOM Level 3
|
|
*/
|
|
PHP_FUNCTION(dom_document_rename_node)
|
|
{
|
|
DOM_NOT_IMPLEMENTED();
|
|
}
|
|
/* }}} end dom_document_rename_node */
|
|
|
|
/* {{{ proto void DOMDocument::__construct([string version], [string encoding]) U */
|
|
PHP_METHOD(domdocument, __construct)
|
|
{
|
|
|
|
zval *id;
|
|
xmlDoc *docp = NULL, *olddoc;
|
|
dom_object *intern;
|
|
char *encoding, *version = NULL;
|
|
int encoding_len = 0, version_len = 0, refcount;
|
|
|
|
php_set_error_handling(EH_THROW, dom_domexception_class_entry TSRMLS_CC);
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O|s&s&", &id, dom_document_class_entry, &version, &version_len, UG(utf8_conv), &encoding, &encoding_len, UG(utf8_conv)) == FAILURE) {
|
|
php_std_error_handling();
|
|
return;
|
|
}
|
|
php_std_error_handling();
|
|
|
|
docp = xmlNewDoc(version);
|
|
|
|
if (!docp) {
|
|
php_dom_throw_error(INVALID_STATE_ERR, 1 TSRMLS_CC);
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (encoding_len > 0) {
|
|
docp->encoding = (const xmlChar*)xmlStrdup(encoding);
|
|
}
|
|
|
|
intern = (dom_object *)zend_object_store_get_object(id TSRMLS_CC);
|
|
if (intern != NULL) {
|
|
olddoc = (xmlDocPtr) dom_object_get_node(intern);
|
|
if (olddoc != NULL) {
|
|
php_libxml_decrement_node_ptr((php_libxml_node_object *) intern TSRMLS_CC);
|
|
refcount = php_libxml_decrement_doc_ref((php_libxml_node_object *)intern TSRMLS_CC);
|
|
if (refcount != 0) {
|
|
olddoc->_private = NULL;
|
|
}
|
|
}
|
|
intern->document = NULL;
|
|
php_libxml_increment_doc_ref((php_libxml_node_object *)intern, docp TSRMLS_CC);
|
|
php_libxml_increment_node_ptr((php_libxml_node_object *)intern, (xmlNodePtr)docp, (void *)intern TSRMLS_CC);
|
|
}
|
|
}
|
|
/* }}} end DOMDocument::__construct */
|
|
|
|
char *_dom_get_valid_file_path(char *source, char *resolved_path, int resolved_path_len TSRMLS_DC) {
|
|
xmlURI *uri;
|
|
xmlChar *escsource;
|
|
char *file_dest;
|
|
int isFileUri = 0;
|
|
|
|
uri = xmlCreateURI();
|
|
escsource = xmlURIEscapeStr(source, ":");
|
|
xmlParseURIReference(uri, escsource);
|
|
xmlFree(escsource);
|
|
|
|
if (uri->scheme != NULL) {
|
|
/* absolute file uris - libxml only supports localhost or empty host */
|
|
if (strncasecmp(source, "file:///",8) == 0) {
|
|
isFileUri = 1;
|
|
#ifdef PHP_WIN32
|
|
source += 8;
|
|
#else
|
|
source += 7;
|
|
#endif
|
|
} else if (strncasecmp(source, "file://localhost/",17) == 0) {
|
|
isFileUri = 1;
|
|
#ifdef PHP_WIN32
|
|
source += 17;
|
|
#else
|
|
source += 16;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
file_dest = source;
|
|
|
|
if ((uri->scheme == NULL || isFileUri)) {
|
|
/* XXX possible buffer overflow if VCWD_REALPATH does not know size of resolved_path */
|
|
if (!VCWD_REALPATH(source, resolved_path) && !expand_filepath(source, resolved_path TSRMLS_CC)) {
|
|
xmlFreeURI(uri);
|
|
return NULL;
|
|
}
|
|
file_dest = resolved_path;
|
|
}
|
|
|
|
xmlFreeURI(uri);
|
|
|
|
return file_dest;
|
|
}
|
|
|
|
|
|
/* {{{ */
|
|
static xmlDocPtr dom_document_parser(zval *id, int mode, char *source, int options TSRMLS_DC) {
|
|
xmlDocPtr ret;
|
|
xmlParserCtxtPtr ctxt = NULL;
|
|
dom_doc_propsptr doc_props;
|
|
dom_object *intern;
|
|
php_libxml_ref_obj *document = NULL;
|
|
int validate, recover, resolve_externals, keep_blanks, substitute_ent;
|
|
int resolved_path_len;
|
|
int old_error_reporting = 0;
|
|
char *directory=NULL, resolved_path[MAXPATHLEN];
|
|
|
|
if (id != NULL) {
|
|
intern = (dom_object *)zend_object_store_get_object(id TSRMLS_CC);
|
|
document = intern->document;
|
|
}
|
|
|
|
doc_props = dom_get_doc_props(document);
|
|
validate = doc_props->validateonparse;
|
|
resolve_externals = doc_props->resolveexternals;
|
|
keep_blanks = doc_props->preservewhitespace;
|
|
substitute_ent = doc_props->substituteentities;
|
|
recover = doc_props->recover;
|
|
|
|
if (document == NULL) {
|
|
efree(doc_props);
|
|
}
|
|
|
|
xmlInitParser();
|
|
|
|
if (mode == DOM_LOAD_FILE) {
|
|
char *file_dest = _dom_get_valid_file_path(source, resolved_path, MAXPATHLEN TSRMLS_CC);
|
|
if (file_dest) {
|
|
ctxt = xmlCreateFileParserCtxt(file_dest);
|
|
}
|
|
|
|
} else {
|
|
ctxt = xmlCreateDocParserCtxt(source);
|
|
}
|
|
|
|
if (ctxt == NULL) {
|
|
return(NULL);
|
|
}
|
|
|
|
/* If loading from memory, we need to set the base directory for the document */
|
|
if (mode != DOM_LOAD_FILE) {
|
|
#if HAVE_GETCWD
|
|
directory = VCWD_GETCWD(resolved_path, MAXPATHLEN);
|
|
#elif HAVE_GETWD
|
|
directory = VCWD_GETWD(resolved_path);
|
|
#endif
|
|
if (directory) {
|
|
if(ctxt->directory != NULL) {
|
|
xmlFree((char *) ctxt->directory);
|
|
}
|
|
resolved_path_len = strlen(resolved_path);
|
|
if (resolved_path[resolved_path_len - 1] != DEFAULT_SLASH) {
|
|
resolved_path[resolved_path_len] = DEFAULT_SLASH;
|
|
resolved_path[++resolved_path_len] = '\0';
|
|
}
|
|
ctxt->directory = (char *) xmlCanonicPath((const xmlChar *) resolved_path);
|
|
}
|
|
}
|
|
|
|
ctxt->vctxt.error = php_libxml_ctx_error;
|
|
ctxt->vctxt.warning = php_libxml_ctx_warning;
|
|
|
|
if (ctxt->sax != NULL) {
|
|
ctxt->sax->error = php_libxml_ctx_error;
|
|
ctxt->sax->warning = php_libxml_ctx_warning;
|
|
}
|
|
|
|
if (validate && ! (options & XML_PARSE_DTDVALID)) {
|
|
options |= XML_PARSE_DTDVALID;
|
|
}
|
|
if (resolve_externals && ! (options & XML_PARSE_DTDATTR)) {
|
|
options |= XML_PARSE_DTDATTR;
|
|
}
|
|
if (substitute_ent && ! (options & XML_PARSE_NOENT)) {
|
|
options |= XML_PARSE_NOENT;
|
|
}
|
|
if (keep_blanks == 0 && ! (options & XML_PARSE_NOBLANKS)) {
|
|
options |= XML_PARSE_NOBLANKS;
|
|
}
|
|
|
|
xmlCtxtUseOptions(ctxt, options);
|
|
|
|
ctxt->recovery = recover;
|
|
if (recover) {
|
|
old_error_reporting = EG(error_reporting);
|
|
EG(error_reporting) = old_error_reporting | E_WARNING;
|
|
}
|
|
|
|
xmlParseDocument(ctxt);
|
|
|
|
if (ctxt->wellFormed || recover) {
|
|
ret = ctxt->myDoc;
|
|
if (ctxt->recovery) {
|
|
EG(error_reporting) = old_error_reporting;
|
|
}
|
|
/* If loading from memory, set the base reference uri for the document */
|
|
if (ret->URL == NULL && ctxt->directory != NULL) {
|
|
ret->URL = xmlStrdup(ctxt->directory);
|
|
}
|
|
} else {
|
|
ret = NULL;
|
|
xmlFreeDoc(ctxt->myDoc);
|
|
ctxt->myDoc = NULL;
|
|
}
|
|
|
|
xmlFreeParserCtxt(ctxt);
|
|
|
|
return(ret);
|
|
}
|
|
/* }}} */
|
|
|
|
/* {{{ static void dom_parse_document(INTERNAL_FUNCTION_PARAMETERS, int mode) */
|
|
static void dom_parse_document(INTERNAL_FUNCTION_PARAMETERS, int mode) {
|
|
zval *id, *rv = NULL;
|
|
xmlDoc *docp = NULL, *newdoc;
|
|
dom_doc_propsptr doc_prop;
|
|
dom_object *intern;
|
|
char *source;
|
|
int source_len, refcount, ret;
|
|
zend_uchar source_type = IS_STRING;
|
|
long options = 0;
|
|
|
|
id = getThis();
|
|
if (id != NULL && ! instanceof_function(Z_OBJCE_P(id), dom_document_class_entry TSRMLS_CC)) {
|
|
id = NULL;
|
|
}
|
|
|
|
if (mode == DOM_LOAD_FILE) {
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "t|l", &source, &source_len, &source_type, &options) == FAILURE) {
|
|
return;
|
|
}
|
|
} else {
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "S|l", &source, &source_len, &options) == FAILURE) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (!source_len) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Empty string supplied as input");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (source_type == IS_UNICODE) {
|
|
if (php_stream_path_encode(NULL, &source, &source_len, (UChar*)source, source_len, REPORT_ERRORS, NULL) == FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
newdoc = dom_document_parser(id, mode, source, options TSRMLS_CC);
|
|
|
|
if (source_type == IS_UNICODE) {
|
|
efree(source);
|
|
}
|
|
|
|
if (!newdoc)
|
|
RETURN_FALSE;
|
|
|
|
if (id != NULL) {
|
|
intern = (dom_object *)zend_object_store_get_object(id TSRMLS_CC);
|
|
if (intern != NULL) {
|
|
docp = (xmlDocPtr) dom_object_get_node(intern);
|
|
doc_prop = NULL;
|
|
if (docp != NULL) {
|
|
php_libxml_decrement_node_ptr((php_libxml_node_object *) intern TSRMLS_CC);
|
|
doc_prop = intern->document->doc_props;
|
|
intern->document->doc_props = NULL;
|
|
refcount = php_libxml_decrement_doc_ref((php_libxml_node_object *)intern TSRMLS_CC);
|
|
if (refcount != 0) {
|
|
docp->_private = NULL;
|
|
}
|
|
}
|
|
intern->document = NULL;
|
|
php_libxml_increment_doc_ref((php_libxml_node_object *)intern, newdoc TSRMLS_CC);
|
|
intern->document->doc_props = doc_prop;
|
|
}
|
|
|
|
php_libxml_increment_node_ptr((php_libxml_node_object *)intern, (xmlNodePtr)newdoc, (void *)intern TSRMLS_CC);
|
|
|
|
RETURN_TRUE;
|
|
} else {
|
|
DOM_RET_OBJ(rv, (xmlNodePtr) newdoc, &ret, NULL);
|
|
}
|
|
}
|
|
/* }}} end dom_parser_document */
|
|
|
|
/* {{{ proto DOMNode dom_document_load(string source [, int options]) U
|
|
URL: http://www.w3.org/TR/DOM-Level-3-LS/load-save.html#LS-DocumentLS-load
|
|
Since: DOM Level 3
|
|
*/
|
|
PHP_METHOD(domdocument, load)
|
|
{
|
|
dom_parse_document(INTERNAL_FUNCTION_PARAM_PASSTHRU, DOM_LOAD_FILE);
|
|
}
|
|
/* }}} end dom_document_load */
|
|
|
|
/* {{{ proto DOMNode dom_document_loadxml(string source [, int options]) U
|
|
URL: http://www.w3.org/TR/DOM-Level-3-LS/load-save.html#LS-DocumentLS-loadXML
|
|
Since: DOM Level 3
|
|
*/
|
|
PHP_METHOD(domdocument, loadXML)
|
|
{
|
|
dom_parse_document(INTERNAL_FUNCTION_PARAM_PASSTHRU, DOM_LOAD_STRING);
|
|
}
|
|
/* }}} end dom_document_loadxml */
|
|
|
|
/* {{{ proto int dom_document_save(string file) U
|
|
Convenience method to save to file
|
|
*/
|
|
PHP_FUNCTION(dom_document_save)
|
|
{
|
|
zval *id;
|
|
xmlDoc *docp;
|
|
int file_len = 0, bytes, format, saveempty = 0;
|
|
dom_object *intern;
|
|
dom_doc_propsptr doc_props;
|
|
char *file;
|
|
zend_uchar file_type;
|
|
long options = 0;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Ot|l", &id, dom_document_class_entry, &file, &file_len, &file_type, &options) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
if (file_len == 0) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid Filename");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
/* encoding handled by property on doc */
|
|
|
|
doc_props = dom_get_doc_props(intern->document);
|
|
format = doc_props->formatoutput;
|
|
if (options & LIBXML_SAVE_NOEMPTYTAG) {
|
|
saveempty = xmlSaveNoEmptyTags;
|
|
xmlSaveNoEmptyTags = 1;
|
|
}
|
|
|
|
if (file_type == IS_UNICODE) {
|
|
if (php_stream_path_encode(NULL, &file, &file_len, (UChar*)file, file_len, REPORT_ERRORS, NULL) == FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
bytes = xmlSaveFormatFileEnc(file, docp, NULL, format);
|
|
|
|
if (file_type == IS_UNICODE) {
|
|
efree(file);
|
|
}
|
|
|
|
if (options & LIBXML_SAVE_NOEMPTYTAG) {
|
|
xmlSaveNoEmptyTags = saveempty;
|
|
}
|
|
if (bytes == -1) {
|
|
RETURN_FALSE;
|
|
}
|
|
RETURN_LONG(bytes);
|
|
}
|
|
/* }}} end dom_document_save */
|
|
|
|
/* {{{ proto string dom_document_savexml([node n]) U
|
|
URL: http://www.w3.org/TR/DOM-Level-3-LS/load-save.html#LS-DocumentLS-saveXML
|
|
Since: DOM Level 3
|
|
*/
|
|
PHP_FUNCTION(dom_document_savexml)
|
|
{
|
|
zval *id, *nodep = NULL;
|
|
xmlDoc *docp;
|
|
xmlNode *node;
|
|
xmlBufferPtr buf;
|
|
xmlChar *mem;
|
|
dom_object *intern, *nodeobj;
|
|
dom_doc_propsptr doc_props;
|
|
int size, format, saveempty = 0;
|
|
long options = 0;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O|O!l", &id, dom_document_class_entry, &nodep, dom_node_class_entry, &options) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
doc_props = dom_get_doc_props(intern->document);
|
|
format = doc_props->formatoutput;
|
|
|
|
if (nodep != NULL) {
|
|
/* Dump contents of Node */
|
|
DOM_GET_OBJ(node, nodep, xmlNodePtr, nodeobj);
|
|
if (node->doc != docp) {
|
|
php_dom_throw_error(WRONG_DOCUMENT_ERR, dom_get_strict_error(intern->document) TSRMLS_CC);
|
|
RETURN_FALSE;
|
|
}
|
|
buf = xmlBufferCreate();
|
|
if (!buf) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Could not fetch buffer");
|
|
RETURN_FALSE;
|
|
}
|
|
if (options & LIBXML_SAVE_NOEMPTYTAG) {
|
|
saveempty = xmlSaveNoEmptyTags;
|
|
xmlSaveNoEmptyTags = 1;
|
|
}
|
|
xmlNodeDump(buf, docp, node, 0, format);
|
|
if (options & LIBXML_SAVE_NOEMPTYTAG) {
|
|
xmlSaveNoEmptyTags = saveempty;
|
|
}
|
|
mem = (xmlChar*) xmlBufferContent(buf);
|
|
if (!mem) {
|
|
xmlBufferFree(buf);
|
|
RETURN_FALSE;
|
|
}
|
|
RETVAL_STRING(mem, 1);
|
|
xmlBufferFree(buf);
|
|
} else {
|
|
if (options & LIBXML_SAVE_NOEMPTYTAG) {
|
|
saveempty = xmlSaveNoEmptyTags;
|
|
xmlSaveNoEmptyTags = 1;
|
|
}
|
|
/* Encoding is handled from the encoding property set on the document */
|
|
xmlDocDumpFormatMemory(docp, &mem, &size, format);
|
|
if (options & LIBXML_SAVE_NOEMPTYTAG) {
|
|
xmlSaveNoEmptyTags = saveempty;
|
|
}
|
|
if (!size) {
|
|
RETURN_FALSE;
|
|
}
|
|
RETVAL_STRINGL(mem, size, 1);
|
|
xmlFree(mem);
|
|
}
|
|
}
|
|
/* }}} end dom_document_savexml */
|
|
|
|
static xmlNodePtr php_dom_free_xinclude_node(xmlNodePtr cur TSRMLS_DC) {
|
|
xmlNodePtr xincnode;
|
|
|
|
xincnode = cur;
|
|
cur = cur->next;
|
|
xmlUnlinkNode(xincnode);
|
|
php_libxml_node_free_resource(xincnode TSRMLS_CC);
|
|
|
|
return cur;
|
|
}
|
|
|
|
static void php_dom_remove_xinclude_nodes(xmlNodePtr cur TSRMLS_DC) {
|
|
while(cur) {
|
|
if (cur->type == XML_XINCLUDE_START) {
|
|
cur = php_dom_free_xinclude_node(cur TSRMLS_CC);
|
|
|
|
/* XML_XINCLUDE_END node will be a sibling of XML_XINCLUDE_START */
|
|
while(cur && cur->type != XML_XINCLUDE_END) {
|
|
cur = cur->next;
|
|
}
|
|
|
|
if (cur && cur->type == XML_XINCLUDE_END) {
|
|
cur = php_dom_free_xinclude_node(cur TSRMLS_CC);
|
|
}
|
|
} else {
|
|
if (cur->type == XML_ELEMENT_NODE) {
|
|
php_dom_remove_xinclude_nodes(cur->children TSRMLS_CC);
|
|
}
|
|
cur = cur->next;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* {{{ proto int dom_document_xinclude([int options]) U
|
|
Substitutues xincludes in a DomDocument */
|
|
PHP_FUNCTION(dom_document_xinclude)
|
|
{
|
|
zval *id;
|
|
xmlDoc *docp;
|
|
xmlNodePtr root;
|
|
long flags = 0;
|
|
int err;
|
|
dom_object *intern;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O|l", &id, dom_document_class_entry, &flags) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
err = xmlXIncludeProcessFlags(docp, flags);
|
|
|
|
/* XML_XINCLUDE_START and XML_XINCLUDE_END nodes need to be removed as these
|
|
are added via xmlXIncludeProcess to mark beginning and ending of xincluded document
|
|
but are not wanted in resulting document - must be done even if err as it could fail after
|
|
having processed some xincludes */
|
|
root = (xmlNodePtr) docp->children;
|
|
while(root && root->type != XML_ELEMENT_NODE && root->type != XML_XINCLUDE_START) {
|
|
root = root->next;
|
|
}
|
|
if (root) {
|
|
php_dom_remove_xinclude_nodes(root TSRMLS_CC);
|
|
}
|
|
|
|
if (err) {
|
|
RETVAL_LONG(err);
|
|
} else {
|
|
RETVAL_FALSE;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
/* {{{ proto boolean dom_document_validate() U
|
|
Since: DOM extended
|
|
*/
|
|
PHP_FUNCTION(dom_document_validate)
|
|
{
|
|
zval *id;
|
|
xmlDoc *docp;
|
|
dom_object *intern;
|
|
xmlValidCtxt *cvp;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O", &id, dom_document_class_entry) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
if (docp->intSubset == NULL) {
|
|
php_error_docref(NULL TSRMLS_CC, E_NOTICE, "No DTD given in XML-Document");
|
|
}
|
|
|
|
cvp = xmlNewValidCtxt();
|
|
|
|
cvp->userData = NULL;
|
|
cvp->error = (xmlValidityErrorFunc) php_libxml_error_handler;
|
|
cvp->warning = (xmlValidityErrorFunc) php_libxml_error_handler;
|
|
|
|
if (xmlValidateDocument(cvp, docp)) {
|
|
RETVAL_TRUE;
|
|
} else {
|
|
RETVAL_FALSE;
|
|
}
|
|
|
|
xmlFreeValidCtxt(cvp);
|
|
|
|
}
|
|
|
|
|
|
#if defined(LIBXML_SCHEMAS_ENABLED)
|
|
static void
|
|
_dom_document_schema_validate(INTERNAL_FUNCTION_PARAMETERS, int type)
|
|
{
|
|
zval *id;
|
|
xmlDoc *docp;
|
|
dom_object *intern;
|
|
char *source = NULL, *valid_file = NULL;
|
|
int source_len = 0;
|
|
xmlSchemaParserCtxtPtr parser;
|
|
xmlSchemaPtr sptr;
|
|
xmlSchemaValidCtxtPtr vptr;
|
|
int is_valid;
|
|
char resolved_path[MAXPATHLEN + 1];
|
|
zend_uchar source_type = IS_STRING;
|
|
|
|
if (type == DOM_LOAD_FILE) {
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Ot", &id, dom_document_class_entry, &source, &source_len, &source_type) == FAILURE) {
|
|
return;
|
|
}
|
|
} else {
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "OS", &id, dom_document_class_entry, &source, &source_len) == FAILURE) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (source_len == 0) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid Schema source");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
switch (type) {
|
|
case DOM_LOAD_FILE:
|
|
if (source_type == IS_UNICODE) {
|
|
if (php_stream_path_encode(NULL, &source, &source_len, (UChar*)source, source_len, REPORT_ERRORS, NULL) == FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
valid_file = _dom_get_valid_file_path(source, resolved_path, MAXPATHLEN TSRMLS_CC);
|
|
if (!valid_file) {
|
|
if (source_type == IS_UNICODE) {
|
|
efree(source);
|
|
}
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid Schema file source");
|
|
RETURN_FALSE;
|
|
}
|
|
parser = xmlSchemaNewParserCtxt(valid_file);
|
|
|
|
if (source_type == IS_UNICODE) {
|
|
efree(source);
|
|
}
|
|
break;
|
|
case DOM_LOAD_STRING:
|
|
parser = xmlSchemaNewMemParserCtxt(source, source_len);
|
|
/* If loading from memory, we need to set the base directory for the document
|
|
but it is not apparent how to do that for schema's */
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
|
|
xmlSchemaSetParserErrors(parser,
|
|
(xmlSchemaValidityErrorFunc) php_libxml_error_handler,
|
|
(xmlSchemaValidityWarningFunc) php_libxml_error_handler,
|
|
parser);
|
|
sptr = xmlSchemaParse(parser);
|
|
xmlSchemaFreeParserCtxt(parser);
|
|
if (!sptr) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid Schema");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(intern);
|
|
|
|
vptr = xmlSchemaNewValidCtxt(sptr);
|
|
if (!vptr) {
|
|
xmlSchemaFree(sptr);
|
|
php_error(E_ERROR, "Invalid Schema Validation Context");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
xmlSchemaSetValidErrors(vptr, php_libxml_error_handler, php_libxml_error_handler, vptr);
|
|
is_valid = xmlSchemaValidateDoc(vptr, docp);
|
|
xmlSchemaFree(sptr);
|
|
xmlSchemaFreeValidCtxt(vptr);
|
|
|
|
if (is_valid == 0) {
|
|
RETURN_TRUE;
|
|
} else {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
/* {{{ proto boolean dom_document_schema_validate_file(string filename) U */
|
|
PHP_FUNCTION(dom_document_schema_validate_file)
|
|
{
|
|
_dom_document_schema_validate(INTERNAL_FUNCTION_PARAM_PASSTHRU, DOM_LOAD_FILE);
|
|
}
|
|
/* }}} end dom_document_schema_validate_file */
|
|
|
|
/* {{{ proto boolean dom_document_schema_validate(string source) U */
|
|
PHP_FUNCTION(dom_document_schema_validate_xml)
|
|
{
|
|
_dom_document_schema_validate(INTERNAL_FUNCTION_PARAM_PASSTHRU, DOM_LOAD_STRING);
|
|
}
|
|
/* }}} end dom_document_schema_validate */
|
|
|
|
|
|
static void
|
|
_dom_document_relaxNG_validate(INTERNAL_FUNCTION_PARAMETERS, int type)
|
|
{
|
|
zval *id;
|
|
xmlDoc *docp;
|
|
dom_object *intern;
|
|
char *source = NULL, *valid_file = NULL;
|
|
int source_len = 0;
|
|
xmlRelaxNGParserCtxtPtr parser;
|
|
xmlRelaxNGPtr sptr;
|
|
xmlRelaxNGValidCtxtPtr vptr;
|
|
int is_valid;
|
|
char resolved_path[MAXPATHLEN + 1];
|
|
zend_uchar source_type = IS_STRING;
|
|
|
|
if (type == DOM_LOAD_FILE) {
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Ot", &id, dom_document_class_entry, &source, &source_len, &source_type) == FAILURE) {
|
|
return;
|
|
}
|
|
} else {
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "OS", &id, dom_document_class_entry, &source, &source_len) == FAILURE) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (source_len == 0) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid Schema source");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
switch (type) {
|
|
case DOM_LOAD_FILE:
|
|
if (source_type == IS_UNICODE) {
|
|
if (php_stream_path_encode(NULL, &source, &source_len, (UChar*)source, source_len, REPORT_ERRORS, NULL) == FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
valid_file = _dom_get_valid_file_path(source, resolved_path, MAXPATHLEN TSRMLS_CC);
|
|
if (!valid_file) {
|
|
if (source_type == IS_UNICODE) {
|
|
efree(source);
|
|
}
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid RelaxNG file source");
|
|
RETURN_FALSE;
|
|
}
|
|
parser = xmlRelaxNGNewParserCtxt(valid_file);
|
|
if (source_type == IS_UNICODE) {
|
|
efree(source);
|
|
}
|
|
break;
|
|
case DOM_LOAD_STRING:
|
|
parser = xmlRelaxNGNewMemParserCtxt(source, source_len);
|
|
/* If loading from memory, we need to set the base directory for the document
|
|
but it is not apparent how to do that for schema's */
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
|
|
xmlRelaxNGSetParserErrors(parser,
|
|
(xmlRelaxNGValidityErrorFunc) php_libxml_error_handler,
|
|
(xmlRelaxNGValidityWarningFunc) php_libxml_error_handler,
|
|
parser);
|
|
sptr = xmlRelaxNGParse(parser);
|
|
xmlRelaxNGFreeParserCtxt(parser);
|
|
if (!sptr) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid RelaxNG");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
docp = (xmlDocPtr) dom_object_get_node(intern);
|
|
|
|
vptr = xmlRelaxNGNewValidCtxt(sptr);
|
|
if (!vptr) {
|
|
xmlRelaxNGFree(sptr);
|
|
php_error(E_ERROR, "Invalid RelaxNG Validation Context");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
xmlRelaxNGSetValidErrors(vptr, php_libxml_error_handler, php_libxml_error_handler, vptr);
|
|
is_valid = xmlRelaxNGValidateDoc(vptr, docp);
|
|
xmlRelaxNGFree(sptr);
|
|
xmlRelaxNGFreeValidCtxt(vptr);
|
|
|
|
if (is_valid == 0) {
|
|
RETURN_TRUE;
|
|
} else {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
/* {{{ proto boolean dom_document_relaxNG_validate_file(string filename) U */
|
|
PHP_FUNCTION(dom_document_relaxNG_validate_file)
|
|
{
|
|
_dom_document_relaxNG_validate(INTERNAL_FUNCTION_PARAM_PASSTHRU, DOM_LOAD_FILE);
|
|
}
|
|
/* }}} end dom_document_relaxNG_validate_file */
|
|
|
|
/* {{{ proto boolean dom_document_relaxNG_validate_xml(string source) U */
|
|
PHP_FUNCTION(dom_document_relaxNG_validate_xml)
|
|
{
|
|
_dom_document_relaxNG_validate(INTERNAL_FUNCTION_PARAM_PASSTHRU, DOM_LOAD_STRING);
|
|
}
|
|
/* }}} end dom_document_relaxNG_validate_xml */
|
|
|
|
#endif
|
|
|
|
#if defined(LIBXML_HTML_ENABLED)
|
|
|
|
static void dom_load_html(INTERNAL_FUNCTION_PARAMETERS, int mode)
|
|
{
|
|
zval *id, *rv = NULL;
|
|
xmlDoc *docp = NULL, *newdoc;
|
|
dom_object *intern;
|
|
dom_doc_propsptr doc_prop;
|
|
char *source;
|
|
int source_len, refcount, ret;
|
|
htmlParserCtxtPtr ctxt;
|
|
zend_uchar source_type = IS_STRING;
|
|
|
|
id = getThis();
|
|
|
|
if (mode == DOM_LOAD_FILE) {
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "t", &source, &source_len, &source_type) == FAILURE) {
|
|
return;
|
|
}
|
|
} else {
|
|
if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "S", &source, &source_len) == FAILURE) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (!source_len) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Empty string supplied as input");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
if (mode == DOM_LOAD_FILE) {
|
|
if (source_type == IS_UNICODE) {
|
|
if (php_stream_path_encode(NULL, &source, &source_len, (UChar*)source, source_len, REPORT_ERRORS, NULL) == FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
ctxt = htmlCreateFileParserCtxt(source, NULL);
|
|
|
|
if (source_type == IS_UNICODE) {
|
|
efree(source);
|
|
}
|
|
} else {
|
|
source_len = xmlStrlen(source);
|
|
ctxt = htmlCreateMemoryParserCtxt(source, source_len);
|
|
}
|
|
|
|
if (!ctxt) {
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
ctxt->vctxt.error = php_libxml_ctx_error;
|
|
ctxt->vctxt.warning = php_libxml_ctx_warning;
|
|
if (ctxt->sax != NULL) {
|
|
ctxt->sax->error = php_libxml_ctx_error;
|
|
ctxt->sax->warning = php_libxml_ctx_warning;
|
|
}
|
|
htmlParseDocument(ctxt);
|
|
newdoc = ctxt->myDoc;
|
|
htmlFreeParserCtxt(ctxt);
|
|
|
|
if (!newdoc)
|
|
RETURN_FALSE;
|
|
|
|
if (id != NULL && instanceof_function(Z_OBJCE_P(id), dom_document_class_entry TSRMLS_CC)) {
|
|
intern = (dom_object *)zend_object_store_get_object(id TSRMLS_CC);
|
|
if (intern != NULL) {
|
|
docp = (xmlDocPtr) dom_object_get_node(intern);
|
|
doc_prop = NULL;
|
|
if (docp != NULL) {
|
|
php_libxml_decrement_node_ptr((php_libxml_node_object *) intern TSRMLS_CC);
|
|
doc_prop = intern->document->doc_props;
|
|
intern->document->doc_props = NULL;
|
|
refcount = php_libxml_decrement_doc_ref((php_libxml_node_object *)intern TSRMLS_CC);
|
|
if (refcount != 0) {
|
|
docp->_private = NULL;
|
|
}
|
|
}
|
|
intern->document = NULL;
|
|
php_libxml_increment_doc_ref((php_libxml_node_object *)intern, newdoc TSRMLS_CC);
|
|
intern->document->doc_props = doc_prop;
|
|
}
|
|
|
|
php_libxml_increment_node_ptr((php_libxml_node_object *)intern, (xmlNodePtr)newdoc, (void *)intern TSRMLS_CC);
|
|
|
|
RETURN_TRUE;
|
|
} else {
|
|
DOM_RET_OBJ(rv, (xmlNodePtr) newdoc, &ret, NULL);
|
|
}
|
|
}
|
|
|
|
/* {{{ proto DOMNode dom_document_load_html_file(string source) U
|
|
Since: DOM extended
|
|
*/
|
|
PHP_METHOD(domdocument, loadHTMLFile)
|
|
{
|
|
dom_load_html(INTERNAL_FUNCTION_PARAM_PASSTHRU, DOM_LOAD_FILE);
|
|
}
|
|
/* }}} end dom_document_load_html_file */
|
|
|
|
/* {{{ proto DOMNode dom_document_load_html(string source) U
|
|
Since: DOM extended
|
|
*/
|
|
PHP_METHOD(domdocument, loadHTML)
|
|
{
|
|
dom_load_html(INTERNAL_FUNCTION_PARAM_PASSTHRU, DOM_LOAD_STRING);
|
|
}
|
|
/* }}} end dom_document_load_html */
|
|
|
|
/* {{{ proto int dom_document_save_html_file(string file) U
|
|
Convenience method to save to file as html
|
|
*/
|
|
PHP_FUNCTION(dom_document_save_html_file)
|
|
{
|
|
zval *id;
|
|
xmlDoc *docp;
|
|
int file_len, bytes, format;
|
|
dom_object *intern;
|
|
dom_doc_propsptr doc_props;
|
|
char *file;
|
|
zend_uchar file_type;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Ot", &id, dom_document_class_entry, &file, &file_len, &file_type) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
if (file_len == 0) {
|
|
php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid Filename");
|
|
RETURN_FALSE;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
if (file_type == IS_UNICODE) {
|
|
if (php_stream_path_encode(NULL, &file, &file_len, (UChar*)file, file_len, REPORT_ERRORS, NULL) == FAILURE) {
|
|
RETURN_FALSE;
|
|
}
|
|
}
|
|
|
|
/* encoding handled by property on doc */
|
|
|
|
doc_props = dom_get_doc_props(intern->document);
|
|
format = doc_props->formatoutput;
|
|
|
|
bytes = htmlSaveFileFormat(file, docp, NULL, format);
|
|
|
|
if (file_type == IS_UNICODE) {
|
|
efree(file);
|
|
}
|
|
|
|
if (bytes == -1) {
|
|
RETURN_FALSE;
|
|
}
|
|
RETURN_LONG(bytes);
|
|
}
|
|
/* }}} end dom_document_save_html_file */
|
|
|
|
/* {{{ proto string dom_document_save_html() U
|
|
Convenience method to output as html
|
|
*/
|
|
PHP_FUNCTION(dom_document_save_html)
|
|
{
|
|
zval *id;
|
|
xmlDoc *docp;
|
|
dom_object *intern;
|
|
xmlChar *mem;
|
|
int size, format;
|
|
dom_doc_propsptr doc_props;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O", &id, dom_document_class_entry) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
#if LIBXML_VERSION >= 20623
|
|
doc_props = dom_get_doc_props(intern->document);
|
|
format = doc_props->formatoutput;
|
|
|
|
htmlDocDumpMemoryFormat(docp, &mem, &size, format);
|
|
#else
|
|
htmlDocDumpMemory(docp, &mem, &size);
|
|
#endif
|
|
|
|
if (!size) {
|
|
if (mem)
|
|
xmlFree(mem);
|
|
RETURN_FALSE;
|
|
}
|
|
RETVAL_STRINGL(mem, size, 1);
|
|
xmlFree(mem);
|
|
}
|
|
/* }}} end dom_document_save_html */
|
|
|
|
#endif /* defined(LIBXML_HTML_ENABLED) */
|
|
|
|
/* {{{ proto boolean DOMDocument::registerNodeClass(string baseclass, string extendedclass)
|
|
Register extended class used to create base node type */
|
|
PHP_METHOD(domdocument, registerNodeClass)
|
|
{
|
|
zval *id;
|
|
xmlDoc *docp;
|
|
char *baseclass = NULL, *extendedclass = NULL;
|
|
int baseclass_len = 0, extendedclass_len = 0;
|
|
zend_class_entry *basece = NULL, *ce = NULL;
|
|
dom_object *intern;
|
|
|
|
if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Oss!", &id, dom_document_class_entry, &baseclass, &baseclass_len, &extendedclass, &extendedclass_len) == FAILURE) {
|
|
return;
|
|
}
|
|
|
|
if (baseclass_len) {
|
|
zend_class_entry **pce;
|
|
if (zend_lookup_class(baseclass, baseclass_len, &pce TSRMLS_CC) == FAILURE) {
|
|
php_error_docref(NULL TSRMLS_CC, E_ERROR, "Class %s does not exist", baseclass);
|
|
return;
|
|
}
|
|
basece = *pce;
|
|
}
|
|
|
|
if (basece == NULL || ! instanceof_function(basece, dom_node_class_entry TSRMLS_CC)) {
|
|
php_error_docref(NULL TSRMLS_CC, E_ERROR, "Class %s is not derived from DOMNode.", baseclass);
|
|
return;
|
|
}
|
|
|
|
if (extendedclass_len) {
|
|
zend_class_entry **pce;
|
|
if (zend_lookup_class(extendedclass, extendedclass_len, &pce TSRMLS_CC) == FAILURE) {
|
|
php_error_docref(NULL TSRMLS_CC, E_ERROR, "Class %s does not exist", extendedclass);
|
|
}
|
|
ce = *pce;
|
|
}
|
|
|
|
if (ce == NULL || instanceof_function(ce, basece TSRMLS_CC)) {
|
|
|
|
DOM_GET_OBJ(docp, id, xmlDocPtr, intern);
|
|
|
|
if (dom_set_doc_classmap(intern->document, basece, ce TSRMLS_CC) == FAILURE) {
|
|
php_error_docref(NULL TSRMLS_CC, E_ERROR, "Class %s could not be registered.", extendedclass);
|
|
}
|
|
RETURN_TRUE;
|
|
} else {
|
|
php_error_docref(NULL TSRMLS_CC, E_ERROR, "Class %s is not derived from %s.", extendedclass, baseclass);
|
|
}
|
|
|
|
RETURN_FALSE;
|
|
}
|
|
/* }}} */
|
|
|
|
#endif /* HAVE_LIBXML && HAVE_DOM */
|