php_packet_soap.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2016 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: Brad Lafountain <rodif_bl@yahoo.com> |
  16. | Shane Caraveo <shane@caraveo.com> |
  17. | Dmitry Stogov <dmitry@zend.com> |
  18. +----------------------------------------------------------------------+
  19. */
  20. /* $Id$ */
  21. #include "php_soap.h"
  22. /* SOAP client calls this function to parse response from SOAP server */
  23. int parse_packet_soap(zval *this_ptr, char *buffer, int buffer_size, sdlFunctionPtr fn, char *fn_name, zval *return_value, zval *soap_headers TSRMLS_DC)
  24. {
  25. char* envelope_ns = NULL;
  26. xmlDocPtr response;
  27. xmlNodePtr trav, env, head, body, resp, cur, fault;
  28. xmlAttrPtr attr;
  29. int param_count = 0;
  30. int soap_version = SOAP_1_1;
  31. HashTable *hdrs = NULL;
  32. ZVAL_NULL(return_value);
  33. /* Response for one-way opearation */
  34. if (buffer_size == 0) {
  35. return TRUE;
  36. }
  37. /* Parse XML packet */
  38. response = soap_xmlParseMemory(buffer, buffer_size);
  39. if (!response) {
  40. add_soap_fault(this_ptr, "Client", "looks like we got no XML document", NULL, NULL TSRMLS_CC);
  41. return FALSE;
  42. }
  43. if (xmlGetIntSubset(response) != NULL) {
  44. add_soap_fault(this_ptr, "Client", "DTD are not supported by SOAP", NULL, NULL TSRMLS_CC);
  45. xmlFreeDoc(response);
  46. return FALSE;
  47. }
  48. /* Get <Envelope> element */
  49. env = NULL;
  50. trav = response->children;
  51. while (trav != NULL) {
  52. if (trav->type == XML_ELEMENT_NODE) {
  53. if (env == NULL && node_is_equal_ex(trav,"Envelope",SOAP_1_1_ENV_NAMESPACE)) {
  54. env = trav;
  55. envelope_ns = SOAP_1_1_ENV_NAMESPACE;
  56. soap_version = SOAP_1_1;
  57. } else if (env == NULL && node_is_equal_ex(trav,"Envelope",SOAP_1_2_ENV_NAMESPACE)) {
  58. env = trav;
  59. envelope_ns = SOAP_1_2_ENV_NAMESPACE;
  60. soap_version = SOAP_1_2;
  61. } else {
  62. add_soap_fault(this_ptr, "VersionMismatch", "Wrong Version", NULL, NULL TSRMLS_CC);
  63. xmlFreeDoc(response);
  64. return FALSE;
  65. }
  66. }
  67. trav = trav->next;
  68. }
  69. if (env == NULL) {
  70. add_soap_fault(this_ptr, "Client", "looks like we got XML without \"Envelope\" element", NULL, NULL TSRMLS_CC);
  71. xmlFreeDoc(response);
  72. return FALSE;
  73. }
  74. attr = env->properties;
  75. while (attr != NULL) {
  76. if (attr->ns == NULL) {
  77. add_soap_fault(this_ptr, "Client", "A SOAP Envelope element cannot have non Namespace qualified attributes", NULL, NULL TSRMLS_CC);
  78. xmlFreeDoc(response);
  79. return FALSE;
  80. } else if (attr_is_equal_ex(attr,"encodingStyle",SOAP_1_2_ENV_NAMESPACE)) {
  81. if (soap_version == SOAP_1_2) {
  82. add_soap_fault(this_ptr, "Client", "encodingStyle cannot be specified on the Envelope", NULL, NULL TSRMLS_CC);
  83. xmlFreeDoc(response);
  84. return FALSE;
  85. } else if (strcmp((char*)attr->children->content, SOAP_1_1_ENC_NAMESPACE) != 0) {
  86. add_soap_fault(this_ptr, "Client", "Unknown data encoding style", NULL, NULL TSRMLS_CC);
  87. xmlFreeDoc(response);
  88. return FALSE;
  89. }
  90. }
  91. attr = attr->next;
  92. }
  93. /* Get <Header> element */
  94. head = NULL;
  95. trav = env->children;
  96. while (trav != NULL && trav->type != XML_ELEMENT_NODE) {
  97. trav = trav->next;
  98. }
  99. if (trav != NULL && node_is_equal_ex(trav,"Header",envelope_ns)) {
  100. head = trav;
  101. trav = trav->next;
  102. }
  103. /* Get <Body> element */
  104. body = NULL;
  105. while (trav != NULL && trav->type != XML_ELEMENT_NODE) {
  106. trav = trav->next;
  107. }
  108. if (trav != NULL && node_is_equal_ex(trav,"Body",envelope_ns)) {
  109. body = trav;
  110. trav = trav->next;
  111. }
  112. while (trav != NULL && trav->type != XML_ELEMENT_NODE) {
  113. trav = trav->next;
  114. }
  115. if (body == NULL) {
  116. add_soap_fault(this_ptr, "Client", "Body must be present in a SOAP envelope", NULL, NULL TSRMLS_CC);
  117. xmlFreeDoc(response);
  118. return FALSE;
  119. }
  120. attr = body->properties;
  121. while (attr != NULL) {
  122. if (attr->ns == NULL) {
  123. if (soap_version == SOAP_1_2) {
  124. add_soap_fault(this_ptr, "Client", "A SOAP Body element cannot have non Namespace qualified attributes", NULL, NULL TSRMLS_CC);
  125. xmlFreeDoc(response);
  126. return FALSE;
  127. }
  128. } else if (attr_is_equal_ex(attr,"encodingStyle",SOAP_1_2_ENV_NAMESPACE)) {
  129. if (soap_version == SOAP_1_2) {
  130. add_soap_fault(this_ptr, "Client", "encodingStyle cannot be specified on the Body", NULL, NULL TSRMLS_CC);
  131. xmlFreeDoc(response);
  132. return FALSE;
  133. } else if (strcmp((char*)attr->children->content, SOAP_1_1_ENC_NAMESPACE) != 0) {
  134. add_soap_fault(this_ptr, "Client", "Unknown data encoding style", NULL, NULL TSRMLS_CC);
  135. xmlFreeDoc(response);
  136. return FALSE;
  137. }
  138. }
  139. attr = attr->next;
  140. }
  141. if (trav != NULL && soap_version == SOAP_1_2) {
  142. add_soap_fault(this_ptr, "Client", "A SOAP 1.2 envelope can contain only Header and Body", NULL, NULL TSRMLS_CC);
  143. xmlFreeDoc(response);
  144. return FALSE;
  145. }
  146. if (head != NULL) {
  147. attr = head->properties;
  148. while (attr != NULL) {
  149. if (attr->ns == NULL) {
  150. add_soap_fault(this_ptr, "Client", "A SOAP Header element cannot have non Namespace qualified attributes", NULL, NULL TSRMLS_CC);
  151. xmlFreeDoc(response);
  152. return FALSE;
  153. } else if (attr_is_equal_ex(attr,"encodingStyle",SOAP_1_2_ENV_NAMESPACE)) {
  154. if (soap_version == SOAP_1_2) {
  155. add_soap_fault(this_ptr, "Client", "encodingStyle cannot be specified on the Header", NULL, NULL TSRMLS_CC);
  156. xmlFreeDoc(response);
  157. return FALSE;
  158. } else if (strcmp((char*)attr->children->content, SOAP_1_1_ENC_NAMESPACE) != 0) {
  159. add_soap_fault(this_ptr, "Client", "Unknown data encoding style", NULL, NULL TSRMLS_CC);
  160. xmlFreeDoc(response);
  161. return FALSE;
  162. }
  163. }
  164. attr = attr->next;
  165. }
  166. }
  167. /* Check if <Body> contains <Fault> element */
  168. fault = get_node_ex(body->children,"Fault",envelope_ns);
  169. if (fault != NULL) {
  170. char *faultcode = NULL, *faultstring = NULL, *faultactor = NULL;
  171. zval *details = NULL;
  172. xmlNodePtr tmp;
  173. if (soap_version == SOAP_1_1) {
  174. tmp = get_node(fault->children, "faultcode");
  175. if (tmp != NULL && tmp->children != NULL) {
  176. faultcode = (char*)tmp->children->content;
  177. }
  178. tmp = get_node(fault->children, "faultstring");
  179. if (tmp != NULL && tmp->children != NULL) {
  180. zval *zv = master_to_zval(get_conversion(IS_STRING), tmp TSRMLS_CC);
  181. faultstring = Z_STRVAL_P(zv);
  182. FREE_ZVAL(zv);
  183. }
  184. tmp = get_node(fault->children, "faultactor");
  185. if (tmp != NULL && tmp->children != NULL) {
  186. zval *zv = master_to_zval(get_conversion(IS_STRING), tmp TSRMLS_CC);
  187. faultactor = Z_STRVAL_P(zv);
  188. FREE_ZVAL(zv);
  189. }
  190. tmp = get_node(fault->children, "detail");
  191. if (tmp != NULL) {
  192. details = master_to_zval(NULL, tmp TSRMLS_CC);
  193. }
  194. } else {
  195. tmp = get_node(fault->children, "Code");
  196. if (tmp != NULL && tmp->children != NULL) {
  197. tmp = get_node(tmp->children, "Value");
  198. if (tmp != NULL && tmp->children != NULL) {
  199. faultcode = (char*)tmp->children->content;
  200. }
  201. }
  202. tmp = get_node(fault->children,"Reason");
  203. if (tmp != NULL && tmp->children != NULL) {
  204. /* TODO: lang attribute */
  205. tmp = get_node(tmp->children,"Text");
  206. if (tmp != NULL && tmp->children != NULL) {
  207. zval *zv = master_to_zval(get_conversion(IS_STRING), tmp TSRMLS_CC);
  208. faultstring = Z_STRVAL_P(zv);
  209. FREE_ZVAL(zv);
  210. }
  211. }
  212. tmp = get_node(fault->children,"Detail");
  213. if (tmp != NULL) {
  214. details = master_to_zval(NULL, tmp TSRMLS_CC);
  215. }
  216. }
  217. add_soap_fault(this_ptr, faultcode, faultstring, faultactor, details TSRMLS_CC);
  218. if (faultstring) {
  219. efree(faultstring);
  220. }
  221. if (faultactor) {
  222. efree(faultactor);
  223. }
  224. if (details) {
  225. Z_DELREF_P(details);
  226. }
  227. xmlFreeDoc(response);
  228. return FALSE;
  229. }
  230. /* Parse content of <Body> element */
  231. array_init(return_value);
  232. resp = body->children;
  233. while (resp != NULL && resp->type != XML_ELEMENT_NODE) {
  234. resp = resp->next;
  235. }
  236. if (resp != NULL) {
  237. if (fn != NULL && fn->binding && fn->binding->bindingType == BINDING_SOAP) {
  238. /* Function has WSDL description */
  239. sdlParamPtr *h_param, param = NULL;
  240. xmlNodePtr val = NULL;
  241. char *name, *ns = NULL;
  242. zval* tmp;
  243. sdlSoapBindingFunctionPtr fnb = (sdlSoapBindingFunctionPtr)fn->bindingAttributes;
  244. int res_count;
  245. hdrs = fnb->output.headers;
  246. if (fn->responseParameters) {
  247. res_count = zend_hash_num_elements(fn->responseParameters);
  248. zend_hash_internal_pointer_reset(fn->responseParameters);
  249. while (zend_hash_get_current_data(fn->responseParameters, (void **)&h_param) == SUCCESS) {
  250. param = (*h_param);
  251. if (fnb->style == SOAP_DOCUMENT) {
  252. if (param->element) {
  253. name = param->element->name;
  254. ns = param->element->namens;
  255. /*
  256. name = param->encode->details.type_str;
  257. ns = param->encode->details.ns;
  258. */
  259. } else {
  260. name = param->paramName;
  261. }
  262. } else {
  263. name = fn->responseName;
  264. /* ns = ? */
  265. }
  266. /* Get value of parameter */
  267. cur = get_node_ex(resp, name, ns);
  268. if (!cur) {
  269. cur = get_node(resp, name);
  270. /* TODO: produce warning invalid ns */
  271. }
  272. if (!cur && fnb->style == SOAP_RPC) {
  273. cur = resp;
  274. }
  275. if (cur) {
  276. if (fnb->style == SOAP_DOCUMENT) {
  277. val = cur;
  278. } else {
  279. val = get_node(cur->children, param->paramName);
  280. if (res_count == 1) {
  281. if (val == NULL) {
  282. val = get_node(cur->children, "return");
  283. }
  284. if (val == NULL) {
  285. val = get_node(cur->children, "result");
  286. }
  287. if (val == NULL && cur->children && cur->children->next == NULL) {
  288. val = cur->children;
  289. }
  290. }
  291. }
  292. }
  293. if (!val) {
  294. /* TODO: may be "nil" is not OK? */
  295. MAKE_STD_ZVAL(tmp);
  296. ZVAL_NULL(tmp);
  297. /*
  298. add_soap_fault(this_ptr, "Client", "Can't find response data", NULL, NULL TSRMLS_CC);
  299. xmlFreeDoc(response);
  300. return FALSE;
  301. */
  302. } else {
  303. /* Decoding value of parameter */
  304. if (param != NULL) {
  305. tmp = master_to_zval(param->encode, val TSRMLS_CC);
  306. } else {
  307. tmp = master_to_zval(NULL, val TSRMLS_CC);
  308. }
  309. }
  310. add_assoc_zval(return_value, param->paramName, tmp);
  311. param_count++;
  312. zend_hash_move_forward(fn->responseParameters);
  313. }
  314. }
  315. } else {
  316. /* Function has no WSDL description */
  317. xmlNodePtr val;
  318. val = resp->children;
  319. while (val != NULL) {
  320. while (val && val->type != XML_ELEMENT_NODE) {
  321. val = val->next;
  322. }
  323. if (val != NULL) {
  324. if (!node_is_equal_ex(val,"result",RPC_SOAP12_NAMESPACE)) {
  325. zval *tmp;
  326. zval **arr;
  327. tmp = master_to_zval(NULL, val TSRMLS_CC);
  328. if (val->name) {
  329. if (zend_hash_find(Z_ARRVAL_P(return_value), (char*)val->name, strlen((char*)val->name)+1, (void**)&arr) == SUCCESS) {
  330. add_next_index_zval(*arr, tmp);
  331. } else if (val->next && get_node(val->next, (char*)val->name)) {
  332. zval *arr;
  333. MAKE_STD_ZVAL(arr);
  334. array_init(arr);
  335. add_next_index_zval(arr, tmp);
  336. add_assoc_zval(return_value, (char*)val->name, arr);
  337. } else {
  338. add_assoc_zval(return_value, (char*)val->name, tmp);
  339. }
  340. } else {
  341. add_next_index_zval(return_value, tmp);
  342. }
  343. ++param_count;
  344. }
  345. val = val->next;
  346. }
  347. }
  348. }
  349. }
  350. if (Z_TYPE_P(return_value) == IS_ARRAY) {
  351. if (param_count == 0) {
  352. zval_dtor(return_value);
  353. ZVAL_NULL(return_value);
  354. } else if (param_count == 1) {
  355. zval *tmp;
  356. zend_hash_internal_pointer_reset(Z_ARRVAL_P(return_value));
  357. zend_hash_get_current_data(Z_ARRVAL_P(return_value), (void**)&tmp);
  358. tmp = *(zval**)tmp;
  359. Z_ADDREF_P(tmp);
  360. zval_dtor(return_value);
  361. *return_value = *tmp;
  362. FREE_ZVAL(tmp);
  363. }
  364. }
  365. if (soap_headers && head) {
  366. trav = head->children;
  367. while (trav != NULL) {
  368. if (trav->type == XML_ELEMENT_NODE) {
  369. encodePtr enc = NULL;
  370. zval* val;
  371. if (hdrs) {
  372. smart_str key = {0};
  373. sdlSoapBindingFunctionHeaderPtr *hdr;
  374. if (trav->ns) {
  375. smart_str_appends(&key, (char*)trav->ns->href);
  376. smart_str_appendc(&key,':');
  377. }
  378. smart_str_appends(&key, (char*)trav->name);
  379. smart_str_0(&key);
  380. if (zend_hash_find(hdrs, key.c, key.len+1, (void**)&hdr) == SUCCESS) {
  381. enc = (*hdr)->encode;
  382. }
  383. smart_str_free(&key);
  384. }
  385. val = master_to_zval(enc, trav TSRMLS_CC);
  386. add_assoc_zval(soap_headers, (char*)trav->name, val);
  387. }
  388. trav = trav->next;
  389. }
  390. }
  391. xmlFreeDoc(response);
  392. return TRUE;
  393. }