xsltprocessor.c 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 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. #ifdef HAVE_CONFIG_H
  20. #include "config.h"
  21. #endif
  22. #include "php.h"
  23. #include "php_xsl.h"
  24. #include "ext/libxml/php_libxml.h"
  25. /* {{{ arginfo */
  26. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_import_stylesheet, 0, 0, 1)
  27. ZEND_ARG_INFO(0, doc)
  28. ZEND_END_ARG_INFO();
  29. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_transform_to_doc, 0, 0, 1)
  30. ZEND_ARG_INFO(0, doc)
  31. ZEND_END_ARG_INFO();
  32. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_transform_to_uri, 0, 0, 2)
  33. ZEND_ARG_INFO(0, doc)
  34. ZEND_ARG_INFO(0, uri)
  35. ZEND_END_ARG_INFO();
  36. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_transform_to_xml, 0, 0, 1)
  37. ZEND_ARG_INFO(0, doc)
  38. ZEND_END_ARG_INFO();
  39. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_set_parameter, 0, 0, 2)
  40. ZEND_ARG_INFO(0, namespace)
  41. ZEND_ARG_INFO(0, name)
  42. ZEND_ARG_INFO(0, value)
  43. ZEND_END_ARG_INFO();
  44. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_get_parameter, 0, 0, 2)
  45. ZEND_ARG_INFO(0, namespace)
  46. ZEND_ARG_INFO(0, name)
  47. ZEND_END_ARG_INFO();
  48. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_remove_parameter, 0, 0, 2)
  49. ZEND_ARG_INFO(0, namespace)
  50. ZEND_ARG_INFO(0, name)
  51. ZEND_END_ARG_INFO();
  52. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_has_exslt_support, 0, 0, 0)
  53. ZEND_END_ARG_INFO();
  54. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_register_php_functions, 0, 0, 0)
  55. ZEND_ARG_INFO(0, restrict)
  56. ZEND_END_ARG_INFO();
  57. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_set_profiling, 0, 0, 1)
  58. ZEND_ARG_INFO(0, filename)
  59. ZEND_END_ARG_INFO();
  60. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_set_security_prefs, 0, 0, 1)
  61. ZEND_ARG_INFO(0, securityPrefs)
  62. ZEND_END_ARG_INFO();
  63. ZEND_BEGIN_ARG_INFO_EX(arginfo_xsl_xsltprocessor_get_security_prefs, 0, 0, 0)
  64. ZEND_END_ARG_INFO();
  65. /* }}} */
  66. /*
  67. * class xsl_xsltprocessor
  68. *
  69. * URL: https://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#
  70. * Since:
  71. */
  72. const zend_function_entry php_xsl_xsltprocessor_class_functions[] = {
  73. PHP_FALIAS(importStylesheet, xsl_xsltprocessor_import_stylesheet, arginfo_xsl_xsltprocessor_import_stylesheet)
  74. PHP_FALIAS(transformToDoc, xsl_xsltprocessor_transform_to_doc, arginfo_xsl_xsltprocessor_transform_to_doc)
  75. PHP_FALIAS(transformToUri, xsl_xsltprocessor_transform_to_uri, arginfo_xsl_xsltprocessor_transform_to_uri)
  76. PHP_FALIAS(transformToXml, xsl_xsltprocessor_transform_to_xml, arginfo_xsl_xsltprocessor_transform_to_xml)
  77. PHP_FALIAS(setParameter, xsl_xsltprocessor_set_parameter, arginfo_xsl_xsltprocessor_set_parameter)
  78. PHP_FALIAS(getParameter, xsl_xsltprocessor_get_parameter, arginfo_xsl_xsltprocessor_get_parameter)
  79. PHP_FALIAS(removeParameter, xsl_xsltprocessor_remove_parameter, arginfo_xsl_xsltprocessor_remove_parameter)
  80. PHP_FALIAS(hasExsltSupport, xsl_xsltprocessor_has_exslt_support, arginfo_xsl_xsltprocessor_has_exslt_support)
  81. PHP_FALIAS(registerPHPFunctions, xsl_xsltprocessor_register_php_functions, arginfo_xsl_xsltprocessor_register_php_functions)
  82. PHP_FALIAS(setProfiling, xsl_xsltprocessor_set_profiling, arginfo_xsl_xsltprocessor_set_profiling)
  83. PHP_FALIAS(setSecurityPrefs, xsl_xsltprocessor_set_security_prefs, arginfo_xsl_xsltprocessor_set_security_prefs)
  84. PHP_FALIAS(getSecurityPrefs, xsl_xsltprocessor_get_security_prefs, arginfo_xsl_xsltprocessor_get_security_prefs)
  85. PHP_FE_END
  86. };
  87. /* {{{ php_xsl_xslt_string_to_xpathexpr()
  88. Translates a string to a XPath Expression */
  89. static char *php_xsl_xslt_string_to_xpathexpr(const char *str)
  90. {
  91. const xmlChar *string = (const xmlChar *)str;
  92. xmlChar *value;
  93. int str_len;
  94. str_len = xmlStrlen(string) + 3;
  95. if (xmlStrchr(string, '"')) {
  96. if (xmlStrchr(string, '\'')) {
  97. php_error_docref(NULL, E_WARNING, "Cannot create XPath expression (string contains both quote and double-quotes)");
  98. return NULL;
  99. }
  100. value = (xmlChar*) safe_emalloc (str_len, sizeof(xmlChar), 0);
  101. snprintf((char*)value, str_len, "'%s'", string);
  102. } else {
  103. value = (xmlChar*) safe_emalloc (str_len, sizeof(xmlChar), 0);
  104. snprintf((char *)value, str_len, "\"%s\"", string);
  105. }
  106. return (char *) value;
  107. }
  108. /* }}} */
  109. /* {{{ php_xsl_xslt_make_params()
  110. Translates a PHP array to a libxslt parameters array */
  111. static char **php_xsl_xslt_make_params(HashTable *parht, int xpath_params)
  112. {
  113. int parsize;
  114. zval *value;
  115. char *xpath_expr;
  116. zend_string *string_key;
  117. char **params = NULL;
  118. int i = 0;
  119. parsize = (2 * zend_hash_num_elements(parht) + 1) * sizeof(char *);
  120. params = (char **)safe_emalloc((2 * zend_hash_num_elements(parht) + 1), sizeof(char *), 0);
  121. memset((char *)params, 0, parsize);
  122. ZEND_HASH_FOREACH_STR_KEY_VAL(parht, string_key, value) {
  123. if (string_key == NULL) {
  124. php_error_docref(NULL, E_WARNING, "Invalid argument or parameter array");
  125. efree(params);
  126. return NULL;
  127. } else {
  128. if (Z_TYPE_P(value) != IS_STRING) {
  129. convert_to_string(value);
  130. }
  131. if (!xpath_params) {
  132. xpath_expr = php_xsl_xslt_string_to_xpathexpr(Z_STRVAL_P(value));
  133. } else {
  134. xpath_expr = estrndup(Z_STRVAL_P(value), Z_STRLEN_P(value));
  135. }
  136. if (xpath_expr) {
  137. params[i++] = estrndup(ZSTR_VAL(string_key), ZSTR_LEN(string_key));
  138. params[i++] = xpath_expr;
  139. }
  140. }
  141. } ZEND_HASH_FOREACH_END();
  142. params[i++] = NULL;
  143. return params;
  144. }
  145. /* }}} */
  146. static void xsl_ext_function_php(xmlXPathParserContextPtr ctxt, int nargs, int type) /* {{{ */
  147. {
  148. xsltTransformContextPtr tctxt;
  149. zval *args;
  150. zval retval;
  151. int result, i;
  152. int error = 0;
  153. zend_fcall_info fci;
  154. zval handler;
  155. xmlXPathObjectPtr obj;
  156. char *str;
  157. xsl_object *intern;
  158. zend_string *callable = NULL;
  159. if (! zend_is_executing()) {
  160. xsltGenericError(xsltGenericErrorContext,
  161. "xsltExtFunctionTest: Function called from outside of PHP\n");
  162. error = 1;
  163. } else {
  164. tctxt = xsltXPathGetTransformContext(ctxt);
  165. if (tctxt == NULL) {
  166. xsltGenericError(xsltGenericErrorContext,
  167. "xsltExtFunctionTest: failed to get the transformation context\n");
  168. error = 1;
  169. } else {
  170. intern = (xsl_object*)tctxt->_private;
  171. if (intern == NULL) {
  172. xsltGenericError(xsltGenericErrorContext,
  173. "xsltExtFunctionTest: failed to get the internal object\n");
  174. error = 1;
  175. }
  176. else if (intern->registerPhpFunctions == 0) {
  177. xsltGenericError(xsltGenericErrorContext,
  178. "xsltExtFunctionTest: PHP Object did not register PHP functions\n");
  179. error = 1;
  180. }
  181. }
  182. }
  183. if (error == 1) {
  184. for (i = nargs - 1; i >= 0; i--) {
  185. obj = valuePop(ctxt);
  186. if (obj) {
  187. xmlXPathFreeObject(obj);
  188. }
  189. }
  190. return;
  191. }
  192. fci.param_count = nargs - 1;
  193. if (fci.param_count > 0) {
  194. args = safe_emalloc(fci.param_count, sizeof(zval), 0);
  195. }
  196. /* Reverse order to pop values off ctxt stack */
  197. for (i = nargs - 2; i >= 0; i--) {
  198. obj = valuePop(ctxt);
  199. if (obj == NULL) {
  200. ZVAL_NULL(&args[i]);
  201. continue;
  202. }
  203. switch (obj->type) {
  204. case XPATH_STRING:
  205. ZVAL_STRING(&args[i], (char *)obj->stringval);
  206. break;
  207. case XPATH_BOOLEAN:
  208. ZVAL_BOOL(&args[i], obj->boolval);
  209. break;
  210. case XPATH_NUMBER:
  211. ZVAL_DOUBLE(&args[i], obj->floatval);
  212. break;
  213. case XPATH_NODESET:
  214. if (type == 1) {
  215. str = (char*)xmlXPathCastToString(obj);
  216. ZVAL_STRING(&args[i], str);
  217. xmlFree(str);
  218. } else if (type == 2) {
  219. int j;
  220. dom_object *domintern = (dom_object *)intern->doc;
  221. if (obj->nodesetval && obj->nodesetval->nodeNr > 0) {
  222. array_init(&args[i]);
  223. for (j = 0; j < obj->nodesetval->nodeNr; j++) {
  224. xmlNodePtr node = obj->nodesetval->nodeTab[j];
  225. zval child;
  226. /* not sure, if we need this... it's copied from xpath.c */
  227. if (node->type == XML_NAMESPACE_DECL) {
  228. xmlNsPtr curns;
  229. xmlNodePtr nsparent;
  230. nsparent = node->_private;
  231. curns = xmlNewNs(NULL, node->name, NULL);
  232. if (node->children) {
  233. curns->prefix = xmlStrdup((xmlChar *)node->children);
  234. }
  235. if (node->children) {
  236. node = xmlNewDocNode(node->doc, NULL, (xmlChar *) node->children, node->name);
  237. } else {
  238. node = xmlNewDocNode(node->doc, NULL, (const xmlChar *) "xmlns", node->name);
  239. }
  240. node->type = XML_NAMESPACE_DECL;
  241. node->parent = nsparent;
  242. node->ns = curns;
  243. } else {
  244. node = xmlDocCopyNode(node, domintern->document->ptr, 1);
  245. }
  246. php_dom_create_object(node, &child, domintern);
  247. add_next_index_zval(&args[i], &child);
  248. }
  249. } else {
  250. ZVAL_EMPTY_ARRAY(&args[i]);
  251. }
  252. }
  253. break;
  254. default:
  255. str = (char *) xmlXPathCastToString(obj);
  256. ZVAL_STRING(&args[i], str);
  257. xmlFree(str);
  258. }
  259. xmlXPathFreeObject(obj);
  260. }
  261. fci.size = sizeof(fci);
  262. if (fci.param_count > 0) {
  263. fci.params = args;
  264. } else {
  265. fci.params = NULL;
  266. }
  267. obj = valuePop(ctxt);
  268. if (obj == NULL || obj->stringval == NULL) {
  269. php_error_docref(NULL, E_WARNING, "Handler name must be a string");
  270. xmlXPathFreeObject(obj);
  271. valuePush(ctxt, xmlXPathNewString((const xmlChar *) ""));
  272. if (fci.param_count > 0) {
  273. for (i = 0; i < nargs - 1; i++) {
  274. zval_ptr_dtor(&args[i]);
  275. }
  276. efree(args);
  277. }
  278. return;
  279. }
  280. ZVAL_STRING(&handler, (char *) obj->stringval);
  281. xmlXPathFreeObject(obj);
  282. ZVAL_COPY_VALUE(&fci.function_name, &handler);
  283. fci.object = NULL;
  284. fci.retval = &retval;
  285. fci.no_separation = 0;
  286. /*fci.function_handler_cache = &function_ptr;*/
  287. if (!zend_make_callable(&handler, &callable)) {
  288. php_error_docref(NULL, E_WARNING, "Unable to call handler %s()", ZSTR_VAL(callable));
  289. valuePush(ctxt, xmlXPathNewString((const xmlChar *) ""));
  290. } else if ( intern->registerPhpFunctions == 2 && zend_hash_exists(intern->registered_phpfunctions, callable) == 0) {
  291. php_error_docref(NULL, E_WARNING, "Not allowed to call handler '%s()'", ZSTR_VAL(callable));
  292. /* Push an empty string, so that we at least have an xslt result... */
  293. valuePush(ctxt, xmlXPathNewString((const xmlChar *) ""));
  294. } else {
  295. result = zend_call_function(&fci, NULL);
  296. if (result == FAILURE) {
  297. if (Z_TYPE(handler) == IS_STRING) {
  298. php_error_docref(NULL, E_WARNING, "Unable to call handler %s()", Z_STRVAL(handler));
  299. valuePush(ctxt, xmlXPathNewString((const xmlChar *) ""));
  300. }
  301. /* retval is == NULL, when an exception occurred, don't report anything, because PHP itself will handle that */
  302. } else if (Z_ISUNDEF(retval)) {
  303. } else {
  304. if (Z_TYPE(retval) == IS_OBJECT && instanceof_function(Z_OBJCE(retval), dom_node_class_entry)) {
  305. xmlNode *nodep;
  306. dom_object *obj;
  307. if (intern->node_list == NULL) {
  308. intern->node_list = zend_new_array(0);
  309. }
  310. Z_ADDREF(retval);
  311. zend_hash_next_index_insert(intern->node_list, &retval);
  312. obj = Z_DOMOBJ_P(&retval);
  313. nodep = dom_object_get_node(obj);
  314. valuePush(ctxt, xmlXPathNewNodeSet(nodep));
  315. } else if (Z_TYPE(retval) == IS_TRUE || Z_TYPE(retval) == IS_FALSE) {
  316. valuePush(ctxt, xmlXPathNewBoolean(Z_TYPE(retval) == IS_TRUE));
  317. } else if (Z_TYPE(retval) == IS_OBJECT) {
  318. php_error_docref(NULL, E_WARNING, "A PHP Object cannot be converted to a XPath-string");
  319. valuePush(ctxt, xmlXPathNewString((const xmlChar *) ""));
  320. } else {
  321. convert_to_string_ex(&retval);
  322. valuePush(ctxt, xmlXPathNewString((xmlChar *) Z_STRVAL(retval)));
  323. }
  324. zval_ptr_dtor(&retval);
  325. }
  326. }
  327. zend_string_release_ex(callable, 0);
  328. zval_ptr_dtor(&handler);
  329. if (fci.param_count > 0) {
  330. for (i = 0; i < nargs - 1; i++) {
  331. zval_ptr_dtor(&args[i]);
  332. }
  333. efree(args);
  334. }
  335. }
  336. /* }}} */
  337. void xsl_ext_function_string_php(xmlXPathParserContextPtr ctxt, int nargs) /* {{{ */
  338. {
  339. xsl_ext_function_php(ctxt, nargs, 1);
  340. }
  341. /* }}} */
  342. void xsl_ext_function_object_php(xmlXPathParserContextPtr ctxt, int nargs) /* {{{ */
  343. {
  344. xsl_ext_function_php(ctxt, nargs, 2);
  345. }
  346. /* }}} */
  347. /* {{{ proto void xsl_xsltprocessor_import_stylesheet(domdocument doc)
  348. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#
  349. Since:
  350. */
  351. PHP_FUNCTION(xsl_xsltprocessor_import_stylesheet)
  352. {
  353. zval *id, *docp = NULL;
  354. xmlDoc *doc = NULL, *newdoc = NULL;
  355. xsltStylesheetPtr sheetp, oldsheetp;
  356. xsl_object *intern;
  357. int prevSubstValue, prevExtDtdValue, clone_docu = 0;
  358. xmlNode *nodep = NULL;
  359. zval *cloneDocu, member, rv;
  360. if (zend_parse_method_parameters(ZEND_NUM_ARGS(), getThis(), "Oo", &id, xsl_xsltprocessor_class_entry, &docp) == FAILURE) {
  361. RETURN_FALSE;
  362. }
  363. nodep = php_libxml_import_node(docp);
  364. if (nodep) {
  365. doc = nodep->doc;
  366. }
  367. if (doc == NULL) {
  368. php_error(E_WARNING, "Invalid Document");
  369. RETURN_FALSE;
  370. }
  371. /* libxslt uses _private, so we must copy the imported
  372. stylesheet document otherwise the node proxies will be a mess */
  373. newdoc = xmlCopyDoc(doc, 1);
  374. xmlNodeSetBase((xmlNodePtr) newdoc, (xmlChar *)doc->URL);
  375. prevSubstValue = xmlSubstituteEntitiesDefault(1);
  376. prevExtDtdValue = xmlLoadExtDtdDefaultValue;
  377. xmlLoadExtDtdDefaultValue = XML_DETECT_IDS | XML_COMPLETE_ATTRS;
  378. sheetp = xsltParseStylesheetDoc(newdoc);
  379. xmlSubstituteEntitiesDefault(prevSubstValue);
  380. xmlLoadExtDtdDefaultValue = prevExtDtdValue;
  381. if (!sheetp) {
  382. xmlFreeDoc(newdoc);
  383. RETURN_FALSE;
  384. }
  385. intern = Z_XSL_P(id);
  386. ZVAL_STRING(&member, "cloneDocument");
  387. cloneDocu = zend_std_read_property(id, &member, BP_VAR_IS, NULL, &rv);
  388. if (Z_TYPE_P(cloneDocu) != IS_NULL) {
  389. convert_to_long(cloneDocu);
  390. clone_docu = Z_LVAL_P(cloneDocu);
  391. }
  392. zval_ptr_dtor(&member);
  393. if (clone_docu == 0) {
  394. /* check if the stylesheet is using xsl:key, if yes, we have to clone the document _always_ before a transformation */
  395. nodep = xmlDocGetRootElement(sheetp->doc);
  396. if (nodep && (nodep = nodep->children)) {
  397. while (nodep) {
  398. if (nodep->type == XML_ELEMENT_NODE && xmlStrEqual(nodep->name, (const xmlChar *) "key") && xmlStrEqual(nodep->ns->href, XSLT_NAMESPACE)) {
  399. intern->hasKeys = 1;
  400. break;
  401. }
  402. nodep = nodep->next;
  403. }
  404. }
  405. } else {
  406. intern->hasKeys = clone_docu;
  407. }
  408. if ((oldsheetp = (xsltStylesheetPtr)intern->ptr)) {
  409. /* free wrapper */
  410. if (((xsltStylesheetPtr) intern->ptr)->_private != NULL) {
  411. ((xsltStylesheetPtr) intern->ptr)->_private = NULL;
  412. }
  413. xsltFreeStylesheet((xsltStylesheetPtr) intern->ptr);
  414. intern->ptr = NULL;
  415. }
  416. php_xsl_set_object(id, sheetp);
  417. RETVAL_TRUE;
  418. }
  419. /* }}} end xsl_xsltprocessor_import_stylesheet */
  420. static xmlDocPtr php_xsl_apply_stylesheet(zval *id, xsl_object *intern, xsltStylesheetPtr style, zval *docp) /* {{{ */
  421. {
  422. xmlDocPtr newdocp = NULL;
  423. xmlDocPtr doc = NULL;
  424. xmlNodePtr node = NULL;
  425. xsltTransformContextPtr ctxt;
  426. php_libxml_node_object *object;
  427. char **params = NULL;
  428. int clone;
  429. zval *doXInclude, member, rv;
  430. FILE *f;
  431. int secPrefsError = 0;
  432. int secPrefsValue;
  433. xsltSecurityPrefsPtr secPrefs = NULL;
  434. node = php_libxml_import_node(docp);
  435. if (node) {
  436. doc = node->doc;
  437. }
  438. if (doc == NULL) {
  439. php_error_docref(NULL, E_WARNING, "Invalid Document");
  440. return NULL;
  441. }
  442. if (style == NULL) {
  443. php_error_docref(NULL, E_WARNING, "No stylesheet associated to this object");
  444. return NULL;
  445. }
  446. if (intern->profiling) {
  447. if (php_check_open_basedir(intern->profiling)) {
  448. f = NULL;
  449. } else {
  450. f = VCWD_FOPEN(intern->profiling, "w");
  451. }
  452. } else {
  453. f = NULL;
  454. }
  455. if (intern->parameter) {
  456. params = php_xsl_xslt_make_params(intern->parameter, 0);
  457. }
  458. intern->doc = emalloc(sizeof(php_libxml_node_object));
  459. memset(intern->doc, 0, sizeof(php_libxml_node_object));
  460. if (intern->hasKeys == 1) {
  461. doc = xmlCopyDoc(doc, 1);
  462. } else {
  463. object = Z_LIBXML_NODE_P(docp);
  464. intern->doc->document = object->document;
  465. }
  466. php_libxml_increment_doc_ref(intern->doc, doc);
  467. ctxt = xsltNewTransformContext(style, doc);
  468. ctxt->_private = (void *) intern;
  469. ZVAL_STRING(&member, "doXInclude");
  470. doXInclude = zend_std_read_property(id, &member, BP_VAR_IS, NULL, &rv);
  471. if (Z_TYPE_P(doXInclude) != IS_NULL) {
  472. convert_to_long(doXInclude);
  473. ctxt->xinclude = Z_LVAL_P(doXInclude);
  474. }
  475. zval_ptr_dtor(&member);
  476. secPrefsValue = intern->securityPrefs;
  477. /* if securityPrefs is set to NONE, we don't have to do any checks, but otherwise... */
  478. if (secPrefsValue != XSL_SECPREF_NONE) {
  479. secPrefs = xsltNewSecurityPrefs();
  480. if (secPrefsValue & XSL_SECPREF_READ_FILE ) {
  481. if (0 != xsltSetSecurityPrefs(secPrefs, XSLT_SECPREF_READ_FILE, xsltSecurityForbid)) {
  482. secPrefsError = 1;
  483. }
  484. }
  485. if (secPrefsValue & XSL_SECPREF_WRITE_FILE ) {
  486. if (0 != xsltSetSecurityPrefs(secPrefs, XSLT_SECPREF_WRITE_FILE, xsltSecurityForbid)) {
  487. secPrefsError = 1;
  488. }
  489. }
  490. if (secPrefsValue & XSL_SECPREF_CREATE_DIRECTORY ) {
  491. if (0 != xsltSetSecurityPrefs(secPrefs, XSLT_SECPREF_CREATE_DIRECTORY, xsltSecurityForbid)) {
  492. secPrefsError = 1;
  493. }
  494. }
  495. if (secPrefsValue & XSL_SECPREF_READ_NETWORK) {
  496. if (0 != xsltSetSecurityPrefs(secPrefs, XSLT_SECPREF_READ_NETWORK, xsltSecurityForbid)) {
  497. secPrefsError = 1;
  498. }
  499. }
  500. if (secPrefsValue & XSL_SECPREF_WRITE_NETWORK) {
  501. if (0 != xsltSetSecurityPrefs(secPrefs, XSLT_SECPREF_WRITE_NETWORK, xsltSecurityForbid)) {
  502. secPrefsError = 1;
  503. }
  504. }
  505. if (0 != xsltSetCtxtSecurityPrefs(secPrefs, ctxt)) {
  506. secPrefsError = 1;
  507. }
  508. }
  509. if (secPrefsError == 1) {
  510. php_error_docref(NULL, E_WARNING, "Can't set libxslt security properties, not doing transformation for security reasons");
  511. } else {
  512. newdocp = xsltApplyStylesheetUser(style, doc, (const char**) params, NULL, f, ctxt);
  513. }
  514. if (f) {
  515. fclose(f);
  516. }
  517. xsltFreeTransformContext(ctxt);
  518. if (secPrefs) {
  519. xsltFreeSecurityPrefs(secPrefs);
  520. }
  521. if (intern->node_list != NULL) {
  522. zend_hash_destroy(intern->node_list);
  523. FREE_HASHTABLE(intern->node_list);
  524. intern->node_list = NULL;
  525. }
  526. php_libxml_decrement_doc_ref(intern->doc);
  527. efree(intern->doc);
  528. intern->doc = NULL;
  529. if (params) {
  530. clone = 0;
  531. while(params[clone]) {
  532. efree(params[clone++]);
  533. }
  534. efree(params);
  535. }
  536. return newdocp;
  537. }
  538. /* }}} */
  539. /* {{{ proto domdocument xsl_xsltprocessor_transform_to_doc(domnode doc)
  540. URL: http://www.w3.org/TR/2003/WD-DOM-Level-3-Core-20030226/DOM3-Core.html#
  541. Since:
  542. */
  543. PHP_FUNCTION(xsl_xsltprocessor_transform_to_doc)
  544. {
  545. zval *id, *docp = NULL;
  546. xmlDoc *newdocp;
  547. xsltStylesheetPtr sheetp;
  548. zend_string *ret_class = NULL;
  549. xsl_object *intern;
  550. id = getThis();
  551. intern = Z_XSL_P(id);
  552. sheetp = (xsltStylesheetPtr) intern->ptr;
  553. if (zend_parse_parameters(ZEND_NUM_ARGS(), "o|S!", &docp, &ret_class) == FAILURE) {
  554. RETURN_FALSE;
  555. }
  556. newdocp = php_xsl_apply_stylesheet(id, intern, sheetp, docp);
  557. if (newdocp) {
  558. if (ret_class) {
  559. zend_string *curclass_name;
  560. zend_class_entry *curce, *ce;
  561. php_libxml_node_object *interndoc;
  562. curce = Z_OBJCE_P(docp);
  563. curclass_name = curce->name;
  564. while (curce->parent != NULL) {
  565. curce = curce->parent;
  566. }
  567. ce = zend_lookup_class(ret_class);
  568. if (ce == NULL || !instanceof_function(ce, curce)) {
  569. xmlFreeDoc(newdocp);
  570. php_error_docref(NULL, E_WARNING,
  571. "Expecting class compatible with %s, '%s' given", ZSTR_VAL(curclass_name), ZSTR_VAL(ret_class));
  572. RETURN_FALSE;
  573. }
  574. object_init_ex(return_value, ce);
  575. interndoc = Z_LIBXML_NODE_P(return_value);
  576. php_libxml_increment_doc_ref(interndoc, newdocp);
  577. php_libxml_increment_node_ptr(interndoc, (xmlNodePtr)newdocp, (void *)interndoc);
  578. } else {
  579. php_dom_create_object((xmlNodePtr) newdocp, return_value, NULL);
  580. }
  581. } else {
  582. RETURN_FALSE;
  583. }
  584. }
  585. /* }}} end xsl_xsltprocessor_transform_to_doc */
  586. /* {{{ proto int xsl_xsltprocessor_transform_to_uri(domdocument doc, string uri)
  587. */
  588. PHP_FUNCTION(xsl_xsltprocessor_transform_to_uri)
  589. {
  590. zval *id, *docp = NULL;
  591. xmlDoc *newdocp;
  592. xsltStylesheetPtr sheetp;
  593. int ret;
  594. size_t uri_len;
  595. char *uri;
  596. xsl_object *intern;
  597. id = getThis();
  598. intern = Z_XSL_P(id);
  599. sheetp = (xsltStylesheetPtr) intern->ptr;
  600. if (zend_parse_parameters(ZEND_NUM_ARGS(), "op", &docp, &uri, &uri_len) == FAILURE) {
  601. RETURN_FALSE;
  602. }
  603. newdocp = php_xsl_apply_stylesheet(id, intern, sheetp, docp);
  604. ret = -1;
  605. if (newdocp) {
  606. ret = xsltSaveResultToFilename(uri, newdocp, sheetp, 0);
  607. xmlFreeDoc(newdocp);
  608. }
  609. RETVAL_LONG(ret);
  610. }
  611. /* }}} end xsl_xsltprocessor_transform_to_uri */
  612. /* {{{ proto string xsl_xsltprocessor_transform_to_xml(domdocument doc)
  613. */
  614. PHP_FUNCTION(xsl_xsltprocessor_transform_to_xml)
  615. {
  616. zval *id, *docp = NULL;
  617. xmlDoc *newdocp;
  618. xsltStylesheetPtr sheetp;
  619. int ret;
  620. xmlChar *doc_txt_ptr;
  621. int doc_txt_len;
  622. xsl_object *intern;
  623. id = getThis();
  624. intern = Z_XSL_P(id);
  625. sheetp = (xsltStylesheetPtr) intern->ptr;
  626. if (zend_parse_parameters(ZEND_NUM_ARGS(), "o", &docp) == FAILURE) {
  627. RETURN_FALSE;
  628. }
  629. newdocp = php_xsl_apply_stylesheet(id, intern, sheetp, docp);
  630. ret = -1;
  631. if (newdocp) {
  632. ret = xsltSaveResultToString(&doc_txt_ptr, &doc_txt_len, newdocp, sheetp);
  633. if (doc_txt_ptr && doc_txt_len) {
  634. RETVAL_STRINGL((char *) doc_txt_ptr, doc_txt_len);
  635. xmlFree(doc_txt_ptr);
  636. }
  637. xmlFreeDoc(newdocp);
  638. }
  639. if (ret < 0) {
  640. RETURN_FALSE;
  641. }
  642. }
  643. /* }}} end xsl_xsltprocessor_transform_to_xml */
  644. /* {{{ proto bool xsl_xsltprocessor_set_parameter(string namespace, mixed name [, string value])
  645. */
  646. PHP_FUNCTION(xsl_xsltprocessor_set_parameter)
  647. {
  648. zval *id;
  649. zval *array_value, *entry, new_string;
  650. xsl_object *intern;
  651. char *namespace;
  652. size_t namespace_len;
  653. zend_string *string_key, *name, *value;
  654. DOM_GET_THIS(id);
  655. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS(), "sa", &namespace, &namespace_len, &array_value) == SUCCESS) {
  656. intern = Z_XSL_P(id);
  657. ZEND_HASH_FOREACH_STR_KEY_VAL(Z_ARRVAL_P(array_value), string_key, entry) {
  658. if (string_key == NULL) {
  659. php_error_docref(NULL, E_WARNING, "Invalid parameter array");
  660. RETURN_FALSE;
  661. }
  662. convert_to_string_ex(entry);
  663. Z_TRY_ADDREF_P(entry);
  664. zend_hash_update(intern->parameter, string_key, entry);
  665. } ZEND_HASH_FOREACH_END();
  666. RETURN_TRUE;
  667. } else if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS(), "sSS", &namespace, &namespace_len, &name, &value) == SUCCESS) {
  668. intern = Z_XSL_P(id);
  669. ZVAL_STR_COPY(&new_string, value);
  670. zend_hash_update(intern->parameter, name, &new_string);
  671. RETURN_TRUE;
  672. } else {
  673. WRONG_PARAM_COUNT;
  674. }
  675. }
  676. /* }}} end xsl_xsltprocessor_set_parameter */
  677. /* {{{ proto string xsl_xsltprocessor_get_parameter(string namespace, string name)
  678. */
  679. PHP_FUNCTION(xsl_xsltprocessor_get_parameter)
  680. {
  681. zval *id;
  682. char *namespace;
  683. size_t namespace_len = 0;
  684. zval *value;
  685. zend_string *name;
  686. xsl_object *intern;
  687. DOM_GET_THIS(id);
  688. if (zend_parse_parameters(ZEND_NUM_ARGS(), "sS", &namespace, &namespace_len, &name) == FAILURE) {
  689. RETURN_FALSE;
  690. }
  691. intern = Z_XSL_P(id);
  692. if ((value = zend_hash_find(intern->parameter, name)) != NULL) {
  693. RETURN_STR(zval_get_string(value));
  694. } else {
  695. RETURN_FALSE;
  696. }
  697. }
  698. /* }}} end xsl_xsltprocessor_get_parameter */
  699. /* {{{ proto bool xsl_xsltprocessor_remove_parameter(string namespace, string name)
  700. */
  701. PHP_FUNCTION(xsl_xsltprocessor_remove_parameter)
  702. {
  703. zval *id;
  704. size_t namespace_len = 0;
  705. char *namespace;
  706. zend_string *name;
  707. xsl_object *intern;
  708. DOM_GET_THIS(id);
  709. if (zend_parse_parameters(ZEND_NUM_ARGS(), "sS", &namespace, &namespace_len, &name) == FAILURE) {
  710. RETURN_FALSE;
  711. }
  712. intern = Z_XSL_P(id);
  713. if (zend_hash_del(intern->parameter, name) == SUCCESS) {
  714. RETURN_TRUE;
  715. } else {
  716. RETURN_FALSE;
  717. }
  718. }
  719. /* }}} end xsl_xsltprocessor_remove_parameter */
  720. /* {{{ proto void xsl_xsltprocessor_register_php_functions([mixed $restrict])
  721. */
  722. PHP_FUNCTION(xsl_xsltprocessor_register_php_functions)
  723. {
  724. zval *id;
  725. xsl_object *intern;
  726. zval *array_value, *entry, new_string;
  727. zend_string *name;
  728. DOM_GET_THIS(id);
  729. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS(), "a", &array_value) == SUCCESS) {
  730. intern = Z_XSL_P(id);
  731. ZEND_HASH_FOREACH_VAL(Z_ARRVAL_P(array_value), entry) {
  732. convert_to_string_ex(entry);
  733. ZVAL_LONG(&new_string ,1);
  734. zend_hash_update(intern->registered_phpfunctions, Z_STR_P(entry), &new_string);
  735. } ZEND_HASH_FOREACH_END();
  736. intern->registerPhpFunctions = 2;
  737. } else if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS(), "S", &name) == SUCCESS) {
  738. intern = Z_XSL_P(id);
  739. ZVAL_LONG(&new_string,1);
  740. zend_hash_update(intern->registered_phpfunctions, name, &new_string);
  741. intern->registerPhpFunctions = 2;
  742. } else {
  743. intern = Z_XSL_P(id);
  744. intern->registerPhpFunctions = 1;
  745. }
  746. }
  747. /* }}} end xsl_xsltprocessor_register_php_functions(); */
  748. /* {{{ proto bool xsl_xsltprocessor_set_profiling(string filename) */
  749. PHP_FUNCTION(xsl_xsltprocessor_set_profiling)
  750. {
  751. zval *id;
  752. xsl_object *intern;
  753. char *filename = NULL;
  754. size_t filename_len;
  755. DOM_GET_THIS(id);
  756. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS(), "p!", &filename, &filename_len) == SUCCESS) {
  757. intern = Z_XSL_P(id);
  758. if (intern->profiling) {
  759. efree(intern->profiling);
  760. }
  761. if (filename != NULL) {
  762. intern->profiling = estrndup(filename, filename_len);
  763. } else {
  764. intern->profiling = NULL;
  765. }
  766. RETURN_TRUE;
  767. } else {
  768. WRONG_PARAM_COUNT;
  769. }
  770. }
  771. /* }}} end xsl_xsltprocessor_set_profiling */
  772. /* {{{ proto int xsl_xsltprocessor_set_security_prefs(int securityPrefs) */
  773. PHP_FUNCTION(xsl_xsltprocessor_set_security_prefs)
  774. {
  775. zval *id;
  776. xsl_object *intern;
  777. zend_long securityPrefs, oldSecurityPrefs;
  778. DOM_GET_THIS(id);
  779. if (zend_parse_parameters(ZEND_NUM_ARGS(), "l", &securityPrefs) == FAILURE) {
  780. return;
  781. }
  782. intern = Z_XSL_P(id);
  783. oldSecurityPrefs = intern->securityPrefs;
  784. intern->securityPrefs = securityPrefs;
  785. /* set this to 1 so that we know, it was set through this method. Can be removed, when we remove the ini setting */
  786. intern->securityPrefsSet = 1;
  787. RETURN_LONG(oldSecurityPrefs);
  788. }
  789. /* }}} end xsl_xsltprocessor_set_security_prefs */
  790. /* {{{ proto int xsl_xsltprocessor_get_security_prefs() */
  791. PHP_FUNCTION(xsl_xsltprocessor_get_security_prefs)
  792. {
  793. zval *id;
  794. xsl_object *intern;
  795. DOM_GET_THIS(id);
  796. if (zend_parse_parameters_ex(ZEND_PARSE_PARAMS_QUIET, ZEND_NUM_ARGS(), "") == SUCCESS) {
  797. intern = Z_XSL_P(id);
  798. RETURN_LONG(intern->securityPrefs);
  799. } else {
  800. WRONG_PARAM_COUNT;
  801. }
  802. }
  803. /* }}} end xsl_xsltprocessor_get_security_prefs */
  804. /* {{{ proto bool xsl_xsltprocessor_has_exslt_support()
  805. */
  806. PHP_FUNCTION(xsl_xsltprocessor_has_exslt_support)
  807. {
  808. #if HAVE_XSL_EXSLT
  809. RETURN_TRUE;
  810. #else
  811. RETURN_FALSE;
  812. #endif
  813. }
  814. /* }}} end xsl_xsltprocessor_has_exslt_support(); */
  815. /*
  816. * Local variables:
  817. * tab-width: 4
  818. * c-basic-offset: 4
  819. * End:
  820. * vim600: sw=4 ts=4 fdm=marker
  821. * vim<600: sw=4 ts=4
  822. */