You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1987 lines
50 KiB

18 years ago
18 years ago
18 years ago
18 years ago
21 years ago
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2009 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Authors: Christian Stocker <chregu@php.net> |
  16. | Rob Richards <rrichards@php.net> |
  17. +----------------------------------------------------------------------+
  18. */
  19. /* $Id$ */
  20. #ifdef HAVE_CONFIG_H
  21. #include "config.h"
  22. #endif
  23. #include "php.h"
  24. #if HAVE_LIBXML && HAVE_DOM
  25. #include "php_dom.h"
  26. /* {{{ arginfo */
  27. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_insert_before, 0, 0, 1)
  28. ZEND_ARG_OBJ_INFO(0, newChild, DOMNode, 0)
  29. ZEND_ARG_OBJ_INFO(0, refChild, DOMNode, 1)
  30. ZEND_END_ARG_INFO();
  31. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_replace_child, 0, 0, 2)
  32. ZEND_ARG_OBJ_INFO(0, newChild, DOMNode, 0)
  33. ZEND_ARG_OBJ_INFO(0, oldChild, DOMNode, 0)
  34. ZEND_END_ARG_INFO();
  35. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_remove_child, 0, 0, 1)
  36. ZEND_ARG_OBJ_INFO(0, oldChild, DOMNode, 0)
  37. ZEND_END_ARG_INFO();
  38. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_append_child, 0, 0, 1)
  39. ZEND_ARG_OBJ_INFO(0, newChild, DOMNode, 0)
  40. ZEND_END_ARG_INFO();
  41. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_has_child_nodes, 0, 0, 0)
  42. ZEND_END_ARG_INFO();
  43. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_clone_node, 0, 0, 1)
  44. ZEND_ARG_INFO(0, deep)
  45. ZEND_END_ARG_INFO();
  46. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_normalize, 0, 0, 0)
  47. ZEND_END_ARG_INFO();
  48. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_is_supported, 0, 0, 2)
  49. ZEND_ARG_INFO(0, feature)
  50. ZEND_ARG_INFO(0, version)
  51. ZEND_END_ARG_INFO();
  52. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_has_attributes, 0, 0, 0)
  53. ZEND_END_ARG_INFO();
  54. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_compare_document_position, 0, 0, 1)
  55. ZEND_ARG_OBJ_INFO(0, other, DOMNode, 0)
  56. ZEND_END_ARG_INFO();
  57. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_is_same_node, 0, 0, 1)
  58. ZEND_ARG_OBJ_INFO(0, other, DOMNode, 0)
  59. ZEND_END_ARG_INFO();
  60. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_lookup_prefix, 0, 0, 1)
  61. ZEND_ARG_INFO(0, namespaceURI)
  62. ZEND_END_ARG_INFO();
  63. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_is_default_namespace, 0, 0, 1)
  64. ZEND_ARG_INFO(0, namespaceURI)
  65. ZEND_END_ARG_INFO();
  66. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_lookup_namespace_uri, 0, 0, 1)
  67. ZEND_ARG_INFO(0, prefix)
  68. ZEND_END_ARG_INFO();
  69. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_is_equal_node, 0, 0, 1)
  70. ZEND_ARG_OBJ_INFO(0, arg, DOMNode, 0)
  71. ZEND_END_ARG_INFO();
  72. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_get_feature, 0, 0, 2)
  73. ZEND_ARG_INFO(0, feature)
  74. ZEND_ARG_INFO(0, version)
  75. ZEND_END_ARG_INFO();
  76. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_set_user_data, 0, 0, 3)
  77. ZEND_ARG_INFO(0, key)
  78. ZEND_ARG_INFO(0, data)
  79. ZEND_ARG_INFO(0, handler)
  80. ZEND_END_ARG_INFO();
  81. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_get_user_data, 0, 0, 1)
  82. ZEND_ARG_INFO(0, key)
  83. ZEND_END_ARG_INFO();
  84. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_getNodePath, 0, 0, 0)
  85. ZEND_END_ARG_INFO();
  86. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_getLineNo, 0, 0, 0)
  87. ZEND_END_ARG_INFO();
  88. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_C14N, 0, 0, 0)
  89. ZEND_ARG_INFO(0, exclusive)
  90. ZEND_ARG_INFO(0, with_comments)
  91. ZEND_ARG_ARRAY_INFO(0, xpath, 1)
  92. ZEND_ARG_ARRAY_INFO(0, ns_prefixes, 1)
  93. ZEND_END_ARG_INFO();
  94. ZEND_BEGIN_ARG_INFO_EX(arginfo_dom_node_C14NFile, 0, 0, 1)
  95. ZEND_ARG_INFO(0, uri)
  96. ZEND_ARG_INFO(0, exclusive)
  97. ZEND_ARG_INFO(0, with_comments)
  98. ZEND_ARG_ARRAY_INFO(0, xpath, 1)
  99. ZEND_ARG_ARRAY_INFO(0, ns_prefixes, 1)
  100. ZEND_END_ARG_INFO();
  101. /* }}} */
  102. /*
  103. * class DOMNode
  104. *
  105. * URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1950641247
  106. * Since:
  107. */
  108. const zend_function_entry php_dom_node_class_functions[] = { /* {{{ */
  109. PHP_FALIAS(insertBefore, dom_node_insert_before, arginfo_dom_node_insert_before)
  110. PHP_FALIAS(replaceChild, dom_node_replace_child, arginfo_dom_node_replace_child)
  111. PHP_FALIAS(removeChild, dom_node_remove_child, arginfo_dom_node_remove_child)
  112. PHP_FALIAS(appendChild, dom_node_append_child, arginfo_dom_node_append_child)
  113. PHP_FALIAS(hasChildNodes, dom_node_has_child_nodes, arginfo_dom_node_has_child_nodes)
  114. PHP_FALIAS(cloneNode, dom_node_clone_node, arginfo_dom_node_clone_node)
  115. PHP_FALIAS(normalize, dom_node_normalize, arginfo_dom_node_normalize)
  116. PHP_FALIAS(isSupported, dom_node_is_supported, arginfo_dom_node_is_supported)
  117. PHP_FALIAS(hasAttributes, dom_node_has_attributes, arginfo_dom_node_has_attributes)
  118. PHP_FALIAS(compareDocumentPosition, dom_node_compare_document_position, arginfo_dom_node_compare_document_position)
  119. PHP_FALIAS(isSameNode, dom_node_is_same_node, arginfo_dom_node_is_same_node)
  120. PHP_FALIAS(lookupPrefix, dom_node_lookup_prefix, arginfo_dom_node_lookup_prefix)
  121. PHP_FALIAS(isDefaultNamespace, dom_node_is_default_namespace, arginfo_dom_node_is_default_namespace)
  122. PHP_FALIAS(lookupNamespaceUri, dom_node_lookup_namespace_uri, arginfo_dom_node_lookup_namespace_uri)
  123. PHP_FALIAS(isEqualNode, dom_node_is_equal_node, arginfo_dom_node_is_equal_node)
  124. PHP_FALIAS(getFeature, dom_node_get_feature, arginfo_dom_node_get_feature)
  125. PHP_FALIAS(setUserData, dom_node_set_user_data, arginfo_dom_node_set_user_data)
  126. PHP_FALIAS(getUserData, dom_node_get_user_data, arginfo_dom_node_get_user_data)
  127. PHP_ME(domnode, getNodePath, arginfo_dom_node_getNodePath, ZEND_ACC_PUBLIC)
  128. PHP_ME(domnode, getLineNo, arginfo_dom_node_getLineNo, ZEND_ACC_PUBLIC)
  129. PHP_ME(domnode, C14N, arginfo_dom_node_C14N, ZEND_ACC_PUBLIC)
  130. PHP_ME(domnode, C14NFile, arginfo_dom_node_C14NFile, ZEND_ACC_PUBLIC)
  131. {NULL, NULL, NULL}
  132. };
  133. /* }}} */
  134. static void dom_reconcile_ns(xmlDocPtr doc, xmlNodePtr nodep) /* {{{ */
  135. {
  136. xmlNsPtr nsptr, nsdftptr, curns, prevns = NULL;
  137. if (nodep->type == XML_ELEMENT_NODE) {
  138. /* Following if block primarily used for inserting nodes created via createElementNS */
  139. if (nodep->nsDef != NULL) {
  140. curns = nodep->nsDef;
  141. while (curns) {
  142. nsdftptr = curns->next;
  143. if (curns->href != NULL) {
  144. if((nsptr = xmlSearchNsByHref(doc, nodep->parent, curns->href)) &&
  145. (curns->prefix == NULL || xmlStrEqual(nsptr->prefix, curns->prefix))) {
  146. curns->next = NULL;
  147. if (prevns == NULL) {
  148. nodep->nsDef = nsdftptr;
  149. } else {
  150. prevns->next = nsdftptr;
  151. }
  152. dom_set_old_ns(doc, curns);
  153. curns = prevns;
  154. }
  155. }
  156. prevns = curns;
  157. curns = nsdftptr;
  158. }
  159. }
  160. xmlReconciliateNs(doc, nodep);
  161. }
  162. }
  163. /* }}} */
  164. /* {{{ nodeName string
  165. readonly=yes
  166. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-F68D095
  167. Since:
  168. */
  169. int dom_node_node_name_read(dom_object *obj, zval **retval TSRMLS_DC)
  170. {
  171. xmlNode *nodep;
  172. xmlNsPtr ns;
  173. char *str = NULL;
  174. xmlChar *qname = NULL;
  175. nodep = dom_object_get_node(obj);
  176. if (nodep == NULL) {
  177. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  178. return FAILURE;
  179. }
  180. switch (nodep->type) {
  181. case XML_ATTRIBUTE_NODE:
  182. case XML_ELEMENT_NODE:
  183. ns = nodep->ns;
  184. if (ns != NULL && ns->prefix) {
  185. qname = xmlStrdup(ns->prefix);
  186. qname = xmlStrcat(qname, ":");
  187. qname = xmlStrcat(qname, nodep->name);
  188. str = qname;
  189. } else {
  190. str = (char *) nodep->name;
  191. }
  192. break;
  193. case XML_NAMESPACE_DECL:
  194. ns = nodep->ns;
  195. if (ns != NULL && ns->prefix) {
  196. qname = xmlStrdup("xmlns");
  197. qname = xmlStrcat(qname, ":");
  198. qname = xmlStrcat(qname, nodep->name);
  199. str = qname;
  200. } else {
  201. str = (char *) nodep->name;
  202. }
  203. break;
  204. case XML_DOCUMENT_TYPE_NODE:
  205. case XML_DTD_NODE:
  206. case XML_PI_NODE:
  207. case XML_ENTITY_DECL:
  208. case XML_ENTITY_REF_NODE:
  209. case XML_NOTATION_NODE:
  210. str = (char *) nodep->name;
  211. break;
  212. case XML_CDATA_SECTION_NODE:
  213. str = "#cdata-section";
  214. break;
  215. case XML_COMMENT_NODE:
  216. str = "#comment";
  217. break;
  218. case XML_HTML_DOCUMENT_NODE:
  219. case XML_DOCUMENT_NODE:
  220. str = "#document";
  221. break;
  222. case XML_DOCUMENT_FRAG_NODE:
  223. str = "#document-fragment";
  224. break;
  225. case XML_TEXT_NODE:
  226. str = "#text";
  227. break;
  228. default:
  229. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Invalid Node Type");
  230. }
  231. ALLOC_ZVAL(*retval);
  232. if(str != NULL) {
  233. ZVAL_STRING(*retval, str, 1);
  234. } else {
  235. ZVAL_EMPTY_STRING(*retval);
  236. }
  237. if (qname != NULL) {
  238. xmlFree(qname);
  239. }
  240. return SUCCESS;
  241. }
  242. /* }}} */
  243. /* {{{ nodeValue string
  244. readonly=no
  245. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-F68D080
  246. Since:
  247. */
  248. int dom_node_node_value_read(dom_object *obj, zval **retval TSRMLS_DC)
  249. {
  250. xmlNode *nodep;
  251. char *str = NULL;
  252. nodep = dom_object_get_node(obj);
  253. if (nodep == NULL) {
  254. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  255. return FAILURE;
  256. }
  257. /* Access to Element node is implemented as a convience method */
  258. switch (nodep->type) {
  259. case XML_ATTRIBUTE_NODE:
  260. case XML_TEXT_NODE:
  261. case XML_ELEMENT_NODE:
  262. case XML_COMMENT_NODE:
  263. case XML_CDATA_SECTION_NODE:
  264. case XML_PI_NODE:
  265. str = xmlNodeGetContent(nodep);
  266. break;
  267. case XML_NAMESPACE_DECL:
  268. str = xmlNodeGetContent(nodep->children);
  269. break;
  270. default:
  271. str = NULL;
  272. break;
  273. }
  274. ALLOC_ZVAL(*retval);
  275. if(str != NULL) {
  276. ZVAL_STRING(*retval, str, 1);
  277. xmlFree(str);
  278. } else {
  279. ZVAL_NULL(*retval);
  280. }
  281. return SUCCESS;
  282. }
  283. int dom_node_node_value_write(dom_object *obj, zval *newval TSRMLS_DC)
  284. {
  285. xmlNode *nodep;
  286. zval value_copy;
  287. nodep = dom_object_get_node(obj);
  288. if (nodep == NULL) {
  289. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  290. return FAILURE;
  291. }
  292. /* Access to Element node is implemented as a convience method */
  293. switch (nodep->type) {
  294. case XML_ELEMENT_NODE:
  295. case XML_ATTRIBUTE_NODE:
  296. if (nodep->children) {
  297. node_list_unlink(nodep->children TSRMLS_CC);
  298. }
  299. case XML_TEXT_NODE:
  300. case XML_COMMENT_NODE:
  301. case XML_CDATA_SECTION_NODE:
  302. case XML_PI_NODE:
  303. if (newval->type != IS_STRING) {
  304. if(Z_REFCOUNT_P(newval) > 1) {
  305. value_copy = *newval;
  306. zval_copy_ctor(&value_copy);
  307. newval = &value_copy;
  308. }
  309. convert_to_string(newval);
  310. }
  311. xmlNodeSetContentLen(nodep, Z_STRVAL_P(newval), Z_STRLEN_P(newval) + 1);
  312. if (newval == &value_copy) {
  313. zval_dtor(newval);
  314. }
  315. break;
  316. default:
  317. break;
  318. }
  319. return SUCCESS;
  320. }
  321. /* }}} */
  322. /* {{{ nodeType int
  323. readonly=yes
  324. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-111237558
  325. Since:
  326. */
  327. int dom_node_node_type_read(dom_object *obj, zval **retval TSRMLS_DC)
  328. {
  329. xmlNode *nodep;
  330. nodep = dom_object_get_node(obj);
  331. if (nodep == NULL) {
  332. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  333. return FAILURE;
  334. }
  335. ALLOC_ZVAL(*retval);
  336. /* Specs dictate that they are both type XML_DOCUMENT_TYPE_NODE */
  337. if (nodep->type == XML_DTD_NODE) {
  338. ZVAL_LONG(*retval, XML_DOCUMENT_TYPE_NODE);
  339. } else {
  340. ZVAL_LONG(*retval, nodep->type);
  341. }
  342. return SUCCESS;
  343. }
  344. /* }}} */
  345. /* {{{ parentNode DomNode
  346. readonly=yes
  347. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1060184317
  348. Since:
  349. */
  350. int dom_node_parent_node_read(dom_object *obj, zval **retval TSRMLS_DC)
  351. {
  352. xmlNode *nodep, *nodeparent;
  353. int ret;
  354. nodep = dom_object_get_node(obj);
  355. if (nodep == NULL) {
  356. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  357. return FAILURE;
  358. }
  359. ALLOC_ZVAL(*retval);
  360. nodeparent = nodep->parent;
  361. if (!nodeparent) {
  362. ZVAL_NULL(*retval);
  363. return SUCCESS;
  364. }
  365. if (NULL == (*retval = php_dom_create_object(nodeparent, &ret, NULL, *retval, obj TSRMLS_CC))) {
  366. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot create required DOM object");
  367. return FAILURE;
  368. }
  369. return SUCCESS;
  370. }
  371. /* }}} */
  372. /* {{{ childNodes DomNodeList
  373. readonly=yes
  374. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1451460987
  375. Since:
  376. */
  377. int dom_node_child_nodes_read(dom_object *obj, zval **retval TSRMLS_DC)
  378. {
  379. xmlNode *nodep;
  380. dom_object *intern;
  381. nodep = dom_object_get_node(obj);
  382. if (nodep == NULL) {
  383. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  384. return FAILURE;
  385. }
  386. ALLOC_ZVAL(*retval);
  387. if (dom_node_children_valid(nodep) == FAILURE) {
  388. ZVAL_NULL(*retval);
  389. } else {
  390. php_dom_create_interator(*retval, DOM_NODELIST TSRMLS_CC);
  391. intern = (dom_object *)zend_objects_get_address(*retval TSRMLS_CC);
  392. dom_namednode_iter(obj, XML_ELEMENT_NODE, intern, NULL, NULL, NULL TSRMLS_CC);
  393. }
  394. return SUCCESS;
  395. }
  396. /* }}} */
  397. /* {{{ firstChild DomNode
  398. readonly=yes
  399. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-169727388
  400. Since:
  401. */
  402. int dom_node_first_child_read(dom_object *obj, zval **retval TSRMLS_DC)
  403. {
  404. xmlNode *nodep, *first = NULL;
  405. int ret;
  406. nodep = dom_object_get_node(obj);
  407. if (nodep == NULL) {
  408. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  409. return FAILURE;
  410. }
  411. if (dom_node_children_valid(nodep) == SUCCESS) {
  412. first = nodep->children;
  413. }
  414. ALLOC_ZVAL(*retval);
  415. if (!first) {
  416. ZVAL_NULL(*retval);
  417. return SUCCESS;
  418. }
  419. if (NULL == (*retval = php_dom_create_object(first, &ret, NULL, *retval, obj TSRMLS_CC))) {
  420. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot create required DOM object");
  421. return FAILURE;
  422. }
  423. return SUCCESS;
  424. }
  425. /* }}} */
  426. /* {{{ lastChild DomNode
  427. readonly=yes
  428. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-61AD09FB
  429. Since:
  430. */
  431. int dom_node_last_child_read(dom_object *obj, zval **retval TSRMLS_DC)
  432. {
  433. xmlNode *nodep, *last = NULL;
  434. int ret;
  435. nodep = dom_object_get_node(obj);
  436. if (nodep == NULL) {
  437. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  438. return FAILURE;
  439. }
  440. if (dom_node_children_valid(nodep) == SUCCESS) {
  441. last = nodep->last;
  442. }
  443. ALLOC_ZVAL(*retval);
  444. if (!last) {
  445. ZVAL_NULL(*retval);
  446. return SUCCESS;
  447. }
  448. if (NULL == (*retval = php_dom_create_object(last, &ret, NULL, *retval, obj TSRMLS_CC))) {
  449. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot create required DOM object");
  450. return FAILURE;
  451. }
  452. return SUCCESS;
  453. }
  454. /* }}} */
  455. /* {{{ previousSibling DomNode
  456. readonly=yes
  457. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-640FB3C8
  458. Since:
  459. */
  460. int dom_node_previous_sibling_read(dom_object *obj, zval **retval TSRMLS_DC)
  461. {
  462. xmlNode *nodep, *prevsib;
  463. int ret;
  464. nodep = dom_object_get_node(obj);
  465. if (nodep == NULL) {
  466. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  467. return FAILURE;
  468. }
  469. ALLOC_ZVAL(*retval);
  470. prevsib = nodep->prev;
  471. if (!prevsib) {
  472. ZVAL_NULL(*retval);
  473. return SUCCESS;
  474. }
  475. if (NULL == (*retval = php_dom_create_object(prevsib, &ret, NULL, *retval, obj TSRMLS_CC))) {
  476. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot create required DOM object");
  477. return FAILURE;
  478. }
  479. return SUCCESS;
  480. }
  481. /* }}} */
  482. /* {{{ nextSibling DomNode
  483. readonly=yes
  484. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-6AC54C2F
  485. Since:
  486. */
  487. int dom_node_next_sibling_read(dom_object *obj, zval **retval TSRMLS_DC)
  488. {
  489. xmlNode *nodep, *nextsib;
  490. int ret;
  491. nodep = dom_object_get_node(obj);
  492. if (nodep == NULL) {
  493. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  494. return FAILURE;
  495. }
  496. nextsib = nodep->next;
  497. if (!nextsib) {
  498. return FAILURE;
  499. }
  500. ALLOC_ZVAL(*retval);
  501. if (NULL == (*retval = php_dom_create_object(nextsib, &ret, NULL, *retval, obj TSRMLS_CC))) {
  502. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot create required DOM object");
  503. return FAILURE;
  504. }
  505. return SUCCESS;
  506. }
  507. /* }}} */
  508. /* {{{ attributes DomNamedNodeMap
  509. readonly=yes
  510. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-84CF096
  511. Since:
  512. */
  513. int dom_node_attributes_read(dom_object *obj, zval **retval TSRMLS_DC)
  514. {
  515. xmlNode *nodep;
  516. dom_object *intern;
  517. nodep = dom_object_get_node(obj);
  518. if (nodep == NULL) {
  519. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  520. return FAILURE;
  521. }
  522. ALLOC_ZVAL(*retval);
  523. if (nodep->type == XML_ELEMENT_NODE) {
  524. php_dom_create_interator(*retval, DOM_NAMEDNODEMAP TSRMLS_CC);
  525. intern = (dom_object *)zend_objects_get_address(*retval TSRMLS_CC);
  526. dom_namednode_iter(obj, XML_ATTRIBUTE_NODE, intern, NULL, NULL, NULL TSRMLS_CC);
  527. } else {
  528. ZVAL_NULL(*retval);
  529. }
  530. return SUCCESS;
  531. }
  532. /* }}} */
  533. /* {{{ ownerDocument DomDocument
  534. readonly=yes
  535. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-node-ownerDoc
  536. Since:
  537. */
  538. int dom_node_owner_document_read(dom_object *obj, zval **retval TSRMLS_DC)
  539. {
  540. xmlNode *nodep;
  541. xmlDocPtr docp;
  542. int ret;
  543. nodep = dom_object_get_node(obj);
  544. if (nodep == NULL) {
  545. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  546. return FAILURE;
  547. }
  548. if (nodep->type == XML_DOCUMENT_NODE || nodep->type == XML_HTML_DOCUMENT_NODE) {
  549. ALLOC_ZVAL(*retval);
  550. ZVAL_NULL(*retval);
  551. return SUCCESS;
  552. }
  553. docp = nodep->doc;
  554. if (!docp) {
  555. return FAILURE;
  556. }
  557. ALLOC_ZVAL(*retval);
  558. if (NULL == (*retval = php_dom_create_object((xmlNodePtr) docp, &ret, NULL, *retval, obj TSRMLS_CC))) {
  559. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot create required DOM object");
  560. return FAILURE;
  561. }
  562. return SUCCESS;
  563. }
  564. /* }}} */
  565. /* {{{ namespaceUri string
  566. readonly=yes
  567. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-NodeNSname
  568. Since: DOM Level 2
  569. */
  570. int dom_node_namespace_uri_read(dom_object *obj, zval **retval TSRMLS_DC)
  571. {
  572. xmlNode *nodep;
  573. char *str = NULL;
  574. nodep = dom_object_get_node(obj);
  575. if (nodep == NULL) {
  576. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  577. return FAILURE;
  578. }
  579. switch (nodep->type) {
  580. case XML_ELEMENT_NODE:
  581. case XML_ATTRIBUTE_NODE:
  582. case XML_NAMESPACE_DECL:
  583. if (nodep->ns != NULL) {
  584. str = (char *) nodep->ns->href;
  585. }
  586. break;
  587. default:
  588. str = NULL;
  589. break;
  590. }
  591. ALLOC_ZVAL(*retval);
  592. if(str != NULL) {
  593. ZVAL_STRING(*retval, str, 1);
  594. } else {
  595. ZVAL_NULL(*retval);
  596. }
  597. return SUCCESS;
  598. }
  599. /* }}} */
  600. /* {{{ prefix string
  601. readonly=no
  602. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-NodeNSPrefix
  603. Since: DOM Level 2
  604. */
  605. int dom_node_prefix_read(dom_object *obj, zval **retval TSRMLS_DC)
  606. {
  607. xmlNode *nodep;
  608. xmlNsPtr ns;
  609. char *str = NULL;
  610. nodep = dom_object_get_node(obj);
  611. if (nodep == NULL) {
  612. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  613. return FAILURE;
  614. }
  615. switch (nodep->type) {
  616. case XML_ELEMENT_NODE:
  617. case XML_ATTRIBUTE_NODE:
  618. case XML_NAMESPACE_DECL:
  619. ns = nodep->ns;
  620. if (ns != NULL && ns->prefix) {
  621. str = (char *) ns->prefix;
  622. }
  623. break;
  624. default:
  625. str = NULL;
  626. break;
  627. }
  628. ALLOC_ZVAL(*retval);
  629. if (str == NULL) {
  630. ZVAL_EMPTY_STRING(*retval);
  631. } else {
  632. ZVAL_STRING(*retval, str, 1);
  633. }
  634. return SUCCESS;
  635. }
  636. int dom_node_prefix_write(dom_object *obj, zval *newval TSRMLS_DC)
  637. {
  638. zval value_copy;
  639. xmlNode *nodep, *nsnode = NULL;
  640. xmlNsPtr ns = NULL, curns;
  641. char *strURI;
  642. char *prefix;
  643. nodep = dom_object_get_node(obj);
  644. if (nodep == NULL) {
  645. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  646. return FAILURE;
  647. }
  648. switch (nodep->type) {
  649. case XML_ELEMENT_NODE:
  650. nsnode = nodep;
  651. case XML_ATTRIBUTE_NODE:
  652. if (nsnode == NULL) {
  653. nsnode = nodep->parent;
  654. if (nsnode == NULL) {
  655. nsnode = xmlDocGetRootElement(nodep->doc);
  656. }
  657. }
  658. if (newval->type != IS_STRING) {
  659. if(Z_REFCOUNT_P(newval) > 1) {
  660. value_copy = *newval;
  661. zval_copy_ctor(&value_copy);
  662. newval = &value_copy;
  663. }
  664. convert_to_string(newval);
  665. }
  666. prefix = Z_STRVAL_P(newval);
  667. if (nsnode && nodep->ns != NULL && !xmlStrEqual(nodep->ns->prefix, (xmlChar *)prefix)) {
  668. strURI = (char *) nodep->ns->href;
  669. if (strURI == NULL ||
  670. (!strcmp (prefix, "xml") && strcmp(strURI, XML_XML_NAMESPACE)) ||
  671. (nodep->type == XML_ATTRIBUTE_NODE && !strcmp (prefix, "xmlns") &&
  672. strcmp (strURI, DOM_XMLNS_NAMESPACE)) ||
  673. (nodep->type == XML_ATTRIBUTE_NODE && !strcmp (nodep->name, "xmlns"))) {
  674. ns = NULL;
  675. } else {
  676. curns = nsnode->nsDef;
  677. while (curns != NULL) {
  678. if (xmlStrEqual((xmlChar *)prefix, curns->prefix) && xmlStrEqual(nodep->ns->href, curns->href)) {
  679. ns = curns;
  680. break;
  681. }
  682. curns = curns->next;
  683. }
  684. if (ns == NULL) {
  685. ns = xmlNewNs(nsnode, nodep->ns->href, (xmlChar *)prefix);
  686. }
  687. }
  688. if (ns == NULL) {
  689. if (newval == &value_copy) {
  690. zval_dtor(newval);
  691. }
  692. php_dom_throw_error(NAMESPACE_ERR, dom_get_strict_error(obj->document) TSRMLS_CC);
  693. return FAILURE;
  694. }
  695. xmlSetNs(nodep, ns);
  696. }
  697. if (newval == &value_copy) {
  698. zval_dtor(newval);
  699. }
  700. break;
  701. default:
  702. break;
  703. }
  704. return SUCCESS;
  705. }
  706. /* }}} */
  707. /* {{{ localName string
  708. readonly=yes
  709. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-NodeNSLocalN
  710. Since: DOM Level 2
  711. */
  712. int dom_node_local_name_read(dom_object *obj, zval **retval TSRMLS_DC)
  713. {
  714. xmlNode *nodep;
  715. nodep = dom_object_get_node(obj);
  716. if (nodep == NULL) {
  717. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  718. return FAILURE;
  719. }
  720. ALLOC_ZVAL(*retval);
  721. if (nodep->type == XML_ELEMENT_NODE || nodep->type == XML_ATTRIBUTE_NODE || nodep->type == XML_NAMESPACE_DECL) {
  722. ZVAL_STRING(*retval, (char *) (nodep->name), 1);
  723. } else {
  724. ZVAL_NULL(*retval);
  725. }
  726. return SUCCESS;
  727. }
  728. /* }}} */
  729. /* {{{ baseURI string
  730. readonly=yes
  731. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-baseURI
  732. Since: DOM Level 3
  733. */
  734. int dom_node_base_uri_read(dom_object *obj, zval **retval TSRMLS_DC)
  735. {
  736. xmlNode *nodep;
  737. xmlChar *baseuri;
  738. nodep = dom_object_get_node(obj);
  739. if (nodep == NULL) {
  740. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  741. return FAILURE;
  742. }
  743. ALLOC_ZVAL(*retval);
  744. baseuri = xmlNodeGetBase(nodep->doc, nodep);
  745. if (baseuri) {
  746. ZVAL_STRING(*retval, (char *) (baseuri), 1);
  747. xmlFree(baseuri);
  748. } else {
  749. ZVAL_NULL(*retval);
  750. }
  751. return SUCCESS;
  752. }
  753. /* }}} */
  754. /* {{{ textContent string
  755. readonly=no
  756. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-textContent
  757. Since: DOM Level 3
  758. */
  759. int dom_node_text_content_read(dom_object *obj, zval **retval TSRMLS_DC)
  760. {
  761. xmlNode *nodep;
  762. char *str = NULL;
  763. nodep = dom_object_get_node(obj);
  764. if (nodep == NULL) {
  765. php_dom_throw_error(INVALID_STATE_ERR, 0 TSRMLS_CC);
  766. return FAILURE;
  767. }
  768. str = xmlNodeGetContent(nodep);
  769. ALLOC_ZVAL(*retval);
  770. if(str != NULL) {
  771. ZVAL_STRING(*retval, str, 1);
  772. xmlFree(str);
  773. } else {
  774. ZVAL_EMPTY_STRING(*retval);
  775. }
  776. return SUCCESS;
  777. }
  778. int dom_node_text_content_write(dom_object *obj, zval *newval TSRMLS_DC)
  779. {
  780. return SUCCESS;
  781. }
  782. /* }}} */
  783. static xmlNodePtr _php_dom_insert_fragment(xmlNodePtr nodep, xmlNodePtr prevsib, xmlNodePtr nextsib, xmlNodePtr fragment, dom_object *intern, dom_object *childobj TSRMLS_DC) /* {{{ */
  784. {
  785. xmlNodePtr newchild, node;
  786. newchild = fragment->children;
  787. if (newchild) {
  788. if (prevsib == NULL) {
  789. nodep->children = newchild;
  790. } else {
  791. prevsib->next = newchild;
  792. }
  793. newchild->prev = prevsib;
  794. if (nextsib == NULL) {
  795. nodep->last = fragment->last;
  796. } else {
  797. fragment->last->next = nextsib;
  798. nextsib->prev = fragment->last;
  799. }
  800. node = newchild;
  801. while (node != NULL) {
  802. node->parent = nodep;
  803. if (node->doc != nodep->doc) {
  804. xmlSetTreeDoc(node, nodep->doc);
  805. if (node->_private != NULL) {
  806. childobj = node->_private;
  807. childobj->document = intern->document;
  808. php_libxml_increment_doc_ref((php_libxml_node_object *)childobj, NULL TSRMLS_CC);
  809. }
  810. }
  811. if (node == fragment->last) {
  812. break;
  813. }
  814. node = node->next;
  815. }
  816. fragment->children = NULL;
  817. fragment->last = NULL;
  818. }
  819. return newchild;
  820. }
  821. /* }}} */
  822. /* {{{ proto domnode dom_node_insert_before(DomNode newChild, DomNode refChild);
  823. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-952280727
  824. Since:
  825. */
  826. PHP_FUNCTION(dom_node_insert_before)
  827. {
  828. zval *id, *node, *ref = NULL, *rv = NULL;
  829. xmlNodePtr child, new_child, parentp, refp;
  830. dom_object *intern, *childobj, *refpobj;
  831. int ret, stricterror;
  832. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "OO|O!", &id, dom_node_class_entry, &node, dom_node_class_entry, &ref, dom_node_class_entry) == FAILURE) {
  833. return;
  834. }
  835. DOM_GET_OBJ(parentp, id, xmlNodePtr, intern);
  836. if (dom_node_children_valid(parentp) == FAILURE) {
  837. RETURN_FALSE;
  838. }
  839. DOM_GET_OBJ(child, node, xmlNodePtr, childobj);
  840. new_child = NULL;
  841. stricterror = dom_get_strict_error(intern->document);
  842. if (dom_node_is_read_only(parentp) == SUCCESS ||
  843. (child->parent != NULL && dom_node_is_read_only(child->parent) == SUCCESS)) {
  844. php_dom_throw_error(NO_MODIFICATION_ALLOWED_ERR, stricterror TSRMLS_CC);
  845. RETURN_FALSE;
  846. }
  847. if (dom_hierarchy(parentp, child) == FAILURE) {
  848. php_dom_throw_error(HIERARCHY_REQUEST_ERR, stricterror TSRMLS_CC);
  849. RETURN_FALSE;
  850. }
  851. if (child->doc != parentp->doc && child->doc != NULL) {
  852. php_dom_throw_error(WRONG_DOCUMENT_ERR, stricterror TSRMLS_CC);
  853. RETURN_FALSE;
  854. }
  855. if (child->type == XML_DOCUMENT_FRAG_NODE && child->children == NULL) {
  856. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Document Fragment is empty");
  857. RETURN_FALSE;
  858. }
  859. if (child->doc == NULL && parentp->doc != NULL) {
  860. childobj->document = intern->document;
  861. php_libxml_increment_doc_ref((php_libxml_node_object *)childobj, NULL TSRMLS_CC);
  862. }
  863. if (ref != NULL) {
  864. DOM_GET_OBJ(refp, ref, xmlNodePtr, refpobj);
  865. if (refp->parent != parentp) {
  866. php_dom_throw_error(NOT_FOUND_ERR, stricterror TSRMLS_CC);
  867. RETURN_FALSE;
  868. }
  869. if (child->parent != NULL) {
  870. xmlUnlinkNode(child);
  871. }
  872. if (child->type == XML_TEXT_NODE && (refp->type == XML_TEXT_NODE ||
  873. (refp->prev != NULL && refp->prev->type == XML_TEXT_NODE))) {
  874. if (child->doc == NULL) {
  875. xmlSetTreeDoc(child, parentp->doc);
  876. }
  877. new_child = child;
  878. new_child->parent = refp->parent;
  879. new_child->next = refp;
  880. new_child->prev = refp->prev;
  881. refp->prev = new_child;
  882. if (new_child->prev != NULL) {
  883. new_child->prev->next = new_child;
  884. }
  885. if (new_child->parent != NULL) {
  886. if (new_child->parent->children == refp) {
  887. new_child->parent->children = new_child;
  888. }
  889. }
  890. } else if (child->type == XML_ATTRIBUTE_NODE) {
  891. xmlAttrPtr lastattr;
  892. if (child->ns == NULL)
  893. lastattr = xmlHasProp(refp->parent, child->name);
  894. else
  895. lastattr = xmlHasNsProp(refp->parent, child->name, child->ns->href);
  896. if (lastattr != NULL && lastattr->type != XML_ATTRIBUTE_DECL) {
  897. if (lastattr != (xmlAttrPtr) child) {
  898. xmlUnlinkNode((xmlNodePtr) lastattr);
  899. php_libxml_node_free_resource((xmlNodePtr) lastattr TSRMLS_CC);
  900. } else {
  901. DOM_RET_OBJ(rv, child, &ret, intern);
  902. return;
  903. }
  904. }
  905. } else if (child->type == XML_DOCUMENT_FRAG_NODE) {
  906. new_child = _php_dom_insert_fragment(parentp, refp->prev, refp, child, intern, childobj TSRMLS_CC);
  907. }
  908. if (new_child == NULL) {
  909. new_child = xmlAddPrevSibling(refp, child);
  910. }
  911. } else {
  912. if (child->parent != NULL){
  913. xmlUnlinkNode(child);
  914. }
  915. if (child->type == XML_TEXT_NODE && parentp->last != NULL && parentp->last->type == XML_TEXT_NODE) {
  916. child->parent = parentp;
  917. if (child->doc == NULL) {
  918. xmlSetTreeDoc(child, parentp->doc);
  919. }
  920. new_child = child;
  921. if (parentp->children == NULL) {
  922. parentp->children = child;
  923. parentp->last = child;
  924. } else {
  925. child = parentp->last;
  926. child->next = new_child;
  927. new_child->prev = child;
  928. parentp->last = new_child;
  929. }
  930. } else if (child->type == XML_ATTRIBUTE_NODE) {
  931. xmlAttrPtr lastattr;
  932. if (child->ns == NULL)
  933. lastattr = xmlHasProp(parentp, child->name);
  934. else
  935. lastattr = xmlHasNsProp(parentp, child->name, child->ns->href);
  936. if (lastattr != NULL && lastattr->type != XML_ATTRIBUTE_DECL) {
  937. if (lastattr != (xmlAttrPtr) child) {
  938. xmlUnlinkNode((xmlNodePtr) lastattr);
  939. php_libxml_node_free_resource((xmlNodePtr) lastattr TSRMLS_CC);
  940. } else {
  941. DOM_RET_OBJ(rv, child, &ret, intern);
  942. return;
  943. }
  944. }
  945. } else if (child->type == XML_DOCUMENT_FRAG_NODE) {
  946. new_child = _php_dom_insert_fragment(parentp, parentp->last, NULL, child, intern, childobj TSRMLS_CC);
  947. }
  948. if (new_child == NULL) {
  949. new_child = xmlAddChild(parentp, child);
  950. }
  951. }
  952. if (NULL == new_child) {
  953. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Couldn't add newnode as the previous sibling of refnode");
  954. RETURN_FALSE;
  955. }
  956. dom_reconcile_ns(parentp->doc, new_child);
  957. DOM_RET_OBJ(rv, new_child, &ret, intern);
  958. }
  959. /* }}} end dom_node_insert_before */
  960. /* {{{ proto DomNode dom_node_replace_child(DomNode newChild, DomNode oldChild);
  961. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-785887307
  962. Since:
  963. */
  964. PHP_FUNCTION(dom_node_replace_child)
  965. {
  966. zval *id, *newnode, *oldnode;
  967. xmlNodePtr children, newchild, oldchild, nodep;
  968. dom_object *intern, *newchildobj, *oldchildobj;
  969. int foundoldchild = 0, stricterror;
  970. int ret;
  971. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "OOO", &id, dom_node_class_entry, &newnode, dom_node_class_entry, &oldnode, dom_node_class_entry) == FAILURE) {
  972. return;
  973. }
  974. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  975. if (dom_node_children_valid(nodep) == FAILURE) {
  976. RETURN_FALSE;
  977. }
  978. DOM_GET_OBJ(newchild, newnode, xmlNodePtr, newchildobj);
  979. DOM_GET_OBJ(oldchild, oldnode, xmlNodePtr, oldchildobj);
  980. children = nodep->children;
  981. if (!children) {
  982. RETURN_FALSE;
  983. }
  984. stricterror = dom_get_strict_error(intern->document);
  985. if (dom_node_is_read_only(nodep) == SUCCESS ||
  986. (newchild->parent != NULL && dom_node_is_read_only(newchild->parent) == SUCCESS)) {
  987. php_dom_throw_error(NO_MODIFICATION_ALLOWED_ERR, stricterror TSRMLS_CC);
  988. RETURN_FALSE;
  989. }
  990. if (newchild->doc != nodep->doc && newchild->doc != NULL) {
  991. php_dom_throw_error(WRONG_DOCUMENT_ERR, stricterror TSRMLS_CC);
  992. RETURN_FALSE;
  993. }
  994. if (dom_hierarchy(nodep, newchild) == FAILURE) {
  995. php_dom_throw_error(HIERARCHY_REQUEST_ERR, stricterror TSRMLS_CC);
  996. RETURN_FALSE;
  997. }
  998. /* check for the old child and whether the new child is already a child */
  999. while (children) {
  1000. if (children == oldchild) {
  1001. foundoldchild = 1;
  1002. break;
  1003. }
  1004. children = children->next;
  1005. }
  1006. if (foundoldchild) {
  1007. xmlNodePtr node;
  1008. zval *rv = NULL;
  1009. if (newchild->type == XML_DOCUMENT_FRAG_NODE) {
  1010. xmlNodePtr prevsib, nextsib;
  1011. prevsib = oldchild->prev;
  1012. nextsib = oldchild->next;
  1013. xmlUnlinkNode(oldchild);
  1014. newchild = _php_dom_insert_fragment(nodep, prevsib, nextsib, newchild, intern, newchildobj TSRMLS_CC);
  1015. if (newchild) {
  1016. dom_reconcile_ns(nodep->doc, newchild);
  1017. }
  1018. } else if (oldchild != newchild) {
  1019. if (newchild->doc == NULL && nodep->doc != NULL) {
  1020. xmlSetTreeDoc(newchild, nodep->doc);
  1021. newchildobj->document = intern->document;
  1022. php_libxml_increment_doc_ref((php_libxml_node_object *)newchildobj, NULL TSRMLS_CC);
  1023. }
  1024. node = xmlReplaceNode(oldchild, newchild);
  1025. dom_reconcile_ns(nodep->doc, newchild);
  1026. }
  1027. DOM_RET_OBJ(rv, oldchild, &ret, intern);
  1028. return;
  1029. } else {
  1030. php_dom_throw_error(NOT_FOUND_ERR, dom_get_strict_error(intern->document) TSRMLS_CC);
  1031. RETURN_FALSE;
  1032. }
  1033. }
  1034. /* }}} end dom_node_replace_child */
  1035. /* {{{ proto DomNode dom_node_remove_child(DomNode oldChild);
  1036. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-1734834066
  1037. Since:
  1038. */
  1039. PHP_FUNCTION(dom_node_remove_child)
  1040. {
  1041. zval *id, *node;
  1042. xmlNodePtr children, child, nodep;
  1043. dom_object *intern, *childobj;
  1044. int ret, stricterror;
  1045. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "OO", &id, dom_node_class_entry, &node, dom_node_class_entry) == FAILURE) {
  1046. return;
  1047. }
  1048. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  1049. if (dom_node_children_valid(nodep) == FAILURE) {
  1050. RETURN_FALSE;
  1051. }
  1052. DOM_GET_OBJ(child, node, xmlNodePtr, childobj);
  1053. stricterror = dom_get_strict_error(intern->document);
  1054. if (dom_node_is_read_only(nodep) == SUCCESS ||
  1055. (child->parent != NULL && dom_node_is_read_only(child->parent) == SUCCESS)) {
  1056. php_dom_throw_error(NO_MODIFICATION_ALLOWED_ERR, stricterror TSRMLS_CC);
  1057. RETURN_FALSE;
  1058. }
  1059. children = nodep->children;
  1060. if (!children) {
  1061. php_dom_throw_error(NOT_FOUND_ERR, stricterror TSRMLS_CC);
  1062. RETURN_FALSE;
  1063. }
  1064. while (children) {
  1065. if (children == child) {
  1066. zval *rv = NULL;
  1067. xmlUnlinkNode(child);
  1068. DOM_RET_OBJ(rv, child, &ret, intern);
  1069. return;
  1070. }
  1071. children = children->next;
  1072. }
  1073. php_dom_throw_error(NOT_FOUND_ERR, stricterror TSRMLS_CC);
  1074. RETURN_FALSE
  1075. }
  1076. /* }}} end dom_node_remove_child */
  1077. /* {{{ proto DomNode dom_node_append_child(DomNode newChild);
  1078. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-184E7107
  1079. Since:
  1080. */
  1081. PHP_FUNCTION(dom_node_append_child)
  1082. {
  1083. zval *id, *node, *rv = NULL;
  1084. xmlNodePtr child, nodep, new_child = NULL;
  1085. dom_object *intern, *childobj;
  1086. int ret, stricterror;
  1087. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "OO", &id, dom_node_class_entry, &node, dom_node_class_entry) == FAILURE) {
  1088. return;
  1089. }
  1090. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  1091. if (dom_node_children_valid(nodep) == FAILURE) {
  1092. RETURN_FALSE;
  1093. }
  1094. DOM_GET_OBJ(child, node, xmlNodePtr, childobj);
  1095. stricterror = dom_get_strict_error(intern->document);
  1096. if (dom_node_is_read_only(nodep) == SUCCESS ||
  1097. (child->parent != NULL && dom_node_is_read_only(child->parent) == SUCCESS)) {
  1098. php_dom_throw_error(NO_MODIFICATION_ALLOWED_ERR, stricterror TSRMLS_CC);
  1099. RETURN_FALSE;
  1100. }
  1101. if (dom_hierarchy(nodep, child) == FAILURE) {
  1102. php_dom_throw_error(HIERARCHY_REQUEST_ERR, stricterror TSRMLS_CC);
  1103. RETURN_FALSE;
  1104. }
  1105. if (!(child->doc == NULL || child->doc == nodep->doc)) {
  1106. php_dom_throw_error(WRONG_DOCUMENT_ERR, stricterror TSRMLS_CC);
  1107. RETURN_FALSE;
  1108. }
  1109. if (child->type == XML_DOCUMENT_FRAG_NODE && child->children == NULL) {
  1110. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Document Fragment is empty");
  1111. RETURN_FALSE;
  1112. }
  1113. if (child->doc == NULL && nodep->doc != NULL) {
  1114. childobj->document = intern->document;
  1115. php_libxml_increment_doc_ref((php_libxml_node_object *)childobj, NULL TSRMLS_CC);
  1116. }
  1117. if (child->parent != NULL){
  1118. xmlUnlinkNode(child);
  1119. }
  1120. if (child->type == XML_TEXT_NODE && nodep->last != NULL && nodep->last->type == XML_TEXT_NODE) {
  1121. child->parent = nodep;
  1122. if (child->doc == NULL) {
  1123. xmlSetTreeDoc(child, nodep->doc);
  1124. }
  1125. new_child = child;
  1126. if (nodep->children == NULL) {
  1127. nodep->children = child;
  1128. nodep->last = child;
  1129. } else {
  1130. child = nodep->last;
  1131. child->next = new_child;
  1132. new_child->prev = child;
  1133. nodep->last = new_child;
  1134. }
  1135. } else if (child->type == XML_ATTRIBUTE_NODE) {
  1136. xmlAttrPtr lastattr;
  1137. if (child->ns == NULL)
  1138. lastattr = xmlHasProp(nodep, child->name);
  1139. else
  1140. lastattr = xmlHasNsProp(nodep, child->name, child->ns->href);
  1141. if (lastattr != NULL && lastattr->type != XML_ATTRIBUTE_DECL) {
  1142. if (lastattr != (xmlAttrPtr) child) {
  1143. xmlUnlinkNode((xmlNodePtr) lastattr);
  1144. php_libxml_node_free_resource((xmlNodePtr) lastattr TSRMLS_CC);
  1145. }
  1146. }
  1147. } else if (child->type == XML_DOCUMENT_FRAG_NODE) {
  1148. new_child = _php_dom_insert_fragment(nodep, nodep->last, NULL, child, intern, childobj TSRMLS_CC);
  1149. }
  1150. if (new_child == NULL) {
  1151. new_child = xmlAddChild(nodep, child);
  1152. if (new_child == NULL) {
  1153. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Couldn't append node");
  1154. RETURN_FALSE;
  1155. }
  1156. }
  1157. dom_reconcile_ns(nodep->doc, new_child);
  1158. DOM_RET_OBJ(rv, new_child, &ret, intern);
  1159. }
  1160. /* }}} end dom_node_append_child */
  1161. /* {{{ proto boolean dom_node_has_child_nodes();
  1162. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-810594187
  1163. Since:
  1164. */
  1165. PHP_FUNCTION(dom_node_has_child_nodes)
  1166. {
  1167. zval *id;
  1168. xmlNode *nodep;
  1169. dom_object *intern;
  1170. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O", &id, dom_node_class_entry) == FAILURE) {
  1171. return;
  1172. }
  1173. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  1174. if (dom_node_children_valid(nodep) == FAILURE) {
  1175. RETURN_FALSE;
  1176. }
  1177. if (nodep->children) {
  1178. RETURN_TRUE;
  1179. } else {
  1180. RETURN_FALSE;
  1181. }
  1182. }
  1183. /* }}} end dom_node_has_child_nodes */
  1184. /* {{{ proto DomNode dom_node_clone_node(boolean deep);
  1185. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-3A0ED0A4
  1186. Since:
  1187. */
  1188. PHP_FUNCTION(dom_node_clone_node)
  1189. {
  1190. zval *rv = NULL;
  1191. zval *id;
  1192. xmlNode *n, *node;
  1193. int ret;
  1194. dom_object *intern;
  1195. long recursive = 0;
  1196. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O|l", &id, dom_node_class_entry, &recursive) == FAILURE) {
  1197. return;
  1198. }
  1199. DOM_GET_OBJ(n, id, xmlNodePtr, intern);
  1200. node = xmlDocCopyNode(n, n->doc, recursive);
  1201. if (!node) {
  1202. RETURN_FALSE;
  1203. }
  1204. /* When deep is false Element nodes still require the attributes
  1205. Following taken from libxml as xmlDocCopyNode doesnt do this */
  1206. if (n->type == XML_ELEMENT_NODE && recursive == 0) {
  1207. if (n->nsDef != NULL) {
  1208. node->nsDef = xmlCopyNamespaceList(n->nsDef);
  1209. }
  1210. if (n->ns != NULL) {
  1211. xmlNsPtr ns;
  1212. ns = xmlSearchNs(n->doc, node, n->ns->prefix);
  1213. if (ns == NULL) {
  1214. ns = xmlSearchNs(n->doc, n, n->ns->prefix);
  1215. if (ns != NULL) {
  1216. xmlNodePtr root = node;
  1217. while (root->parent != NULL) {
  1218. root = root->parent;
  1219. }
  1220. node->ns = xmlNewNs(root, ns->href, ns->prefix);
  1221. }
  1222. } else {
  1223. node->ns = ns;
  1224. }
  1225. }
  1226. if (n->properties != NULL) {
  1227. node->properties = xmlCopyPropList(node, n->properties);
  1228. }
  1229. }
  1230. /* If document cloned we want a new document proxy */
  1231. if (node->doc != n->doc) {
  1232. intern = NULL;
  1233. }
  1234. DOM_RET_OBJ(rv, node, &ret, intern);
  1235. }
  1236. /* }}} end dom_node_clone_node */
  1237. /* {{{ proto void dom_node_normalize();
  1238. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-normalize
  1239. Since:
  1240. */
  1241. PHP_FUNCTION(dom_node_normalize)
  1242. {
  1243. zval *id;
  1244. xmlNode *nodep;
  1245. dom_object *intern;
  1246. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O", &id, dom_node_class_entry) == FAILURE) {
  1247. return;
  1248. }
  1249. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  1250. dom_normalize(nodep TSRMLS_CC);
  1251. }
  1252. /* }}} end dom_node_normalize */
  1253. /* {{{ proto boolean dom_node_is_supported(string feature, string version);
  1254. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-Level-2-Core-Node-supports
  1255. Since: DOM Level 2
  1256. */
  1257. PHP_FUNCTION(dom_node_is_supported)
  1258. {
  1259. zval *id;
  1260. int feature_len, version_len;
  1261. char *feature, *version;
  1262. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Oss", &id, dom_node_class_entry, &feature, &feature_len, &version, &version_len) == FAILURE) {
  1263. return;
  1264. }
  1265. if (dom_has_feature(feature, version)) {
  1266. RETURN_TRUE;
  1267. } else {
  1268. RETURN_FALSE;
  1269. }
  1270. }
  1271. /* }}} end dom_node_is_supported */
  1272. /* {{{ proto boolean dom_node_has_attributes();
  1273. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#core-ID-NodeHasAttrs
  1274. Since: DOM Level 2
  1275. */
  1276. PHP_FUNCTION(dom_node_has_attributes)
  1277. {
  1278. zval *id;
  1279. xmlNode *nodep;
  1280. dom_object *intern;
  1281. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O", &id, dom_node_class_entry) == FAILURE) {
  1282. return;
  1283. }
  1284. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  1285. if (nodep->type != XML_ELEMENT_NODE)
  1286. RETURN_FALSE;
  1287. if (nodep->properties) {
  1288. RETURN_TRUE;
  1289. } else {
  1290. RETURN_FALSE;
  1291. }
  1292. }
  1293. /* }}} end dom_node_has_attributes */
  1294. /* {{{ proto short dom_node_compare_document_position(DomNode other);
  1295. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-compareDocumentPosition
  1296. Since: DOM Level 3
  1297. */
  1298. PHP_FUNCTION(dom_node_compare_document_position)
  1299. {
  1300. DOM_NOT_IMPLEMENTED();
  1301. }
  1302. /* }}} end dom_node_compare_document_position */
  1303. /* {{{ proto boolean dom_node_is_same_node(DomNode other);
  1304. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-isSameNode
  1305. Since: DOM Level 3
  1306. */
  1307. PHP_FUNCTION(dom_node_is_same_node)
  1308. {
  1309. zval *id, *node;
  1310. xmlNodePtr nodeotherp, nodep;
  1311. dom_object *intern, *nodeotherobj;
  1312. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "OO", &id, dom_node_class_entry, &node, dom_node_class_entry) == FAILURE) {
  1313. return;
  1314. }
  1315. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  1316. DOM_GET_OBJ(nodeotherp, node, xmlNodePtr, nodeotherobj);
  1317. if (nodep == nodeotherp) {
  1318. RETURN_TRUE;
  1319. } else {
  1320. RETURN_FALSE;
  1321. }
  1322. }
  1323. /* }}} end dom_node_is_same_node */
  1324. /* {{{ proto string dom_node_lookup_prefix(string namespaceURI);
  1325. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-lookupNamespacePrefix
  1326. Since: DOM Level 3
  1327. */
  1328. PHP_FUNCTION(dom_node_lookup_prefix)
  1329. {
  1330. zval *id;
  1331. xmlNodePtr nodep, lookupp = NULL;
  1332. dom_object *intern;
  1333. xmlNsPtr nsptr;
  1334. int uri_len = 0;
  1335. char *uri;
  1336. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os", &id, dom_node_class_entry, &uri, &uri_len) == FAILURE) {
  1337. return;
  1338. }
  1339. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  1340. if (uri_len > 0) {
  1341. switch (nodep->type) {
  1342. case XML_ELEMENT_NODE:
  1343. lookupp = nodep;
  1344. break;
  1345. case XML_DOCUMENT_NODE:
  1346. case XML_HTML_DOCUMENT_NODE:
  1347. lookupp = xmlDocGetRootElement((xmlDocPtr) nodep);
  1348. break;
  1349. case XML_ENTITY_NODE :
  1350. case XML_NOTATION_NODE:
  1351. case XML_DOCUMENT_FRAG_NODE:
  1352. case XML_DOCUMENT_TYPE_NODE:
  1353. case XML_DTD_NODE:
  1354. RETURN_NULL();
  1355. break;
  1356. default:
  1357. lookupp = nodep->parent;
  1358. }
  1359. if (lookupp != NULL && (nsptr = xmlSearchNsByHref(lookupp->doc, lookupp, uri))) {
  1360. if (nsptr->prefix != NULL) {
  1361. RETURN_STRING((char *) nsptr->prefix, 1);
  1362. }
  1363. }
  1364. }
  1365. RETURN_NULL();
  1366. }
  1367. /* }}} end dom_node_lookup_prefix */
  1368. /* {{{ proto boolean dom_node_is_default_namespace(string namespaceURI);
  1369. URL: http://www.w3.org/TR/DOM-Level-3-Core/core.html#Node3-isDefaultNamespace
  1370. Since: DOM Level 3
  1371. */
  1372. PHP_FUNCTION(dom_node_is_default_namespace)
  1373. {
  1374. zval *id;
  1375. xmlNodePtr nodep;
  1376. dom_object *intern;
  1377. xmlNsPtr nsptr;
  1378. int uri_len = 0;
  1379. char *uri;
  1380. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os", &id, dom_node_class_entry, &uri, &uri_len) == FAILURE) {
  1381. return;
  1382. }
  1383. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  1384. if (nodep->type == XML_DOCUMENT_NODE || nodep->type == XML_HTML_DOCUMENT_NODE) {
  1385. nodep = xmlDocGetRootElement((xmlDocPtr) nodep);
  1386. }
  1387. if (nodep && uri_len > 0) {
  1388. nsptr = xmlSearchNs(nodep->doc, nodep, NULL);
  1389. if (nsptr && xmlStrEqual(nsptr->href, uri)) {
  1390. RETURN_TRUE;
  1391. }
  1392. }
  1393. RETURN_FALSE;
  1394. }
  1395. /* }}} end dom_node_is_default_namespace */
  1396. /* {{{ proto string dom_node_lookup_namespace_uri(string prefix);
  1397. URL: http://www.w3.org/TR/DOM-Level-3-Core/core.html#Node3-lookupNamespaceURI
  1398. Since: DOM Level 3
  1399. */
  1400. PHP_FUNCTION(dom_node_lookup_namespace_uri)
  1401. {
  1402. zval *id;
  1403. xmlNodePtr nodep;
  1404. dom_object *intern;
  1405. xmlNsPtr nsptr;
  1406. int prefix_len = 0;
  1407. char *prefix=NULL;
  1408. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "Os!", &id, dom_node_class_entry, &prefix, &prefix_len) == FAILURE) {
  1409. return;
  1410. }
  1411. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  1412. if (nodep->type == XML_DOCUMENT_NODE || nodep->type == XML_HTML_DOCUMENT_NODE) {
  1413. nodep = xmlDocGetRootElement((xmlDocPtr) nodep);
  1414. if (nodep == NULL) {
  1415. RETURN_NULL();
  1416. }
  1417. }
  1418. nsptr = xmlSearchNs(nodep->doc, nodep, prefix);
  1419. if (nsptr && nsptr->href != NULL) {
  1420. RETURN_STRING((char *) nsptr->href, 1);
  1421. }
  1422. RETURN_NULL();
  1423. }
  1424. /* }}} end dom_node_lookup_namespace_uri */
  1425. /* {{{ proto boolean dom_node_is_equal_node(DomNode arg);
  1426. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-isEqualNode
  1427. Since: DOM Level 3
  1428. */
  1429. PHP_FUNCTION(dom_node_is_equal_node)
  1430. {
  1431. DOM_NOT_IMPLEMENTED();
  1432. }
  1433. /* }}} end dom_node_is_equal_node */
  1434. /* {{{ proto DomNode dom_node_get_feature(string feature, string version);
  1435. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-getFeature
  1436. Since: DOM Level 3
  1437. */
  1438. PHP_FUNCTION(dom_node_get_feature)
  1439. {
  1440. DOM_NOT_IMPLEMENTED();
  1441. }
  1442. /* }}} end dom_node_get_feature */
  1443. /* {{{ proto DomUserData dom_node_set_user_data(string key, DomUserData data, userdatahandler handler);
  1444. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-setUserData
  1445. Since: DOM Level 3
  1446. */
  1447. PHP_FUNCTION(dom_node_set_user_data)
  1448. {
  1449. DOM_NOT_IMPLEMENTED();
  1450. }
  1451. /* }}} end dom_node_set_user_data */
  1452. /* {{{ proto DomUserData dom_node_get_user_data(string key);
  1453. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#Node3-getUserData
  1454. Since: DOM Level 3
  1455. */
  1456. PHP_FUNCTION(dom_node_get_user_data)
  1457. {
  1458. DOM_NOT_IMPLEMENTED();
  1459. }
  1460. /* }}} end dom_node_get_user_data */
  1461. static void dom_canonicalization(INTERNAL_FUNCTION_PARAMETERS, int mode) /* {{{ */
  1462. {
  1463. zval *id;
  1464. zval *xpath_array=NULL, *ns_prefixes=NULL;
  1465. xmlNodePtr nodep;
  1466. xmlDocPtr docp;
  1467. xmlNodeSetPtr nodeset = NULL;
  1468. dom_object *intern;
  1469. zend_bool exclusive=0, with_comments=0;
  1470. xmlChar **inclusive_ns_prefixes = NULL;
  1471. char *file = NULL;
  1472. int ret = -1, file_len = 0;
  1473. xmlOutputBufferPtr buf;
  1474. xmlXPathContextPtr ctxp=NULL;
  1475. xmlXPathObjectPtr xpathobjp=NULL;
  1476. if (mode == 0) {
  1477. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(),
  1478. "O|bba!a!", &id, dom_node_class_entry, &exclusive, &with_comments,
  1479. &xpath_array, &ns_prefixes) == FAILURE) {
  1480. return;
  1481. }
  1482. } else {
  1483. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(),
  1484. "Os|bba!a!", &id, dom_node_class_entry, &file, &file_len, &exclusive,
  1485. &with_comments, &xpath_array, &ns_prefixes) == FAILURE) {
  1486. return;
  1487. }
  1488. }
  1489. DOM_GET_OBJ(nodep, id, xmlNodePtr, intern);
  1490. docp = nodep->doc;
  1491. if (! docp) {
  1492. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Node must be associated with a document");
  1493. RETURN_FALSE;
  1494. }
  1495. if (xpath_array == NULL) {
  1496. if (nodep->type != XML_DOCUMENT_NODE) {
  1497. ctxp = xmlXPathNewContext(docp);
  1498. ctxp->node = nodep;
  1499. xpathobjp = xmlXPathEvalExpression("(.//. | .//@* | .//namespace::*)", ctxp);
  1500. ctxp->node = NULL;
  1501. if (xpathobjp && xpathobjp->type == XPATH_NODESET) {
  1502. nodeset = xpathobjp->nodesetval;
  1503. } else {
  1504. if (xpathobjp) {
  1505. xmlXPathFreeObject(xpathobjp);
  1506. }
  1507. xmlXPathFreeContext(ctxp);
  1508. php_error_docref(NULL TSRMLS_CC, E_WARNING, "XPath query did not return a nodeset.");
  1509. RETURN_FALSE;
  1510. }
  1511. }
  1512. } else {
  1513. /*xpath query from xpath_array */
  1514. HashTable *ht = Z_ARRVAL_P(xpath_array);
  1515. zval **tmp;
  1516. char *xquery;
  1517. if (zend_hash_find(ht, "query", sizeof("query"), (void**)&tmp) == SUCCESS &&
  1518. Z_TYPE_PP(tmp) == IS_STRING) {
  1519. xquery = Z_STRVAL_PP(tmp);
  1520. } else {
  1521. php_error_docref(NULL TSRMLS_CC, E_WARNING, "'query' missing from xpath array or is not a string");
  1522. RETURN_FALSE;
  1523. }
  1524. ctxp = xmlXPathNewContext(docp);
  1525. ctxp->node = nodep;
  1526. if (zend_hash_find(ht, "namespaces", sizeof("namespaces"), (void**)&tmp) == SUCCESS &&
  1527. Z_TYPE_PP(tmp) == IS_ARRAY) {
  1528. zval **tmpns;
  1529. while (zend_hash_get_current_data(Z_ARRVAL_PP(tmp), (void **)&tmpns) == SUCCESS) {
  1530. if (Z_TYPE_PP(tmpns) == IS_STRING) {
  1531. char *prefix;
  1532. ulong idx;
  1533. int prefix_key_len;
  1534. if (zend_hash_get_current_key_ex(Z_ARRVAL_PP(tmp),
  1535. &prefix, &prefix_key_len, &idx, 0, NULL) == HASH_KEY_IS_STRING) {
  1536. xmlXPathRegisterNs(ctxp, prefix, Z_STRVAL_PP(tmpns));
  1537. }
  1538. }
  1539. zend_hash_move_forward(Z_ARRVAL_PP(tmp));
  1540. }
  1541. }
  1542. xpathobjp = xmlXPathEvalExpression(xquery, ctxp);
  1543. ctxp->node = NULL;
  1544. if (xpathobjp && xpathobjp->type == XPATH_NODESET) {
  1545. nodeset = xpathobjp->nodesetval;
  1546. } else {
  1547. if (xpathobjp) {
  1548. xmlXPathFreeObject(xpathobjp);
  1549. }
  1550. xmlXPathFreeContext(ctxp);
  1551. php_error_docref(NULL TSRMLS_CC, E_WARNING, "XPath query did not return a nodeset.");
  1552. RETURN_FALSE;
  1553. }
  1554. }
  1555. if (ns_prefixes != NULL) {
  1556. if (exclusive) {
  1557. zval **tmpns;
  1558. int nscount = 0;
  1559. inclusive_ns_prefixes = safe_emalloc(zend_hash_num_elements(Z_ARRVAL_P(ns_prefixes)) + 1,
  1560. sizeof(xmlChar *), 0);
  1561. while (zend_hash_get_current_data(Z_ARRVAL_P(ns_prefixes), (void **)&tmpns) == SUCCESS) {
  1562. if (Z_TYPE_PP(tmpns) == IS_STRING) {
  1563. inclusive_ns_prefixes[nscount++] = Z_STRVAL_PP(tmpns);
  1564. }
  1565. zend_hash_move_forward(Z_ARRVAL_P(ns_prefixes));
  1566. }
  1567. inclusive_ns_prefixes[nscount] = NULL;
  1568. } else {
  1569. php_error_docref(NULL TSRMLS_CC, E_NOTICE,
  1570. "Inclusive namespace prefixes only allowed in exlcusive mode.");
  1571. }
  1572. }
  1573. if (mode == 1) {
  1574. buf = xmlOutputBufferCreateFilename(file, NULL, 0);
  1575. } else {
  1576. buf = xmlAllocOutputBuffer(NULL);
  1577. }
  1578. if (buf != NULL) {
  1579. ret = xmlC14NDocSaveTo(docp, nodeset, exclusive, inclusive_ns_prefixes,
  1580. with_comments, buf);
  1581. }
  1582. if (inclusive_ns_prefixes != NULL) {
  1583. efree(inclusive_ns_prefixes);
  1584. }
  1585. if (xpathobjp != NULL) {
  1586. xmlXPathFreeObject(xpathobjp);
  1587. }
  1588. if (ctxp != NULL) {
  1589. xmlXPathFreeContext(ctxp);
  1590. }
  1591. if (buf == NULL || ret < 0) {
  1592. RETVAL_FALSE;
  1593. } else {
  1594. if (mode == 0) {
  1595. ret = buf->buffer->use;
  1596. if (ret > 0) {
  1597. RETVAL_STRINGL((char *) buf->buffer->content, ret, 1);
  1598. } else {
  1599. RETVAL_EMPTY_STRING();
  1600. }
  1601. }
  1602. }
  1603. if (buf) {
  1604. int bytes;
  1605. bytes = xmlOutputBufferClose(buf);
  1606. if (mode == 1 && (ret >= 0)) {
  1607. RETURN_LONG(bytes);
  1608. }
  1609. }
  1610. }
  1611. /* }}} */
  1612. /* {{{ proto string DOMNode::C14N([bool exclusive [, bool with_comments [, array xpath [, array ns_prefixes]]]])
  1613. Canonicalize nodes to a string */
  1614. PHP_METHOD(domnode, C14N)
  1615. {
  1616. dom_canonicalization(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
  1617. }
  1618. /* }}} */
  1619. /* {{{ proto int DOMNode::C14NFile(string uri [, bool exclusive [, bool with_comments [, array xpath [, array ns_prefixes]]]])
  1620. Canonicalize nodes to a file */
  1621. PHP_METHOD(domnode, C14NFile)
  1622. {
  1623. dom_canonicalization(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
  1624. }
  1625. /* }}} */
  1626. /* {{{ proto int DOMNode::getNodePath()
  1627. Gets an xpath for a node */
  1628. PHP_METHOD(domnode, getNodePath)
  1629. {
  1630. zval *id;
  1631. xmlNode *nodep;
  1632. dom_object *intern;
  1633. char *value;
  1634. DOM_GET_THIS_OBJ(nodep, id, xmlNodePtr, intern);
  1635. value = xmlGetNodePath(nodep);
  1636. if (value == NULL) {
  1637. RETURN_NULL();
  1638. } else {
  1639. RETVAL_STRING(value, 1);
  1640. xmlFree(value);
  1641. }
  1642. }
  1643. /* }}} */
  1644. /* {{{ proto int DOMNode::getLineNo()
  1645. Gets line number for a node */
  1646. PHP_METHOD(domnode, getLineNo)
  1647. {
  1648. zval *id;
  1649. xmlNode *nodep;
  1650. dom_object *intern;
  1651. if (zend_parse_parameters_none() == FAILURE) {
  1652. return;
  1653. }
  1654. DOM_GET_THIS_OBJ(nodep, id, xmlNodePtr, intern);
  1655. RETURN_LONG(xmlGetLineNo(nodep));
  1656. }
  1657. /* }}} */
  1658. #endif
  1659. /*
  1660. * Local variables:
  1661. * tab-width: 4
  1662. * c-basic-offset: 4
  1663. * End:
  1664. * vim600: noet sw=4 ts=4 fdm=marker
  1665. * vim<600: noet sw=4 ts=4
  1666. */