Reduce memory usage for XML-RPC requests. Props Demitrious Kelly. fixes #10698

git-svn-id: http://svn.automattic.com/wordpress/trunk@12263 1a063a9b-81f0-0310-95a4-ce76da25c4cd
This commit is contained in:
ryan 2009-11-23 16:53:36 +00:00
parent bc718b550a
commit 38bcfbb49b
1 changed files with 22 additions and 15 deletions

View File

@ -153,12 +153,14 @@ class IXR_Message {
var $_currentTagContents;
// The XML parser
var $_parser;
function IXR_Message ($message) {
$this->message = $message;
function IXR_Message (&$message) {
$this->message = &$message;
}
function parse() {
// first remove the XML declaration
$this->message = preg_replace('/<\?xml.*?\?'.'>/', '', $this->message);
// this method avoids the RAM usage of preg_replace on very large messages
$header = preg_replace( '/<\?xml.*?\?'.'>/', '', substr( $this->message, 0, 100 ), 1 );
$this->message = substr_replace($this->message, $header, 0, 100);
if (trim($this->message) == '') {
return false;
}
@ -169,12 +171,17 @@ class IXR_Message {
xml_set_object($this->_parser, $this);
xml_set_element_handler($this->_parser, 'tag_open', 'tag_close');
xml_set_character_data_handler($this->_parser, 'cdata');
if (!xml_parse($this->_parser, $this->message)) {
/* die(sprintf('XML error: %s at line %d',
xml_error_string(xml_get_error_code($this->_parser)),
xml_get_current_line_number($this->_parser))); */
$chunk_size = 262144; // 256Kb, parse in chunks to avoid the RAM usage on very large messages
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') {
@ -304,7 +311,7 @@ class IXR_Server {
header( 'Content-Type: text/plain' );
die('XML-RPC server accepts POST requests only.');
}
$data = $HTTP_RAW_POST_DATA;
$data = &$HTTP_RAW_POST_DATA;
}
$this->message = new IXR_Message($data);
if (!$this->message->parse()) {