Port to PEAR and more | // | Authors: Dietrich Ayala Original Author | // +----------------------------------------------------------------------+ // // $Id: Server.php,v 1.44.2.1 2003/12/20 21:21:59 sklar Exp $ // require_once 'SOAP/Base.php'; require_once 'SOAP/Fault.php'; require_once 'SOAP/Parser.php'; require_once 'SOAP/Value.php'; require_once 'SOAP/WSDL.php'; $soap_server_fault = null; function SOAP_ServerErrorHandler($errno, $errmsg, $filename, $linenum, $vars) { global $soap_server_fault; $detail = "Errno: $errno\nFilename: $filename\nLineno: $linenum\n"; // XXX very strange behaviour with error handling if we =& here. $soap_server_fault = new SOAP_Fault($errmsg, 'Server', 'PHP', $detail); } /** * SOAP::Server * SOAP Server Class * * originaly based on SOAPx4 by Dietrich Ayala http://dietrich.ganx4.com/soapx4 * * @access public * @version $Id: Server.php,v 1.44.2.1 2003/12/20 21:21:59 sklar Exp $ * @package SOAP::Client * @author Shane Caraveo Conversion to PEAR and updates * @author Dietrich Ayala Original Author */ class SOAP_Server extends SOAP_Base { /** * * @var array */ var $dispatch_map = array(); // create empty dispatch map var $dispatch_objects = array(); var $soapobject = NULL; var $call_methodname = NULL; var $callHandler = NULL; var $callValidation = true; /** * * @var string */ var $headers = ''; /** * * @var string */ var $request = ''; /** * * @var string XML-Encoding */ var $xml_encoding = SOAP_DEFAULT_ENCODING; var $response_encoding = 'UTF-8'; var $result = 'successful'; // for logging interop results to db var $endpoint = ''; // the uri to ME! var $service = ''; //soapaction header var $method_namespace = NULL; var $__options = array('use'=>'encoded','style'=>'rpc','parameters'=>0); function SOAP_Server($options=NULL) { ini_set('track_errors',1); parent::SOAP_Base('Server'); if (is_array($options)) { if (isset($options['use'])) $this->__options['use'] = $options['use']; if (isset($options['style'])) $this->__options['style'] = $options['style']; if (isset($options['parameters'])) $this->__options['parameters'] = $options['parameters']; } $this->_section5 = TRUE; // assume we encode with section 5 if ($this->__options['use']=='literal') $this->_section5 = FALSE; } function _getContentEncoding($content_type) { // get the character encoding of the incoming request // treat incoming data as UTF-8 if no encoding set $this->xml_encoding = 'UTF-8'; if (strpos($content_type,'=')) { $enc = strtoupper(str_replace('"',"",substr(strstr($content_type,'='),1))); if (!in_array($enc, $this->_encodings)) { return FALSE; } $this->xml_encoding = $enc; } return TRUE; } // parses request and posts response function service($data, $endpoint = '', $test = FALSE) { $response = NULL; $attachments = array(); $headers = array(); $useEncoding = 'DIME'; // figure out our endpoint $this->endpoint = $endpoint; if (!$test && !$this->endpoint) { // we'll try to build our endpoint $this->endpoint = 'http://'.$_SERVER['SERVER_NAME']; if ($_SERVER['SERVER_PORT']) $this->endpoint .= ':'.$_SERVER['SERVER_PORT']; $this->endpoint .= $_SERVER['SCRIPT_NAME']; } // get the character encoding of the incoming request // treat incoming data as UTF-8 if no encoding set if (isset($_SERVER['CONTENT_TYPE'])) { if (strcasecmp($_SERVER['CONTENT_TYPE'],'application/dime')==0) { $this->_decodeDIMEMessage($data,$headers,$attachments); $useEncoding = 'DIME'; } else if (stristr($_SERVER['CONTENT_TYPE'],'multipart/related')) { // this is a mime message, lets decode it. $data = 'Content-Type: '.stripslashes($_SERVER['CONTENT_TYPE'])."\r\n\r\n".$data; $this->_decodeMimeMessage($data,$headers,$attachments); $useEncoding = 'Mime'; } if (!isset($headers['content-type'])) { $headers['content-type'] = stripslashes($_SERVER['CONTENT_TYPE']); } if (!$this->fault && !$this->_getContentEncoding($headers['content-type'])) { $this->xml_encoding = SOAP_DEFAULT_ENCODING; // an encoding we don't understand, return a fault $this->_raiseSoapFault('Unsupported encoding, use one of ISO-8859-1, US-ASCII, UTF-8','','','Server'); } } // if this is not a POST with Content-Type text/xml, try to return a WSDL file if (!$this->fault && !$test && ($_SERVER['REQUEST_METHOD'] != 'POST' || strncmp($headers['content-type'],'text/xml',8) != 0)) { // this is not possibly a valid soap request, try to return a WSDL file $this->_raiseSoapFault("Invalid SOAP request, must be POST with content-type: text/xml, got: ".(isset($headers['content-type'])?$headers['content-type']:'Nothing!'),'','','Server'); } if (!$this->fault) { // $response is a soap_msg object $soap_msg = $this->parseRequest($data, $attachments); // handle Mime or DIME encoding // XXX DIME Encoding should move to the transport, do it here for now // and for ease of getting it done if (count($this->__attachments)) { if ($useEncoding == 'Mime') { $soap_msg = $this->_makeMimeMessage($soap_msg); } else { // default is dime $soap_msg = $this->_makeDIMEMessage($soap_msg); $header['Content-Type'] = 'application/dime'; } if (PEAR::isError($soap_msg)) { return $this->raiseSoapFault($soap_msg); } } if (is_array($soap_msg)) { $response = $soap_msg['body']; if (count($soap_msg['headers'])) { $header = $soap_msg['headers']; } } else { $response = $soap_msg; } } // make distinction between the different choice of installation, // running php as cgi or as a module if(stristr(php_sapi_name(),'cgi')==0) $hdrs_type = 'Status:'; else $hdrs_type = 'HTTP/1.1'; if ($this->fault) { $hdrs = "$hdrs_type 500 Soap Fault\r\n"; $response = $this->fault->message(); } else { $hdrs = "$hdrs_type 200 OK\r\n"; } header($hdrs); $header['Server'] = SOAP_LIBRARY_NAME; if (!isset($header['Content-Type'])) $header['Content-Type'] = "text/xml; charset=$this->response_encoding"; $header['Content-Length'] = strlen($response); reset($header); foreach ($header as $k => $v) { header("$k: $v"); $hdrs .= "$k: $v\r\n"; } $this->response = $hdrs . "\r\n" . $response; print $response; } function &callMethod($methodname, &$args) { global $soap_server_fault; $soap_server_fault = null; if ($this->callHandler) { return @call_user_func_array($this->callHandler,array($methodname,$args)); } set_error_handler('SOAP_ServerErrorHandler'); if ($args) { // call method with parameters if (isset($this->soapobject) && is_object($this->soapobject)) { $ret = @call_user_func_array(array(&$this->soapobject, $methodname),$args); } else { $ret = @call_user_func_array($methodname,$args); } } else { // call method w/ no parameters if (is_object($this->soapobject)) { $ret = @call_user_func(array(&$this->soapobject, $methodname)); } else { $ret = @call_user_func($methodname); } } restore_error_handler(); return is_null($soap_server_fault) ? $ret : $soap_server_fault; } // create soap_val object w/ return values from method, use method signature to determine type function buildResult(&$method_response, &$return_type, $return_name='return', $namespace = '') { if (gettype($method_response) == 'object' && is_a($method_response,'soap_value')) { $return_val = array($method_response); } else { if (is_array($return_type) && is_array($method_response)) { $i = 0; foreach ($return_type as $key => $type) { if (is_numeric($key)) $key = 'item'; if (is_a($method_response[$i],'soap_value')) { $return_val[] = $method_response[$i++]; } else { $qn =& new QName($key, $namespace); $return_val[] =& new SOAP_Value($qn->fqn(),$type,$method_response[$i++]); } } } else { if (is_array($return_type)) { $keys = array_keys($return_type); if (!is_numeric($keys[0])) $return_name = $keys[0]; $values = array_values($return_type); $return_type = $values[0]; } $qn =& new QName($return_name, $namespace); $return_val = array(); $return_val[] =& new SOAP_Value($qn->fqn(),$return_type,$method_response); } } return $return_val; } function parseRequest($data = '', $attachments = null) { // parse response, get soap parser obj $parser =& new SOAP_Parser($data,$this->xml_encoding,$attachments); // if fault occurred during message parsing if ($parser->fault) { $this->fault = $parser->fault; return null; } //******************************************************* // handle message headers $request_headers = $parser->getHeaders(); $header_results = array(); if ($request_headers) { if (!is_a($request_headers,'soap_value')) { $this->_raiseSoapFault("parser did not return SOAP_Value object: $request_headers",'','','Server'); return null; } if ($request_headers->value) { // handle headers now foreach ($request_headers->value as $header_val) { $f_exists = $this->validateMethod($header_val->name, $header_val->namespace); # XXX this does not take into account message routing yet $myactor = ( !$header_val->actor || $header_val->actor == 'http://schemas.xmlsoap.org/soap/actor/next' || $header_val->actor == $this->endpoint); if (!$f_exists && $header_val->mustunderstand && $myactor) { $this->_raiseSoapFault("I don't understand header $header_val->name.",'','','MustUnderstand'); return null; } // we only handle the header if it's for us $isok = $f_exists && $myactor; if ($isok) { # call our header now! $header_method = $header_val->name; $header_data = array($this->_decode($header_val)); // if there are parameters to pass $hr =& $this->callMethod($header_method, $header_data); # if they return a fault, then it's all over! if (PEAR::isError($hr)) { $this->_raiseSoapFault($hr); return null; } $header_results[] = array_shift($this->buildResult($hr, $this->return_type, $header_method, $header_val->namespace)); } } } } //******************************************************* // handle the method call // evaluate message, getting back a SOAP_Value object $this->call_methodname = $this->methodname = $parser->root_struct_name[0]; // figure out the method_namespace $this->method_namespace = $parser->message[$parser->root_struct[0]]['namespace']; if ($this->_wsdl) { $this->_setSchemaVersion($this->_wsdl->xsd); $dataHandler = $this->_wsdl->getDataHandler($this->methodname,$this->method_namespace); if ($dataHandler) $this->call_methodname = $this->methodname = $dataHandler; $this->_portName = $this->_wsdl->getPortName($this->methodname); if (PEAR::isError($this->_portName)) { return $this->_raiseSoapFault($this->_portName); } $opData = $this->_wsdl->getOperationData($this->_portName, $this->methodname); if (PEAR::isError($opData)) { return $this->_raiseSoapFault($opData); } $this->__options['style'] = $opData['style']; $this->__options['use'] = $opData['output']['use']; $this->__options['parameters'] = $opData['parameters']; } // does method exist? if (!$this->methodname || !$this->validateMethod($this->methodname,$this->method_namespace)) { $this->_raiseSoapFault("method '{{$this->method_namespace}}$this->methodname' not defined in service",'','','Server'); return NULL; } if (!$request_val = $parser->getResponse()) { return NULL; } if (!is_a($request_val,'soap_value')) { $this->_raiseSoapFault("parser did not return SOAP_Value object: $request_val",'','','Server'); return NULL; } // verify that SOAP_Value objects in request match the methods signature if (!$this->verifyMethod($request_val)) { // verifyMethod creates the fault return NULL; } // need to set special error detection inside the value class // so as to differentiate between no params passed, and an error decoding $request_data = $this->__decodeRequest($request_val); if (PEAR::isError($request_data)) { return $this->_raiseSoapFault($request_data); } $method_response =& $this->callMethod($this->call_methodname, $request_data); if (PEAR::isError($method_response)) { $this->_raiseSoapFault($method_response); return null; } if ($this->__options['parameters'] || !$method_response || $this->__options['style']=='rpc') { // get the method result if (is_null($method_response)) $return_val = NULL; else $return_val = $this->buildResult($method_response, $this->return_type); $qn =& new QName($this->methodname.'Response',$this->method_namespace); $methodValue =& new SOAP_Value($qn->fqn(), 'Struct', $return_val); } else { $methodValue =& $method_response; } return $this->_makeEnvelope($methodValue, $header_results, $this->response_encoding); } function &__decodeRequest($request,$shift=false) { if (!$request) return NULL; // check for valid response if (PEAR::isError($request)) { return $this->_raiseSoapFault($request); } else if (!is_a($request,'soap_value')) { return $this->_raiseSoapFault("Invalid data in server::__decodeRequest"); } // decode to native php datatype $requestArray = $this->_decode($request); // fault? if (PEAR::isError($requestArray)) { return $this->_raiseSoapFault($requestArray); } if (is_object($requestArray)&& get_class($requestArray) == 'stdClass') { $requestArray = get_object_vars($requestArray); } else if ($this->__options['style']=='document') { $requestArray = array($requestArray); } if (is_array($requestArray)) { if (isset($requestArray['faultcode']) || isset($requestArray['SOAP-ENV:faultcode'])) { $faultcode = $faultstring = $faultdetail = $faultactor = ''; foreach ($requestArray as $k => $v) { if (stristr($k,'faultcode')) $faultcode = $v; if (stristr($k,'faultstring')) $faultstring = $v; if (stristr($k,'detail')) $faultdetail = $v; if (stristr($k,'faultactor')) $faultactor = $v; } return $this->_raiseSoapFault($faultstring, $faultdetail, $faultactor, $faultcode); } // return array of return values if ($shift && count($requestArray) == 1) { return array_shift($requestArray); } return $requestArray; } return $requestArray; } function verifyMethod($request) { if (!$this->callValidation) return TRUE; //return true; $params = $request->value; // get the dispatch map if one exists $map = NULL; if (array_key_exists($this->methodname, $this->dispatch_map)) { $map = $this->dispatch_map[$this->methodname]; } else if (isset($this->soapobject)) { if (method_exists($this->soapobject, '__dispatch')) { $map = $this->soapobject->__dispatch($this->methodname); } else if (method_exists($this->soapobject, $this->methodname)) { // no map, all public functions are soap functions return TRUE; } } if (!$map) { $this->_raiseSoapFault("soap request specified an unhandled method '$this->methodname'",'','','Client'); return FALSE; } // if we aliased the soap method name to a php function, // change the call_method so we do the right thing. if (array_key_exists('alias',$map) && !empty($map['alias'])) { $this->call_methodname = $map['alias']; } // if there are input parameters required... if ($sig = $map['in']) { $this->input_value = count($sig); $this->return_type = $this->getReturnType($map['out']); if (is_array($params)) { // validate the number of parameters if (count($params) == count($sig)) { // make array of param types foreach ($params as $param) { $p[] = strtolower($param->type); } $sig_t = array_values($sig); // validate each param's type for($i=0; $i < count($p); $i++) { // type not match // if soap types do not match, we ok it if the mapped php types match // this allows using plain php variables to work (ie. stuff like Decimal would fail otherwise) // we only error if the types exist in our type maps, and they differ if (strcasecmp($sig_t[$i],$p[$i])!=0 && (isset($this->_typemap[SOAP_XML_SCHEMA_VERSION][$sig_t[$i]]) && strcasecmp($this->_typemap[SOAP_XML_SCHEMA_VERSION][$sig_t[$i]],$this->_typemap[SOAP_XML_SCHEMA_VERSION][$p[$i]])!=0)) { $param = $params[$i]; $this->_raiseSoapFault("soap request contained mismatching parameters of name $param->name had type [{$p[$i]}], which did not match signature's type: [{$sig_t[$i]}], matched? ".(strcasecmp($sig_t[$i],$p[$i])),'','','Client'); return false; } } return true; // oops, wrong number of paramss } else { $this->_raiseSoapFault("soap request contained incorrect number of parameters. method '$this->methodname' required ".count($sig).' and request provided '.count($params),'','','Client'); return false; } // oops, no params... } else { $this->_raiseSoapFault("soap request contained incorrect number of parameters. method '$this->methodname' requires ".count($sig).' parameters, and request provided none','','','Client'); return false; } // no params } // we'll try it anyway return true; } // get string return type from dispatch map function getReturnType($returndata) { if (is_array($returndata)) { if (count($returndata) > 1) { return $returndata; } $type = array_shift($returndata); return $type; } return false; } function validateMethod($methodname, $namespace = NULL) { unset($this->soapobject); if (!$this->callValidation) return TRUE; # no soap access to private functions if ($methodname[0] == '_') return FALSE; /* if it's in our function list, ok */ if (array_key_exists($methodname, $this->dispatch_map) && (!$namespace || !array_key_exists('namespace', $this->dispatch_map[$methodname]) || $namespace == $this->dispatch_map[$methodname]['namespace'])) { if (array_key_exists('namespace', $this->dispatch_map[$methodname])) $this->method_namespace = $this->dispatch_map[$methodname]['namespace']; return TRUE; } /* if it's in an object, it's ok */ if (isset($this->dispatch_objects[$namespace])) { $c = count($this->dispatch_objects[$namespace]); for ($i=0; $i < $c; $i++) { $obj =& $this->dispatch_objects[$namespace][$i]; // if we have a dispatch map, and the function is not // in the dispatch map, then it is not callable! if (method_exists($obj, '__dispatch')) { if ($obj->__dispatch($methodname)) { $this->method_namespace = $namespace; $this->soapobject =& $obj; return TRUE; } } else if (method_exists($obj, $methodname)) { $this->method_namespace = $namespace; $this->soapobject =& $obj; return TRUE; } } } return FALSE; } function addObjectMap(&$obj, $namespace = null, $service_name = 'Default', $service_desc = '') { if (!$namespace) { if (isset($obj->namespace)) { // XXX a bit of backwards compatibility $namespace = $obj->namespace; } else { $this->_raiseSoapFault('No namespace provided for class!','','','Server'); return false; } } if (!isset($this->dispatch_objects[$namespace])) { $this->dispatch_objects[$namespace] = array(); } $this->dispatch_objects[$namespace][] =& $obj; // Create internal WSDL structures for object // XXX Because some internal workings of PEAR::SOAP decide whether to // do certain things by the presence or absence of _wsdl, we should // only create a _wsdl structure if we know we can fill it; if // __dispatch_map or __typedef for the object is missing, we should // avoid creating it. Later, when we are using PHP 5 introspection, // we will be able to make the data for all objects without any extra // information from the developers, and this condition should be // dropped. // XXX Known issue: if imported WSDL (bindWSDL) or another WSDL source // is used to add _wsdl structure information, then addObjectWSDL is // used, there is a high possibility of _wsdl data corruption; // therefore you should avoid using __dispatch_map/__typedef definitions // AND other WSDL data sources in the same service. We exclude classes // that don't have __typedefs to allow external WSDL files to be used // with classes with no internal type definitions (the types are defined // in the WSDL file). When addObjectWSDL is refactored to not cause // corruption, this restriction can be relaxed. // In summry, if you add an object with both a dispatch map and type // definitions, then previous WSDL file operation and type definitions // will be overwritten. if (isset($obj->__dispatch_map) && isset($obj->__typedef)) { $this->addObjectWSDL($obj, $namespace, $service_name, $service_desc); } return true; } // add a method to the dispatch map function addToMap($methodname, $in, $out, $namespace = NULL, $alias=NULL) { if (!function_exists($methodname)) { $this->_raiseSoapFault("error mapping function\n",'','','Server'); return false; } $this->dispatch_map[$methodname]['in'] = $in; $this->dispatch_map[$methodname]['out'] = $out; $this->dispatch_map[$methodname]['alias'] = $alias; if ($namespace) $this->dispatch_map[$methodname]['namespace'] = $namespace; return true; } function setCallHandler($callHandler, $validation=true) { $this->callHandler = $callHandler; $this->callValidation = $validation; } /** * @deprecated use bindWSDL from now on */ function bind($wsdl_url) { $this->bindWSDL($wsdl_url); } /** * @param string a url to a WSDL resource * @return void */ function bindWSDL($wsdl_url) { // instantiate wsdl class $this->_wsdl =& new SOAP_WSDL($wsdl_url); if ($this->_wsdl->fault) { $this->_raiseSoapFault($this->_wsdl->fault); } } /** * @return void */ function addObjectWSDL(&$wsdl_obj, $targetNamespace, $service_name, $service_desc = '') { if (!isset($this->_wsdl)) { $this->_wsdl =& new SOAP_WSDL; } $this->_wsdl->parseObject($wsdl_obj, $targetNamespace, $service_name, $service_desc); if ($this->_wsdl->fault) { $this->_raiseSoapFault($this->_wsdl->fault); } } } ?>