| 
									
										
										
										
											2019-03-12 09:27:46 +00:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * IXR_MESSAGE | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @package IXR | 
					
						
							|  |  |  |  * @since 1.5.0 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class IXR_Message | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     var $message     = false; | 
					
						
							|  |  |  |     var $messageType = false;  // methodCall / methodResponse / fault
 | 
					
						
							|  |  |  |     var $faultCode   = false; | 
					
						
							|  |  |  |     var $faultString = false; | 
					
						
							|  |  |  |     var $methodName  = ''; | 
					
						
							|  |  |  |     var $params      = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Current variable stacks
 | 
					
						
							|  |  |  |     var $_arraystructs = array();   // The stack used to keep track of the current array/struct
 | 
					
						
							|  |  |  |     var $_arraystructstypes = array(); // Stack keeping track of if things are structs or array
 | 
					
						
							|  |  |  |     var $_currentStructName = array();  // A stack as well
 | 
					
						
							|  |  |  |     var $_param; | 
					
						
							|  |  |  |     var $_value; | 
					
						
							|  |  |  |     var $_currentTag; | 
					
						
							|  |  |  |     var $_currentTagContents; | 
					
						
							|  |  |  |     // The XML parser
 | 
					
						
							|  |  |  |     var $_parser; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * PHP5 constructor. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  |     function __construct( $message ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->message =& $message; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * PHP4 constructor. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	public function IXR_Message( $message ) { | 
					
						
							|  |  |  | 		self::__construct( $message ); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function parse() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if ( ! function_exists( 'xml_parser_create' ) ) { | 
					
						
							|  |  |  |             trigger_error( __( "PHP's XML extension is not available. Please contact your hosting provider to enable PHP's XML extension." ) ); | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // first remove the XML declaration
 | 
					
						
							|  |  |  |         // merged from WP #10698 - this method avoids the RAM usage of preg_replace on very large messages
 | 
					
						
							|  |  |  |         $header = preg_replace( '/<\?xml.*?\?'.'>/s', '', substr( $this->message, 0, 100 ), 1 ); | 
					
						
							|  |  |  |         $this->message = trim( substr_replace( $this->message, $header, 0, 100 ) ); | 
					
						
							|  |  |  |         if ( '' == $this->message ) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Then remove the DOCTYPE
 | 
					
						
							|  |  |  |         $header = preg_replace( '/^<!DOCTYPE[^>]*+>/i', '', substr( $this->message, 0, 200 ), 1 ); | 
					
						
							|  |  |  |         $this->message = trim( substr_replace( $this->message, $header, 0, 200 ) ); | 
					
						
							|  |  |  |         if ( '' == $this->message ) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Check that the root tag is valid
 | 
					
						
							|  |  |  |         $root_tag = substr( $this->message, 0, strcspn( substr( $this->message, 0, 20 ), "> \t\r\n" ) ); | 
					
						
							|  |  |  |         if ( '<!DOCTYPE' === strtoupper( $root_tag ) ) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if ( ! in_array( $root_tag, array( '<methodCall', '<methodResponse', '<fault' ) ) ) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Bail if there are too many elements to parse
 | 
					
						
							|  |  |  |         $element_limit = 30000; | 
					
						
							|  |  |  |         if ( function_exists( 'apply_filters' ) ) { | 
					
						
							|  |  |  |             /** | 
					
						
							|  |  |  |              * Filters the number of elements to parse in an XML-RPC response. | 
					
						
							|  |  |  |              * | 
					
						
							|  |  |  |              * @since 4.0.0 | 
					
						
							|  |  |  |              * | 
					
						
							|  |  |  |              * @param int $element_limit Default elements limit. | 
					
						
							|  |  |  |              */ | 
					
						
							|  |  |  |             $element_limit = apply_filters( 'xmlrpc_element_limit', $element_limit ); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if ( $element_limit && 2 * $element_limit < substr_count( $this->message, '<' ) ) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->_parser = xml_parser_create(); | 
					
						
							|  |  |  |         // Set XML parser to take the case of tags in to account
 | 
					
						
							|  |  |  |         xml_parser_set_option($this->_parser, XML_OPTION_CASE_FOLDING, false); | 
					
						
							|  |  |  |         // Set XML parser callback functions
 | 
					
						
							|  |  |  |         xml_set_object($this->_parser, $this); | 
					
						
							|  |  |  |         xml_set_element_handler($this->_parser, 'tag_open', 'tag_close'); | 
					
						
							|  |  |  |         xml_set_character_data_handler($this->_parser, 'cdata'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // 256Kb, parse in chunks to avoid the RAM usage on very large messages
 | 
					
						
							|  |  |  |         $chunk_size = 262144; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /** | 
					
						
							| 
									
										
										
										
											2019-05-08 08:05:39 +01:00
										 |  |  |          * Filters the chunk size that can be used to parse an XML-RPC response message. | 
					
						
							| 
									
										
										
										
											2019-03-12 09:27:46 +00:00
										 |  |  |          * | 
					
						
							|  |  |  |          * @since 4.4.0 | 
					
						
							|  |  |  |          * | 
					
						
							|  |  |  |          * @param int $chunk_size Chunk size to parse in bytes. | 
					
						
							|  |  |  |          */ | 
					
						
							|  |  |  |         $chunk_size = apply_filters( 'xmlrpc_chunk_parsing_size', $chunk_size ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $final = false; | 
					
						
							|  |  |  |         do { | 
					
						
							|  |  |  |             if (strlen($this->message) <= $chunk_size) { | 
					
						
							|  |  |  |                 $final = true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             $part = substr($this->message, 0, $chunk_size); | 
					
						
							|  |  |  |             $this->message = substr($this->message, $chunk_size); | 
					
						
							|  |  |  |             if (!xml_parse($this->_parser, $part, $final)) { | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if ($final) { | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } while (true); | 
					
						
							|  |  |  |         xml_parser_free($this->_parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Grab the error messages, if any
 | 
					
						
							|  |  |  |         if ($this->messageType == 'fault') { | 
					
						
							|  |  |  |             $this->faultCode = $this->params[0]['faultCode']; | 
					
						
							|  |  |  |             $this->faultString = $this->params[0]['faultString']; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function tag_open($parser, $tag, $attr) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->_currentTagContents = ''; | 
					
						
							|  |  |  |         $this->currentTag = $tag; | 
					
						
							|  |  |  |         switch($tag) { | 
					
						
							|  |  |  |             case 'methodCall': | 
					
						
							|  |  |  |             case 'methodResponse': | 
					
						
							|  |  |  |             case 'fault': | 
					
						
							|  |  |  |                 $this->messageType = $tag; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |                 /* Deal with stacks of arrays and structs */ | 
					
						
							|  |  |  |             case 'data':    // data is to all intents and puposes more interesting than array
 | 
					
						
							|  |  |  |                 $this->_arraystructstypes[] = 'array'; | 
					
						
							|  |  |  |                 $this->_arraystructs[] = array(); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 'struct': | 
					
						
							|  |  |  |                 $this->_arraystructstypes[] = 'struct'; | 
					
						
							|  |  |  |                 $this->_arraystructs[] = array(); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function cdata($parser, $cdata) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $this->_currentTagContents .= $cdata; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function tag_close($parser, $tag) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $valueFlag = false; | 
					
						
							|  |  |  |         switch($tag) { | 
					
						
							|  |  |  |             case 'int': | 
					
						
							|  |  |  |             case 'i4': | 
					
						
							|  |  |  |                 $value = (int)trim($this->_currentTagContents); | 
					
						
							|  |  |  |                 $valueFlag = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 'double': | 
					
						
							|  |  |  |                 $value = (double)trim($this->_currentTagContents); | 
					
						
							|  |  |  |                 $valueFlag = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 'string': | 
					
						
							|  |  |  |                 $value = (string)trim($this->_currentTagContents); | 
					
						
							|  |  |  |                 $valueFlag = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 'dateTime.iso8601': | 
					
						
							|  |  |  |                 $value = new IXR_Date(trim($this->_currentTagContents)); | 
					
						
							|  |  |  |                 $valueFlag = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 'value': | 
					
						
							|  |  |  |                 // "If no type is indicated, the type is string."
 | 
					
						
							|  |  |  |                 if (trim($this->_currentTagContents) != '') { | 
					
						
							|  |  |  |                     $value = (string)$this->_currentTagContents; | 
					
						
							|  |  |  |                     $valueFlag = true; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 'boolean': | 
					
						
							|  |  |  |                 $value = (boolean)trim($this->_currentTagContents); | 
					
						
							|  |  |  |                 $valueFlag = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 'base64': | 
					
						
							|  |  |  |                 $value = base64_decode($this->_currentTagContents); | 
					
						
							|  |  |  |                 $valueFlag = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |                 /* Deal with stacks of arrays and structs */ | 
					
						
							|  |  |  |             case 'data': | 
					
						
							|  |  |  |             case 'struct': | 
					
						
							|  |  |  |                 $value = array_pop($this->_arraystructs); | 
					
						
							|  |  |  |                 array_pop($this->_arraystructstypes); | 
					
						
							|  |  |  |                 $valueFlag = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 'member': | 
					
						
							|  |  |  |                 array_pop($this->_currentStructName); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 'name': | 
					
						
							|  |  |  |                 $this->_currentStructName[] = trim($this->_currentTagContents); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 'methodName': | 
					
						
							|  |  |  |                 $this->methodName = trim($this->_currentTagContents); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($valueFlag) { | 
					
						
							|  |  |  |             if (count($this->_arraystructs) > 0) { | 
					
						
							|  |  |  |                 // Add value to struct or array
 | 
					
						
							|  |  |  |                 if ($this->_arraystructstypes[count($this->_arraystructstypes)-1] == 'struct') { | 
					
						
							|  |  |  |                     // Add to struct
 | 
					
						
							|  |  |  |                     $this->_arraystructs[count($this->_arraystructs)-1][$this->_currentStructName[count($this->_currentStructName)-1]] = $value; | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     // Add to array
 | 
					
						
							|  |  |  |                     $this->_arraystructs[count($this->_arraystructs)-1][] = $value; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 // Just add as a parameter
 | 
					
						
							|  |  |  |                 $this->params[] = $value; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $this->_currentTagContents = ''; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |