2020-04-16 15:26:03 +03:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* MIT License
|
|
|
|
*
|
2021-12-06 14:37:43 +03:00
|
|
|
* Copyright (c) 2021 DIGITAL RETAIL TECHNOLOGIES SL
|
2020-04-16 15:26:03 +03:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
* SOFTWARE.
|
|
|
|
*
|
|
|
|
* DISCLAIMER
|
|
|
|
*
|
|
|
|
* Do not edit or add to this file if you wish to upgrade PrestaShop to newer
|
|
|
|
* versions in the future. If you wish to customize PrestaShop for your
|
|
|
|
* needs please refer to http://www.prestashop.com for more information.
|
|
|
|
*
|
2021-12-06 14:37:43 +03:00
|
|
|
* @author DIGITAL RETAIL TECHNOLOGIES SL <mail@simlachat.com>
|
|
|
|
* @copyright 2021 DIGITAL RETAIL TECHNOLOGIES SL
|
|
|
|
* @license https://opensource.org/licenses/MIT The MIT License
|
2020-04-16 15:26:03 +03:00
|
|
|
*
|
|
|
|
* Don't forget to prefix your containers with your own identifier
|
|
|
|
* to avoid any conflicts with others containers.
|
|
|
|
*/
|
2021-12-06 14:37:43 +03:00
|
|
|
|
2020-04-16 15:26:03 +03:00
|
|
|
if (!defined('_PS_VERSION_')) {
|
|
|
|
exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Class RetailcrmLogger
|
2021-11-03 16:19:39 +07:00
|
|
|
*
|
2020-04-16 15:26:03 +03:00
|
|
|
* @author DIGITAL RETAIL TECHNOLOGIES SL <mail@simlachat.com>
|
|
|
|
* @license GPL
|
2021-11-03 16:19:39 +07:00
|
|
|
*
|
|
|
|
* @see https://retailcrm.ru
|
2020-04-16 15:26:03 +03:00
|
|
|
*/
|
|
|
|
class RetailcrmLogger
|
|
|
|
{
|
2020-04-23 15:49:29 +03:00
|
|
|
static $cloneToStdout;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set to true if you want all output to be cloned into STDOUT
|
|
|
|
*
|
|
|
|
* @param bool $cloneToStdout
|
|
|
|
*/
|
|
|
|
public static function setCloneToStdout($cloneToStdout)
|
|
|
|
{
|
|
|
|
self::$cloneToStdout = $cloneToStdout;
|
|
|
|
}
|
|
|
|
|
2020-04-16 15:26:03 +03:00
|
|
|
/**
|
|
|
|
* Write entry to log
|
|
|
|
*
|
|
|
|
* @param string $caller
|
|
|
|
* @param string $message
|
|
|
|
*/
|
|
|
|
public static function writeCaller($caller, $message)
|
|
|
|
{
|
2020-04-23 15:49:29 +03:00
|
|
|
$result = sprintf(
|
|
|
|
'[%s] @ [%s] %s' . PHP_EOL,
|
|
|
|
date(DATE_RFC3339),
|
|
|
|
$caller,
|
|
|
|
$message
|
|
|
|
);
|
|
|
|
|
2020-04-16 15:26:03 +03:00
|
|
|
error_log(
|
2020-04-23 15:49:29 +03:00
|
|
|
$result,
|
2020-04-16 15:26:03 +03:00
|
|
|
3,
|
2020-04-23 15:49:29 +03:00
|
|
|
static::getLogFile()
|
2020-04-16 15:26:03 +03:00
|
|
|
);
|
2020-04-23 15:49:29 +03:00
|
|
|
|
|
|
|
if (self::$cloneToStdout) {
|
|
|
|
self::output($result, '');
|
|
|
|
}
|
2020-04-16 15:26:03 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Write entry to log without caller name
|
|
|
|
*
|
|
|
|
* @param string $message
|
|
|
|
*/
|
|
|
|
public static function writeNoCaller($message)
|
|
|
|
{
|
2020-04-23 15:49:29 +03:00
|
|
|
$result = sprintf(
|
|
|
|
'[%s] %s' . PHP_EOL,
|
|
|
|
date(DATE_RFC3339),
|
|
|
|
$message
|
|
|
|
);
|
|
|
|
|
2020-04-16 15:26:03 +03:00
|
|
|
error_log(
|
2020-04-23 15:49:29 +03:00
|
|
|
$result,
|
2020-04-16 15:26:03 +03:00
|
|
|
3,
|
2020-04-23 15:49:29 +03:00
|
|
|
static::getLogFile()
|
2020-04-16 15:26:03 +03:00
|
|
|
);
|
2020-04-23 15:49:29 +03:00
|
|
|
|
|
|
|
if (self::$cloneToStdout) {
|
|
|
|
self::output($result, '');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Output message to stdout
|
|
|
|
*
|
|
|
|
* @param string $message
|
|
|
|
* @param string $end
|
|
|
|
*/
|
|
|
|
public static function output($message = '', $end = PHP_EOL)
|
|
|
|
{
|
2021-11-03 16:19:39 +07:00
|
|
|
if ('cli' == php_sapi_name()) {
|
2020-04-23 15:49:29 +03:00
|
|
|
echo $message . $end;
|
|
|
|
}
|
2020-04-16 15:26:03 +03:00
|
|
|
}
|
|
|
|
|
2021-07-27 13:25:44 +03:00
|
|
|
/**
|
|
|
|
* Output error info to stdout
|
|
|
|
*
|
|
|
|
* @param $exception
|
|
|
|
* @param string $header
|
|
|
|
*/
|
|
|
|
public static function printException($exception, $header = 'Error while executing a job: ', $toOutput = true)
|
|
|
|
{
|
|
|
|
$method = $toOutput ? 'output' : 'writeNoCaller';
|
|
|
|
|
|
|
|
self::$method(sprintf('%s%s', $header, $exception->getMessage()));
|
|
|
|
self::$method(sprintf('%s:%d', $exception->getFile(), $exception->getLine()));
|
|
|
|
self::$method('');
|
|
|
|
self::$method($exception->getTraceAsString());
|
|
|
|
}
|
|
|
|
|
2020-04-16 15:26:03 +03:00
|
|
|
/**
|
|
|
|
* Write debug log record
|
|
|
|
*
|
|
|
|
* @param string $caller
|
|
|
|
* @param mixed $message
|
|
|
|
*/
|
|
|
|
public static function writeDebug($caller, $message)
|
|
|
|
{
|
|
|
|
if (RetailcrmTools::isDebug()) {
|
|
|
|
static::writeNoCaller(sprintf(
|
|
|
|
'(DEBUG) <%s> %s',
|
|
|
|
$caller,
|
|
|
|
print_r($message, true)
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-08 16:09:45 +03:00
|
|
|
/**
|
|
|
|
* Debug log record with multiple entries
|
|
|
|
*
|
2021-11-03 16:19:39 +07:00
|
|
|
* @param string $caller
|
2020-07-08 16:09:45 +03:00
|
|
|
* @param array|string $messages
|
|
|
|
*/
|
|
|
|
public static function writeDebugArray($caller, $messages)
|
|
|
|
{
|
|
|
|
if (RetailcrmTools::isDebug()) {
|
|
|
|
if (!empty($caller) && !empty($messages)) {
|
|
|
|
$result = is_array($messages) ? substr(
|
|
|
|
array_reduce(
|
|
|
|
$messages,
|
|
|
|
function ($carry, $item) {
|
|
|
|
$carry .= ' ' . print_r($item, true);
|
2021-11-03 16:19:39 +07:00
|
|
|
|
2020-07-08 16:09:45 +03:00
|
|
|
return $carry;
|
|
|
|
}
|
|
|
|
),
|
|
|
|
1
|
|
|
|
) : $messages;
|
|
|
|
|
|
|
|
self::writeDebug($caller, $result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-16 15:26:03 +03:00
|
|
|
/**
|
2020-04-23 15:49:29 +03:00
|
|
|
* Returns log file path
|
2020-04-16 15:26:03 +03:00
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2020-04-23 15:49:29 +03:00
|
|
|
public static function getLogFile()
|
2020-04-16 15:26:03 +03:00
|
|
|
{
|
|
|
|
if (!defined('_PS_ROOT_DIR_')) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
2021-05-28 14:32:43 +03:00
|
|
|
return self::getLogDir() . '/retailcrm_' . self::getLogFilePrefix() . '_' . date('Y_m_d') . '.log';
|
|
|
|
}
|
|
|
|
|
2021-07-27 13:25:44 +03:00
|
|
|
/**
|
|
|
|
* Returns log files directory based on current Prestashop version
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2021-05-28 14:32:43 +03:00
|
|
|
public static function getLogDir()
|
|
|
|
{
|
2021-06-16 11:17:59 +03:00
|
|
|
$logDir = version_compare(_PS_VERSION_, '1.7', '<') ? '/log' : '/var/logs';
|
|
|
|
|
|
|
|
return _PS_ROOT_DIR_ . $logDir;
|
2020-04-23 15:49:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns log file prefix based on current environment
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private static function getLogFilePrefix()
|
|
|
|
{
|
2021-11-03 16:19:39 +07:00
|
|
|
if ('cli' == php_sapi_name()) {
|
2020-04-23 15:49:29 +03:00
|
|
|
if (isset($_SERVER['TERM'])) {
|
|
|
|
return 'cli';
|
|
|
|
} else {
|
|
|
|
return 'cron';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 'web';
|
2020-04-16 15:26:03 +03:00
|
|
|
}
|
2021-05-28 14:32:43 +03:00
|
|
|
|
|
|
|
/**
|
2021-07-27 13:25:44 +03:00
|
|
|
* Removes module log files that are older than 30 days
|
2021-05-28 14:32:43 +03:00
|
|
|
*/
|
|
|
|
public static function clearObsoleteLogs()
|
|
|
|
{
|
2021-07-27 13:25:44 +03:00
|
|
|
$logFiles = self::getLogFiles();
|
2021-06-21 16:39:50 +03:00
|
|
|
|
2021-07-27 13:25:44 +03:00
|
|
|
foreach ($logFiles as $logFile) {
|
|
|
|
if (filemtime($logFile) < strtotime('-30 days')) {
|
|
|
|
unlink($logFile);
|
2021-05-28 14:32:43 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-27 13:25:44 +03:00
|
|
|
/**
|
|
|
|
* Retrieves log files basic info for advanced tab
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
2021-05-28 14:32:43 +03:00
|
|
|
public static function getLogFilesInfo()
|
|
|
|
{
|
|
|
|
$fileNames = [];
|
2021-07-27 13:25:44 +03:00
|
|
|
$logFiles = self::getLogFiles();
|
|
|
|
|
|
|
|
foreach ($logFiles as $logFile) {
|
2021-11-03 16:19:39 +07:00
|
|
|
$fileNames[] = [
|
|
|
|
'name' => basename($logFile),
|
|
|
|
'path' => $logFile,
|
|
|
|
'size' => number_format(filesize($logFile), 0, '.', ' ') . ' bytes',
|
|
|
|
'modified' => date('Y-m-d H:i:s', filemtime($logFile)),
|
|
|
|
];
|
|
|
|
}
|
2021-07-27 13:25:44 +03:00
|
|
|
|
|
|
|
return $fileNames;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieves log files paths
|
|
|
|
*
|
|
|
|
* @return Generator|void
|
|
|
|
*/
|
|
|
|
private static function getLogFiles()
|
|
|
|
{
|
2021-05-28 14:32:43 +03:00
|
|
|
$logDir = self::getLogDir();
|
|
|
|
|
2021-06-21 16:39:50 +03:00
|
|
|
if (!is_dir($logDir)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-05-28 14:32:43 +03:00
|
|
|
$handle = opendir($logDir);
|
2021-07-27 13:25:44 +03:00
|
|
|
while (($file = readdir($handle)) !== false) {
|
2021-11-03 16:19:39 +07:00
|
|
|
if (false !== self::checkFileName($file)) {
|
2021-07-27 13:25:44 +03:00
|
|
|
yield "$logDir/$file";
|
2021-05-28 14:32:43 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-27 13:25:44 +03:00
|
|
|
closedir($handle);
|
2021-05-28 14:32:43 +03:00
|
|
|
}
|
|
|
|
|
2021-07-27 13:25:44 +03:00
|
|
|
/**
|
|
|
|
* Checks if given logs filename relates to the module
|
|
|
|
*
|
|
|
|
* @param string $file
|
2021-11-03 16:19:39 +07:00
|
|
|
*
|
2021-07-27 13:25:44 +03:00
|
|
|
* @return false|string
|
|
|
|
*/
|
2021-05-28 14:32:43 +03:00
|
|
|
public static function checkFileName($file)
|
|
|
|
{
|
|
|
|
$logDir = self::getLogDir();
|
|
|
|
if (preg_match('/^retailcrm[a-zA-Z0-9-_]+.log$/', $file)) {
|
|
|
|
$path = "$logDir/$file";
|
|
|
|
if (is_file($path)) {
|
|
|
|
return $path;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2021-10-12 20:55:46 +07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Reduces error array into string
|
|
|
|
*
|
|
|
|
* @param $errors
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function reduceErrors($errors)
|
|
|
|
{
|
|
|
|
$reduced = '';
|
|
|
|
|
|
|
|
if (is_array($errors)) {
|
|
|
|
foreach ($errors as $key => $error) {
|
|
|
|
$reduced .= sprintf('%s => %s\n', $key, $error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $reduced;
|
|
|
|
}
|
2021-11-03 16:19:39 +07:00
|
|
|
}
|