Blame | Letzte Änderung | Log anzeigen | RSS feed
<?php/* vim: set expandtab tabstop=4 shiftwidth=4 softtabstop=4: *//*** PHP implementation of the XML-RPC protocol** This is a PEAR-ified version of Useful inc's XML-RPC for PHP.* It has support for HTTP transport, proxies and authentication.** PHP versions 4 and 5** @category Web Services* @package XML_RPC* @author Edd Dumbill <edd@usefulinc.com>* @author Stig Bakken <stig@php.net>* @author Martin Jansen <mj@php.net>* @author Daniel Convissor <danielc@php.net>* @copyright 1999-2001 Edd Dumbill, 2001-2010 The PHP Group* @license http://www.php.net/license/3_01.txt PHP License* @version SVN: $Id: RPC.php 315594 2011-08-27 01:03:57Z danielc $* @link http://pear.php.net/package/XML_RPC*/if (!function_exists('xml_parser_create')) {include_once 'PEAR.php';PEAR::loadExtension('xml');}/**#@+* Error constants*//*** Parameter values don't match parameter types*/define('XML_RPC_ERROR_INVALID_TYPE', 101);/*** Parameter declared to be numeric but the values are not*/define('XML_RPC_ERROR_NON_NUMERIC_FOUND', 102);/*** Communication error*/define('XML_RPC_ERROR_CONNECTION_FAILED', 103);/*** The array or struct has already been started*/define('XML_RPC_ERROR_ALREADY_INITIALIZED', 104);/*** Incorrect parameters submitted*/define('XML_RPC_ERROR_INCORRECT_PARAMS', 105);/*** Programming error by developer*/define('XML_RPC_ERROR_PROGRAMMING', 106);/**#@-*//*** Data types* @global string $GLOBALS['XML_RPC_I4']*/$GLOBALS['XML_RPC_I4'] = 'i4';/*** Data types* @global string $GLOBALS['XML_RPC_Int']*/$GLOBALS['XML_RPC_Int'] = 'int';/*** Data types* @global string $GLOBALS['XML_RPC_Boolean']*/$GLOBALS['XML_RPC_Boolean'] = 'boolean';/*** Data types* @global string $GLOBALS['XML_RPC_Double']*/$GLOBALS['XML_RPC_Double'] = 'double';/*** Data types* @global string $GLOBALS['XML_RPC_String']*/$GLOBALS['XML_RPC_String'] = 'string';/*** Data types* @global string $GLOBALS['XML_RPC_DateTime']*/$GLOBALS['XML_RPC_DateTime'] = 'dateTime.iso8601';/*** Data types* @global string $GLOBALS['XML_RPC_Base64']*/$GLOBALS['XML_RPC_Base64'] = 'base64';/*** Data types* @global string $GLOBALS['XML_RPC_Array']*/$GLOBALS['XML_RPC_Array'] = 'array';/*** Data types* @global string $GLOBALS['XML_RPC_Struct']*/$GLOBALS['XML_RPC_Struct'] = 'struct';/*** Data type meta-types* @global array $GLOBALS['XML_RPC_Types']*/$GLOBALS['XML_RPC_Types'] = array($GLOBALS['XML_RPC_I4'] => 1,$GLOBALS['XML_RPC_Int'] => 1,$GLOBALS['XML_RPC_Boolean'] => 1,$GLOBALS['XML_RPC_String'] => 1,$GLOBALS['XML_RPC_Double'] => 1,$GLOBALS['XML_RPC_DateTime'] => 1,$GLOBALS['XML_RPC_Base64'] => 1,$GLOBALS['XML_RPC_Array'] => 2,$GLOBALS['XML_RPC_Struct'] => 3,);/*** Error message numbers* @global array $GLOBALS['XML_RPC_err']*/$GLOBALS['XML_RPC_err'] = array('unknown_method' => 1,'invalid_return' => 2,'incorrect_params' => 3,'introspect_unknown' => 4,'http_error' => 5,'not_response_object' => 6,'invalid_request' => 7,);/*** Error message strings* @global array $GLOBALS['XML_RPC_str']*/$GLOBALS['XML_RPC_str'] = array('unknown_method' => 'Unknown method','invalid_return' => 'Invalid return payload: enable debugging to examine incoming payload','incorrect_params' => 'Incorrect parameters passed to method','introspect_unknown' => 'Can\'t introspect: method unknown','http_error' => 'Didn\'t receive 200 OK from remote server.','not_response_object' => 'The requested method didn\'t return an XML_RPC_Response object.','invalid_request' => 'Invalid request payload',);/*** Default XML encoding (ISO-8859-1, UTF-8 or US-ASCII)* @global string $GLOBALS['XML_RPC_defencoding']*/$GLOBALS['XML_RPC_defencoding'] = 'UTF-8';/*** User error codes start at 800* @global int $GLOBALS['XML_RPC_erruser']*/$GLOBALS['XML_RPC_erruser'] = 800;/*** XML parse error codes start at 100* @global int $GLOBALS['XML_RPC_errxml']*/$GLOBALS['XML_RPC_errxml'] = 100;/*** Compose backslashes for escaping regexp* @global string $GLOBALS['XML_RPC_backslash']*/$GLOBALS['XML_RPC_backslash'] = chr(92) . chr(92);/*** Should we automatically base64 encode strings that contain characters* which can cause PHP's SAX-based XML parser to break?* @global boolean $GLOBALS['XML_RPC_auto_base64']*/$GLOBALS['XML_RPC_auto_base64'] = false;/*** Valid parents of XML elements* @global array $GLOBALS['XML_RPC_valid_parents']*/$GLOBALS['XML_RPC_valid_parents'] = array('BOOLEAN' => array('VALUE'),'I4' => array('VALUE'),'INT' => array('VALUE'),'STRING' => array('VALUE'),'DOUBLE' => array('VALUE'),'DATETIME.ISO8601' => array('VALUE'),'BASE64' => array('VALUE'),'ARRAY' => array('VALUE'),'STRUCT' => array('VALUE'),'PARAM' => array('PARAMS'),'METHODNAME' => array('METHODCALL'),'PARAMS' => array('METHODCALL', 'METHODRESPONSE'),'MEMBER' => array('STRUCT'),'NAME' => array('MEMBER'),'DATA' => array('ARRAY'),'FAULT' => array('METHODRESPONSE'),'VALUE' => array('MEMBER', 'DATA', 'PARAM', 'FAULT'),);/*** Stores state during parsing** quick explanation of components:* + ac = accumulates values* + qt = decides if quotes are needed for evaluation* + cm = denotes struct or array (comma needed)* + isf = indicates a fault* + lv = indicates "looking for a value": implements the logic* to allow values with no types to be strings* + params = stores parameters in method calls* + method = stores method name** @global array $GLOBALS['XML_RPC_xh']*/$GLOBALS['XML_RPC_xh'] = array();/*** Start element handler for the XML parser** @return void*/function XML_RPC_se($parser_resource, $name, $attrs){global $XML_RPC_xh, $XML_RPC_valid_parents;$parser = (int) $parser_resource;// if invalid xmlrpc already detected, skip all processingif ($XML_RPC_xh[$parser]['isf'] >= 2) {return;}// check for correct element nesting// top level element can only be of 2 typesif (count($XML_RPC_xh[$parser]['stack']) == 0) {if ($name != 'METHODRESPONSE' && $name != 'METHODCALL') {$XML_RPC_xh[$parser]['isf'] = 2;$XML_RPC_xh[$parser]['isf_reason'] = 'missing top level xmlrpc element';return;}} else {// not top level element: see if parent is OKif (!in_array($XML_RPC_xh[$parser]['stack'][0], $XML_RPC_valid_parents[$name])) {$name = preg_replace('@[^a-zA-Z0-9._-]@', '', $name);$XML_RPC_xh[$parser]['isf'] = 2;$XML_RPC_xh[$parser]['isf_reason'] = "xmlrpc element $name cannot be child of {$XML_RPC_xh[$parser]['stack'][0]}";return;}}switch ($name) {case 'STRUCT':$XML_RPC_xh[$parser]['cm']++;// turn quoting off$XML_RPC_xh[$parser]['qt'] = 0;$cur_val = array();$cur_val['value'] = array();$cur_val['members'] = 1;array_unshift($XML_RPC_xh[$parser]['valuestack'], $cur_val);break;case 'ARRAY':$XML_RPC_xh[$parser]['cm']++;// turn quoting off$XML_RPC_xh[$parser]['qt'] = 0;$cur_val = array();$cur_val['value'] = array();$cur_val['members'] = 0;array_unshift($XML_RPC_xh[$parser]['valuestack'], $cur_val);break;case 'NAME':$XML_RPC_xh[$parser]['ac'] = '';break;case 'FAULT':$XML_RPC_xh[$parser]['isf'] = 1;break;case 'PARAM':$XML_RPC_xh[$parser]['valuestack'] = array();break;case 'VALUE':$XML_RPC_xh[$parser]['lv'] = 1;$XML_RPC_xh[$parser]['vt'] = $GLOBALS['XML_RPC_String'];$XML_RPC_xh[$parser]['ac'] = '';$XML_RPC_xh[$parser]['qt'] = 0;// look for a value: if this is still 1 by the// time we reach the first data segment then the type is string// by implication and we need to add in a quotebreak;case 'I4':case 'INT':case 'STRING':case 'BOOLEAN':case 'DOUBLE':case 'DATETIME.ISO8601':case 'BASE64':$XML_RPC_xh[$parser]['ac'] = ''; // reset the accumulatorif ($name == 'DATETIME.ISO8601' || $name == 'STRING') {$XML_RPC_xh[$parser]['qt'] = 1;if ($name == 'DATETIME.ISO8601') {$XML_RPC_xh[$parser]['vt'] = $GLOBALS['XML_RPC_DateTime'];}} elseif ($name == 'BASE64') {$XML_RPC_xh[$parser]['qt'] = 2;} else {// No quoting is required here -- but// at the end of the element we must check// for data format errors.$XML_RPC_xh[$parser]['qt'] = 0;}break;case 'MEMBER':$XML_RPC_xh[$parser]['ac'] = '';break;case 'DATA':case 'METHODCALL':case 'METHODNAME':case 'METHODRESPONSE':case 'PARAMS':// valid elements that add little to processingbreak;}// Save current element to stackarray_unshift($XML_RPC_xh[$parser]['stack'], $name);if ($name != 'VALUE') {$XML_RPC_xh[$parser]['lv'] = 0;}}/*** End element handler for the XML parser** @return void*/function XML_RPC_ee($parser_resource, $name){global $XML_RPC_xh;$parser = (int) $parser_resource;if ($XML_RPC_xh[$parser]['isf'] >= 2) {return;}// push this element from stack// NB: if XML validates, correct opening/closing is guaranteed and// we do not have to check for $name == $curr_elem.// we also checked for proper nesting at start of elements...$curr_elem = array_shift($XML_RPC_xh[$parser]['stack']);switch ($name) {case 'STRUCT':case 'ARRAY':$cur_val = array_shift($XML_RPC_xh[$parser]['valuestack']);$XML_RPC_xh[$parser]['value'] = $cur_val['value'];$XML_RPC_xh[$parser]['vt'] = strtolower($name);$XML_RPC_xh[$parser]['cm']--;break;case 'NAME':$XML_RPC_xh[$parser]['valuestack'][0]['name'] = $XML_RPC_xh[$parser]['ac'];break;case 'BOOLEAN':// special case here: we translate boolean 1 or 0 into PHP// constants true or falseif ($XML_RPC_xh[$parser]['ac'] == '1') {$XML_RPC_xh[$parser]['ac'] = 'true';} else {$XML_RPC_xh[$parser]['ac'] = 'false';}$XML_RPC_xh[$parser]['vt'] = strtolower($name);// Drop through intentionally.case 'I4':case 'INT':case 'STRING':case 'DOUBLE':case 'DATETIME.ISO8601':case 'BASE64':if ($XML_RPC_xh[$parser]['qt'] == 1) {// we use double quotes rather than single so backslashification works OK$XML_RPC_xh[$parser]['value'] = $XML_RPC_xh[$parser]['ac'];} elseif ($XML_RPC_xh[$parser]['qt'] == 2) {$XML_RPC_xh[$parser]['value'] = base64_decode($XML_RPC_xh[$parser]['ac']);} elseif ($name == 'BOOLEAN') {$XML_RPC_xh[$parser]['value'] = $XML_RPC_xh[$parser]['ac'];} else {// we have an I4, INT or a DOUBLE// we must check that only 0123456789-.<space> are characters hereif (!preg_match("@^[+-]?[0123456789 \t\.]+$@", $XML_RPC_xh[$parser]['ac'])) {XML_RPC_Base::raiseError('Non-numeric value received in INT or DOUBLE',XML_RPC_ERROR_NON_NUMERIC_FOUND);$XML_RPC_xh[$parser]['value'] = XML_RPC_ERROR_NON_NUMERIC_FOUND;} else {// it's ok, add it on$XML_RPC_xh[$parser]['value'] = $XML_RPC_xh[$parser]['ac'];}}$XML_RPC_xh[$parser]['ac'] = '';$XML_RPC_xh[$parser]['qt'] = 0;$XML_RPC_xh[$parser]['lv'] = 3; // indicate we've found a valuebreak;case 'VALUE':if ($XML_RPC_xh[$parser]['vt'] == $GLOBALS['XML_RPC_String']) {if (strlen($XML_RPC_xh[$parser]['ac']) > 0) {$XML_RPC_xh[$parser]['value'] = $XML_RPC_xh[$parser]['ac'];} elseif ($XML_RPC_xh[$parser]['lv'] == 1) {// The <value> element was empty.$XML_RPC_xh[$parser]['value'] = '';}}$temp = new XML_RPC_Value($XML_RPC_xh[$parser]['value'], $XML_RPC_xh[$parser]['vt']);$cur_val = array_shift($XML_RPC_xh[$parser]['valuestack']);if (is_array($cur_val)) {if ($cur_val['members']==0) {$cur_val['value'][] = $temp;} else {$XML_RPC_xh[$parser]['value'] = $temp;}array_unshift($XML_RPC_xh[$parser]['valuestack'], $cur_val);} else {$XML_RPC_xh[$parser]['value'] = $temp;}break;case 'MEMBER':$XML_RPC_xh[$parser]['ac'] = '';$XML_RPC_xh[$parser]['qt'] = 0;$cur_val = array_shift($XML_RPC_xh[$parser]['valuestack']);if (is_array($cur_val)) {if ($cur_val['members']==1) {$cur_val['value'][$cur_val['name']] = $XML_RPC_xh[$parser]['value'];}array_unshift($XML_RPC_xh[$parser]['valuestack'], $cur_val);}break;case 'DATA':$XML_RPC_xh[$parser]['ac'] = '';$XML_RPC_xh[$parser]['qt'] = 0;break;case 'PARAM':$XML_RPC_xh[$parser]['params'][] = $XML_RPC_xh[$parser]['value'];break;case 'METHODNAME':case 'RPCMETHODNAME':$XML_RPC_xh[$parser]['method'] = preg_replace("@^[\n\r\t ]+@", '',$XML_RPC_xh[$parser]['ac']);break;}// if it's a valid type name, set the typeif (isset($GLOBALS['XML_RPC_Types'][strtolower($name)])) {$XML_RPC_xh[$parser]['vt'] = strtolower($name);}}/*** Character data handler for the XML parser** @return void*/function XML_RPC_cd($parser_resource, $data){global $XML_RPC_xh, $XML_RPC_backslash;$parser = (int) $parser_resource;if ($XML_RPC_xh[$parser]['lv'] != 3) {// "lookforvalue==3" means that we've found an entire value// and should discard any further character dataif ($XML_RPC_xh[$parser]['lv'] == 1) {// if we've found text and we're just in a <value> then// turn quoting on, as this will be a string$XML_RPC_xh[$parser]['qt'] = 1;// and say we've found a value$XML_RPC_xh[$parser]['lv'] = 2;}// replace characters that eval would// do special things withif (!isset($XML_RPC_xh[$parser]['ac'])) {$XML_RPC_xh[$parser]['ac'] = '';}$XML_RPC_xh[$parser]['ac'] .= $data;}}/*** The common methods and properties for all of the XML_RPC classes** @category Web Services* @package XML_RPC* @author Edd Dumbill <edd@usefulinc.com>* @author Stig Bakken <stig@php.net>* @author Martin Jansen <mj@php.net>* @author Daniel Convissor <danielc@php.net>* @copyright 1999-2001 Edd Dumbill, 2001-2010 The PHP Group* @license http://www.php.net/license/3_01.txt PHP License* @version Release: @package_version@* @link http://pear.php.net/package/XML_RPC*/class XML_RPC_Base {/*** PEAR Error handling** @return object PEAR_Error object*/function raiseError($msg, $code){include_once 'PEAR.php';if (is_object(@$this)) {return PEAR::raiseError(get_class($this) . ': ' . $msg, $code);} else {return PEAR::raiseError('XML_RPC: ' . $msg, $code);}}/*** Tell whether something is a PEAR_Error object** @param mixed $value the item to check** @return bool whether $value is a PEAR_Error object or not** @access public*/function isError($value){return is_object($value) && is_a($value, 'PEAR_Error');}}/*** The methods and properties for submitting XML RPC requests** @category Web Services* @package XML_RPC* @author Edd Dumbill <edd@usefulinc.com>* @author Stig Bakken <stig@php.net>* @author Martin Jansen <mj@php.net>* @author Daniel Convissor <danielc@php.net>* @copyright 1999-2001 Edd Dumbill, 2001-2010 The PHP Group* @license http://www.php.net/license/3_01.txt PHP License* @version Release: @package_version@* @link http://pear.php.net/package/XML_RPC*/class XML_RPC_Client extends XML_RPC_Base {/*** The path and name of the RPC server script you want the request to go to* @var string*/var $path = '';/*** The name of the remote server to connect to* @var string*/var $server = '';/*** The protocol to use in contacting the remote server* @var string*/var $protocol = 'http://';/*** The port for connecting to the remote server** The default is 80 for http:// connections* and 443 for https:// and ssl:// connections.** @var integer*/var $port = 80;/*** A user name for accessing the RPC server* @var string* @see XML_RPC_Client::setCredentials()*/var $username = '';/*** A password for accessing the RPC server* @var string* @see XML_RPC_Client::setCredentials()*/var $password = '';/*** The name of the proxy server to use, if any* @var string*/var $proxy = '';/*** The protocol to use in contacting the proxy server, if any* @var string*/var $proxy_protocol = 'http://';/*** The port for connecting to the proxy server** The default is 8080 for http:// connections* and 443 for https:// and ssl:// connections.** @var integer*/var $proxy_port = 8080;/*** A user name for accessing the proxy server* @var string*/var $proxy_user = '';/*** A password for accessing the proxy server* @var string*/var $proxy_pass = '';/*** The error number, if any* @var integer*/var $errno = 0;/*** The error message, if any* @var string*/var $errstr = '';/*** The current debug mode (1 = on, 0 = off)* @var integer*/var $debug = 0;/*** The HTTP headers for the current request.* @var string*/var $headers = '';/*** Sets the object's properties** @param string $path the path and name of the RPC server script* you want the request to go to* @param string $server the URL of the remote server to connect to.* If this parameter doesn't specify a* protocol and $port is 443, ssl:// is* assumed.* @param integer $port a port for connecting to the remote server.* Defaults to 80 for http:// connections and* 443 for https:// and ssl:// connections.* @param string $proxy the URL of the proxy server to use, if any.* If this parameter doesn't specify a* protocol and $port is 443, ssl:// is* assumed.* @param integer $proxy_port a port for connecting to the remote server.* Defaults to 8080 for http:// connections and* 443 for https:// and ssl:// connections.* @param string $proxy_user a user name for accessing the proxy server* @param string $proxy_pass a password for accessing the proxy server** @return void*/function XML_RPC_Client($path, $server, $port = 0,$proxy = '', $proxy_port = 0,$proxy_user = '', $proxy_pass = ''){$this->path = $path;$this->proxy_user = $proxy_user;$this->proxy_pass = $proxy_pass;preg_match('@^(http://|https://|ssl://)?(.*)$@', $server, $match);if ($match[1] == '') {if ($port == 443) {$this->server = $match[2];$this->protocol = 'ssl://';$this->port = 443;} else {$this->server = $match[2];if ($port) {$this->port = $port;}}} elseif ($match[1] == 'http://') {$this->server = $match[2];if ($port) {$this->port = $port;}} else {$this->server = $match[2];$this->protocol = 'ssl://';if ($port) {$this->port = $port;} else {$this->port = 443;}}if ($proxy) {preg_match('@^(http://|https://|ssl://)?(.*)$@', $proxy, $match);if ($match[1] == '') {if ($proxy_port == 443) {$this->proxy = $match[2];$this->proxy_protocol = 'ssl://';$this->proxy_port = 443;} else {$this->proxy = $match[2];if ($proxy_port) {$this->proxy_port = $proxy_port;}}} elseif ($match[1] == 'http://') {$this->proxy = $match[2];if ($proxy_port) {$this->proxy_port = $proxy_port;}} else {$this->proxy = $match[2];$this->proxy_protocol = 'ssl://';if ($proxy_port) {$this->proxy_port = $proxy_port;} else {$this->proxy_port = 443;}}}}/*** Change the current debug mode** @param int $in where 1 = on, 0 = off** @return void*/function setDebug($in){if ($in) {$this->debug = 1;} else {$this->debug = 0;}}/*** Sets whether strings that contain characters which may cause PHP's* SAX-based XML parser to break should be automatically base64 encoded** This is is a workaround for systems that don't have PHP's mbstring* extension available.** @param int $in where 1 = on, 0 = off** @return void*/function setAutoBase64($in){if ($in) {$GLOBALS['XML_RPC_auto_base64'] = true;} else {$GLOBALS['XML_RPC_auto_base64'] = false;}}/*** Set username and password properties for connecting to the RPC server** @param string $u the user name* @param string $p the password** @return void** @see XML_RPC_Client::$username, XML_RPC_Client::$password*/function setCredentials($u, $p){$this->username = $u;$this->password = $p;}/*** Transmit the RPC request via HTTP 1.0 protocol** @param object $msg the XML_RPC_Message object* @param int $timeout how many seconds to wait for the request** @return object an XML_RPC_Response object. 0 is returned if any* problems happen.** @see XML_RPC_Message, XML_RPC_Client::XML_RPC_Client(),* XML_RPC_Client::setCredentials()*/function send($msg, $timeout = 0){if (!is_object($msg) || !is_a($msg, 'XML_RPC_Message')) {$this->errstr = 'send()\'s $msg parameter must be an'. ' XML_RPC_Message object.';$this->raiseError($this->errstr, XML_RPC_ERROR_PROGRAMMING);return 0;}$msg->debug = $this->debug;return $this->sendPayloadHTTP10($msg, $this->server, $this->port,$timeout, $this->username,$this->password);}/*** Transmit the RPC request via HTTP 1.0 protocol** Requests should be sent using XML_RPC_Client send() rather than* calling this method directly.** @param object $msg the XML_RPC_Message object* @param string $server the server to send the request to* @param int $port the server port send the request to* @param int $timeout how many seconds to wait for the request* before giving up* @param string $username a user name for accessing the RPC server* @param string $password a password for accessing the RPC server** @return object an XML_RPC_Response object. 0 is returned if any* problems happen.** @access protected* @see XML_RPC_Client::send()*/function sendPayloadHTTP10($msg, $server, $port, $timeout = 0,$username = '', $password = ''){// Pre-emptive BC hacks for fools calling sendPayloadHTTP10() directlyif ($username != $this->username) {$this->setCredentials($username, $password);}// Only create the payload if it was not created previouslyif (empty($msg->payload)) {$msg->createPayload();}$this->createHeaders($msg);$op = $this->headers . "\r\n\r\n";$op .= $msg->payload;if ($this->debug) {print "\n<pre>---SENT---\n";print $op;print "\n---END---</pre>\n";}/** If we're using a proxy open a socket to the proxy server* instead to the xml-rpc server*/if ($this->proxy) {if ($this->proxy_protocol == 'http://') {$protocol = '';} else {$protocol = $this->proxy_protocol;}if ($timeout > 0) {$fp = @fsockopen($protocol . $this->proxy, $this->proxy_port,$this->errno, $this->errstr, $timeout);} else {$fp = @fsockopen($protocol . $this->proxy, $this->proxy_port,$this->errno, $this->errstr);}} else {if ($this->protocol == 'http://') {$protocol = '';} else {$protocol = $this->protocol;}if ($timeout > 0) {$fp = @fsockopen($protocol . $server, $port,$this->errno, $this->errstr, $timeout);} else {$fp = @fsockopen($protocol . $server, $port,$this->errno, $this->errstr);}}/** Just raising the error without returning it is strange,* but keep it here for backwards compatibility.*/if (!$fp && $this->proxy) {$this->raiseError('Connection to proxy server '. $this->proxy . ':' . $this->proxy_port. ' failed. ' . $this->errstr,XML_RPC_ERROR_CONNECTION_FAILED);return 0;} elseif (!$fp) {$this->raiseError('Connection to RPC server '. $server . ':' . $port. ' failed. ' . $this->errstr,XML_RPC_ERROR_CONNECTION_FAILED);return 0;}if ($timeout) {/** Using socket_set_timeout() because stream_set_timeout()* was introduced in 4.3.0, but we need to support 4.2.0.*/socket_set_timeout($fp, $timeout);}if (!fputs($fp, $op, strlen($op))) {$this->errstr = 'Write error';return 0;}$resp = $msg->parseResponseFile($fp);$meta = socket_get_status($fp);if ($meta['timed_out']) {fclose($fp);$this->errstr = 'RPC server did not send response before timeout.';$this->raiseError($this->errstr, XML_RPC_ERROR_CONNECTION_FAILED);return 0;}fclose($fp);return $resp;}/*** Determines the HTTP headers and puts it in the $headers property** @param object $msg the XML_RPC_Message object** @return boolean TRUE if okay, FALSE if the message payload isn't set.** @access protected*/function createHeaders($msg){if (empty($msg->payload)) {return false;}if ($this->proxy) {$this->headers = 'POST ' . $this->protocol . $this->server;if ($this->proxy_port) {$this->headers .= ':' . $this->port;}} else {$this->headers = 'POST ';}$this->headers .= $this->path. " HTTP/1.0\r\n";$this->headers .= "User-Agent: PEAR XML_RPC\r\n";$this->headers .= 'Host: ' . $this->server . "\r\n";if ($this->proxy && $this->proxy_user) {$this->headers .= 'Proxy-Authorization: Basic '. base64_encode("$this->proxy_user:$this->proxy_pass"). "\r\n";}// thanks to Grant Rauscher <grant7@firstworld.net> for thisif ($this->username) {$this->headers .= 'Authorization: Basic '. base64_encode("$this->username:$this->password"). "\r\n";}$this->headers .= "Content-Type: text/xml\r\n";$this->headers .= 'Content-Length: ' . strlen($msg->payload);return true;}}/*** The methods and properties for interpreting responses to XML RPC requests** @category Web Services* @package XML_RPC* @author Edd Dumbill <edd@usefulinc.com>* @author Stig Bakken <stig@php.net>* @author Martin Jansen <mj@php.net>* @author Daniel Convissor <danielc@php.net>* @copyright 1999-2001 Edd Dumbill, 2001-2010 The PHP Group* @license http://www.php.net/license/3_01.txt PHP License* @version Release: @package_version@* @link http://pear.php.net/package/XML_RPC*/class XML_RPC_Response extends XML_RPC_Base{var $xv;var $fn;var $fs;var $hdrs;/*** @return void*/function XML_RPC_Response($val, $fcode = 0, $fstr = ''){if ($fcode != 0) {$this->fn = $fcode;$this->fs = htmlspecialchars($fstr);} else {$this->xv = $val;}}/*** @return int the error code*/function faultCode(){if (isset($this->fn)) {return $this->fn;} else {return 0;}}/*** @return string the error string*/function faultString(){return $this->fs;}/*** @return mixed the value*/function value(){return $this->xv;}/*** @return string the error message in XML format*/function serialize(){$rs = "<methodResponse>\n";if ($this->fn) {$rs .= "<fault><value><struct><member><name>faultCode</name><value><int>" . $this->fn . "</int></value></member><member><name>faultString</name><value><string>" . $this->fs . "</string></value></member></struct></value></fault>";} else {$rs .= "<params>\n<param>\n" . $this->xv->serialize() ."</param>\n</params>";}$rs .= "\n</methodResponse>";return $rs;}}/*** The methods and properties for composing XML RPC messages** @category Web Services* @package XML_RPC* @author Edd Dumbill <edd@usefulinc.com>* @author Stig Bakken <stig@php.net>* @author Martin Jansen <mj@php.net>* @author Daniel Convissor <danielc@php.net>* @copyright 1999-2001 Edd Dumbill, 2001-2010 The PHP Group* @license http://www.php.net/license/3_01.txt PHP License* @version Release: @package_version@* @link http://pear.php.net/package/XML_RPC*/class XML_RPC_Message extends XML_RPC_Base{/*** Should the payload's content be passed through mb_convert_encoding()?** @see XML_RPC_Message::setConvertPayloadEncoding()* @since Property available since Release 1.5.1* @var boolean*/var $convert_payload_encoding = false;/*** The current debug mode (1 = on, 0 = off)* @var integer*/var $debug = 0;/*** The encoding to be used for outgoing messages** Defaults to the value of <var>$GLOBALS['XML_RPC_defencoding']</var>** @var string* @see XML_RPC_Message::setSendEncoding(),* $GLOBALS['XML_RPC_defencoding'], XML_RPC_Message::xml_header()*/var $send_encoding = '';/*** The method presently being evaluated* @var string*/var $methodname = '';/*** @var array*/var $params = array();/*** The XML message being generated* @var string*/var $payload = '';/*** Should extra line breaks be removed from the payload?* @since Property available since Release 1.4.6* @var boolean*/var $remove_extra_lines = true;/*** The XML response from the remote server* @since Property available since Release 1.4.6* @var string*/var $response_payload = '';/*** @return void*/function XML_RPC_Message($meth, $pars = 0){$this->methodname = $meth;if (is_array($pars) && sizeof($pars) > 0) {for ($i = 0; $i < sizeof($pars); $i++) {$this->addParam($pars[$i]);}}}/*** Produces the XML declaration including the encoding attribute** The encoding is determined by this class' <var>$send_encoding</var>* property. If the <var>$send_encoding</var> property is not set, use* <var>$GLOBALS['XML_RPC_defencoding']</var>.** @return string the XML declaration and <methodCall> element** @see XML_RPC_Message::setSendEncoding(),* XML_RPC_Message::$send_encoding, $GLOBALS['XML_RPC_defencoding']*/function xml_header(){global $XML_RPC_defencoding;if (!$this->send_encoding) {$this->send_encoding = $XML_RPC_defencoding;}return '<?xml version="1.0" encoding="' . $this->send_encoding . '"?>'. "\n<methodCall>\n";}/*** @return string the closing </methodCall> tag*/function xml_footer(){return "</methodCall>\n";}/*** Fills the XML_RPC_Message::$payload property** Part of the process makes sure all line endings are in DOS format* (CRLF), which is probably required by specifications.** If XML_RPC_Message::setConvertPayloadEncoding() was set to true,* the payload gets passed through mb_convert_encoding()* to ensure the payload matches the encoding set in the* XML declaration. The encoding type can be manually set via* XML_RPC_Message::setSendEncoding().** @return void** @uses XML_RPC_Message::xml_header(), XML_RPC_Message::xml_footer()* @see XML_RPC_Message::setSendEncoding(), $GLOBALS['XML_RPC_defencoding'],* XML_RPC_Message::setConvertPayloadEncoding()*/function createPayload(){$this->payload = $this->xml_header();$this->payload .= '<methodName>' . $this->methodname . "</methodName>\n";$this->payload .= "<params>\n";for ($i = 0; $i < sizeof($this->params); $i++) {$p = $this->params[$i];$this->payload .= "<param>\n" . $p->serialize() . "</param>\n";}$this->payload .= "</params>\n";$this->payload .= $this->xml_footer();if ($this->remove_extra_lines) {$this->payload = preg_replace("@[\r\n]+@", "\r\n", $this->payload);} else {$this->payload = preg_replace("@\r\n|\n|\r|\n\r@", "\r\n", $this->payload);}if ($this->convert_payload_encoding) {$this->payload = mb_convert_encoding($this->payload, $this->send_encoding);}}/*** @return string the name of the method*/function method($meth = ''){if ($meth != '') {$this->methodname = $meth;}return $this->methodname;}/*** @return string the payload*/function serialize(){$this->createPayload();return $this->payload;}/*** @return void*/function addParam($par){$this->params[] = $par;}/*** Obtains an XML_RPC_Value object for the given parameter** @param int $i the index number of the parameter to obtain** @return object the XML_RPC_Value object.* If the parameter doesn't exist, an XML_RPC_Response object.** @since Returns XML_RPC_Response object on error since Release 1.3.0*/function getParam($i){global $XML_RPC_err, $XML_RPC_str;if (isset($this->params[$i])) {return $this->params[$i];} else {$this->raiseError('The submitted request did not contain this parameter',XML_RPC_ERROR_INCORRECT_PARAMS);return new XML_RPC_Response(0, $XML_RPC_err['incorrect_params'],$XML_RPC_str['incorrect_params']);}}/*** @return int the number of parameters*/function getNumParams(){return sizeof($this->params);}/*** Sets whether the payload's content gets passed through* mb_convert_encoding()** Returns PEAR_ERROR object if mb_convert_encoding() isn't available.** @param int $in where 1 = on, 0 = off** @return void** @see XML_RPC_Message::setSendEncoding()* @since Method available since Release 1.5.1*/function setConvertPayloadEncoding($in){if ($in && !function_exists('mb_convert_encoding')) {return $this->raiseError('mb_convert_encoding() is not available',XML_RPC_ERROR_PROGRAMMING);}$this->convert_payload_encoding = $in;}/*** Sets the XML declaration's encoding attribute** @param string $type the encoding type (ISO-8859-1, UTF-8 or US-ASCII)** @return void** @see XML_RPC_Message::setConvertPayloadEncoding(), XML_RPC_Message::xml_header()* @since Method available since Release 1.2.0*/function setSendEncoding($type){$this->send_encoding = $type;}/*** Determine the XML's encoding via the encoding attribute* in the XML declaration** If the encoding parameter is not set or is not ISO-8859-1, UTF-8* or US-ASCII, $XML_RPC_defencoding will be returned.** @param string $data the XML that will be parsed** @return string the encoding to be used** @link http://php.net/xml_parser_create* @since Method available since Release 1.2.0*/function getEncoding($data){global $XML_RPC_defencoding;if (preg_match('@<\?xml[^>]*\s*encoding\s*=\s*[\'"]([^"\']*)[\'"]@',$data, $match)){$match[1] = trim(strtoupper($match[1]));switch ($match[1]) {case 'ISO-8859-1':case 'UTF-8':case 'US-ASCII':return $match[1];break;default:return $XML_RPC_defencoding;}} else {return $XML_RPC_defencoding;}}/*** @return object a new XML_RPC_Response object*/function parseResponseFile($fp){$ipd = '';while ($data = @fread($fp, 8192)) {$ipd .= $data;}return $this->parseResponse($ipd);}/*** @return object a new XML_RPC_Response object*/function parseResponse($data = ''){global $XML_RPC_xh, $XML_RPC_err, $XML_RPC_str, $XML_RPC_defencoding;$encoding = $this->getEncoding($data);$parser_resource = xml_parser_create($encoding);$parser = (int) $parser_resource;$XML_RPC_xh = array();$XML_RPC_xh[$parser] = array();$XML_RPC_xh[$parser]['cm'] = 0;$XML_RPC_xh[$parser]['isf'] = 0;$XML_RPC_xh[$parser]['ac'] = '';$XML_RPC_xh[$parser]['qt'] = '';$XML_RPC_xh[$parser]['stack'] = array();$XML_RPC_xh[$parser]['valuestack'] = array();xml_parser_set_option($parser_resource, XML_OPTION_CASE_FOLDING, true);xml_set_element_handler($parser_resource, 'XML_RPC_se', 'XML_RPC_ee');xml_set_character_data_handler($parser_resource, 'XML_RPC_cd');$hdrfnd = 0;if ($this->debug) {print "\n<pre>---GOT---\n";print isset($_SERVER['SERVER_PROTOCOL']) ? htmlspecialchars($data) : $data;print "\n---END---</pre>\n";}// See if response is a 200 or a 100 then a 200, else raise error.// But only do this if we're using the HTTP protocol.if (preg_match('@^HTTP@', $data) &&!preg_match('@^HTTP/[0-9\.]+ 200 @', $data) &&!preg_match('@^HTTP/[0-9\.]+ 10[0-9]([A-Z ]+)?[\r\n]+HTTP/[0-9\.]+ 200@', $data)){$errstr = substr($data, 0, strpos($data, "\n") - 1);error_log('HTTP error, got response: ' . $errstr);$r = new XML_RPC_Response(0, $XML_RPC_err['http_error'],$XML_RPC_str['http_error'] . ' (' .$errstr . ')');xml_parser_free($parser_resource);return $r;}// gotta get rid of headers hereif (!$hdrfnd && ($brpos = strpos($data,"\r\n\r\n"))) {$XML_RPC_xh[$parser]['ha'] = substr($data, 0, $brpos);$data = substr($data, $brpos + 4);$hdrfnd = 1;}/** be tolerant of junk after methodResponse* (e.g. javascript automatically inserted by free hosts)* thanks to Luca Mariano <luca.mariano@email.it>*/$data = substr($data, 0, strpos($data, "</methodResponse>") + 17);$this->response_payload = $data;if (!xml_parse($parser_resource, $data, sizeof($data))) {// thanks to Peter Kocks <peter.kocks@baygate.com>if (xml_get_current_line_number($parser_resource) == 1) {$errstr = 'XML error at line 1, check URL';} else {$errstr = sprintf('XML error: %s at line %d',xml_error_string(xml_get_error_code($parser_resource)),xml_get_current_line_number($parser_resource));}error_log($errstr);$r = new XML_RPC_Response(0, $XML_RPC_err['invalid_return'],$XML_RPC_str['invalid_return']);xml_parser_free($parser_resource);return $r;}xml_parser_free($parser_resource);if ($this->debug) {print "\n<pre>---PARSED---\n";var_dump($XML_RPC_xh[$parser]['value']);print "---END---</pre>\n";}if ($XML_RPC_xh[$parser]['isf'] > 1) {$r = new XML_RPC_Response(0, $XML_RPC_err['invalid_return'],$XML_RPC_str['invalid_return'].' '.$XML_RPC_xh[$parser]['isf_reason']);} elseif (!is_object($XML_RPC_xh[$parser]['value'])) {// then something odd has happened// and it's time to generate a client side error// indicating something odd went on$r = new XML_RPC_Response(0, $XML_RPC_err['invalid_return'],$XML_RPC_str['invalid_return']);} else {$v = $XML_RPC_xh[$parser]['value'];if ($XML_RPC_xh[$parser]['isf']) {$f = $v->structmem('faultCode');$fs = $v->structmem('faultString');$r = new XML_RPC_Response($v, $f->scalarval(),$fs->scalarval());} else {$r = new XML_RPC_Response($v);}}$r->hdrs = preg_split("@\r?\n@", $XML_RPC_xh[$parser]['ha']);return $r;}}/*** The methods and properties that represent data in XML RPC format** @category Web Services* @package XML_RPC* @author Edd Dumbill <edd@usefulinc.com>* @author Stig Bakken <stig@php.net>* @author Martin Jansen <mj@php.net>* @author Daniel Convissor <danielc@php.net>* @copyright 1999-2001 Edd Dumbill, 2001-2010 The PHP Group* @license http://www.php.net/license/3_01.txt PHP License* @version Release: @package_version@* @link http://pear.php.net/package/XML_RPC*/class XML_RPC_Value extends XML_RPC_Base{var $me = array();var $mytype = 0;/*** @return void*/function XML_RPC_Value($val = -1, $type = ''){$this->me = array();$this->mytype = 0;if ($val != -1 || $type != '') {if ($type == '') {$type = 'string';}if (!array_key_exists($type, $GLOBALS['XML_RPC_Types'])) {// XXX// need some way to report this error} elseif ($GLOBALS['XML_RPC_Types'][$type] == 1) {$this->addScalar($val, $type);} elseif ($GLOBALS['XML_RPC_Types'][$type] == 2) {$this->addArray($val);} elseif ($GLOBALS['XML_RPC_Types'][$type] == 3) {$this->addStruct($val);}}}/*** @return int returns 1 if successful or 0 if there are problems*/function addScalar($val, $type = 'string'){if ($this->mytype == 1) {$this->raiseError('Scalar can have only one value',XML_RPC_ERROR_INVALID_TYPE);return 0;}$typeof = $GLOBALS['XML_RPC_Types'][$type];if ($typeof != 1) {$this->raiseError("Not a scalar type (${typeof})",XML_RPC_ERROR_INVALID_TYPE);return 0;}if ($type == $GLOBALS['XML_RPC_Boolean']) {if (strcasecmp($val, 'true') == 0|| $val == 1|| ($val == true && strcasecmp($val, 'false'))){$val = 1;} else {$val = 0;}}if ($this->mytype == 2) {// we're adding to an array here$ar = $this->me['array'];$ar[] = new XML_RPC_Value($val, $type);$this->me['array'] = $ar;} else {// a scalar, so set the value and remember we're scalar$this->me[$type] = $val;$this->mytype = $typeof;}return 1;}/*** @return int returns 1 if successful or 0 if there are problems*/function addArray($vals){if ($this->mytype != 0) {$this->raiseError('Already initialized as a [' . $this->kindOf() . ']',XML_RPC_ERROR_ALREADY_INITIALIZED);return 0;}$this->mytype = $GLOBALS['XML_RPC_Types']['array'];$this->me['array'] = $vals;return 1;}/*** @return int returns 1 if successful or 0 if there are problems*/function addStruct($vals){if ($this->mytype != 0) {$this->raiseError('Already initialized as a [' . $this->kindOf() . ']',XML_RPC_ERROR_ALREADY_INITIALIZED);return 0;}$this->mytype = $GLOBALS['XML_RPC_Types']['struct'];$this->me['struct'] = $vals;return 1;}/*** @return void*/function dump($ar){reset($ar);foreach ($ar as $key => $val) {echo "$key => $val<br />";if ($key == 'array') {foreach ($val as $key2 => $val2) {echo "-- $key2 => $val2<br />";}}}}/*** @return string the data type of the current value*/function kindOf(){switch ($this->mytype) {case 3:return 'struct';case 2:return 'array';case 1:return 'scalar';default:return 'undef';}}/*** @return string the data in XML format*/function serializedata($typ, $val){$rs = '';if (!array_key_exists($typ, $GLOBALS['XML_RPC_Types'])) {// XXX// need some way to report this errorreturn;}switch ($GLOBALS['XML_RPC_Types'][$typ]) {case 3:// struct$rs .= "<struct>\n";reset($val);foreach ($val as $key2 => $val2) {$rs .= "<member><name>" . htmlspecialchars($key2) . "</name>\n";$rs .= $this->serializeval($val2);$rs .= "</member>\n";}$rs .= '</struct>';break;case 2:// array$rs .= "<array>\n<data>\n";foreach ($val as $value) {$rs .= $this->serializeval($value);}$rs .= "</data>\n</array>";break;case 1:switch ($typ) {case $GLOBALS['XML_RPC_Base64']:$rs .= "<${typ}>" . base64_encode($val) . "</${typ}>";break;case $GLOBALS['XML_RPC_Boolean']:$rs .= "<${typ}>" . ($val ? '1' : '0') . "</${typ}>";break;case $GLOBALS['XML_RPC_String']:$rs .= "<${typ}>" . htmlspecialchars($val). "</${typ}>";break;default:$rs .= "<${typ}>${val}</${typ}>";}}return $rs;}/*** @return string the data in XML format*/function serialize(){return $this->serializeval($this);}/*** @return string the data in XML format*/function serializeval($o){if (!is_object($o) || empty($o->me) || !is_array($o->me)) {return '';}$ar = $o->me;reset($ar);list($typ, $val) = each($ar);return '<value>' . $this->serializedata($typ, $val) . "</value>\n";}/*** @return mixed the contents of the element requested*/function structmem($m){return $this->me['struct'][$m];}/*** @return void*/function structreset(){reset($this->me['struct']);}/*** @return the key/value pair of the struct's current element*/function structeach(){return each($this->me['struct']);}/*** @return mixed the current value*/function getval(){// UNSTABLEreset($this->me);$b = current($this->me);// contributed by I Sofer, 2001-03-24// add support for nested arrays to scalarval// i've created a new method here, so as to// preserve back compatibilityif (is_array($b)) {foreach ($b as $id => $cont) {$b[$id] = $cont->scalarval();}}// add support for structures directly encoding php objectsif (is_object($b)) {$t = get_object_vars($b);foreach ($t as $id => $cont) {$t[$id] = $cont->scalarval();}foreach ($t as $id => $cont) {$b->$id = $cont;}}// end contribreturn $b;}/*** @return mixed the current element's scalar value. If the value is* not scalar, FALSE is returned.*/function scalarval(){reset($this->me);$v = current($this->me);if (!is_scalar($v)) {$v = false;}return $v;}/*** @return string*/function scalartyp(){reset($this->me);$a = key($this->me);if ($a == $GLOBALS['XML_RPC_I4']) {$a = $GLOBALS['XML_RPC_Int'];}return $a;}/*** @return mixed the struct's current element*/function arraymem($m){return $this->me['array'][$m];}/*** @return int the number of elements in the array*/function arraysize(){reset($this->me);list($a, $b) = each($this->me);return sizeof($b);}/*** Determines if the item submitted is an XML_RPC_Value object** @param mixed $val the variable to be evaluated** @return bool TRUE if the item is an XML_RPC_Value object** @static* @since Method available since Release 1.3.0*/function isValue($val){return (strtolower(get_class($val)) == 'xml_rpc_value');}}/*** Return an ISO8601 encoded string** While timezones ought to be supported, the XML-RPC spec says:** "Don't assume a timezone. It should be specified by the server in its* documentation what assumptions it makes about timezones."** This routine always assumes localtime unless $utc is set to 1, in which* case UTC is assumed and an adjustment for locale is made when encoding.** @return string the formatted date*/function XML_RPC_iso8601_encode($timet, $utc = 0){if (!$utc) {$t = strftime('%Y%m%dT%H:%M:%S', $timet);} else {if (function_exists('gmstrftime')) {// gmstrftime doesn't exist in some versions// of PHP$t = gmstrftime('%Y%m%dT%H:%M:%S', $timet);} else {$t = strftime('%Y%m%dT%H:%M:%S', $timet - date('Z'));}}return $t;}/*** Convert a datetime string into a Unix timestamp** While timezones ought to be supported, the XML-RPC spec says:** "Don't assume a timezone. It should be specified by the server in its* documentation what assumptions it makes about timezones."** This routine always assumes localtime unless $utc is set to 1, in which* case UTC is assumed and an adjustment for locale is made when encoding.** @return int the unix timestamp of the date submitted*/function XML_RPC_iso8601_decode($idate, $utc = 0){$t = 0;if (preg_match('@([0-9]{4})([0-9]{2})([0-9]{2})T([0-9]{2}):([0-9]{2}):([0-9]{2})@', $idate, $regs)) {if ($utc) {$t = gmmktime($regs[4], $regs[5], $regs[6], $regs[2], $regs[3], $regs[1]);} else {$t = mktime($regs[4], $regs[5], $regs[6], $regs[2], $regs[3], $regs[1]);}}return $t;}/*** Converts an XML_RPC_Value object into native PHP types** @param object $XML_RPC_val the XML_RPC_Value object to decode** @return mixed the PHP values*/function XML_RPC_decode($XML_RPC_val){$kind = $XML_RPC_val->kindOf();if ($kind == 'scalar') {return $XML_RPC_val->scalarval();} elseif ($kind == 'array') {$size = $XML_RPC_val->arraysize();$arr = array();for ($i = 0; $i < $size; $i++) {$arr[] = XML_RPC_decode($XML_RPC_val->arraymem($i));}return $arr;} elseif ($kind == 'struct') {$XML_RPC_val->structreset();$arr = array();while (list($key, $value) = $XML_RPC_val->structeach()) {$arr[$key] = XML_RPC_decode($value);}return $arr;}}/*** Converts native PHP types into an XML_RPC_Value object** @param mixed $php_val the PHP value or variable you want encoded** @return object the XML_RPC_Value object*/function XML_RPC_encode($php_val){$type = gettype($php_val);$XML_RPC_val = new XML_RPC_Value;switch ($type) {case 'array':if (empty($php_val)) {$XML_RPC_val->addArray($php_val);break;}$tmp = array_diff(array_keys($php_val), range(0, count($php_val)-1));if (empty($tmp)) {$arr = array();foreach ($php_val as $k => $v) {$arr[$k] = XML_RPC_encode($v);}$XML_RPC_val->addArray($arr);break;}// fall though if it's not an enumerated arraycase 'object':$arr = array();foreach ($php_val as $k => $v) {$arr[$k] = XML_RPC_encode($v);}$XML_RPC_val->addStruct($arr);break;case 'integer':$XML_RPC_val->addScalar($php_val, $GLOBALS['XML_RPC_Int']);break;case 'double':$XML_RPC_val->addScalar($php_val, $GLOBALS['XML_RPC_Double']);break;case 'string':case 'NULL':if (preg_match('@^[0-9]{8}\T{1}[0-9]{2}\:[0-9]{2}\:[0-9]{2}$@', $php_val)) {$XML_RPC_val->addScalar($php_val, $GLOBALS['XML_RPC_DateTime']);} elseif ($GLOBALS['XML_RPC_auto_base64']&& preg_match("@[^ -~\t\r\n]@", $php_val)){// Characters other than alpha-numeric, punctuation, SP, TAB,// LF and CR break the XML parser, encode value via Base 64.$XML_RPC_val->addScalar($php_val, $GLOBALS['XML_RPC_Base64']);} else {$XML_RPC_val->addScalar($php_val, $GLOBALS['XML_RPC_String']);}break;case 'boolean':// Add support for encoding/decoding of booleans, since they// are supported in PHP// by <G_Giunta_2001-02-29>$XML_RPC_val->addScalar($php_val, $GLOBALS['XML_RPC_Boolean']);break;case 'unknown type':default:$XML_RPC_val = false;}return $XML_RPC_val;}/** Local variables:* tab-width: 4* c-basic-offset: 4* c-hanging-comment-ender-p: nil* End:*/?>