New file |
0,0 → 1,478 |
<?php if ( ! defined('BASEPATH')) exit('No direct script access allowed'); |
/** |
* CodeIgniter |
* |
* An open source application development framework for PHP 4.3.2 or newer |
* |
* @package CodeIgniter |
* @author ExpressionEngine Dev Team |
* @copyright Copyright (c) 2008, EllisLab, Inc. |
* @license http://codeigniter.com/user_guide/license.html |
* @link http://codeigniter.com |
* @since Version 1.0 |
* @filesource |
*/ |
|
// ------------------------------------------------------------------------ |
|
/** |
* Output Class |
* |
* Responsible for sending final output to browser |
* |
* @package CodeIgniter |
* @subpackage Libraries |
* @category Output |
* @author ExpressionEngine Dev Team |
* @link http://codeigniter.com/user_guide/libraries/output.html |
*/ |
class CI_Output { |
|
var $final_output; |
var $cache_expiration = 0; |
var $headers = array(); |
var $enable_profiler = FALSE; |
|
|
function CI_Output() |
{ |
log_message('debug', "Output Class Initialized"); |
} |
|
// -------------------------------------------------------------------- |
|
/** |
* Get Output |
* |
* Returns the current output string |
* |
* @access public |
* @return string |
*/ |
function get_output() |
{ |
return $this->final_output; |
} |
|
// -------------------------------------------------------------------- |
|
/** |
* Set Output |
* |
* Sets the output string |
* |
* @access public |
* @param string |
* @return void |
*/ |
function set_output($output) |
{ |
$this->final_output = $output; |
} |
|
// -------------------------------------------------------------------- |
|
/** |
* Append Output |
* |
* Appends data onto the output string |
* |
* @access public |
* @param string |
* @return void |
*/ |
function append_output($output) |
{ |
if ($this->final_output == '') |
{ |
$this->final_output = $output; |
} |
else |
{ |
$this->final_output .= $output; |
} |
} |
|
// -------------------------------------------------------------------- |
|
/** |
* Set Header |
* |
* Lets you set a server header which will be outputted with the final display. |
* |
* Note: If a file is cached, headers will not be sent. We need to figure out |
* how to permit header data to be saved with the cache data... |
* |
* @access public |
* @param string |
* @return void |
*/ |
function set_header($header, $replace = TRUE) |
{ |
$this->headers[] = array($header, $replace); |
} |
|
// -------------------------------------------------------------------- |
|
/** |
* Set HTTP Status Header |
* |
* @access public |
* @param int the status code |
* @param string |
* @return void |
*/ |
function set_status_header($code = '200', $text = '') |
{ |
$stati = array( |
'200' => 'OK', |
'201' => 'Created', |
'202' => 'Accepted', |
'203' => 'Non-Authoritative Information', |
'204' => 'No Content', |
'205' => 'Reset Content', |
'206' => 'Partial Content', |
|
'300' => 'Multiple Choices', |
'301' => 'Moved Permanently', |
'302' => 'Found', |
'304' => 'Not Modified', |
'305' => 'Use Proxy', |
'307' => 'Temporary Redirect', |
|
'400' => 'Bad Request', |
'401' => 'Unauthorized', |
'403' => 'Forbidden', |
'404' => 'Not Found', |
'405' => 'Method Not Allowed', |
'406' => 'Not Acceptable', |
'407' => 'Proxy Authentication Required', |
'408' => 'Request Timeout', |
'409' => 'Conflict', |
'410' => 'Gone', |
'411' => 'Length Required', |
'412' => 'Precondition Failed', |
'413' => 'Request Entity Too Large', |
'414' => 'Request-URI Too Long', |
'415' => 'Unsupported Media Type', |
'416' => 'Requested Range Not Satisfiable', |
'417' => 'Expectation Failed', |
|
'500' => 'Internal Server Error', |
'501' => 'Not Implemented', |
'502' => 'Bad Gateway', |
'503' => 'Service Unavailable', |
'504' => 'Gateway Timeout', |
'505' => 'HTTP Version Not Supported' |
); |
|
if ($code == '' OR ! is_numeric($code)) |
{ |
show_error('Status codes must be numeric'); |
} |
|
if (isset($stati[$code]) AND $text == '') |
{ |
$text = $stati[$code]; |
} |
|
if ($text == '') |
{ |
show_error('No status text available. Please check your status code number or supply your own message text.'); |
} |
|
$server_protocol = (isset($_SERVER['SERVER_PROTOCOL'])) ? $_SERVER['SERVER_PROTOCOL'] : FALSE; |
|
if (substr(php_sapi_name(), 0, 3) == 'cgi') |
{ |
header("Status: {$code} {$text}", TRUE); |
} |
elseif ($server_protocol == 'HTTP/1.1' OR $server_protocol == 'HTTP/1.0') |
{ |
header($server_protocol." {$code} {$text}", TRUE, $code); |
} |
else |
{ |
header("HTTP/1.1 {$code} {$text}", TRUE, $code); |
} |
} |
|
// -------------------------------------------------------------------- |
|
/** |
* Enable/disable Profiler |
* |
* @access public |
* @param bool |
* @return void |
*/ |
function enable_profiler($val = TRUE) |
{ |
$this->enable_profiler = (is_bool($val)) ? $val : TRUE; |
} |
|
// -------------------------------------------------------------------- |
|
/** |
* Set Cache |
* |
* @access public |
* @param integer |
* @return void |
*/ |
function cache($time) |
{ |
$this->cache_expiration = ( ! is_numeric($time)) ? 0 : $time; |
} |
|
// -------------------------------------------------------------------- |
|
/** |
* Display Output |
* |
* All "view" data is automatically put into this variable by the controller class: |
* |
* $this->final_output |
* |
* This function sends the finalized output data to the browser along |
* with any server headers and profile data. It also stops the |
* benchmark timer so the page rendering speed and memory usage can be shown. |
* |
* @access public |
* @return mixed |
*/ |
function _display($output = '') |
{ |
// Note: We use globals because we can't use $CI =& get_instance() |
// since this function is sometimes called by the caching mechanism, |
// which happens before the CI super object is available. |
global $BM, $CFG; |
|
// -------------------------------------------------------------------- |
|
// Set the output data |
if ($output == '') |
{ |
$output =& $this->final_output; |
} |
|
// -------------------------------------------------------------------- |
|
// Do we need to write a cache file? |
if ($this->cache_expiration > 0) |
{ |
$this->_write_cache($output); |
} |
|
// -------------------------------------------------------------------- |
|
// Parse out the elapsed time and memory usage, |
// then swap the pseudo-variables with the data |
|
$elapsed = '' ; //$BM->elapsed_time('total_execution_time_start', 'total_execution_time_end'); |
$output = str_replace('{elapsed_time}', $elapsed, $output); |
|
$memory = '' ;//( ! function_exists('memory_get_usage')) ? '0' : round(memory_get_usage()/1024/1024, 2).'MB'; |
$output = str_replace('{memory_usage}', $memory, $output); |
|
// -------------------------------------------------------------------- |
|
// Is compression requested? |
/*if ($CFG->item('compress_output') === TRUE) |
{ |
if (extension_loaded('zlib')) |
{ |
if (isset($_SERVER['HTTP_ACCEPT_ENCODING']) AND strpos($_SERVER['HTTP_ACCEPT_ENCODING'], 'gzip') !== FALSE) |
{ |
ob_start('ob_gzhandler'); |
} |
} |
}*/ |
|
// -------------------------------------------------------------------- |
|
// Are there any server headers to send? |
if (count($this->headers) > 0) |
{ |
foreach ($this->headers as $header) |
{ |
@header($header[0], $header[1]); |
} |
} |
|
// -------------------------------------------------------------------- |
|
// Does the get_instance() function exist? |
// If not we know we are dealing with a cache file so we'll |
// simply echo out the data and exit. |
if ( ! function_exists('get_instance')) |
{ |
echo $output; |
log_message('debug', "Final output sent to browser"); |
log_message('debug', "Total execution time: ".$elapsed); |
return TRUE; |
} |
|
// -------------------------------------------------------------------- |
|
// Grab the super object. We'll need it in a moment... |
$CI =& get_instance(); |
|
// Do we need to generate profile data? |
// If so, load the Profile class and run it. |
if ($this->enable_profiler == TRUE) |
{ |
$CI->load->library('profiler'); |
|
// If the output data contains closing </body> and </html> tags |
// we will remove them and add them back after we insert the profile data |
if (preg_match("|</body>.*?</html>|is", $output)) |
{ |
$output = preg_replace("|</body>.*?</html>|is", '', $output); |
$output .= $CI->profiler->run(); |
$output .= '</body></html>'; |
} |
else |
{ |
$output .= $CI->profiler->run(); |
} |
} |
|
// -------------------------------------------------------------------- |
|
// Does the controller contain a function named _output()? |
// If so send the output there. Otherwise, echo it. |
if (method_exists($CI, '_output')) |
{ |
$CI->_output($output); |
} |
else |
{ |
echo $output; // Send it to the browser! |
} |
|
log_message('debug', "Final output sent to browser"); |
log_message('debug', "Total execution time: ".$elapsed); |
} |
|
// -------------------------------------------------------------------- |
|
/** |
* Write a Cache File |
* |
* @access public |
* @return void |
*/ |
function _write_cache($output) |
{ |
$CI =& get_instance(); |
$path = $CI->config->item('cache_path'); |
|
$cache_path = ($path == '') ? BASEPATH.'cache/' : $path; |
|
if ( ! is_dir($cache_path) OR ! is_really_writable($cache_path)) |
{ |
return; |
} |
|
$uri = $CI->config->item('base_url'). |
$CI->config->item('index_page'). |
$CI->uri->uri_string(); |
|
$cache_path .= md5($uri); |
|
if ( ! $fp = @fopen($cache_path, FOPEN_WRITE_CREATE_DESTRUCTIVE)) |
{ |
log_message('error', "Unable to write cache file: ".$cache_path); |
return; |
} |
|
$expire = time() + ($this->cache_expiration * 60); |
|
if (flock($fp, LOCK_EX)) |
{ |
fwrite($fp, $expire.'TS--->'.$output); |
flock($fp, LOCK_UN); |
} |
else |
{ |
log_message('error', "Unable to secure a file lock for file at: ".$cache_path); |
return; |
} |
fclose($fp); |
@chmod($cache_path, DIR_WRITE_MODE); |
|
log_message('debug', "Cache file written: ".$cache_path); |
} |
|
// -------------------------------------------------------------------- |
|
/** |
* Update/serve a cached file |
* |
* @access public |
* @return void |
*/ |
function _display_cache(&$CFG, &$URI) |
{ |
$cache_path = ($CFG->item('cache_path') == '') ? BASEPATH.'cache/' : $CFG->item('cache_path'); |
|
if ( ! is_dir($cache_path) OR ! is_really_writable($cache_path)) |
{ |
return FALSE; |
} |
|
// Build the file path. The file name is an MD5 hash of the full URI |
$uri = $CFG->item('base_url'). |
$CFG->item('index_page'). |
$URI->uri_string; |
|
$filepath = $cache_path.md5($uri); |
|
if ( ! @file_exists($filepath)) |
{ |
return FALSE; |
} |
|
if ( ! $fp = @fopen($filepath, FOPEN_READ)) |
{ |
return FALSE; |
} |
|
flock($fp, LOCK_SH); |
|
$cache = ''; |
if (filesize($filepath) > 0) |
{ |
$cache = fread($fp, filesize($filepath)); |
} |
|
flock($fp, LOCK_UN); |
fclose($fp); |
|
// Strip out the embedded timestamp |
if ( ! preg_match("/(\d+TS--->)/", $cache, $match)) |
{ |
return FALSE; |
} |
|
// Has the file expired? If so we'll delete it. |
if (time() >= trim(str_replace('TS--->', '', $match['1']))) |
{ |
@unlink($filepath); |
log_message('debug', "Cache file has expired. File deleted"); |
return FALSE; |
} |
|
// Display the cache |
$this->_display(str_replace($match['0'], '', $cache)); |
log_message('debug', "Cache file is current. Sending it to browser."); |
return TRUE; |
} |
|
|
} |
// END Output Class |
|
/* End of file Output.php */ |
/* Location: ./system/libraries/Output.php */ |