Вход Регистрация
Файл: core/classes/class.ajax.php
Строк: 551
<?php
/**
 * JsHttpRequest: PHP backend for JavaScript DHTML loader.
 * (C) Dmitry Koterov, http://en.dklab.ru
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 * See http://www.gnu.org/copyleft/lesser.html
 *
 * Do not remove this comment if you want to use the script!
 * Не удаляйте данный комментарий, если вы хотите использовать скрипт!
 *
 * This backend library also supports POST requests additionally to GET.
 *
 * @author Dmitry Koterov
 * @version 5.x $Id$
 */

class JsHttpRequest
{
    var 
$SCRIPT_ENCODING DEFAULT_CHARSET;
    var 
$SCRIPT_DECODE_MODE '';
    var 
$LOADER null;
    var 
$ID null;

    
// Internal; uniq value.
    
var $_uniqHash;
    
// Magic number for display_error checking.
    
var $_magic 14623;
    
// Previous display_errors value.
    
var $_prevDisplayErrors null;
    
// Internal: response content-type depending on loader type.
    
var $_contentTypes = array(
        
"script" => "text/javascript",
        
"xml"    => "text/plain"// In XMLHttpRequest mode we must return text/plain - stupid Opera 8.0. :(
        
"form"   => "text/html",
        
""       => "text/plain"// for unknown loader
    
);
    
// Internal: conversion to UTF-8 JSON cancelled because of non-ascii key.
    
var $_toUtfFailed false;
    
// Internal: list of characters 128...255 (for strpbrk() ASCII check).
    
var $_nonAsciiChars '';
    
// Which Unicode conversion function is available?
    
var $_unicodeConvMethod null;
    
// Emergency memory buffer to be freed on memory_limit error.
    
var $_emergBuffer null;


    
/**
     * Constructor.
     *
     * Create new JsHttpRequest backend object and attach it
     * to script output buffer. As a result - script will always return
     * correct JavaScript code, even in case of fatal errors.
     *
     * QUERY_STRING is in form of: PHPSESSID=<sid>&a=aaa&b=bbb&JsHttpRequest=<id>-<loader>
     * where <id> is a request ID, <loader> is a loader name, <sid> - a session ID (if present),
     * PHPSESSID - session parameter name (by default = "PHPSESSID").
     *
     * If an object is created WITHOUT an active AJAX query, it is simply marked as
     * non-active. Use statuc method isActive() to check.
     */
    
function JsHttpRequest($enc)
    {
        global 
$JsHttpRequest_Active;

        
// Parse QUERY_STRING.
        
if (preg_match('/^(.*)(?:&|^)JsHttpRequest=(?:(d+)-)?([^&]+)((?:&|$).*)$/s', @$_SERVER['QUERY_STRING'], $m)) {
            
$this->ID $m[2];
            
$this->LOADER strtolower($m[3]);
            
$_SERVER['QUERY_STRING'] = preg_replace('/^&+|&+$/s'''preg_replace('/(^|&)'.session_name().'=[^&]*&?/s''&'$m[1] . $m[4]));
            unset(
                
$_GET['JsHttpRequest'],
                
$_REQUEST['JsHttpRequest'],
                
$_GET[session_name()],
                
$_POST[session_name()],
                
$_REQUEST[session_name()]
            );
            
// Detect Unicode conversion method.
            
$this->_unicodeConvMethod function_exists('mb_convert_encoding')? 'mb' : (function_exists('iconv')? 'iconv' null);

            
// Fill an emergency buffer. We erase it at the first line of OB processor
            // to free some memory. This memory may be used on memory_limit error.
            
$this->_emergBuffer str_repeat('a'1024 200);

            
// Intercept fatal errors via display_errors (seems it is the only way).
            
$this->_uniqHash md5('JsHttpRequest' microtime() . getmypid());
            
$this->_prevDisplayErrors ini_get('display_errors');
            
ini_set('display_errors'$this->_magic); //
            
ini_set('error_prepend_string'$this->_uniqHash ini_get('error_prepend_string'));
            
ini_set('error_append_string',  ini_get('error_append_string') . $this->_uniqHash);

            
// Start OB handling early.
            
ob_start(array(&$this"_obHandler"));
            
$JsHttpRequest_Active false;

            
// Set up the encoding.
            
$this->setEncoding($enc);

            
// Check if headers are already sent (see Content-Type library usage).
            // If true - generate a debug message and exit.
            
$file $line null;
            
$headersSent version_compare(PHP_VERSION"4.3.0") < 0headers_sent() : headers_sent($file$line);
            if (
$headersSent) {
                
trigger_error(
                    
"HTTP headers are already sent" . ($line !== null" in $file on line $line" somewhere in the script") . ". "
                    
"Possibly you have an extra space (or a newline) before the first line of the script or any library. "
                    
"Please note that JsHttpRequest uses its own Content-Type header and fails if "
                    
"this header cannot be set. See header() function documentation for more details",
                    
E_USER_ERROR
                
);
                exit();
            }
        } else {
            
$this->ID 0;
            
$this->LOADER 'unknown';
            
$JsHttpRequest_Active false;
        }
    }


    
/**
     * Static function.
     * Returns true if JsHttpRequest output processor is currently active.
     *
     * @return boolean    True if the library is active, false otherwise.
     */
    
function isActive()
    {
        return !empty(
$GLOBALS['JsHttpRequest_Active']);
    }


    
/**
     * string getJsCode()
     *
     * Return JavaScript part of the library.
     */
    
function getJsCode()
    {
        return 
file_get_contents(dirname(__FILE__) . '/JsHttpRequest.js');
    }


    
/**
     * void setEncoding(string $encoding)
     *
     * Set an active script encoding & correct QUERY_STRING according to it.
     * Examples:
     *   "windows-1251"          - set plain encoding (non-windows characters,
     *                             e.g. hieroglyphs, are totally ignored)
     *   "windows-1251 entities" - set windows encoding, BUT additionally replace:
     *                             "&"         ->  "&amp;"
     *                             hieroglyph  ->  &#XXXX; entity
     */
    
function setEncoding($enc)
    {
        
// Parse an encoding.
        
preg_match('/^(S*)(?:s+(S*))$/'$enc$p);
        
$this->SCRIPT_ENCODING    strtolower(!empty($p[1])? $p[1] : $enc);
        
$this->SCRIPT_DECODE_MODE = !empty($p[2])? $p[2] : '';
        
// Manually parse QUERY_STRING because of damned Unicode's %uXXXX.
        
$this->_correctSuperglobals();
    }


    
/**
     * string quoteInput(string $input)
     *
     * Quote a string according to the input decoding mode.
     * If entities are used (see setEncoding()), no '&' character is quoted,
     * only '"', '>' and '<' (we presume that '&' is already quoted by
     * an input reader function).
     *
     * Use this function INSTEAD of htmlspecialchars() for $_GET data
     * in your scripts.
     */
    
function quoteInput($s)
    {
        if (
$this->SCRIPT_DECODE_MODE == 'entities')
            return 
str_replace(array('"''<''>'), array('&quot;''&lt;''&gt;'), $s);
        else
            return 
htmlspecialchars($s);
    }


    
/**
     * Convert a PHP scalar, array or hash to JS scalar/array/hash. This function is
     * an analog of json_encode(), but it can work with a non-UTF8 input and does not
     * analyze the passed data. Output format must be fully JSON compatible.
     *
     * @param mixed $a   Any structure to convert to JS.
     * @return string    JavaScript equivalent structure.
     */
    
function php2js($a=false)
    {
        if (
is_null($a)) return 'null';
        if (
$a === false) return 'false';
        if (
$a === true) return 'true';
        if (
is_scalar($a)) {
            if (
is_float($a)) {
                
// Always use "." for floats.
                
$a str_replace(",""."strval($a));
            }
            
// All scalars are converted to strings to avoid indeterminism.
            // PHP's "1" and 1 are equal for all PHP operators, but
            // JS's "1" and 1 are not. So if we pass "1" or 1 from the PHP backend,
            // we should get the same result in the JS frontend (string).
            // Character replacements for JSON.
            
static $jsonReplaces = array(
                array(
"\", "/", "n", "t", "r", "b", "f", '"'),
                array('
\\', '\/', '\n', '\t', '\r', '\b', '\f', '"')
            );
            return '"' . str_replace($jsonReplaces[0], $jsonReplaces[1], $a) . '"';
        }
        
$isList = true;
        for (
$i = 0, reset($a); $i < count($a); $i++, next($a)) {
            if (key(
$a) !== $i) {
                
$isList = false;
                break;
            }
        }
        
$result = array();
        if (
$isList) {
            foreach (
$a as $v) {
                
$result[] = JsHttpRequest::php2js($v);
            }
            return '[ ' . join(', ', 
$result) . ' ]';
        } else {
            foreach (
$a as $k => $v) {
                
$result[] = JsHttpRequest::php2js($k) . ': ' . JsHttpRequest::php2js($v);
            }
            return '{ ' . join(', ', 
$result) . ' }';
        }
    }


    /**
     * Internal methods.
     */

    /**
     * Parse & decode QUERY_STRING.
     */
    function _correctSuperglobals()
    {
        // In case of FORM loader we may go to nirvana, everything is already parsed by PHP.
        if (
$this->LOADER == 'form') return;

        // ATTENTION!!!
        // HTTP_RAW_POST_DATA is only accessible when Content-Type of POST request
        // is NOT default "
application/x-www-form-urlencoded"!!!
        // Library frontend sets "
application/octet-stream" for that purpose,
        // see JavaScript code. In PHP 5.2.2.HTTP_RAW_POST_DATA is not set sometimes;
        // in such cases - read the POST data manually from the STDIN stream.
        
$rawPost = strcasecmp($_SERVER['REQUEST_METHOD'], 'POST') == 0? (isset($GLOBALS['HTTP_RAW_POST_DATA'])? $GLOBALS['HTTP_RAW_POST_DATA'] : @file_get_contents("php://input")) : null;
        
$source = array(
            
'_GET' => !empty($_SERVER['QUERY_STRING'])? $_SERVER['QUERY_STRING'] : null,
            
'_POST'=> $rawPost,
        );
        foreach (
$source as $dst=>$src) {
            
// First correct all 2-byte entities.
            
$s preg_replace('/%(?!5B)(?!5D)([0-9a-f]{2})/si''%u00\1'$src);
            
// Now we can use standard parse_str() with no worry!
            
$data null;
            
parse_str($s$data);
            
$GLOBALS[$dst] = $this->_ucs2EntitiesDecode($data);
        }
        
$GLOBALS['HTTP_GET_VARS'] = $_GET// deprecated vars
        
$GLOBALS['HTTP_POST_VARS'] = $_POST;
        
$_REQUEST =
            (isset(
$_COOKIE)? $_COOKIE : array()) +
            (isset(
$_POST)? $_POST : array()) +
            (isset(
$_GET)? $_GET : array());
        if (
ini_get('register_globals')) {
            
// TODO?
        
}
    }


    
/**
     * Called in case of error too!
     */
    
function _obHandler($text)
    {
        unset(
$this->_emergBuffer); // free a piece of memory for memory_limit error
        
unset($GLOBALS['JsHttpRequest_Active']);

        
// Check for error & fetch a resulting data.
        
if (preg_match("/{$this->_uniqHash}(.*?){$this->_uniqHash}/sx"$text$m)) {
            if (!
ini_get('display_errors') || (!$this->_prevDisplayErrors && ini_get('display_errors') == $this->_magic)) {
                
// Display_errors:
                // 1. disabled manually after the library initialization, or
                // 2. was initially disabled and is not changed
                
$text str_replace($m[0], ''$text); // strip whole error message
            
} else {
                
$text str_replace($this->_uniqHash''$text);
            }
        }
        if (
$m && preg_match('/bFatal error(<.*?>)?:/i'$m[1])) {
            
// On fatal errors - force null result (generate 500 error).
            
$this->RESULT null;
        } else {
            
// Make a resulting hash.
            
if (!isset($this->RESULT)) {
                
$this->RESULT = isset($GLOBALS['_RESULT'])? $GLOBALS['_RESULT'] : null;
            }
        }

        
$encoding $this->SCRIPT_ENCODING;
        
$result = array(
            
'id'   => $this->ID,
            
'js'   => $this->RESULT,
            
'text' => $text,
        );
        if (
function_exists('array_walk_recursive') && function_exists('json_encode') && $this->_unicodeConvMethod) {
            
$encoding "UTF-8";
            
$this->_nonAsciiChars join(""array_map('chr'range(128255)));
            
$this->_toUtfFailed false;
            
array_walk_recursive($result, array(&$this'_toUtf8_callback'), $this->SCRIPT_ENCODING);
            if (!
$this->_toUtfFailed) {
                
// If some key contains non-ASCII character, convert everything manually.
                
$text json_encode($result);
            } else {
                
$text $this->php2js($result);
            }
        } else {
            
$text $this->php2js($result);
        }

        
// Content-type header.
        // In XMLHttpRequest mode we must return text/plain - damned stupid Opera 8.0. :(
        
$ctype = !empty($this->_contentTypes[$this->LOADER])? $this->_contentTypes[$this->LOADER] : $this->_contentTypes[''];
        
header("Content-type: $ctype; charset=$encoding");

        if (
$this->LOADER != "xml") {
            
// In non-XML mode we cannot use plain JSON. So - wrap with JS function call.
            // If top.JsHttpRequestGlobal is not defined, loading is aborted and
            // iframe is removed, so - do not call dataReady().
            
$text ""
                
. ($this->LOADER == "form"'top && top.JsHttpRequestGlobal && top.JsHttpRequestGlobal' 'JsHttpRequest')
                . 
".dataReady(" $text ")n"
                
"";
            if (
$this->LOADER == "form") {
                
$text '<script type="text/javascript" language="JavaScript"><!--' "n$text'//--></script>';
            }
        }

        return 
$text;
    }


    
/**
     * Internal function, used in array_walk_recursive() before json_encode() call.
     * If a key contains non-ASCII characters, this function sets $this->_toUtfFailed = true,
     * becaues array_walk_recursive() cannot modify array keys.
     */
    
function _toUtf8_callback(&$v$k$fromEnc)
    {
        if (
$v === null || is_bool($v)) return;
        if (
$this->_toUtfFailed || !is_scalar($v) || strpbrk($k$this->_nonAsciiChars) !== false) {
            
$this->_toUtfFailed true;
        } else {
            
$v $this->_unicodeConv($fromEnc'UTF-8'$v);
        }
    }


    
/**
     * Decode all %uXXXX entities in string or array (recurrent).
     * String must not contain %XX entities - they are ignored!
     */
    
function _ucs2EntitiesDecode($data)
    {
        if (
is_array($data)) {
            
$d = array();
            foreach (
$data as $k=>$v) {
                
$d[$this->_ucs2EntitiesDecode($k)] = $this->_ucs2EntitiesDecode($v);
            }
            return 
$d;
        } else {
            if (
strpos($data'%u') !== false) { // improve speed
                
$data preg_replace_callback('/%u([0-9A-F]{1,4})/si', array(&$this'_ucs2EntitiesDecodeCallback'), $data);
            }
            return 
$data;
        }
    }


    
/**
     * Decode one %uXXXX entity (RE callback).
     */
    
function _ucs2EntitiesDecodeCallback($p)
    {
        
$hex $p[1];
        
$dec hexdec($hex);
        if (
$dec === "38" && $this->SCRIPT_DECODE_MODE == 'entities') {
            
// Process "&" separately in "entities" decode mode.
            
$c "&amp;";
        } else {
            if (
$this->_unicodeConvMethod) {
                
$c = @$this->_unicodeConv('UCS-2BE'$this->SCRIPT_ENCODINGpack('n'$dec));
            } else {
                
$c $this->_decUcs2Decode($dec$this->SCRIPT_ENCODING);
            }
            if (!
strlen($c)) {
                if (
$this->SCRIPT_DECODE_MODE == 'entities') {
                    
$c '&#' $dec ';';
                } else {
                    
$c '?';
                }
            }
        }
        return 
$c;
    }


    
/**
     * Wrapper for iconv() or mb_convert_encoding() functions.
     * This function will generate fatal error if none of these functons available!
     *
     * @see iconv()
     */
    
function _unicodeConv($fromEnc$toEnc$v)
    {
        if (
$this->_unicodeConvMethod == 'iconv') {
            return 
iconv($fromEnc$toEnc$v);
        }
        return 
mb_convert_encoding($v$toEnc$fromEnc);
    }


    
/**
     * If there is no ICONV, try to decode 1-byte characters manually
     * (for most popular charsets only).
     */

    /**
     * Convert from UCS-2BE decimal to $toEnc.
     */
    
function _decUcs2Decode($code$toEnc)
    {
        if (
$code 128) return chr($code);
        if (isset(
$this->_encTables[$toEnc])) {
            
// TODO: possible speedup by using array_flip($this->_encTables) and later hash access in the constructor.
            
$p array_search($code$this->_encTables[$toEnc]);
            if (
$p !== false) return chr(128 $p);
        }
        return 
"";
    }


    
/**
     * UCS-2BE -> 1-byte encodings (from #128).
     */
    
var $_encTables = array(
        
'windows-1251' => array(
            
0x04020x04030x201A0x04530x201E0x20260x20200x2021,
            
0x20AC0x20300x04090x20390x040A0x040C0x040B0x040F,
            
0x04520x20180x20190x201C0x201D0x20220x20130x2014,
            
0x00980x21220x04590x203A0x045A0x045C0x045B0x045F,
            
0x00A00x040E0x045E0x04080x00A40x04900x00A60x00A7,
            
0x04010x00A90x04040x00AB0x00AC0x00AD0x00AE0x0407,
            
0x00B00x00B10x04060x04560x04910x00B50x00B60x00B7,
            
0x04510x21160x04540x00BB0x04580x04050x04550x0457,
            
0x04100x04110x04120x04130x04140x04150x04160x0417,
            
0x04180x04190x041A0x041B0x041C0x041D0x041E0x041F,
            
0x04200x04210x04220x04230x04240x04250x04260x0427,
            
0x04280x04290x042A0x042B0x042C0x042D0x042E0x042F,
            
0x04300x04310x04320x04330x04340x04350x04360x0437,
            
0x04380x04390x043A0x043B0x043C0x043D0x043E0x043F,
            
0x04400x04410x04420x04430x04440x04450x04460x0447,
            
0x04480x04490x044A0x044B0x044C0x044D0x044E0x044F,
        ),
        
'koi8-r' => array(
            
0x25000x25020x250C0x25100x25140x25180x251C0x2524,
            
0x252C0x25340x253C0x25800x25840x25880x258C0x2590,
            
0x25910x25920x25930x23200x25A00x22190x221A0x2248,
            
0x22640x22650x00A00x23210x00B00x00B20x00B70x00F7,
            
0x25500x25510x25520x04510x25530x25540x25550x2556,
            
0x25570x25580x25590x255A0x255B0x255C0x255d0x255E,
            
0x255F0x25600x25610x04010x25620x25630x25640x2565,
            
0x25660x25670x25680x25690x256A0x256B0x256C0x00A9,
            
0x044E0x04300x04310x04460x04340x04350x04440x0433,
            
0x04450x04380x04390x043A0x043B0x043C0x043d0x043E,
            
0x043F0x044F0x04400x04410x04420x04430x04360x0432,
            
0x044C0x044B0x04370x04480x044d0x04490x04470x044A,
            
0x042E0x04100x04110x04260x04140x04150x04240x0413,
            
0x04250x04180x04190x041A0x041B0x041C0x041d0x041E,
            
0x041F0x042F0x04200x04210x04220x04230x04160x0412,
            
0x042C0x042B0x04170x04280x042d0x04290x04270x042A
        
),
    );
}
?>
Онлайн: 2
Реклама