PNG  IHDR;IDATxܻn0K )(pA 7LeG{ §㻢|ذaÆ 6lذaÆ 6lذaÆ 6lom$^yذag5bÆ 6lذaÆ 6lذa{ 6lذaÆ `}HFkm,mӪôô! x|'ܢ˟;E:9&ᶒ}{v]n&6 h_tڠ͵-ҫZ;Z$.Pkž)!o>}leQfJTu іچ\X=8Rن4`Vwl>nG^is"ms$ui?wbs[m6K4O.4%/bC%t Mז -lG6mrz2s%9s@-k9=)kB5\+͂Zsٲ Rn~GRC wIcIn7jJhۛNCS|j08yiHKֶۛkɈ+;SzL/F*\Ԕ#"5m2[S=gnaPeғL lذaÆ 6l^ḵaÆ 6lذaÆ 6lذa; _ذaÆ 6lذaÆ 6lذaÆ RIENDB` * @author Bernhard Posselt * @author Joas Schilling * @author Jörn Friedrich Dreyer * @author Juan Pablo Villafañez * @author Lukas Reschke * @author Morris Jobke * @author Olivier Paroz * @author Robin Appelman * @author Roeland Jago Douma * @author Thomas Müller * @author Victor Dubiniuk * * @copyright Copyright (c) 2018, ownCloud GmbH * @license AGPL-3.0 * * This code is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License, version 3, * as published by the Free Software Foundation. * * 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, version 3, * along with this program. If not, see * */ namespace OC; use InterfaSys\LogNormalizer\Normalizer; use OCP\ILogger; use OCP\IUserSession; use OCP\Util; use Symfony\Component\EventDispatcher\GenericEvent; use Symfony\Component\EventDispatcher\EventDispatcherInterface; /** * logging utilities * * This is a stand in, this should be replaced by a Psr\Log\LoggerInterface * compatible logger. See https://github.com/php-fig/fig-standards/blob/master/accepted/PSR-3-logger-interface.md * for the full interface specification. * * MonoLog is an example implementing this interface. */ class Log implements ILogger { /** @var string */ private $logger; /** @var SystemConfig */ private $config; /** @var EventDispatcherInterface */ private $eventDispatcher; /** @var boolean|null cache the result of the log condition check for the request */ private $logConditionSatisfied; /** @var Normalizer */ private $normalizer; /** * Flag whether we are within the event block * * @var bool */ private $inEvent = false; protected static $methodsWithSensitiveParameters = [ // Session/User 'login', 'checkPassword', 'updatePrivateKeyPassword', 'validateUserPass', 'loginWithPassword', 'createSessionToken', // TokenProvider 'getToken', 'isTokenPassword', 'getPassword', 'decryptPassword', 'logClientIn', 'generateToken', 'validateToken', // TwoFactorAuth 'solveChallenge', 'verifyChallenge', //ICrypto 'calculateHMAC', 'encrypt', 'decrypt', //LoginController 'tryLogin', //User_LDAP\Connection 'bind', //User_LDAP\LDAP 'invokeLDAPMethod', //User_LDAP\Access 'areCredentialsValid', //UserManagement\Controller\UsersController 'setPassword', //any closure can leak passwords '{closure}' ]; /** * @param null $logger The logger that should be used * @param SystemConfig|null $config the system config object * @param null $normalizer * @param EventDispatcherInterface|null $eventDispatcher event dispatcher */ public function __construct( $logger = null, SystemConfig $config = null, $normalizer = null, EventDispatcherInterface $eventDispatcher = null ) { // FIXME: Add this for backwards compatibility, should be fixed at some point probably if ($config === null) { $config = \OC::$server->getSystemConfig(); } $this->config = $config; // FIXME: Add this for backwards compatibility, should be fixed at some point probably if ($logger === null) { $this->logger = 'OC\\Log\\'.\ucfirst($this->config->getValue('log_type', 'owncloud')); \call_user_func([$this->logger, 'init']); } else { $this->logger = $logger; } if ($normalizer === null) { $this->normalizer = new Normalizer(); } else { $this->normalizer = $normalizer; } if ($eventDispatcher === null) { $this->eventDispatcher = \OC::$server->getEventDispatcher(); } else { $this->eventDispatcher = $eventDispatcher; } } /** * System is unusable. * * @param string $message * @param array $context * @return void */ public function emergency($message, array $context = []) { $this->log(Util::FATAL, $message, $context); } /** * Action must be taken immediately. * * Example: Entire website down, database unavailable, etc. This should * trigger the SMS alerts and wake you up. * * @param string $message * @param array $context * @return void */ public function alert($message, array $context = []) { $this->log(Util::ERROR, $message, $context); } /** * Critical conditions. * * Example: Application component unavailable, unexpected exception. * * @param string $message * @param array $context * @return void */ public function critical($message, array $context = []) { $this->log(Util::ERROR, $message, $context); } /** * Runtime errors that do not require immediate action but should typically * be logged and monitored. * * @param string $message * @param array $context * @return void */ public function error($message, array $context = []) { $this->log(Util::ERROR, $message, $context); } /** * Exceptional occurrences that are not errors. * * Example: Use of deprecated APIs, poor use of an API, undesirable things * that are not necessarily wrong. * * @param string $message * @param array $context * @return void */ public function warning($message, array $context = []) { $this->log(Util::WARN, $message, $context); } /** * Normal but significant events. * * @param string $message * @param array $context * @return void */ public function notice($message, array $context = []) { $this->log(Util::INFO, $message, $context); } /** * Interesting events. * * Example: User logs in, SQL logs. * * @param string $message * @param array $context * @return void */ public function info($message, array $context = []) { $this->log(Util::INFO, $message, $context); } /** * Detailed debug information. * * @param string $message * @param array $context * @return void */ public function debug($message, array $context = []) { $this->log(Util::DEBUG, $message, $context); } /** * Logs with an arbitrary level. * * @param mixed $level * @param string $message * @param array $context * @return void */ public function log($level, $message, array $context = []) { $minLevel = \min($this->config->getValue('loglevel', Util::WARN), Util::FATAL); $logConditions = $this->config->getValue('log.conditions', []); if (empty($logConditions)) { $logConditions[] = $this->config->getValue('log.condition', []); } $logConditionFile = null; $extraFields = []; if (isset($context['extraFields'])) { $extraFields = $context['extraFields']; unset($context['extraFields']); } $exception = null; if (isset($context['exception'])) { $exception = $context['exception']; unset($context['exception']); } \array_walk($context, [$this->normalizer, 'format']); if (isset($context['app'])) { $app = $context['app']; /** * check log condition based on the context of each log message * once this is met -> change the required log level to debug */ if (!empty($logConditions)) { foreach ($logConditions as $logCondition) { if (!empty($logCondition['apps']) && \in_array($app, $logCondition['apps'], true)) { $minLevel = Util::DEBUG; if (!empty($logCondition['logfile'])) { $logConditionFile = $logCondition['logfile']; break; } } } } } else { $app = 'no app in context'; } // interpolate $message as defined in PSR-3 $replace = []; foreach ($context as $key => $val) { $replace['{' . $key . '}'] = $val; } // interpolate replacement values into the message and return $formattedMessage = \strtr($message, $replace); /** * check for a special log condition - this enables an increased log on * a per request/user base */ if ($this->logConditionSatisfied === null) { // default to false to just process this once per request $this->logConditionSatisfied = false; if (!empty($logConditions)) { foreach ($logConditions as $logCondition) { // check for secret token in the request if (!empty($logCondition['shared_secret'])) { $request = \OC::$server->getRequest(); // if token is found in the request change set the log condition to satisfied if ($request && \hash_equals($logCondition['shared_secret'], $request->getParam('log_secret', ''))) { $this->logConditionSatisfied = true; if (!empty($logCondition['logfile'])) { $logConditionFile = $logCondition['logfile']; } break; } } // check for user if (!empty($logCondition['users'])) { $userSession = \OC::$server->getUserSession(); if ($userSession instanceof IUserSession) { $user = $userSession->getUser(); // if the user matches set the log condition to satisfied if ($user !== null && \in_array($user->getUID(), $logCondition['users'], true)) { $this->logConditionSatisfied = true; if (!empty($logCondition['logfile'])) { $logConditionFile = $logCondition['logfile']; } break; } } } } } } // if log condition is satisfied change the required log level to DEBUG if ($this->logConditionSatisfied) { $minLevel = Util::DEBUG; } $skipEvents = false; // avoid infinite loop in case an event handler logs something if ($this->inEvent) { $skipEvents = true; } $eventArgs = [ 'app' => $app, 'loglevel' => $level, 'message' => $message, 'formattedMessage' => $formattedMessage, 'context' => $context, 'extraFields' => $extraFields, 'exception' => $exception ]; // note: regardless of log level we let listeners receive messages if (!$skipEvents) { $this->inEvent = true; $event = new GenericEvent(null); $event->setArguments($eventArgs); $this->eventDispatcher->dispatch($event, 'log.beforewrite'); } if ($level >= $minLevel) { $logger = $this->logger; // check if logger supports extra fields if (!empty($extraFields) && \is_callable([$logger, 'writeExtra'])) { \call_user_func([$logger, 'writeExtra'], $app, $formattedMessage, $level, $logConditionFile, $extraFields); } else { \call_user_func([$logger, 'write'], $app, $formattedMessage, $level, $logConditionFile); } } if (!$skipEvents) { $event = new GenericEvent(null); $event->setArguments($eventArgs); try { $this->eventDispatcher->dispatch($event, 'log.afterwrite'); } finally { $this->inEvent = false; } } } /** * Logs an exception very detailed * * @param \Exception | \Throwable $exception * @param array $context * @return void * @since 8.2.0 */ public function logException($exception, array $context = []) { $context['exception'] = $exception; $level = Util::ERROR; if (isset($context['level'])) { $level = $context['level']; } $exception = [ 'Exception' => \get_class($exception), 'Message' => $exception->getMessage(), 'Code' => $exception->getCode(), 'Trace' => $exception->getTraceAsString(), 'File' => $exception->getFile(), 'Line' => $exception->getLine(), ]; $exception['Trace'] = self::replaceSensitiveData($exception['Trace']); if (\OC::$server->getUserSession() && \OC::$server->getUserSession()->isLoggedIn()) { $context['userid'] = \OC::$server->getUserSession()->getUser()->getUID(); } $msg = isset($context['message']) ? $context['message'] : 'Exception'; $msg .= ': ' . \json_encode($exception); $this->log($level, $msg, $context); // also log previous exception if ($context['exception']->getPrevious()) { $context['message'] = 'Caused by'; $this->logException($context['exception']->getPrevious(), $context); } } /** * @param $trace * @return array|string|string[]|null */ public static function replaceSensitiveData($trace) { return \preg_replace('!(' . \implode('|', self::$methodsWithSensitiveParameters) . ')\(.*\)!', '$1(*** sensitive parameters replaced ***)', $trace); } }