Improve logging and error handling (#3059)

* refactor: logging and errror handling
This commit is contained in:
Dag 2022-10-16 17:55:43 +02:00 committed by GitHub
parent e21394d2d3
commit ffbc107687
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 143 additions and 93 deletions

View File

@ -163,18 +163,17 @@ class DisplayAction implements ActionInterface
// Create "new" error message every 24 hours // Create "new" error message every 24 hours
$request['_error_time'] = urlencode((int)(time() / 86400)); $request['_error_time'] = urlencode((int)(time() / 86400));
$message = sprintf('Bridge returned error %s! (%s)', $e->getCode(), $request['_error_time']); $itemTitle = sprintf('Bridge returned error %s! (%s)', $e->getCode(), $request['_error_time']);
$item->setTitle($message); $item->setTitle($itemTitle);
$item->setURI(get_current_url()); $item->setURI(get_current_url());
$item->setTimestamp(time()); $item->setTimestamp(time());
$message = create_sane_exception_message($e); $content = render_template(__DIR__ . '/../templates/bridge-error.html.php', [
$content = render_template('bridge-error.html.php', [ 'message' => create_sane_exception_message($e),
'message' => $message, 'trace' => trace_from_exception($e),
'stacktrace' => create_sane_stacktrace($e),
'searchUrl' => self::createGithubSearchUrl($bridge), 'searchUrl' => self::createGithubSearchUrl($bridge),
'issueUrl' => self::createGithubIssueUrl($bridge, $e, $message), 'issueUrl' => self::createGithubIssueUrl($bridge, $e, create_sane_exception_message($e)),
'bridge' => $bridge, 'maintainer' => $bridge->getMaintainer(),
]); ]);
$item->setContent($content); $item->setContent($content);
@ -211,7 +210,7 @@ class DisplayAction implements ActionInterface
'body' => sprintf( 'body' => sprintf(
"```\n%s\n\n%s\n\nQuery string: %s\nVersion: %s\nOs: %s\nPHP version: %s\n```", "```\n%s\n\n%s\n\nQuery string: %s\nVersion: %s\nOs: %s\nPHP version: %s\n```",
$message, $message,
implode("\n", create_sane_stacktrace($e)), implode("\n", trace_to_call_points(trace_from_exception($e))),
$_SERVER['QUERY_STRING'] ?? '', $_SERVER['QUERY_STRING'] ?? '',
Configuration::getVersion(), Configuration::getVersion(),
PHP_OS_FAMILY, PHP_OS_FAMILY,

View File

@ -19,7 +19,8 @@ class PatreonBridge extends BridgeAbstract
public function collectData() public function collectData()
{ {
$html = getSimpleHTMLDOMCached($this->getURI(), 86400); $url = $this->getURI();
$html = getSimpleHTMLDOMCached($url);
$regex = '#/api/campaigns/([0-9]+)#'; $regex = '#/api/campaigns/([0-9]+)#';
if (preg_match($regex, $html->save(), $matches) > 0) { if (preg_match($regex, $html->save(), $matches) > 0) {
$campaign_id = $matches[1]; $campaign_id = $matches[1];

View File

@ -20,7 +20,7 @@ final class AuthenticationMiddleware
$password = $_SERVER['PHP_AUTH_PW'] ?? null; $password = $_SERVER['PHP_AUTH_PW'] ?? null;
if ($user === null || $password === null) { if ($user === null || $password === null) {
$this->renderAuthenticationDialog(); print $this->renderAuthenticationDialog();
exit; exit;
} }
if ( if (
@ -29,14 +29,14 @@ final class AuthenticationMiddleware
) { ) {
return; return;
} }
$this->renderAuthenticationDialog(); print $this->renderAuthenticationDialog();
exit; exit;
} }
private function renderAuthenticationDialog(): void private function renderAuthenticationDialog(): string
{ {
http_response_code(401); http_response_code(401);
header('WWW-Authenticate: Basic realm="RSS-Bridge"'); header('WWW-Authenticate: Basic realm="RSS-Bridge"');
print render('error.html.php', ['message' => 'Please authenticate in order to access this instance !']); return render('access-denied.html.php');
} }
} }

View File

@ -101,13 +101,12 @@ class Debug
if (!self::isEnabled()) { if (!self::isEnabled()) {
return; return;
} }
$backtrace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS, 3); $e = new \Exception();
$lastFrame = end($backtrace); $trace = trace_from_exception($e);
$file = trim_path_prefix($lastFrame['file']); // Drop the current frame
$line = $lastFrame['line']; array_pop($trace);
$class = $lastFrame['class'] ?? ''; $lastFrame = $trace[array_key_last($trace)];
$function = $lastFrame['function']; $text = sprintf('%s(%s): %s', $lastFrame['file'], $lastFrame['line'], $message);
$text = sprintf('%s:%s %s->%s() %s', $file, $line, $class, $function, $message); Logger::debug($text);
Logger::info($text);
} }
} }

View File

@ -38,11 +38,11 @@ class FormatFactory
if (! preg_match('/^[a-zA-Z0-9-]*$/', $name)) { if (! preg_match('/^[a-zA-Z0-9-]*$/', $name)) {
throw new \InvalidArgumentException('Format name invalid!'); throw new \InvalidArgumentException('Format name invalid!');
} }
$name = $this->sanitizeFormatName($name); $sanitizedName = $this->sanitizeFormatName($name);
if ($name === null) { if ($sanitizedName === null) {
throw new \InvalidArgumentException('Unknown format given!'); throw new \InvalidArgumentException(sprintf('Unknown format given `%s`', $name));
} }
$className = '\\' . $name . 'Format'; $className = '\\' . $sanitizedName . 'Format';
return new $className(); return new $className();
} }

View File

@ -4,6 +4,11 @@ declare(strict_types=1);
final class Logger final class Logger
{ {
public static function debug(string $message, array $context = [])
{
self::log('DEBUG', $message, $context);
}
public static function info(string $message, array $context = []): void public static function info(string $message, array $context = []): void
{ {
self::log('INFO', $message, $context); self::log('INFO', $message, $context);
@ -23,22 +28,20 @@ final class Logger
{ {
if (isset($context['e'])) { if (isset($context['e'])) {
$context['message'] = create_sane_exception_message($context['e']); $context['message'] = create_sane_exception_message($context['e']);
$context['file'] = trim_path_prefix($context['e']->getFile());
$context['line'] = $context['e']->getLine();
$context['code'] = $context['e']->getCode(); $context['code'] = $context['e']->getCode();
$context['url'] = get_current_url(); $context['url'] = get_current_url();
$context['trace'] = create_sane_stacktrace($context['e']); $context['trace'] = trace_to_call_points(trace_from_exception($context['e']));
unset($context['e']); unset($context['e']);
// Don't log these records
$ignoredExceptions = [ $ignoredExceptions = [
'Exception Exception: You must specify a format!', 'Exception Exception: You must specify a format',
'Exception InvalidArgumentException: Format name invalid!', 'Exception InvalidArgumentException: Format name invalid',
'Exception InvalidArgumentException: Unknown format given!', 'Exception InvalidArgumentException: Unknown format given',
'Exception InvalidArgumentException: Bridge name invalid!', 'Exception InvalidArgumentException: Bridge name invalid',
'Exception Exception: twitter: No results for this query', 'Exception Exception: twitter: No results for this query',
]; ];
foreach ($ignoredExceptions as $ignoredException) { foreach ($ignoredExceptions as $ignoredException) {
if (str_starts_with($context['message'], $ignoredException)) { if (str_starts_with($context['message'], $ignoredException)) {
// Don't log this record because it's usually a bot
return; return;
} }
} }

View File

@ -16,9 +16,9 @@ final class RssBridge
} catch (\Throwable $e) { } catch (\Throwable $e) {
Logger::error('Exception in main', ['e' => $e]); Logger::error('Exception in main', ['e' => $e]);
http_response_code(500); http_response_code(500);
print render('error.html.php', [ print render(__DIR__ . '/../templates/error.html.php', [
'message' => create_sane_exception_message($e), 'message' => create_sane_exception_message($e),
'stacktrace' => create_sane_stacktrace($e), 'trace' => trace_from_exception($e),
]); ]);
} }
} }
@ -38,6 +38,7 @@ final class RssBridge
return false; return false;
} }
$text = sprintf('%s at %s line %s', $message, trim_path_prefix($file), $line); $text = sprintf('%s at %s line %s', $message, trim_path_prefix($file), $line);
// Drop the current frame
Logger::warning($text); Logger::warning($text);
if (Debug::isEnabled()) { if (Debug::isEnabled()) {
print sprintf('<pre>%s</pre>', $text); print sprintf('<pre>%s</pre>', $text);

View File

@ -14,6 +14,9 @@
function render(string $template, array $context = []): string function render(string $template, array $context = []): string
{ {
if ($template === 'base.html.php') {
throw new \Exception('Do not render base.html.php into itself');
}
$context['page'] = render_template($template, $context); $context['page'] = render_template($template, $context);
return render_template('base.html.php', $context); return render_template('base.html.php', $context);
} }

View File

@ -48,7 +48,7 @@ function get_current_url(): string
function create_sane_exception_message(\Throwable $e): string function create_sane_exception_message(\Throwable $e): string
{ {
return sprintf( return sprintf(
'Exception %s: %s at %s line %s', 'Exception %s: %s in %s line %s',
get_class($e), get_class($e),
$e->getMessage(), $e->getMessage(),
trim_path_prefix($e->getFile()), trim_path_prefix($e->getFile()),
@ -56,7 +56,15 @@ function create_sane_exception_message(\Throwable $e): string
); );
} }
function create_sane_stacktrace(\Throwable $e): array /**
* Returns e.g. https://github.com/RSS-Bridge/rss-bridge/blob/master/bridges/AO3Bridge.php#L8
*/
function render_github_url(string $file, int $line, string $revision = 'master'): string
{
return sprintf('https://github.com/RSS-Bridge/rss-bridge/blob/%s/%s#L%s', $revision, $file, $line);
}
function trace_from_exception(\Throwable $e): array
{ {
$frames = array_reverse($e->getTrace()); $frames = array_reverse($e->getTrace());
$frames[] = [ $frames[] = [
@ -64,19 +72,50 @@ function create_sane_stacktrace(\Throwable $e): array
'line' => $e->getLine(), 'line' => $e->getLine(),
]; ];
$trace = []; $trace = [];
foreach ($frames as $i => $frame) { foreach ($frames as $frame) {
$file = $frame['file'] ?? '(no file)'; $trace[] = [
$line = $frame['line'] ?? '(no line)'; 'file' => trim_path_prefix($frame['file'] ?? ''),
$trace[] = sprintf( 'line' => $frame['line'] ?? null,
'#%s %s:%s', 'class' => $frame['class'] ?? null,
$i, 'type' => $frame['type'] ?? null,
trim_path_prefix($file), 'function' => $frame['function'] ?? null,
$line, ];
);
} }
return $trace; return $trace;
} }
function trace_to_call_points(array $trace): array
{
return array_map(fn($frame) => frame_to_call_point($frame), $trace);
}
function frame_to_call_point(array $frame): string
{
if ($frame['class']) {
return sprintf(
'%s(%s): %s%s%s()',
$frame['file'],
$frame['line'],
$frame['class'],
$frame['type'],
$frame['function'],
);
} elseif ($frame['function']) {
return sprintf(
'%s(%s): %s()',
$frame['file'],
$frame['line'],
$frame['function'],
);
} else {
return sprintf(
'%s(%s)',
$frame['file'],
$frame['line'],
);
}
}
/** /**
* Trim path prefix for privacy/security reasons * Trim path prefix for privacy/security reasons
* *

View File

@ -50,6 +50,12 @@ a:hover {
text-decoration: underline; text-decoration: underline;
} }
h1,h2 {
margin-bottom: 10px;
}
p {
margin-bottom: 10px;
}
/* Header */ /* Header */
header { header {
@ -328,12 +334,6 @@ h5 {
margin-bottom: 6px; margin-bottom: 6px;
} }
.advice {
margin-left: auto;
margin-right: auto;
display: table;
}
.advice > li { .advice > li {
text-align: left; text-align: left;
} }

View File

@ -0,0 +1,6 @@
<section>
<h1>
Please authenticate in order to access this instance
</h1>
</section>

View File

@ -1,21 +1,17 @@
<section> <section>
<p class="exception-message"> <p class="exception-message">
<?= e($message ?? '') ?> <?= e($message) ?>
</p> </p>
<?php if (isset($stacktrace)): ?> <?php foreach ($trace as $i => $frame) : ?>
<?php foreach ($stacktrace as $frame): ?> #<?= $i ?> <?= e(frame_to_call_point($frame)) ?>
<code> <br>
<?= e($frame) ?> <?php endforeach; ?>
</code>
<br>
<?php endforeach; ?>
<?php endif; ?>
<br> <br>
<p> <p>
Query string: <?= e($_SERVER['QUERY_STRING'] ?? '') ?> Query string: <?= e(urldecode($_SERVER['QUERY_STRING']) ?? '') ?>
</p> </p>
<p> <p>
Version: <?= raw(Configuration::getVersion()) ?> Version: <?= raw(Configuration::getVersion()) ?>
@ -24,26 +20,19 @@
OS: <?= raw(PHP_OS_FAMILY) ?> OS: <?= raw(PHP_OS_FAMILY) ?>
</p> </p>
<p> <p>
PHP version: <?= raw(phpversion() ?: 'Unknown'()) ?> PHP version: <?= raw(PHP_VERSION ?: 'Unknown') ?>
</p> </p>
<div class="advice">
<ul class="advice">
<li>Press Return to check your input parameters</li>
<li>Press F5 to retry</li>
<li>Check if this issue was already reported on <a href="<?= raw($searchUrl) ?>">GitHub</a> (give it a thumbs-up)</li>
<li>Open a <a href="<?= raw($issueUrl) ?>">GitHub Issue</a> if this error persists</li>
</ul>
</div>
<a href="<?= raw($searchUrl) ?>" title="Opens GitHub to search for similar issues"> <a href="<?= raw($searchUrl) ?>" title="Opens GitHub to search for similar issues">
<button>Search GitHub Issues</button> <button>Find similar bugs</button>
</a> </a>
<a href="<?= raw($issueUrl) ?>" title="After clicking this button you can review the issue before submitting it"> <a href="<?= raw($issueUrl) ?>" title="After clicking this button you can review the issue before submitting it">
<button>Open GitHub Issue</button> <button>Create GitHub Issue</button>
</a> </a>
<p class="maintainer"><?= e($bridge->getMaintainer()) ?></p> <p class="maintainer">
<?= e($maintainer) ?>
</p>
</section> </section>

View File

@ -1,25 +1,35 @@
<div style="width: 60%; margin: 30px auto"> <div style="width: 60%; margin: 30px auto">
<h1> <h1>Something went wrong</h1>
<?= e($title ?? 'Something went wrong') ?>
</h1>
<br> <p>
<?= e($message) ?> <?= e($message) ?>
<br> </p>
<br>
<?php if (isset($stacktrace)): ?> <h2>Stacktrace</h2>
<h2>Stacktrace</h2>
<?php foreach ($trace as $i => $frame) : ?>
<code>
#<?= $i ?> <?= e(frame_to_call_point($frame)) ?>
</code>
<br> <br>
<?php endforeach; ?>
<?php foreach ($stacktrace as $frame) : ?> <br>
<code>
<?= e($frame) ?>
</code>
<br>
<?php endforeach; ?>
<?php endif; ?>
<h2>Context</h2>
<p>
Query string: <?= e(urldecode($_SERVER['QUERY_STRING'] ?? '')) ?>
</p>
<p>
Version: <?= raw(Configuration::getVersion()) ?>
</p>
<p>
OS: <?= raw(PHP_OS_FAMILY) ?>
</p>
<p>
PHP version: <?= raw(PHP_VERSION ?: 'Unknown') ?>
</p>
</div> </div>