Skip to content
Extraits de code Groupes Projets
Valider cd8c762a rédigé par Matthieu Aubry's avatar Matthieu Aubry Validation de GitHub
Parcourir les fichiers

When safe mode is displayed, and Super User was not logged in already, let...

When safe mode is displayed, and Super User was not logged in already, let Super User display the full safe mode to troubleshoot further (#11082)

* When safe mode is displayed, and Super User was not logged in already, let Super User display the full safe mode to troubleshoot further

* meant for a different PR

* Catch exceptions during CSS/Less compiling

* when there's an error during Twig template processing, or during LESS css compiling, display the safe mode

* Deactivate plugin as super user when authenticated with i_am_super_user

* when user is not logged as Super User, Make deactivate plugin work by forwarding the i_am_super_user URL parameter

* Display plugin version in safe mode fixes https://github.com/piwik/piwik/issues/11043

* Added a text to indicate users to first disable third party plugins

* Updated UI test

* clarify that salt is a secret

* reuse helper method

* minor

* Catching PHP7 errors and making it work on php5
parent 3ce6d87a
Aucune branche associée trouvée
Aucune étiquette associée trouvée
Aucune requête de fusion associée trouvée
...@@ -13,6 +13,7 @@ use lessc; ...@@ -13,6 +13,7 @@ use lessc;
use Piwik\AssetManager\UIAsset; use Piwik\AssetManager\UIAsset;
use Piwik\AssetManager\UIAssetMerger; use Piwik\AssetManager\UIAssetMerger;
use Piwik\Common; use Piwik\Common;
use Piwik\Exception\StylesheetLessCompileException;
use Piwik\Piwik; use Piwik\Piwik;
class StylesheetUIAssetMerger extends UIAssetMerger class StylesheetUIAssetMerger extends UIAssetMerger
...@@ -41,7 +42,11 @@ class StylesheetUIAssetMerger extends UIAssetMerger ...@@ -41,7 +42,11 @@ class StylesheetUIAssetMerger extends UIAssetMerger
$concatenatedAssets = $this->getConcatenatedAssets(); $concatenatedAssets = $this->getConcatenatedAssets();
$this->lessCompiler->setFormatter('classic'); $this->lessCompiler->setFormatter('classic');
$compiled = $this->lessCompiler->compile($concatenatedAssets); try {
$compiled = $this->lessCompiler->compile($concatenatedAssets);
} catch(\Exception $e) {
throw new StylesheetLessCompileException($e->getMessage());
}
foreach ($this->cssAssetsToReplace as $asset) { foreach ($this->cssAssetsToReplace as $asset) {
// to fix #10173 // to fix #10173
......
<?php
/**
* Piwik - free/libre analytics platform
*
* @link http://piwik.org
* @license http://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
*
*/
namespace Piwik\Exception;
class StylesheetLessCompileException extends Exception
{
}
...@@ -15,6 +15,7 @@ use Piwik\Container\StaticContainer; ...@@ -15,6 +15,7 @@ use Piwik\Container\StaticContainer;
use Piwik\Exception\AuthenticationFailedException; use Piwik\Exception\AuthenticationFailedException;
use Piwik\Exception\DatabaseSchemaIsNewerThanCodebaseException; use Piwik\Exception\DatabaseSchemaIsNewerThanCodebaseException;
use Piwik\Exception\PluginDeactivatedException; use Piwik\Exception\PluginDeactivatedException;
use Piwik\Exception\StylesheetLessCompileException;
use Piwik\Http\ControllerResolver; use Piwik\Http\ControllerResolver;
use Piwik\Http\Router; use Piwik\Http\Router;
use Piwik\Plugins\CoreAdminHome\CustomLogo; use Piwik\Plugins\CoreAdminHome\CustomLogo;
...@@ -73,11 +74,11 @@ class FrontController extends Singleton ...@@ -73,11 +74,11 @@ class FrontController extends Singleton
/** /**
* @param $lastError * @param $lastError
* @return mixed|void * @return string
* @throws AuthenticationFailedException * @throws AuthenticationFailedException
* @throws Exception * @throws Exception
*/ */
private static function generateSafeModeOutput($lastError) private static function generateSafeModeOutputFromError($lastError)
{ {
Common::sendResponseCode(500); Common::sendResponseCode(500);
...@@ -93,6 +94,20 @@ class FrontController extends Singleton ...@@ -93,6 +94,20 @@ class FrontController extends Singleton
return $message; return $message;
} }
/**
* @param Exception $e
* @return string
*/
private static function generateSafeModeOutputFromException($e)
{
$error = array(
'message' => $e->getMessage(),
'file' => $e->getFile(),
'line' => $e->getLine()
);
return self::generateSafeModeOutputFromError($error);
}
/** /**
* Executes the requested plugin controller method. * Executes the requested plugin controller method.
* *
...@@ -133,6 +148,15 @@ class FrontController extends Singleton ...@@ -133,6 +148,15 @@ class FrontController extends Singleton
* @param \Piwik\NoAccessException $exception The exception that was caught. * @param \Piwik\NoAccessException $exception The exception that was caught.
*/ */
Piwik::postEvent('User.isNotAuthorized', array($exception), $pending = true); Piwik::postEvent('User.isNotAuthorized', array($exception), $pending = true);
} catch (\Twig_Error_Runtime $e) {
echo $this->generateSafeModeOutputFromException($e);
exit;
} catch(StylesheetLessCompileException $e) {
echo $this->generateSafeModeOutputFromException($e);
exit;
} catch(\Error $e) {
echo $this->generateSafeModeOutputFromException($e);
exit;
} }
} }
...@@ -202,7 +226,7 @@ class FrontController extends Singleton ...@@ -202,7 +226,7 @@ class FrontController extends Singleton
{ {
$lastError = error_get_last(); $lastError = error_get_last();
if (!empty($lastError) && $lastError['type'] == E_ERROR) { if (!empty($lastError) && $lastError['type'] == E_ERROR) {
$message = self::generateSafeModeOutput($lastError); $message = self::generateSafeModeOutputFromError($lastError);
echo $message; echo $message;
} }
} }
......
...@@ -20,7 +20,9 @@ class SettingsPiwik ...@@ -20,7 +20,9 @@ class SettingsPiwik
const OPTION_PIWIK_URL = 'piwikUrl'; const OPTION_PIWIK_URL = 'piwikUrl';
/** /**
* Get salt from [General] section * Get salt from [General] section. Should ONLY be used as a seed to create hashes
*
* NOTE: Keep this salt secret! Never output anywhere or share it etc.
* *
* @return string * @return string
*/ */
......
...@@ -702,3 +702,12 @@ if (!function_exists('dump')) { ...@@ -702,3 +702,12 @@ if (!function_exists('dump')) {
} }
} }
/**
* Need to catch that PHP7 error object on php5
*/
if( !class_exists('\Error')) {
class Error {
}
}
\ No newline at end of file
...@@ -11,6 +11,7 @@ namespace Piwik\Plugins\CorePluginsAdmin; ...@@ -11,6 +11,7 @@ namespace Piwik\Plugins\CorePluginsAdmin;
use Exception; use Exception;
use Piwik\API\Request; use Piwik\API\Request;
use Piwik\Common; use Piwik\Common;
use Piwik\Config;
use Piwik\Container\StaticContainer; use Piwik\Container\StaticContainer;
use Piwik\Exception\MissingFilePermissionException; use Piwik\Exception\MissingFilePermissionException;
use Piwik\Filechecks; use Piwik\Filechecks;
...@@ -22,6 +23,7 @@ use Piwik\Plugin; ...@@ -22,6 +23,7 @@ use Piwik\Plugin;
use Piwik\Plugins\Marketplace\Marketplace; use Piwik\Plugins\Marketplace\Marketplace;
use Piwik\Plugins\Marketplace\Controller as MarketplaceController; use Piwik\Plugins\Marketplace\Controller as MarketplaceController;
use Piwik\Plugins\Marketplace\Plugins; use Piwik\Plugins\Marketplace\Plugins;
use Piwik\SettingsPiwik;
use Piwik\Translation\Translator; use Piwik\Translation\Translator;
use Piwik\Url; use Piwik\Url;
use Piwik\Version; use Piwik\Version;
...@@ -296,22 +298,27 @@ class Controller extends Plugin\ControllerAdmin ...@@ -296,22 +298,27 @@ class Controller extends Plugin\ControllerAdmin
return $message; return $message;
} }
if (Common::isPhpCliMode()) { // TODO: I can't find how this will ever get called / safeMode is never set for Console if (Common::isPhpCliMode()) {
throw new Exception("Error: " . var_export($lastError, true)); throw new Exception("Error: " . var_export($lastError, true));
} }
$view = new View('@CorePluginsAdmin/safemode'); $view = new View('@CorePluginsAdmin/safemode');
$view->lastError = $lastError; $view->lastError = $lastError;
$view->isAllowedToTroubleshootAsSuperUser = $this->isAllowedToTroubleshootAsSuperUser();
$view->isSuperUser = Piwik::hasUserSuperUserAccess(); $view->isSuperUser = Piwik::hasUserSuperUserAccess();
$view->isAnonymousUser = Piwik::isUserIsAnonymous(); $view->isAnonymousUser = Piwik::isUserIsAnonymous();
$view->plugins = $this->pluginManager->loadAllPluginsAndGetTheirInfo(); $view->plugins = $this->pluginManager->loadAllPluginsAndGetTheirInfo();
$view->deactivateNonce = Nonce::getNonce(static::DEACTIVATE_NONCE); $view->deactivateNonce = Nonce::getNonce(static::DEACTIVATE_NONCE);
$view->deactivateIAmSuperUserSalt = Common::getRequestVar('i_am_super_user', '', 'string');
$view->uninstallNonce = Nonce::getNonce(static::UNINSTALL_NONCE); $view->uninstallNonce = Nonce::getNonce(static::UNINSTALL_NONCE);
$view->emailSuperUser = implode(',', Piwik::getAllSuperUserAccessEmailAddresses()); $view->emailSuperUser = implode(',', Piwik::getAllSuperUserAccessEmailAddresses());
$view->piwikVersion = Version::VERSION; $view->piwikVersion = Version::VERSION;
$view->showVersion = !Common::getRequestVar('tests_hide_piwik_version', 0); $view->showVersion = !Common::getRequestVar('tests_hide_piwik_version', 0);
$view->pluginCausesIssue = ''; $view->pluginCausesIssue = '';
// When the CSS merger in StylesheetUIAssetMerger throws an exception, safe mode is displayed.
// This flag prevents an infinite loop where safemode would try to re-generate the cache buster which requires CSS merger..
$view->disableCacheBuster();
if (!empty($lastError['file'])) { if (!empty($lastError['file'])) {
preg_match('/piwik\/plugins\/(.*)\//', $lastError['file'], $matches); preg_match('/piwik\/plugins\/(.*)\//', $lastError['file'], $matches);
...@@ -367,11 +374,13 @@ class Controller extends Plugin\ControllerAdmin ...@@ -367,11 +374,13 @@ class Controller extends Plugin\ControllerAdmin
public function deactivate($redirectAfter = true) public function deactivate($redirectAfter = true)
{ {
$pluginName = $this->initPluginModification(static::DEACTIVATE_NONCE); if($this->isAllowedToTroubleshootAsSuperUser()) {
$this->dieIfPluginsAdminIsDisabled(); Piwik::doAsSuperUser(function() use ($redirectAfter) {
$this->doDeactivatePlugin($redirectAfter);
$this->pluginManager->deactivatePlugin($pluginName); });
$this->redirectAfterModification($redirectAfter); } else {
$this->doDeactivatePlugin($redirectAfter);
}
} }
public function uninstall($redirectAfter = true) public function uninstall($redirectAfter = true)
...@@ -455,4 +464,34 @@ class Controller extends Plugin\ControllerAdmin ...@@ -455,4 +464,34 @@ class Controller extends Plugin\ControllerAdmin
} catch (Exception $e) {} } catch (Exception $e) {}
} }
/**
* Let Super User troubleshoot in safe mode, even when Login is broken, with this special trick
*
* @return bool
* @throws Exception
*/
protected function isAllowedToTroubleshootAsSuperUser()
{
$isAllowedToTroubleshootAsSuperUser = false;
$salt = SettingsPiwik::getSalt();
if (!empty($salt)) {
$saltFromRequest = Common::getRequestVar('i_am_super_user', '', 'string');
$isAllowedToTroubleshootAsSuperUser = ($salt == $saltFromRequest);
}
return $isAllowedToTroubleshootAsSuperUser;
}
/**
* @param $redirectAfter
* @throws Exception
*/
protected function doDeactivatePlugin($redirectAfter)
{
$pluginName = $this->initPluginModification(static::DEACTIVATE_NONCE);
$this->dieIfPluginsAdminIsDisabled();
$this->pluginManager->deactivatePlugin($pluginName);
$this->redirectAfterModification($redirectAfter);
}
} }
...@@ -28,7 +28,7 @@ ...@@ -28,7 +28,7 @@
<div style="width: 640px"> <div style="width: 640px">
{% if not isAnonymousUser %} {% if isAllowedToTroubleshootAsSuperUser or not isAnonymousUser %}
<p> <p>
The following error just broke Piwik{% if showVersion %} (v{{ piwikVersion }}){% endif %}: The following error just broke Piwik{% if showVersion %} (v{{ piwikVersion }}){% endif %}:
<pre>{{ lastError.message }}</pre> <pre>{{ lastError.message }}</pre>
...@@ -58,12 +58,13 @@ ...@@ -58,12 +58,13 @@
{% endif %} {% endif %}
{% if isSuperUser %} {% if isAllowedToTroubleshootAsSuperUser or isSuperUser %}
<h3>Further troubleshooting</h3> <h3>Further troubleshooting</h3>
<p> <p>
If this error continues to happen, you may be able to fix this issue by disabling one or more of If this error continues to happen, you may be able to fix this issue by disabling one or more of
the Third-Party plugins. You can enable them again in the the Third-Party plugins. If you don't know which plugin is causing this error, we recommend to first disable any plugin not created by "Piwik" and not created by "InnoCraft".
You can enable plugin again afterwards in the
<a rel="noreferrer" target="_blank" href="index.php?module=CorePluginsAdmin&action=plugins">Plugins</a> <a rel="noreferrer" target="_blank" href="index.php?module=CorePluginsAdmin&action=plugins">Plugins</a>
or <a target="_blank" href="index.php?module=CorePluginsAdmin&action=themes">Themes</a> page under or <a target="_blank" href="index.php?module=CorePluginsAdmin&action=themes">Themes</a> page under
settings at any time. settings at any time.
...@@ -79,7 +80,10 @@ ...@@ -79,7 +80,10 @@
{{ pluginName }} {{ pluginName }}
</td> </td>
<td> <td>
<a href="index.php?module=CorePluginsAdmin&action=deactivate&pluginName={{ pluginName }}&nonce={{ deactivateNonce }}" {{ plugin.info.version|default('') }}
</td>
<td>
<a href="index.php?module=CorePluginsAdmin&action=deactivate&pluginName={{ pluginName }}&nonce={{ deactivateNonce }}{% if deactivateIAmSuperUserSalt is not empty %}&i_am_super_user={{ deactivateIAmSuperUserSalt }}{% endif %}"
target="_blank">deactivate</a> target="_blank">deactivate</a>
</td> </td>
</tr> </tr>
...@@ -115,7 +119,7 @@ ...@@ -115,7 +119,7 @@
{% elseif isAnonymousUser %} {% elseif isAnonymousUser %}
<p>Please contact the system administrator, or login to Piwik to learn more.</p> <p>Please contact the system administrator, or <a href="?module={{ loginModule }}">login to Piwik</a> to learn more.</p>
{% else %} {% else %}
<p> <p>
...@@ -125,6 +129,16 @@ ...@@ -125,6 +129,16 @@
</p> </p>
{% endif %} {% endif %}
{% if not isAllowedToTroubleshootAsSuperUser and not isSuperUser %}
<p>If you are Super User, but cannot login because of this error, you can still troubleshoot further. Follow these steps:
<br/>1) open the config/config.ini.php file and look for the <code>salt</code> value under <code>[General]</code>.
<br/>2) edit this current URL you are viewing and add the following text (replacing <code>salt_value_from_config</code> by the <code>salt</code> value from the config file):
<br/><br/><code>index.php?i_am_super_user=salt_value_from_config&....</code>
</p>
{% endif %}
</div> </div>
</body> </body>
......
Le fichier a été supprimé par une entrée .gitattributes, ou son encodage n'est pas pris en charge.
0% Chargement en cours ou .
You are about to add 0 people to the discussion. Proceed with caution.
Terminez d'abord l'édition de ce message.
Veuillez vous inscrire ou vous pour commenter