2004-01-29 17:27:06 +08:00
|
|
|
/*
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| PHP Version 5 |
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| Copyright (c) 1997-2004 The PHP Group |
|
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
| This source file is subject to version 3.0 of the PHP license, |
|
|
|
|
| that is bundled with this package in the file LICENSE, and is |
|
|
|
|
| available through the world-wide-web at the following url: |
|
|
|
|
| http://www.php.net/license/3_0.txt. |
|
|
|
|
| If you did not receive a copy of the PHP license and are unable to |
|
|
|
|
| obtain it through the world-wide-web, please send a note to |
|
|
|
|
| license@php.net so we can mail you a copy immediately. |
|
|
|
|
+----------------------------------------------------------------------+
|
2004-01-29 19:51:11 +08:00
|
|
|
| Authors: Brad Lafountain <rodif_bl@yahoo.com> |
|
|
|
|
| Shane Caraveo <shane@caraveo.com> |
|
|
|
|
| Dmitry Stogov <dmitry@zend.com> |
|
2004-01-29 17:27:06 +08:00
|
|
|
+----------------------------------------------------------------------+
|
|
|
|
*/
|
|
|
|
/* $Id$ */
|
|
|
|
|
2002-07-08 07:03:43 +08:00
|
|
|
#include "php_soap.h"
|
|
|
|
|
2004-01-06 00:44:01 +08:00
|
|
|
/* SOAP client calls this function to parse response from SOAP server */
|
|
|
|
int parse_packet_soap(zval *this_ptr, char *buffer, int buffer_size, sdlFunctionPtr fn, char *fn_name, zval *return_value TSRMLS_DC)
|
2002-07-08 07:03:43 +08:00
|
|
|
{
|
2004-01-19 08:10:41 +08:00
|
|
|
char* envelope_ns = NULL;
|
2002-07-08 07:03:43 +08:00
|
|
|
xmlDocPtr response;
|
2004-01-09 22:16:30 +08:00
|
|
|
xmlNodePtr trav, env, head, body, resp, cur, fault;
|
2004-01-30 23:07:19 +08:00
|
|
|
xmlAttrPtr attr;
|
2004-01-06 00:44:01 +08:00
|
|
|
int param_count = 0;
|
2004-01-13 17:31:50 +08:00
|
|
|
int old_error_reporting;
|
2004-01-27 00:19:29 +08:00
|
|
|
int soap_version;
|
2002-07-08 07:03:43 +08:00
|
|
|
|
2004-01-06 00:44:01 +08:00
|
|
|
ZVAL_NULL(return_value);
|
|
|
|
|
2004-01-13 17:31:50 +08:00
|
|
|
old_error_reporting = EG(error_reporting);
|
|
|
|
EG(error_reporting) &= ~(E_WARNING|E_NOTICE|E_USER_WARNING|E_USER_NOTICE);
|
|
|
|
|
2004-01-06 00:44:01 +08:00
|
|
|
/* Parse XML packet */
|
2002-07-08 07:03:43 +08:00
|
|
|
response = xmlParseMemory(buffer, buffer_size);
|
|
|
|
xmlCleanupParser();
|
2004-01-23 17:15:58 +08:00
|
|
|
|
2004-01-13 17:31:50 +08:00
|
|
|
EG(error_reporting) = old_error_reporting;
|
2004-01-23 17:15:58 +08:00
|
|
|
|
2002-11-22 21:41:51 +08:00
|
|
|
if (!response) {
|
2004-01-15 18:59:24 +08:00
|
|
|
add_soap_fault(this_ptr, "Client", "looks like we got no XML document", NULL, NULL TSRMLS_CC);
|
2004-01-06 00:44:01 +08:00
|
|
|
return FALSE;
|
2002-11-22 21:41:51 +08:00
|
|
|
}
|
2004-01-14 23:35:27 +08:00
|
|
|
if (xmlGetIntSubset(response) != NULL) {
|
2004-01-15 18:59:24 +08:00
|
|
|
add_soap_fault(this_ptr, "Client", "DTD are not supported by SOAP", NULL, NULL TSRMLS_CC);
|
2004-01-30 23:07:19 +08:00
|
|
|
xmlFreeDoc(response);
|
2004-01-14 23:35:27 +08:00
|
|
|
return FALSE;
|
|
|
|
}
|
2002-07-08 07:03:43 +08:00
|
|
|
|
2004-01-06 00:44:01 +08:00
|
|
|
/* Get <Envelope> element */
|
|
|
|
env = NULL;
|
2002-07-08 07:03:43 +08:00
|
|
|
trav = response->children;
|
2004-01-06 00:44:01 +08:00
|
|
|
while (trav != NULL) {
|
2004-01-08 23:27:50 +08:00
|
|
|
if (trav->type == XML_ELEMENT_NODE) {
|
2004-01-15 18:59:24 +08:00
|
|
|
if (env == NULL && node_is_equal_ex(trav,"Envelope",SOAP_1_1_ENV_NAMESPACE)) {
|
2004-01-10 02:22:03 +08:00
|
|
|
env = trav;
|
2004-01-15 18:59:24 +08:00
|
|
|
envelope_ns = SOAP_1_1_ENV_NAMESPACE;
|
2004-01-27 00:19:29 +08:00
|
|
|
soap_version = SOAP_1_1;
|
2004-01-15 18:59:24 +08:00
|
|
|
} else if (env == NULL && node_is_equal_ex(trav,"Envelope",SOAP_1_2_ENV_NAMESPACE)) {
|
2004-01-10 02:22:03 +08:00
|
|
|
env = trav;
|
2004-01-15 18:59:24 +08:00
|
|
|
envelope_ns = SOAP_1_2_ENV_NAMESPACE;
|
2004-01-27 00:19:29 +08:00
|
|
|
soap_version = SOAP_1_2;
|
2004-01-10 02:22:03 +08:00
|
|
|
} else {
|
2004-01-30 23:07:19 +08:00
|
|
|
add_soap_fault(this_ptr, "VersionMismatch", "Wrong Version", NULL, NULL TSRMLS_CC);
|
2004-01-09 22:16:30 +08:00
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
2004-01-10 02:22:03 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
trav = trav->next;
|
2004-01-06 00:44:01 +08:00
|
|
|
}
|
|
|
|
if (env == NULL) {
|
2004-01-30 23:07:19 +08:00
|
|
|
add_soap_fault(this_ptr, "Client", "looks like we got XML without \"Envelope\" element", NULL, NULL TSRMLS_CC);
|
2004-01-06 00:44:01 +08:00
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2002-07-08 07:03:43 +08:00
|
|
|
|
2004-01-30 23:07:19 +08:00
|
|
|
attr = env->properties;
|
|
|
|
while (attr != NULL) {
|
|
|
|
if (attr->ns == NULL) {
|
|
|
|
add_soap_fault(this_ptr, "Client", "A SOAP Envelope element cannot have non Namespace qualified attributes", NULL, NULL TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
2004-02-03 00:19:39 +08:00
|
|
|
} else if (attr_is_equal_ex(attr,"encodingStyle",SOAP_1_2_ENV_NAMESPACE)) {
|
|
|
|
if (soap_version == SOAP_1_2) {
|
|
|
|
add_soap_fault(this_ptr, "Client", "encodingStyle cannot be specified on the Envelope", NULL, NULL TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
} else if (strcmp(attr->children->content,SOAP_1_1_ENC_NAMESPACE) != 0) {
|
|
|
|
add_soap_fault(this_ptr, "Client", "Unknown data encoding style", NULL, NULL TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2004-01-30 23:07:19 +08:00
|
|
|
}
|
|
|
|
attr = attr->next;
|
|
|
|
}
|
|
|
|
|
2004-01-09 22:16:30 +08:00
|
|
|
/* Get <Header> element */
|
|
|
|
head = NULL;
|
|
|
|
trav = env->children;
|
|
|
|
while (trav != NULL && trav->type != XML_ELEMENT_NODE) {
|
2004-01-10 02:22:03 +08:00
|
|
|
trav = trav->next;
|
2004-01-09 22:16:30 +08:00
|
|
|
}
|
|
|
|
if (trav != NULL && node_is_equal_ex(trav,"Header",envelope_ns)) {
|
2004-01-10 02:22:03 +08:00
|
|
|
head = trav;
|
|
|
|
trav = trav->next;
|
2004-01-09 22:16:30 +08:00
|
|
|
}
|
|
|
|
|
2004-01-06 00:44:01 +08:00
|
|
|
/* Get <Body> element */
|
|
|
|
body = NULL;
|
2004-01-30 23:07:19 +08:00
|
|
|
while (trav != NULL && trav->type != XML_ELEMENT_NODE) {
|
|
|
|
trav = trav->next;
|
|
|
|
}
|
|
|
|
if (trav != NULL && node_is_equal_ex(trav,"Body",envelope_ns)) {
|
|
|
|
body = trav;
|
|
|
|
trav = trav->next;
|
|
|
|
}
|
|
|
|
while (trav != NULL && trav->type != XML_ELEMENT_NODE) {
|
2004-01-10 02:22:03 +08:00
|
|
|
trav = trav->next;
|
2004-01-06 00:44:01 +08:00
|
|
|
}
|
|
|
|
if (body == NULL) {
|
2004-01-30 23:07:19 +08:00
|
|
|
add_soap_fault(this_ptr, "Client", "Body must be present in a SOAP envelope", NULL, NULL TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2004-02-03 00:19:39 +08:00
|
|
|
attr = body->properties;
|
|
|
|
while (attr != NULL) {
|
|
|
|
if (attr->ns == NULL) {
|
|
|
|
if (soap_version == SOAP_1_2) {
|
|
|
|
add_soap_fault(this_ptr, "Client", "A SOAP Body element cannot have non Namespace qualified attributes", NULL, NULL TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
} else if (attr_is_equal_ex(attr,"encodingStyle",SOAP_1_2_ENV_NAMESPACE)) {
|
|
|
|
if (soap_version == SOAP_1_2) {
|
|
|
|
add_soap_fault(this_ptr, "Client", "encodingStyle cannot be specified on the Body", NULL, NULL TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
} else if (strcmp(attr->children->content,SOAP_1_1_ENC_NAMESPACE) != 0) {
|
|
|
|
add_soap_fault(this_ptr, "Client", "Unknown data encoding style", NULL, NULL TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
attr = attr->next;
|
2004-01-30 23:07:19 +08:00
|
|
|
}
|
|
|
|
if (trav != NULL && soap_version == SOAP_1_2) {
|
|
|
|
add_soap_fault(this_ptr, "Client", "A SOAP 1.2 envelope can contain only Header and Body", NULL, NULL TSRMLS_CC);
|
2004-01-06 00:44:01 +08:00
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2002-08-14 13:26:25 +08:00
|
|
|
|
2004-02-03 00:19:39 +08:00
|
|
|
if (head != NULL) {
|
|
|
|
attr = head->properties;
|
|
|
|
while (attr != NULL) {
|
|
|
|
if (attr->ns == NULL) {
|
|
|
|
add_soap_fault(this_ptr, "Client", "A SOAP Header element cannot have non Namespace qualified attributes", NULL, NULL TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
} else if (attr_is_equal_ex(attr,"encodingStyle",SOAP_1_2_ENV_NAMESPACE)) {
|
|
|
|
if (soap_version == SOAP_1_2) {
|
|
|
|
add_soap_fault(this_ptr, "Client", "encodingStyle cannot be specified on the Header", NULL, NULL TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
} else if (strcmp(attr->children->content,SOAP_1_1_ENC_NAMESPACE) != 0) {
|
|
|
|
add_soap_fault(this_ptr, "Client", "Unknown data encoding style", NULL, NULL TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
attr = attr->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-06 00:44:01 +08:00
|
|
|
/* Check if <Body> contains <Fault> element */
|
2004-01-08 23:27:50 +08:00
|
|
|
fault = get_node_ex(body->children,"Fault",envelope_ns);
|
2004-01-10 02:22:03 +08:00
|
|
|
if (fault != NULL) {
|
2004-01-06 00:44:01 +08:00
|
|
|
char *faultcode = NULL, *faultstring = NULL, *faultactor = NULL;
|
|
|
|
zval *details = NULL;
|
|
|
|
xmlNodePtr tmp;
|
2002-08-14 13:26:25 +08:00
|
|
|
|
2004-01-27 00:19:29 +08:00
|
|
|
if (soap_version == SOAP_1_1) {
|
|
|
|
tmp = get_node(fault->children,"faultcode");
|
|
|
|
if (tmp != NULL && tmp->children != NULL) {
|
|
|
|
faultcode = tmp->children->content;
|
|
|
|
}
|
2002-08-14 13:26:25 +08:00
|
|
|
|
2004-01-27 00:19:29 +08:00
|
|
|
tmp = get_node(fault->children,"faultstring");
|
|
|
|
if (tmp != NULL && tmp->children != NULL) {
|
|
|
|
faultstring = tmp->children->content;
|
|
|
|
}
|
2002-08-14 13:26:25 +08:00
|
|
|
|
2004-01-27 00:19:29 +08:00
|
|
|
tmp = get_node(fault->children,"faultactor");
|
|
|
|
if (tmp != NULL && tmp->children != NULL) {
|
|
|
|
faultactor = tmp->children->content;
|
|
|
|
}
|
2002-08-14 13:26:25 +08:00
|
|
|
|
2004-01-27 00:19:29 +08:00
|
|
|
tmp = get_node(fault->children,"detail");
|
|
|
|
if (tmp != NULL) {
|
|
|
|
details = master_to_zval(NULL, tmp);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
tmp = get_node(fault->children,"Code");
|
|
|
|
if (tmp != NULL && tmp->children != NULL) {
|
2004-01-27 01:11:05 +08:00
|
|
|
tmp = get_node(tmp->children,"Value");
|
|
|
|
if (tmp != NULL && tmp->children != NULL) {
|
|
|
|
faultcode = tmp->children->content;
|
|
|
|
}
|
2004-01-27 00:19:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
tmp = get_node(fault->children,"Reason");
|
|
|
|
if (tmp != NULL && tmp->children != NULL) {
|
2004-01-27 01:11:05 +08:00
|
|
|
/* TODO: lang attribute */
|
|
|
|
tmp = get_node(tmp->children,"Text");
|
|
|
|
if (tmp != NULL && tmp->children != NULL) {
|
|
|
|
faultstring = tmp->children->content;
|
|
|
|
}
|
2004-01-27 00:19:29 +08:00
|
|
|
}
|
2002-08-14 13:26:25 +08:00
|
|
|
|
2004-01-27 00:19:29 +08:00
|
|
|
tmp = get_node(fault->children,"Detail");
|
|
|
|
if (tmp != NULL) {
|
|
|
|
details = master_to_zval(NULL, tmp);
|
|
|
|
}
|
|
|
|
}
|
2004-01-06 00:44:01 +08:00
|
|
|
add_soap_fault(this_ptr, faultcode, faultstring, faultactor, details TSRMLS_CC);
|
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2002-08-14 13:26:25 +08:00
|
|
|
|
2004-01-06 00:44:01 +08:00
|
|
|
/* Parse content of <Body> element */
|
2004-01-13 23:58:01 +08:00
|
|
|
array_init(return_value);
|
2004-01-06 00:44:01 +08:00
|
|
|
resp = body->children;
|
2004-01-13 23:58:01 +08:00
|
|
|
while (resp != NULL && resp->type != XML_ELEMENT_NODE) {
|
|
|
|
resp = resp->next;
|
|
|
|
}
|
|
|
|
if (resp != NULL) {
|
|
|
|
if (fn != NULL && fn->binding && fn->binding->bindingType == BINDING_SOAP) {
|
|
|
|
/* Function has WSDL description */
|
|
|
|
sdlParamPtr *h_param, param = NULL;
|
|
|
|
xmlNodePtr val = NULL;
|
|
|
|
char *name, *ns = NULL;
|
|
|
|
zval* tmp;
|
2004-01-06 00:44:01 +08:00
|
|
|
sdlSoapBindingFunctionPtr fnb = (sdlSoapBindingFunctionPtr)fn->bindingAttributes;
|
|
|
|
int res_count = zend_hash_num_elements(fn->responseParameters);
|
|
|
|
|
|
|
|
zend_hash_internal_pointer_reset(fn->responseParameters);
|
|
|
|
while (zend_hash_get_current_data(fn->responseParameters, (void **)&h_param) == SUCCESS) {
|
|
|
|
param = (*h_param);
|
|
|
|
if (fnb->style == SOAP_DOCUMENT) {
|
|
|
|
name = param->encode->details.type_str;
|
|
|
|
ns = param->encode->details.ns;
|
|
|
|
} else {
|
|
|
|
name = fn->responseName;
|
|
|
|
/* ns = ? */
|
|
|
|
}
|
2002-08-14 13:26:25 +08:00
|
|
|
|
2004-01-06 00:44:01 +08:00
|
|
|
/* Get value of parameter */
|
|
|
|
cur = get_node_ex(resp, name, ns);
|
|
|
|
if (!cur) {
|
|
|
|
cur = get_node(resp, name);
|
|
|
|
/* TODO: produce warning invalid ns */
|
|
|
|
}
|
|
|
|
if (cur) {
|
|
|
|
if (fnb->style == SOAP_DOCUMENT) {
|
|
|
|
val = cur;
|
|
|
|
} else {
|
|
|
|
val = get_node(cur->children, param->paramName);
|
|
|
|
if (val == NULL && res_count == 1) {
|
|
|
|
val = get_node(cur->children, "return");
|
|
|
|
}
|
2004-01-09 22:16:30 +08:00
|
|
|
if (val == NULL && res_count == 1) {
|
|
|
|
val = get_node(cur->children, "result");
|
|
|
|
}
|
2004-01-06 00:44:01 +08:00
|
|
|
}
|
|
|
|
}
|
2002-08-14 13:26:25 +08:00
|
|
|
|
2004-01-06 00:44:01 +08:00
|
|
|
if (!val) {
|
|
|
|
/* TODO: may be "nil" is not OK? */
|
|
|
|
MAKE_STD_ZVAL(tmp);
|
|
|
|
ZVAL_NULL(tmp);
|
|
|
|
/*
|
2004-01-15 18:59:24 +08:00
|
|
|
add_soap_fault(this_ptr, "Client", "Can't find response data", NULL, NULL TSRMLS_CC);
|
2004-01-06 00:44:01 +08:00
|
|
|
xmlFreeDoc(response);
|
|
|
|
return FALSE;
|
|
|
|
*/
|
|
|
|
} else {
|
|
|
|
/* Decoding value of parameter */
|
|
|
|
if (param != NULL) {
|
|
|
|
tmp = master_to_zval(param->encode, val);
|
|
|
|
} else {
|
2004-01-26 17:51:07 +08:00
|
|
|
tmp = master_to_zval(NULL, val);
|
2002-07-08 07:03:43 +08:00
|
|
|
}
|
|
|
|
}
|
2004-01-06 00:44:01 +08:00
|
|
|
add_assoc_zval(return_value, param->paramName, tmp);
|
|
|
|
|
|
|
|
param_count++;
|
|
|
|
|
|
|
|
zend_hash_move_forward(fn->responseParameters);
|
|
|
|
}
|
2004-01-13 23:58:01 +08:00
|
|
|
} else {
|
|
|
|
/* Function hasn't WSDL description */
|
2004-01-06 00:44:01 +08:00
|
|
|
xmlNodePtr val;
|
2004-01-13 23:58:01 +08:00
|
|
|
val = resp->children;
|
2004-01-06 00:44:01 +08:00
|
|
|
while (val != NULL) {
|
2004-01-10 02:22:03 +08:00
|
|
|
while (val && val->type != XML_ELEMENT_NODE) {
|
2004-01-06 00:44:01 +08:00
|
|
|
val = val->next;
|
|
|
|
}
|
|
|
|
if (val != NULL) {
|
2004-01-26 23:13:45 +08:00
|
|
|
if (!node_is_equal_ex(val,"result",RPC_SOAP12_NAMESPACE)) {
|
|
|
|
zval *tmp;
|
|
|
|
|
|
|
|
tmp = master_to_zval(NULL, val);
|
|
|
|
if (val->name) {
|
|
|
|
add_assoc_zval(return_value, (char*)val->name, tmp);
|
|
|
|
} else {
|
|
|
|
add_next_index_zval(return_value, tmp);
|
|
|
|
}
|
|
|
|
++param_count;
|
2002-07-08 07:03:43 +08:00
|
|
|
}
|
2004-01-06 00:44:01 +08:00
|
|
|
val = val->next;
|
2002-07-08 07:03:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-01-06 00:44:01 +08:00
|
|
|
|
|
|
|
if (Z_TYPE_P(return_value) == IS_ARRAY) {
|
|
|
|
if (param_count == 0) {
|
2004-01-10 02:22:03 +08:00
|
|
|
zval_dtor(return_value);
|
|
|
|
ZVAL_NULL(return_value);
|
2004-01-06 00:44:01 +08:00
|
|
|
} else if (param_count == 1) {
|
2004-01-10 02:22:03 +08:00
|
|
|
zval *tmp = *(zval**)Z_ARRVAL_P(return_value)->pListHead->pData;
|
|
|
|
tmp->refcount++;
|
|
|
|
zval_dtor(return_value);
|
|
|
|
*return_value = *tmp;
|
|
|
|
FREE_ZVAL(tmp);
|
2004-01-06 00:44:01 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-07-08 07:03:43 +08:00
|
|
|
xmlFreeDoc(response);
|
|
|
|
return TRUE;
|
|
|
|
}
|