2021-03-27 18:28:09 +01:00
|
|
|
<?php
|
|
|
|
/**
|
2023-01-01 09:36:24 -05:00
|
|
|
* @copyright Copyright (C) 2010-2023, the Friendica project
|
2021-03-27 18:28:09 +01:00
|
|
|
*
|
|
|
|
* @license GNU AGPL version 3 or any later version
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
2021-05-24 22:05:02 +02:00
|
|
|
|
2021-03-27 18:28:09 +01:00
|
|
|
namespace Friendica\Object\Log;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse a log line and offer some utility methods
|
|
|
|
*/
|
2021-08-20 09:47:53 +02:00
|
|
|
class ParsedLogLine
|
2021-03-27 18:28:09 +01:00
|
|
|
{
|
|
|
|
const REGEXP = '/^(\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}[^ ]*) (\w+) \[(\w*)\]: (.*)/';
|
|
|
|
|
2021-05-24 21:47:10 +02:00
|
|
|
/** @var int */
|
2021-03-27 18:28:09 +01:00
|
|
|
public $id = 0;
|
2021-05-24 21:47:10 +02:00
|
|
|
|
|
|
|
/** @var string */
|
2021-03-27 18:28:09 +01:00
|
|
|
public $date = null;
|
2021-05-24 21:47:10 +02:00
|
|
|
|
|
|
|
/** @var string */
|
2021-03-27 18:28:09 +01:00
|
|
|
public $context = null;
|
2021-05-24 21:47:10 +02:00
|
|
|
|
|
|
|
/** @var string */
|
2021-03-27 18:28:09 +01:00
|
|
|
public $level = null;
|
2021-05-24 21:47:10 +02:00
|
|
|
|
|
|
|
/** @var string */
|
2021-03-27 18:28:09 +01:00
|
|
|
public $message = null;
|
2021-05-24 21:47:10 +02:00
|
|
|
|
|
|
|
/** @var string */
|
2021-03-27 18:28:09 +01:00
|
|
|
public $data = null;
|
2021-05-24 21:47:10 +02:00
|
|
|
|
|
|
|
/** @var string */
|
2021-03-27 18:28:09 +01:00
|
|
|
public $source = null;
|
|
|
|
|
2021-05-24 21:47:10 +02:00
|
|
|
/** @var string */
|
|
|
|
public $logline;
|
|
|
|
|
2021-03-27 18:28:09 +01:00
|
|
|
/**
|
2021-05-24 21:47:10 +02:00
|
|
|
* @param int line id
|
2021-03-27 18:28:09 +01:00
|
|
|
* @param string $logline Source log line to parse
|
|
|
|
*/
|
|
|
|
public function __construct(int $id, string $logline)
|
|
|
|
{
|
|
|
|
$this->id = $id;
|
|
|
|
$this->parse($logline);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function parse($logline)
|
|
|
|
{
|
2021-08-20 09:47:53 +02:00
|
|
|
$this->logline = $logline;
|
|
|
|
|
2021-07-20 18:03:20 +02:00
|
|
|
// if data is empty is serialized as '[]'. To ease the parsing
|
|
|
|
// let's replace it with '{""}'. It will be replaced by null later
|
|
|
|
$logline = str_replace(' [] - {', ' {""} - {', $logline);
|
|
|
|
|
2021-05-24 22:05:02 +02:00
|
|
|
|
2021-08-20 09:47:53 +02:00
|
|
|
if (strstr($logline, ' - {') === false) {
|
|
|
|
// the log line is not well formed
|
|
|
|
$jsonsource = null;
|
|
|
|
} else {
|
|
|
|
// here we hope that there will not be the string ' - {' inside the $jsonsource value
|
|
|
|
list($logline, $jsonsource) = explode(' - {', $logline);
|
|
|
|
$jsonsource = '{' . $jsonsource;
|
|
|
|
}
|
2021-05-24 22:05:02 +02:00
|
|
|
|
2021-08-20 09:47:53 +02:00
|
|
|
$jsondata = null;
|
2021-03-27 18:28:09 +01:00
|
|
|
if (strpos($logline, '{"') > 0) {
|
|
|
|
list($logline, $jsondata) = explode('{"', $logline, 2);
|
2021-05-24 22:05:02 +02:00
|
|
|
|
2021-03-27 18:28:09 +01:00
|
|
|
$jsondata = '{"' . $jsondata;
|
|
|
|
}
|
2021-07-20 18:03:20 +02:00
|
|
|
|
2021-03-27 18:28:09 +01:00
|
|
|
preg_match(self::REGEXP, $logline, $matches);
|
2021-05-24 22:05:02 +02:00
|
|
|
|
2021-08-20 09:47:53 +02:00
|
|
|
if (count($matches) == 0) {
|
|
|
|
// regexp not matching
|
|
|
|
$this->message = $this->logline;
|
|
|
|
} else {
|
|
|
|
$this->date = $matches[1];
|
|
|
|
$this->context = $matches[2];
|
|
|
|
$this->level = $matches[3];
|
|
|
|
$this->message = $matches[4];
|
|
|
|
$this->data = $jsondata == '{""}' ? null : $jsondata;
|
|
|
|
$this->source = $jsonsource;
|
|
|
|
$this->tryfixjson();
|
|
|
|
}
|
2021-05-24 21:47:10 +02:00
|
|
|
|
2021-08-20 09:47:53 +02:00
|
|
|
$this->message = trim($this->message);
|
2021-03-27 18:28:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-05-24 21:47:10 +02:00
|
|
|
* Fix message / data split
|
2021-05-24 22:05:02 +02:00
|
|
|
*
|
2021-03-27 18:28:09 +01:00
|
|
|
* In log boundary between message and json data is not specified.
|
|
|
|
* If message contains '{' the parser thinks there starts the json data.
|
|
|
|
* This method try to parse the found json and if it fails, search for next '{'
|
|
|
|
* in json data and retry
|
|
|
|
*/
|
2021-08-20 09:47:53 +02:00
|
|
|
private function tryfixjson()
|
2021-03-27 18:28:09 +01:00
|
|
|
{
|
2021-07-20 18:03:20 +02:00
|
|
|
if (is_null($this->data) || $this->data == '') {
|
2021-03-27 18:28:09 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
try {
|
2021-05-24 21:47:10 +02:00
|
|
|
$d = json_decode($this->data, true, 512, JSON_THROW_ON_ERROR);
|
2021-03-27 18:28:09 +01:00
|
|
|
} catch (\JsonException $e) {
|
|
|
|
// try to find next { in $str and move string before to 'message'
|
|
|
|
|
2021-05-24 21:47:10 +02:00
|
|
|
$pos = strpos($this->data, '{', 1);
|
2021-08-20 09:47:53 +02:00
|
|
|
if ($pos === false) {
|
|
|
|
$this->message .= $this->data;
|
|
|
|
$this->data = null;
|
|
|
|
return;
|
|
|
|
}
|
2021-03-27 18:28:09 +01:00
|
|
|
|
2021-05-24 21:47:10 +02:00
|
|
|
$this->message .= substr($this->data, 0, $pos);
|
|
|
|
$this->data = substr($this->data, $pos);
|
2021-08-20 09:47:53 +02:00
|
|
|
$this->tryfixjson();
|
2021-03-27 18:28:09 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return decoded `data` as array suitable for template
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
2021-08-20 09:47:53 +02:00
|
|
|
public function getData()
|
2021-05-24 22:05:02 +02:00
|
|
|
{
|
2021-03-27 18:28:09 +01:00
|
|
|
$data = json_decode($this->data, true);
|
|
|
|
if ($data) {
|
2021-05-24 22:05:02 +02:00
|
|
|
foreach ($data as $k => $v) {
|
|
|
|
$data[$k] = print_r($v, true);
|
2021-03-27 18:28:09 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return $data;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return decoded `source` as array suitable for template
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
2021-08-20 09:47:53 +02:00
|
|
|
public function getSource()
|
2021-05-24 22:05:02 +02:00
|
|
|
{
|
2021-03-27 18:28:09 +01:00
|
|
|
return json_decode($this->source, true);
|
|
|
|
}
|
|
|
|
}
|