2001-07-16 08:05:33 +02:00
< ? php
2001-08-18 16:37:52 +02:00
// SOAP server class
2001-07-16 08:05:33 +02:00
2001-08-18 16:37:52 +02:00
// for example usage, see the test_server.php file.
2001-07-16 08:05:33 +02:00
2001-08-18 16:37:52 +02:00
/* changelog :
2001 - 07 - 05
- detection of character encoding in Content - Type header . server
will now call the soap_parser object with the specified encoding
- server will now return the Content - Type header with the sender ' s encoding type specified
must still learn more bout encoding , and figure out what i have to do to be able to
make sure that my response is * actually * encoded correctly
2001 - 07 - 21
- force proper error reporting for windows compatibility
2001 - 07 - 27
- get_all_headers () check for windows compatibility
*/
/* $Id$ */
2001-07-30 05:40:08 +02:00
2001-08-18 16:37:52 +02:00
// make errors handle properly in windows
error_reporting ( 2039 );
2001-07-30 05:40:08 +02:00
2001-08-18 16:37:52 +02:00
class soap_server
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
function soap_server ()
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
// create empty dispatch map
$this -> dispatch_map = array ();
$this -> debug_flag = True ;
$this -> debug_str = '' ;
$this -> headers = '' ;
$this -> request = '' ;
$this -> xml_encoding = 'UTF-8' ;
$this -> fault = false ;
$this -> fault_code = '' ;
$this -> fault_str = '' ;
$this -> fault_actor = '' ;
// for logging interop results to db
$this -> result = 'successful' ;
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
// parses request and posts response
function service ( $data )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
// $response is a soap_msg object
$response = $this -> parse_request ( $data );
$this -> debug ( " parsed request and got an object of this class ' " . get_class ( $response ) . " ' " );
$this -> debug ( " server sending... " );
// pass along the debug string
if ( $this -> debug_flag )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
$response -> debug ( $this -> debug_str );
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
$payload = $response -> serialize ();
// print headers
if ( $this -> fault )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
$header [] = " HTTP/1.0 500 Internal Server Error \r \n " ;
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
else
2001-07-30 05:40:08 +02:00
{
2001-08-18 16:37:52 +02:00
$header [] = " HTTP/1.0 200 OK \r \n " ;
$header [] = " Status: 200 \r \n " ;
2001-07-30 05:40:08 +02:00
}
2001-08-18 16:37:52 +02:00
$header [] = " Server: SOAPx4 Server v0.344359s \r \n " ;
$header [] = " Connection: Close \r \n " ;
$header [] = " Content-Type: text/xml; charset= $this->xml_encoding\r\n " ;
$header [] = " Content-Length: " . strlen ( $payload ) . " \r \n \r \n " ;
@ reset ( $header );
while ( list ( $null , $hdr ) = @ each ( $header ))
/* foreach($header as $hdr) */
2001-07-30 05:40:08 +02:00
{
2001-08-18 16:37:52 +02:00
header ( $hdr );
2001-07-30 05:40:08 +02:00
}
2001-08-18 16:37:52 +02:00
print $payload ;
2001-07-16 08:05:33 +02:00
}
2001-08-02 10:31:48 +02:00
2001-08-18 16:37:52 +02:00
function parse_request ( $data = '' )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
global $HTTP_SERVER_VARS ;
2001-07-16 22:36:22 +02:00
2001-08-18 16:37:52 +02:00
$this -> debug ( 'entering parse_request() on ' . date ( 'H:i Y-m-d' ));
$request_uri = $HTTP_SERVER_VARS [ 'REQUEST_URI' ];
$this -> debug ( " request uri: $request_uri " );
// get headers
if ( function_exists ( 'getallheaders' ))
{
$this -> headers = getallheaders ();
while ( list ( $k , $v ) = @ each ( $this -> headers ))
/* foreach($this->headers as $k=>$v) */
{
$dump .= " $k : $v\r\n " ;
}
// get SOAPAction header
if ( $this -> headers [ 'SOAPAction' ])
{
$this -> SOAPAction = str_replace ( '"' , '' , $this -> headers [ 'SOAPAction' ]);
$this -> service = $this -> SOAPAction ;
}
// get character encoding
if ( ereg ( " = " , $this -> headers [ 'Content-Type' ]))
{
$this -> xml_encoding = str_replace ( " \" " , " " , substr ( strstr ( $this -> headers [ " Content-Type " ], " = " ), 1 ));
}
elseif ( ereg ( " ^text/xml " , $this -> headers [ 'Content-Type' ]))
{
$this -> xml_encoding = 'us-ascii' ;
}
$this -> debug ( " got encoding: $this->xml_encoding " );
}
$this -> request = $dump . " \r \n \r \n " . $data ;
// parse response, get soap parser obj
$parser = CreateObject ( 'phpgwapi.soap_parser' , $data );
// get/set methodname
$this -> methodname = $parser -> root_struct_name ;
$this -> debug ( " method name: $this->methodname " );
// does method exist?
if ( function_exists ( $this -> methodname ))
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
$this -> debug ( " method ' $this->methodname ' exists " );
}
else
{
/* phpgroupware customization - createobject based on methodname */
list ( $app , $class , $method ) = explode ( '.' , $this -> methodname );
$obj = CreateObject ( sprintf ( '%s.%s' , $app , $class ));
/*
// "method not found" fault here
$this -> debug ( " method ' $obj->method ' not found! " );
$this -> result = " fault: method not found " ;
$this -> make_fault ( " Server " , " method ' $obj->method ' not defined in service ' $this->service ' " );
return $this -> fault ();
*/
}
// if fault occurred during message parsing
if ( $parser -> fault ())
{
// parser debug
$this -> debug ( $parser -> debug_str );
$this -> result = 'fault: error in msg parsing or eval' ;
$this -> make_fault ( " Server " , " error in msg parsing or eval: \n " . $parser -> get_response ());
// return soapresp
2001-07-16 08:05:33 +02:00
return $this -> fault ();
2001-08-18 16:37:52 +02:00
// else successfully parsed request into soapval object
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
else
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
// get eval_str
$this -> debug ( 'calling parser->get_response()' );
// evaluate it, getting back a soapval object
if ( ! $request_val = $parser -> get_response ())
2001-07-16 22:36:22 +02:00
{
2001-08-18 16:37:52 +02:00
return $this -> fault ();
2001-07-16 22:36:22 +02:00
}
2001-08-18 16:37:52 +02:00
// parser debug
$this -> debug ( $parser -> debug_str );
if ( get_class ( $request_val ) == 'soapval' )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
if ( is_object ( $obj ))
2001-07-16 22:36:22 +02:00
{
2001-08-18 16:37:52 +02:00
/* Add the function to the server map */
$in = " array(' " . implode ( " ',' " , $obj -> soap_functions [ $method ][ 'in' ]) . " ') " ;
$out = " array(' " . implode ( " ',' " , $obj -> soap_functions [ $method ][ 'out' ]) . " ') " ;
$evalmap = " \$ this->add_to_map( \$ this->methodname, $in , $out ); " ;
eval ( $evalmap );
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
/* verify that soapval objects in request match the methods signature */
if ( $this -> verify_method ( $request_val ))
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
$this -> debug ( " request data - name: $request_val->name , type: $request_val->type , value: $request_val->value " );
if ( $this -> input_value )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
/* decode the soapval object, and pass resulting values to the requested method */
if ( ! $request_data = $request_val -> decode ())
2001-07-16 22:36:22 +02:00
{
2001-08-18 16:37:52 +02:00
$this -> make_fault ( " Server " , " Unable to decode response from soapval object into native php type. " );
return $this -> fault ();
2001-07-16 22:36:22 +02:00
}
2001-08-18 16:37:52 +02:00
$this -> debug ( " request data: $request_data " );
}
/* if there are return values */
if ( $this -> return_type = $this -> get_return_type ())
{
$this -> debug ( " got return type: ' $this->return_type ' " );
/* if there are parameters to pass */
if ( $request_data )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
if ( is_object ( $obj ))
{
$code = " \$ method_response = call_user_method( \$ method, \$ obj, " ;
$this -> debug ( " about to call method ' $class -> $method ' " );
}
else
{
$code = " \$ method_response = call_user_func( \$ this->methodname, " ;
$this -> debug ( " about to call method ' $this->methodname ' " );
}
/* call method with parameters */
$this -> debug ( " about to call method ' $class -> $method ' " );
while ( list ( $x , $y ) = each ( $request_data ))
{
$code .= " \$ request_data[ $x ] " . ',' ;
}
$code = substr ( $code , 0 , - 1 ) . ');' ;
if ( eval ( $code ))
{
if ( is_object ( $obj ))
{
$this -> make_fault ( " Server " , " Method call failed for ' $obj->method ' with params: " . join ( " , " , $request_data ));
}
else
{
$this -> make_fault ( " Server " , " Method call failed for ' $this->methodname ' with params: " . join ( " , " , $request_data ));
}
return $this -> fault ();
}
2001-07-16 22:36:22 +02:00
}
2001-08-18 16:37:52 +02:00
else
2001-07-16 22:36:22 +02:00
{
2001-08-18 16:37:52 +02:00
/* call method w/ no parameters */
2001-07-16 22:36:22 +02:00
if ( is_object ( $obj ))
{
2001-08-18 16:37:52 +02:00
$this -> debug ( " about to call method ' $obj->method ' " );
if ( ! $method_response = call_user_method ( $method , $obj ))
{
$this -> make_fault ( " Server " , " Method call failed for ' $obj->method ' with no params " );
return $this -> fault ();
}
2001-07-16 22:36:22 +02:00
}
else
{
2001-08-18 16:37:52 +02:00
$this -> debug ( " about to call method ' $this->methodname ' " );
if ( ! $method_response = call_user_func ( $this -> methodname ))
{
$this -> make_fault ( " Server " , " Method call failed for ' $this->methodname ' with no params " );
return $this -> fault ();
}
2001-07-16 22:36:22 +02:00
}
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
/* no return values */
2001-07-16 08:05:33 +02:00
}
else
{
2001-08-18 16:37:52 +02:00
if ( $request_data )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
/* call method with parameters */
$code = " \$ method_response = call_user_method( \$ method, \$ obj, " ;
while ( list ( $x , $y ) = each ( $request_data ))
2001-07-16 22:36:22 +02:00
{
2001-08-18 16:37:52 +02:00
$code .= " \$ request_data[ $x ] " . ',' ;
2001-07-16 22:36:22 +02:00
}
2001-08-18 16:37:52 +02:00
$code = substr ( $code , 0 , - 1 ) . ');' ;
$this -> debug ( " about to call method ' $obj->method ' " );
eval ( $code );
2001-07-16 22:36:22 +02:00
}
else
{
2001-08-18 16:37:52 +02:00
/* call method w/ no parameters */
$this -> debug ( " about to call method ' $obj->method ' " );
call_user_method ( $method , $obj );
2001-07-16 08:05:33 +02:00
}
}
2001-08-18 16:37:52 +02:00
/* return fault */
if ( get_class ( $method_response ) == 'soapmsg' )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
if ( eregi ( 'fault' , $method_response -> value -> name ))
2001-07-16 22:36:22 +02:00
{
2001-08-18 16:37:52 +02:00
$this -> fault = True ;
2001-07-16 22:36:22 +02:00
}
2001-08-18 16:37:52 +02:00
$return_msg = $method_response ;
2001-07-16 08:05:33 +02:00
}
else
{
2001-08-18 16:37:52 +02:00
/* return soapval object */
if ( get_class ( $method_response ) == 'soapval' )
{
$return_val = $method_response ;
/* create soap_val object w/ return values from method, use method signature to determine type */
}
else
{
$return_val = CreateObject ( 'phpgwapi.soapval' , $method , $this -> return_type , $method_response );
}
$this -> debug ( $return_val -> debug_str );
/* response object is a soap_msg object */
$return_msg = CreateObject ( 'phpgwapi.soapmsg' , $method . 'Response' , array ( $return_val ), $this -> service );
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
if ( $this -> debug_flag )
2001-07-30 05:40:08 +02:00
{
2001-08-18 16:37:52 +02:00
$return_msg -> debug_flag = true ;
2001-07-30 05:40:08 +02:00
}
2001-08-18 16:37:52 +02:00
$this -> result = 'successful' ;
return $return_msg ;
2001-07-16 08:05:33 +02:00
}
else
{
2001-08-18 16:37:52 +02:00
// debug
$this -> debug ( " ERROR: request not verified against method signature " );
$this -> result = " fault: request failed validation against method signature " ;
// return soapresp
return $this -> fault ();
2001-07-16 08:05:33 +02:00
}
}
else
{
// debug
2001-08-18 16:37:52 +02:00
$this -> debug ( " ERROR: parser did not return soapval object: $request_val " . get_class ( $request_val ));
$this -> result = " fault: parser did not return soapval object: $request_val " ;
// return fault
$this -> make_fault ( 'Server' , " parser did not return soapval object: $request_val " );
2001-07-16 08:05:33 +02:00
return $this -> fault ();
}
}
}
2001-08-18 16:37:52 +02:00
function verify_method ( $request )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
//return true;
$this -> debug ( 'entered verify_method() w/ request name: ' . $request -> name );
$params = $request -> value ;
// if there are input parameters required...
if ( $sig = $this -> dispatch_map [ $this -> methodname ][ " in " ])
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
$this -> input_value = count ( $sig );
if ( is_array ( $params ))
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
$this -> debug ( 'entered verify_method() with ' . count ( $params ) . ' parameters' );
2001-08-02 10:59:08 +02:00
@ reset ( $params );
2001-08-18 16:37:52 +02:00
while ( list ( $null , $v ) = @ each ( $params ))
/* foreach($params as $v) */
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
$this -> debug ( " param ' $v->name ' of type ' $v->type ' " );
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
// validate the number of parameters
if ( count ( $params ) == count ( $sig ))
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
$this -> debug ( " got correct number of parameters: " . count ( $sig ));
// make array of param types
@ reset ( $params );
while ( list ( $null , $param ) = @ each ( $params ))
/* foreach($params as $param) */
{
$p [] = strtolower ( $param -> type );
}
// validate each param's type
for ( $i = 0 ; $i < count ( $p ); $i ++ )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
// type not match
if ( strtolower ( $sig [ $i ]) != strtolower ( $p [ $i ]))
{
$this -> debug ( " mismatched parameter types: $sig[$i] != $p[$i] " );
$this -> make_fault ( 'Client' , " soap request contained mismatching parameters of name $v->name had type $p[$i] , which did not match signature's type: $sig[$i] " );
return false ;
}
$this -> debug ( " parameter type match: $sig[$i] = $p[$i] " );
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
return true ;
// oops, wrong number of paramss
}
else
{
$this -> debug ( 'oops, wrong number of parameter!' );
$this -> make_fault ( 'Client' , " soap request contained incorrect number of parameters. method ' $this->methodname ' required " . count ( $sig ) . " and request provided " . count ( $params ));
return false ;
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
// oops, no params...
2001-07-16 08:05:33 +02:00
}
else
{
2001-08-18 16:37:52 +02:00
$this -> debug ( " oops, no parameters sent! Method ' $this->methodname ' requires " . count ( $sig ) . " input parameters! " );
$this -> make_fault ( 'Client' , " soap request contained incorrect number of parameters. method ' $this->methodname ' requires " . count ( $sig ) . " parameters, and request provided none " );
2001-07-16 08:05:33 +02:00
return false ;
}
2001-08-18 16:37:52 +02:00
// no params
}
elseif ( ( count ( $params ) == 0 ) && ( count ( $sig ) <= 1 ) )
{
$this -> input_values = 0 ;
return true ;
2001-07-16 08:05:33 +02:00
}
else
{
2001-08-18 16:37:52 +02:00
//$this->debug("well, request passed parameters to a method that requires none?");
//$this->make_fault("Client","method '$this->methodname' requires no parameters. The request passed in ".count($params).": ".@implode(" param: ",$params) );
return true ;
2001-07-16 08:05:33 +02:00
}
}
2001-08-18 16:37:52 +02:00
// get string return type from dispatch map
function get_return_type ()
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
if ( count ( $this -> dispatch_map [ $this -> methodname ][ 'out' ]) >= 1 )
{
$type = array_shift ( $this -> dispatch_map [ $this -> methodname ][ 'out' ]);
$this -> debug ( " got return type from dispatch map: ' $type ' " );
return $type ;
}
return false ;
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
// dbg
function debug ( $string )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
if ( $this -> debug_flag )
{
$this -> debug_str .= " $string\n " ;
}
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
// add a method to the dispatch map
function add_to_map ( $methodname , $in , $out )
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
$this -> dispatch_map [ $methodname ][ 'in' ] = $in ;
$this -> dispatch_map [ $methodname ][ 'out' ] = $out ;
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
// set up a fault
function fault ()
2001-07-16 08:05:33 +02:00
{
2001-08-18 16:37:52 +02:00
return CreateObject ( 'phpgwapi.soapmsg' ,
'Fault' ,
array (
'faultcode' => $this -> fault_code ,
'faultstring' => $this -> fault_str ,
'faultactor' => $this -> fault_actor ,
'faultdetail' => $this -> fault_detail . $this -> debug_str
),
'http://schemas.xmlsoap.org/soap/envelope/'
);
2001-07-16 08:05:33 +02:00
}
2001-08-18 16:37:52 +02:00
function make_fault ( $fault_code , $fault_string )
{
$this -> fault_code = $fault_code ;
$this -> fault_str = $fault_string ;
$this -> fault = true ;
}
2001-07-16 08:05:33 +02:00
}
?>