2020-03-10 00:04:17 +08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This file is part of MailSo.
|
|
|
|
*
|
|
|
|
* (c) 2014 Usenko Timur
|
|
|
|
*
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace MailSo\Log;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @category MailSo
|
|
|
|
* @package Log
|
|
|
|
*/
|
|
|
|
abstract class Driver
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $sDatePattern;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $sName;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $aPrefixes;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
protected $bGuidPrefix;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $sTimeOffset;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
protected $bTimePrefix;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
protected $bTypedPrefix;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $sNewLine;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
private $iWriteOnTimeoutOnly;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
private $bWriteOnErrorOnly;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
private $bWriteOnPhpErrorOnly;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
private $bFlushCache;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
private $aCache;
|
|
|
|
|
|
|
|
protected function __construct()
|
|
|
|
{
|
|
|
|
$this->sDatePattern = 'H:i:s';
|
|
|
|
$this->sName = 'INFO';
|
|
|
|
$this->sNewLine = "\r\n";
|
|
|
|
$this->bTimePrefix = true;
|
|
|
|
$this->bTypedPrefix = true;
|
|
|
|
$this->bGuidPrefix = true;
|
|
|
|
|
|
|
|
$this->sTimeOffset = '0';
|
|
|
|
|
|
|
|
$this->iWriteOnTimeoutOnly = 0;
|
|
|
|
$this->bWriteOnErrorOnly = false;
|
|
|
|
$this->bWriteOnPhpErrorOnly = false;
|
|
|
|
$this->bFlushCache = false;
|
|
|
|
$this->aCache = array();
|
|
|
|
|
|
|
|
$this->aPrefixes = array(
|
|
|
|
\MailSo\Log\Enumerations\Type::INFO => '[DATA]',
|
|
|
|
\MailSo\Log\Enumerations\Type::SECURE => '[SECURE]',
|
|
|
|
\MailSo\Log\Enumerations\Type::NOTE => '[NOTE]',
|
|
|
|
\MailSo\Log\Enumerations\Type::TIME => '[TIME]',
|
|
|
|
\MailSo\Log\Enumerations\Type::TIME_DELTA => '[TIME]',
|
|
|
|
\MailSo\Log\Enumerations\Type::MEMORY => '[MEMORY]',
|
|
|
|
\MailSo\Log\Enumerations\Type::NOTICE => '[NOTICE]',
|
|
|
|
\MailSo\Log\Enumerations\Type::WARNING => '[WARNING]',
|
|
|
|
\MailSo\Log\Enumerations\Type::ERROR => '[ERROR]',
|
|
|
|
|
|
|
|
\MailSo\Log\Enumerations\Type::NOTICE_PHP => '[NOTICE]',
|
|
|
|
\MailSo\Log\Enumerations\Type::WARNING_PHP => '[WARNING]',
|
|
|
|
\MailSo\Log\Enumerations\Type::ERROR_PHP => '[ERROR]',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
public function SetTimeOffset(string $sTimeOffset) : self
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
$this->sTimeOffset = (string) $sTimeOffset;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
public function DisableGuidPrefix() : self
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
$this->bGuidPrefix = false;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
public function DisableTimePrefix() : self
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
$this->bTimePrefix = false;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
public function WriteOnErrorOnly(bool $bValue) : self
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
$this->bWriteOnErrorOnly = !!$bValue;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
public function WriteOnPhpErrorOnly(bool $bValue) : self
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
$this->bWriteOnPhpErrorOnly = !!$bValue;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
public function WriteOnTimeoutOnly(int $iTimeout) : self
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
$this->iWriteOnTimeoutOnly = (int) $iTimeout;
|
|
|
|
if (0 > $this->iWriteOnTimeoutOnly)
|
|
|
|
{
|
|
|
|
$this->iWriteOnTimeoutOnly = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
public function DisableTypedPrefix() : self
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
$this->bTypedPrefix = false;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
abstract protected function writeImplementation($mDesc) : bool;
|
2020-03-10 00:04:17 +08:00
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
protected function writeEmptyLineImplementation() : bool
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
return $this->writeImplementation('');
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
protected function loggerLineImplementation(string $sTimePrefix, string $sDesc,
|
|
|
|
int $iType = \MailSo\Log\Enumerations\Type::INFO, string $sName = '') : string
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
return \ltrim(
|
|
|
|
($this->bTimePrefix ? '['.$sTimePrefix.']' : '').
|
|
|
|
($this->bGuidPrefix ? '['.\MailSo\Log\Logger::Guid().']' : '').
|
|
|
|
($this->bTypedPrefix ? ' '.$this->getTypedPrefix($iType, $sName) : '')
|
|
|
|
).$sDesc;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
protected function clearImplementation() : bool
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
protected function getTimeWithMicroSec() : string
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
$aMicroTimeItems = \explode(' ', \microtime());
|
|
|
|
return \MailSo\Log\Logger::DateHelper($this->sDatePattern, $this->sTimeOffset, $aMicroTimeItems[1]).'.'.
|
|
|
|
\str_pad((int) ($aMicroTimeItems[0] * 1000), 3, '0', STR_PAD_LEFT);
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
protected function getTypedPrefix(int $iType, string $sName = '') : string
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
$sName = 0 < \strlen($sName) ? $sName : $this->sName;
|
|
|
|
return isset($this->aPrefixes[$iType]) ? $sName.$this->aPrefixes[$iType].': ' : '';
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
protected function localWriteImplementation($mDesc, bool $bDiplayCrLf = false) : string
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
if ($bDiplayCrLf)
|
|
|
|
{
|
|
|
|
if (\is_array($mDesc))
|
|
|
|
{
|
|
|
|
foreach ($mDesc as &$sLine)
|
|
|
|
{
|
|
|
|
$sLine = \strtr($sLine, array("\r" => '\r', "\n" => '\n'.$this->sNewLine));
|
|
|
|
$sLine = \rtrim($sLine);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$mDesc = \strtr($mDesc, array("\r" => '\r', "\n" => '\n'.$this->sNewLine));
|
|
|
|
$mDesc = \rtrim($mDesc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->writeImplementation($mDesc);
|
|
|
|
}
|
|
|
|
|
|
|
|
final public function Write($sDesc, $iType = \MailSo\Log\Enumerations\Type::INFO, $sName = '', $bDiplayCrLf = false)
|
|
|
|
{
|
|
|
|
$bResult = true;
|
|
|
|
if (!$this->bFlushCache && ($this->bWriteOnErrorOnly || $this->bWriteOnPhpErrorOnly || 0 < $this->iWriteOnTimeoutOnly))
|
|
|
|
{
|
|
|
|
$bErrorPhp = false;
|
|
|
|
|
|
|
|
$bError = $this->bWriteOnErrorOnly && \in_array($iType, array(
|
|
|
|
\MailSo\Log\Enumerations\Type::NOTICE,
|
|
|
|
\MailSo\Log\Enumerations\Type::NOTICE_PHP,
|
|
|
|
\MailSo\Log\Enumerations\Type::WARNING,
|
|
|
|
\MailSo\Log\Enumerations\Type::WARNING_PHP,
|
|
|
|
\MailSo\Log\Enumerations\Type::ERROR,
|
|
|
|
\MailSo\Log\Enumerations\Type::ERROR_PHP
|
|
|
|
));
|
|
|
|
|
|
|
|
if (!$bError)
|
|
|
|
{
|
|
|
|
$bErrorPhp = $this->bWriteOnPhpErrorOnly && \in_array($iType, array(
|
|
|
|
\MailSo\Log\Enumerations\Type::NOTICE_PHP,
|
|
|
|
\MailSo\Log\Enumerations\Type::WARNING_PHP,
|
|
|
|
\MailSo\Log\Enumerations\Type::ERROR_PHP
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($bError || $bErrorPhp)
|
|
|
|
{
|
|
|
|
$sFlush = '--- FlushLogCache: '.($bError ? 'WriteOnErrorOnly' : 'WriteOnPhpErrorOnly');
|
|
|
|
if (isset($this->aCache[0]) && empty($this->aCache[0]))
|
|
|
|
{
|
|
|
|
$this->aCache[0] = $sFlush;
|
|
|
|
\array_unshift($this->aCache, '');
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
\array_unshift($this->aCache, $sFlush);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->aCache[] = '--- FlushLogCache: Trigger';
|
|
|
|
$this->aCache[] = $this->loggerLineImplementation($this->getTimeWithMicroSec(), $sDesc, $iType, $sName);
|
|
|
|
|
|
|
|
$this->bFlushCache = true;
|
|
|
|
$bResult = $this->localWriteImplementation($this->aCache, $bDiplayCrLf);
|
|
|
|
$this->aCache = array();
|
|
|
|
}
|
2020-03-16 20:08:53 +08:00
|
|
|
else if (0 < $this->iWriteOnTimeoutOnly && \time() - $_SERVER['REQUEST_TIME_FLOAT'] > $this->iWriteOnTimeoutOnly)
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
2020-03-16 20:08:53 +08:00
|
|
|
$sFlush = '--- FlushLogCache: WriteOnTimeoutOnly ['.(\time() - $_SERVER['REQUEST_TIME_FLOAT']).'sec]';
|
2020-03-10 00:04:17 +08:00
|
|
|
if (isset($this->aCache[0]) && empty($this->aCache[0]))
|
|
|
|
{
|
|
|
|
$this->aCache[0] = $sFlush;
|
|
|
|
\array_unshift($this->aCache, '');
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
\array_unshift($this->aCache, $sFlush);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->aCache[] = '--- FlushLogCache: Trigger';
|
|
|
|
$this->aCache[] = $this->loggerLineImplementation($this->getTimeWithMicroSec(), $sDesc, $iType, $sName);
|
|
|
|
|
|
|
|
$this->bFlushCache = true;
|
|
|
|
$bResult = $this->localWriteImplementation($this->aCache, $bDiplayCrLf);
|
|
|
|
$this->aCache = array();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$this->aCache[] = $this->loggerLineImplementation($this->getTimeWithMicroSec(), $sDesc, $iType, $sName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$bResult = $this->localWriteImplementation(
|
|
|
|
$this->loggerLineImplementation($this->getTimeWithMicroSec(), $sDesc, $iType, $sName), $bDiplayCrLf);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $bResult;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
public function GetNewLine() : string
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
return $this->sNewLine;
|
|
|
|
}
|
|
|
|
|
2020-03-16 20:08:53 +08:00
|
|
|
final public function Clear() : bool
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
return $this->clearImplementation();
|
|
|
|
}
|
|
|
|
|
2020-03-11 01:45:00 +08:00
|
|
|
final public function WriteEmptyLine() : void
|
2020-03-10 00:04:17 +08:00
|
|
|
{
|
|
|
|
if (!$this->bFlushCache && ($this->bWriteOnErrorOnly || $this->bWriteOnPhpErrorOnly || 0 < $this->iWriteOnTimeoutOnly))
|
|
|
|
{
|
|
|
|
$this->aCache[] = '';
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$this->writeEmptyLineImplementation();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|