node.c 40 KB

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