CodeIgniter/Creamture Log Viewer
<?php | |
/** | |
* Enhaced Log viewer for CodeIgniter | |
* SeViR @2015 | |
* based in old version of 2010 | |
*/ | |
class Viewer{ | |
//Please for security set your external ip | |
private static $allow_ips = array( | |
'127.0.0.1', | |
'84.124.50.10' | |
); | |
//default logs path | |
private static $log_path = '../application/logs/'; | |
private static $initialized = false; | |
public static $assets = array( | |
'error_icon' => '%3D%3D', | |
'load_icon' => '', | |
'debug_icon' => '', | |
'info_icon' => '', | |
'favicon' => '' | |
); | |
private function __construct() {} | |
private static function initialize() | |
{ | |
if (self::$initialized) | |
return; | |
self::disable_ob(); | |
if (file_exists(dirname( __FILE__).'/system/logs/')){ | |
self::$log_path = dirname( __FILE__).'/system/logs/'; | |
}else if (file_exists(dirname( __FILE__).'/../application/logs/')){ | |
self::$log_path = dirname( __FILE__).'/../application/logs/'; | |
} | |
self::$initialized = true; | |
} | |
//reads all log file | |
public static function readLog($file){ | |
self::initialize(); | |
echo '<link href="//maxcdn.bootstrapcdn.com/bootstrap/3.3.2/css/bootstrap.min.css" rel="stylesheet">'; | |
echo '<style>pre:empty, pre:nth-of-type(1){ display:none; }body{margin: 60px 20px 20px 20px;}</style>'; | |
if(strpos($file, './') !== FALSE) die('invalid access'); | |
if (file_exists(self::$log_path.$file)){ | |
echo '<pre>'.str_replace( | |
array( | |
'ERROR -', | |
'INFO -', | |
'DEBUG -' | |
), | |
array( | |
'</pre><pre class="alert alert-warning"><img alt="error" src="'.self::$assets['error_icon'].'" />', | |
'</pre><pre class="info"><img alt="info" src="'.self::$assets['info_icon'].'" />', | |
'</pre><pre class="debug"><img alt="debug" src="'.self::$assets['debug_icon'].'" />', | |
),file_get_contents(self::$log_path.$file) | |
).'</pre>'; | |
} | |
} | |
//Check if you access from a valid IP | |
public static function checkAccess(){ | |
self::initialize(); | |
return in_array($_SERVER['REMOTE_ADDR'], self::$allow_ips); | |
} | |
//rename the old logfile | |
public static function clearLog(){ | |
self::initialize(); | |
$the_file = 'log-'.date('Y-m-d').'.php'; | |
if (file_exists(self::$log_path.$the_file)) rename(self::$log_path.$the_file,self::$log_path.str_replace('.php', '', $the_file).'_'.date('Gis').'.php'); | |
header('Location: '.str_replace('?'.$_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI'])); | |
} | |
//compress logs in a ZipFile and it removes them | |
public static function compressLogs(){ | |
self::initialize(); | |
$log_files = glob(self::$log_path.'*.php'); | |
$zip = new ZipArchive(); | |
$zip->open(self::$log_path.'backup_'.date('Y-m-d').'_'.date('Gis').'.zip',ZIPARCHIVE::OVERWRITE); | |
foreach($log_files as $file) { | |
$zip->addFile($file,str_replace(dirname($file), '', $file)); | |
} | |
$zip->close(); | |
foreach($log_files as $file) { | |
unlink($file); | |
} | |
header('Location: '.str_replace('?'.$_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI'])); | |
} | |
//compress logs and download it | |
public static function downloadLogs(){ | |
self::initialize(); | |
$log_files = glob(self::$log_path.'*.php'); | |
$zipname = self::$log_path.'backup_'.date('Y-m-d').'_'.date('Gis').'.zip'; | |
$zip = new ZipArchive(); | |
$zip->open($zipname,ZIPARCHIVE::OVERWRITE); | |
foreach($log_files as $file) { | |
$zip->addFile($file,str_replace(dirname($file), '', $file)); | |
} | |
$zip->close(); | |
header('Content-disposition: attachment; filename='.str_replace(dirname($zipname), '', $zipname)); | |
header('Content-type: application/octet-stream'); | |
readfile($zipname); | |
unlink($zipname); | |
} | |
//download a zip log backup file | |
public static function downloadFile($file){ | |
self::initialize(); | |
if((strpos($file, './') !== FALSE) || (strpos($file,'.zip') === FALSE)) die('invalid access'); | |
header('Content-disposition: attachment; filename='.$file); | |
header('Content-type: application/octet-stream'); | |
readfile(self::$log_path.$file); | |
} | |
//get list of logs and zip backups | |
public static function getFiles(){ | |
$log_files = glob(self::$log_path.'*.php'); | |
usort( | |
$log_files, | |
create_function('$a,$b', 'return filemtime($b) - filemtime($a);') | |
); | |
$backups_files = glob(self::$log_path.'*.zip'); | |
usort( | |
$backups_files, | |
create_function('$a,$b', 'return filemtime($b) - filemtime($a);') | |
); | |
return array( | |
'logs'=> $log_files, | |
'backups'=> $backups_files | |
); | |
} | |
//hide progress | |
public static function stopProgress(){ | |
?> | |
<script type="text/javascript"> | |
window.top.document.getElementById('progress').style.display = 'none'; | |
window.scrollTo(0,document.body.scrollHeight); | |
</script> | |
<?php | |
} | |
public static function disable_ob() { | |
// Turn off output buffering | |
ini_set('output_buffering', 'off'); | |
// Turn off PHP output compression | |
ini_set('zlib.output_compression', false); | |
// Implicitly flush the buffer(s) | |
ini_set('implicit_flush', true); | |
ob_implicit_flush(true); | |
// Clear, and turn off output buffering | |
while (ob_get_level() > 0) { | |
// Get the curent level | |
$level = ob_get_level(); | |
// End the buffering | |
ob_end_clean(); | |
// If the current level has not changed, abort | |
if (ob_get_level() == $level) break; | |
} | |
// Disable apache output buffering/compression | |
if (function_exists('apache_setenv')) { | |
apache_setenv('no-gzip', '1'); | |
apache_setenv('dont-vary', '1'); | |
} | |
} | |
public static function realtime(){ | |
echo '<link href="//maxcdn.bootstrapcdn.com/bootstrap/3.3.2/css/bootstrap.min.css" rel="stylesheet">'; | |
?> | |
<style type="text/css">pre:empty, pre:nth-of-type(1){ display:none;}body{margin: 60px 20px 20px 20px;}</style> | |
<script type="text/javascript"> | |
setInterval(function(){ | |
window.scrollBy(0,200); | |
}, 300); | |
</script> | |
<?php | |
$the_file = self::$log_path.'log-'.date('Y-m-d').'.php'; | |
if ( file_exists($the_file)){ | |
ini_set('max_execution_time', 0); | |
$lastpos = 0; | |
while (true) { | |
usleep(300000); //0.3 s | |
clearstatcache(false, $the_file); | |
$len = filesize($the_file); | |
if ($len < $lastpos) { | |
//file deleted or reset | |
$lastpos = $len; | |
} | |
elseif ($len > $lastpos) { | |
$f = fopen($the_file, "rb"); | |
if ($f === false) | |
die(); | |
fseek($f, $lastpos); | |
while (!feof($f)) { | |
$buffer = fread($f, 4096); | |
echo '<pre>'.str_replace( | |
array( | |
'ERROR -', | |
'INFO -', | |
'DEBUG -' | |
), | |
array( | |
'</pre><pre class="alert alert-warning"><img alt="error" src="'.self::$assets['error_icon'].'" />', | |
'</pre><pre class="info"><img alt="info" src="'.self::$assets['info_icon'].'" />', | |
'</pre><pre class="debug"><img alt="debug" src="'.self::$assets['debug_icon'].'" />', | |
),$buffer | |
).'</pre>'; | |
flush(); | |
} | |
$lastpos = ftell($f); | |
fclose($f); | |
} | |
} | |
}else{ | |
echo 'No today log found'; | |
} | |
} | |
} | |
if (!Viewer::checkAccess()) die('Your IP ('.$_SERVER['REMOTE_ADDR'].') is not allowed'); | |
if (isset($_GET['q'])){ | |
$query = $_GET['q']; | |
switch ($query) { | |
case 'phpinfo': | |
echo '<style>body{margin: 60px 20px 20px 20px;}</style>'; | |
phpinfo(); Viewer::stopProgress(); die(); | |
break; | |
case 'file': | |
if(isset($_GET['f'])){ | |
Viewer::readLog($_GET['f']); | |
} | |
Viewer::stopProgress(); | |
die(); | |
break; | |
case 'download': | |
if(isset($_GET['f'])){ | |
Viewer::downloadFile($_GET['f']); | |
} | |
die(); | |
break; | |
case 'clear': | |
Viewer::clearLog(); | |
die(); | |
break; | |
case 'downloadzip': | |
Viewer::downloadLogs(); | |
die(); | |
break; | |
case 'backup': | |
Viewer::compressLogs(); | |
die(); | |
break; | |
case 'realtime': | |
Viewer::realtime(); | |
flush(); | |
die(); | |
break; | |
default: | |
echo "no action available";Viewer::stopProgress();die(); | |
break; | |
} | |
} | |
$files = Viewer::getFiles(); | |
//And now the main view | |
?> | |
<html> | |
<head> | |
<title>Log Viewer for CodeIgniter</title> | |
<link rel='icon' type='image/png' href='<?php echo Viewer::$assets['favicon'];?>'> | |
<link href="//maxcdn.bootstrapcdn.com/bootstrap/3.3.2/css/bootstrap.min.css" rel="stylesheet"> | |
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.11.2/jquery.min.js"></script> | |
<script src="//maxcdn.bootstrapcdn.com/bootstrap/3.3.2/js/bootstrap.min.js"></script> | |
<script> | |
$(function(){ | |
$("a.action").click(function(){ | |
$("#progress").show(); | |
$("#content").attr('src', $(this).attr("data-href") ); | |
}); | |
$("#logfilter, #backupfilter").click(function(){ | |
return false; | |
}).keyup(function(){ | |
if(this.value==''){ | |
$("#"+this.id+" div").show(); | |
}else{ | |
filtertext = this.value; | |
$("#"+$(this).attr("data-id")+" div").each(function(){ | |
console.log($("a:first",this).html()); | |
console.log(filtertext); | |
if($("a:first",this).html().indexOf(filtertext) > -1){ | |
$(this).show(); | |
}else{ | |
$(this).hide(); | |
} | |
}); | |
} | |
}); | |
}); | |
</script> | |
<style> | |
.pr-info{ | |
padding-top: 5px; | |
padding-left: 10px; | |
} | |
.pr-info span{ | |
display: block; | |
font-size: 9px; | |
} | |
#backupfiles div, #logfiles div{ | |
font-size: 10px; | |
margin-left: 10px; | |
} | |
#backupfiles, #logfiles{ | |
max-height: 200px; | |
} | |
#content{ | |
position:absolute; | |
width: 100%; | |
height: 100%; | |
border: none; | |
} | |
#progress{ | |
display:none; | |
} | |
#logfilter, #backupfilter{ | |
margin: 5px 10px; | |
font-size: 10px; | |
height: 25px; | |
} | |
body{ | |
margin:0; | |
} | |
</style> | |
</head> | |
<body> | |
<nav class="navbar navbar-default navbar-fixed-top"> | |
<div class="container-fluid"> | |
<!-- Brand and toggle get grouped for better mobile display --> | |
<div class="navbar-header"> | |
<button type="button" class="navbar-toggle collapsed" data-toggle="collapse" data-target="#bs-example-navbar-collapse-1"> | |
<span class="sr-only">Toggle navigation</span> | |
<span class="icon-bar"></span> | |
<span class="icon-bar"></span> | |
<span class="icon-bar"></span> | |
</button> | |
<a class="navbar-brand" title="refresh" href="<?=$_SERVER['QUERY_STRING'].$_SERVER['REQUEST_URI']; ?>">Log Viewer for CodeIgniter</a> | |
</div> | |
<!-- Collect the nav links, forms, and other content for toggling --> | |
<div class="collapse navbar-collapse" id="bs-example-navbar-collapse-1"> | |
<ul class="nav navbar-nav navbar-fixed"> | |
<li class="pr-info"> | |
<span><strong><?php echo $_SERVER["SERVER_NAME"]?></strong></span> | |
<span>server ip: <?php echo gethostbyname ( $_SERVER["SERVER_NAME"] )?></span> | |
<span>my ip: <?php echo $_SERVER['REMOTE_ADDR'];?></span> | |
</li> | |
<!-- DEPRECATED | |
<li><a title="backup and clear this log file" href="<?php echo str_replace('?'.$_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI']).'?q=clear&f=';?>">Clear</a></li> | |
--> | |
<li><a title="rename today log and create a new" href="<?php echo str_replace('?'.$_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI']).'?q=clear';?>">Clear</a></li> | |
<li><a title="compress into a zip file and download all logs" target="_blank" href="<?php echo str_replace('?'.$_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI']).'?q=downloadzip';?>">Download</a></li> | |
<li><a title="backup into a zip file all logs and removes them" href="<?php echo str_replace('?'.$_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI']).'?q=backup';?>">Backup</a></li> | |
<li class="dropdown"> | |
<a href="#" class="dropdown-toggle" data-toggle="dropdown" role="button" aria-expanded="false">Logs files<span class="caret"></span></a> | |
<ul class="dropdown-menu" role="menu"> | |
<li> | |
<a href="#">Log files</a> | |
<div class="form-group"> | |
<input id="logfilter" data-id="logfiles" type="text" class="form-control" placeholder="Filter for..."> | |
</div><!-- /input-group --> | |
</li> | |
<li id="logfiles"> | |
<?php foreach ($files['logs'] as $file): ?> | |
<div class="filelink"><a class="action log" href="#log" data-href="<?php echo str_replace('?'.$_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI']).'?q=file&f='.basename($file);?>"><?=basename($file)?></a></div> | |
<?php endforeach ?> | |
</li> | |
<li class="divider"></li> | |
<li> | |
<a href="#">Backup files</a> | |
<div class="form-group"> | |
<input id="backupfilter" data-id="backupfiles" type="text" class="form-control" placeholder="Filter for..."/> | |
</div><!-- /input-group --> | |
</li> | |
<li id="backupfiles"> | |
<?php foreach ($files['backups'] as $file): ?> | |
<div class="filelink"><a target="_blank" href="<?php echo str_replace('?'.$_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI']).'?q=download&f='.basename($file);?>"><?=basename($file)?></a></div> | |
<?php endforeach ?> | |
</li> | |
</ul> | |
</li> | |
<li class="dropdown"> | |
<a href="#" class="dropdown-toggle" data-toggle="dropdown" role="button" aria-expanded="false">Extra info<span class="caret"></span></a> | |
<ul class="dropdown-menu" role="menu"> | |
<li><a class="action" href="#phpinfo" data-href="<?php echo str_replace('?'.$_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI']).'?q=phpinfo'?>">Show phpinfo()</a></li> | |
</ul> | |
</li> | |
<li id="progress"><a href="#"><img src="<?php echo Viewer::$assets['load_icon'];?>"></a></li> | |
</ul> | |
</div><!-- /.navbar-collapse --> | |
</div><!-- /.container-fluid --> | |
</nav> | |
<div class=""> | |
<iframe id="content" src="<?php echo str_replace('?'.$_SERVER['QUERY_STRING'], '', $_SERVER['REQUEST_URI']).'?q=realtime'?>"></iframe> | |
</div> | |
</body> | |
</html> |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment