Skip to content

Instantly share code, notes, and snippets.

@sevir
Created January 31, 2015 23:12
Show Gist options
  • Star 1 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save sevir/bbfa3b66dae047978696 to your computer and use it in GitHub Desktop.
Save sevir/bbfa3b66dae047978696 to your computer and use it in GitHub Desktop.
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