Created
January 31, 2015 23:12
-
-
Save sevir/bbfa3b66dae047978696 to your computer and use it in GitHub Desktop.
CodeIgniter/Creamture Log Viewer
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?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