site-accueil-insa/matomo/plugins/CoreUpdater/Controller.php

454 lines
15 KiB
PHP

<?php
/**
* Matomo - free/libre analytics platform
*
* @link https://matomo.org
* @license http://www.gnu.org/licenses/gpl-3.0.html GPL v3 or later
*
*/
namespace Piwik\Plugins\CoreUpdater;
use Exception;
use Piwik\AssetManager;
use Piwik\Common;
use Piwik\Config;
use Piwik\DataTable\Renderer\Json;
use Piwik\DbHelper;
use Piwik\Development;
use Piwik\Filechecks;
use Piwik\FileIntegrity;
use Piwik\Filesystem;
use Piwik\Nonce;
use Piwik\Option;
use Piwik\Piwik;
use Piwik\Plugin\Manager as PluginManager;
use Piwik\Plugins\CoreVue\CoreVue;
use Piwik\Plugins\Marketplace\Plugins;
use Piwik\SettingsPiwik;
use Piwik\SettingsServer;
use Piwik\Updater as DbUpdater;
use Piwik\Updater\Migration\Db as DbMigration;
use Piwik\Version;
use Piwik\View;
use Piwik\View\OneClickDone;
class Controller extends \Piwik\Plugin\Controller
{
private $coreError = false;
private $warningMessages = array();
private $errorMessages = array();
private $deactivatedPlugins = array();
/**
* @var Updater
*/
private $updater;
/**
* @var Plugins
*/
private $marketplacePlugins;
public function __construct(Updater $updater, Plugins $marketplacePlugins = null)
{
$this->updater = $updater;
$this->marketplacePlugins = $marketplacePlugins;
parent::__construct();
}
/**
* Return the base.less compiled to css
*
* @return string
*/
public function getUpdaterCss()
{
Common::sendHeader('Content-Type: text/css');
Common::sendHeader('Cache-Control: max-age=' . (60 * 60));
$files = array(
'plugins/Morpheus/stylesheets/base/bootstrap.css',
'plugins/Morpheus/stylesheets/base/icons.css',
"node_modules/jquery-ui-dist/jquery-ui.theme.min.css",
'node_modules/materialize-css/dist/css/materialize.min.css',
'plugins/Morpheus/stylesheets/base.less',
'plugins/Morpheus/stylesheets/general/_forms.less',
'plugins/Morpheus/stylesheets/simple_structure.css',
'plugins/CoreHome/stylesheets/jquery.ui.autocomplete.css',
'plugins/CoreUpdater/stylesheets/updateLayout.css'
);
return AssetManager::compileCustomStylesheets($files);
}
/**
* Return the base.less compiled to css
*
* @return string
*/
public function getUpdaterJs()
{
Common::sendHeader('Content-Type: application/javascript; charset=UTF-8');
Common::sendHeader('Cache-Control: max-age=' . (60 * 60));
$files = array(
"node_modules/jquery/dist/jquery.min.js",
"node_modules/jquery-ui-dist/jquery-ui.min.js",
'node_modules/materialize-css/dist/js/materialize.min.js',
"plugins/CoreHome/javascripts/materialize-bc.js",
'plugins/Morpheus/javascripts/piwikHelper.js',
"plugins/CoreHome/javascripts/broadcast.js",
'plugins/CoreUpdater/javascripts/updateLayout.js',
'node_modules/angular/angular.min.js',
'node_modules/angular-sanitize/angular-sanitize.min.js',
'node_modules/angular-animate/angular-animate.min.js',
'node_modules/angular-cookies/angular-cookies.min.js',
'node_modules/ng-dialog/js/ngDialog.min.js',
'plugins/CoreHome/angularjs/common/services/service.module.js',
'plugins/CoreHome/angularjs/common/filters/filter.module.js',
'plugins/CoreHome/angularjs/common/filters/translate.js',
'plugins/CoreHome/angularjs/common/directives/directive.module.js',
'plugins/CoreHome/angularjs/piwikApp.config.js',
'plugins/CoreHome/angularjs/piwikApp.js',
'plugins/Installation/javascripts/installation.js',
);
CoreVue::addJsFilesTo($files);
$coreHomeUmd = Development::isEnabled() ? 'CoreHome.umd.js' : 'CoreHome.umd.min.js';
$files[] = "plugins/CoreHome/vue/dist/$coreHomeUmd";
return AssetManager::compileCustomJs($files);
}
public function newVersionAvailable()
{
Piwik::checkUserHasSuperUserAccess();
if (!SettingsPiwik::isAutoUpdateEnabled()) {
throw new Exception('Auto updater is disabled');
}
$this->checkNewVersionIsAvailableOrDie();
$newVersion = $this->updater->getLatestVersion();
$view = new View('@CoreUpdater/newVersionAvailable');
$this->addCustomLogoInfo($view);
$this->setBasicVariablesView($view);
$view->piwik_version = Version::VERSION;
$view->piwik_new_version = $newVersion;
$incompatiblePlugins = $this->getIncompatiblePlugins($newVersion);
$marketplacePlugins = array();
try {
if (!empty($incompatiblePlugins) && $this->marketplacePlugins) {
$marketplacePlugins = $this->marketplacePlugins->getAllAvailablePluginNames();
}
} catch (\Exception $e) {}
$view->marketplacePlugins = $marketplacePlugins;
$view->incompatiblePlugins = $incompatiblePlugins;
$view->piwik_latest_version_url = $this->updater->getArchiveUrl($newVersion);
$view->can_auto_update = Filechecks::canAutoUpdate();
$view->makeWritableCommands = Filechecks::getAutoUpdateMakeWritableMessage();
$view->nonce = Nonce::getNonce('oneClickUpdate');
return $view->render();
}
public function oneClickUpdate()
{
Piwik::checkUserHasSuperUserAccess();
if (!SettingsPiwik::isAutoUpdateEnabled()) {
throw new Exception('Auto updater is disabled');
}
Nonce::checkNonce('oneClickUpdate');
$view = new OneClickDone(Piwik::getCurrentUserTokenAuth());
$useHttps = Common::getRequestVar('https', 1, 'int');
try {
$messages = $this->updater->updatePiwik($useHttps);
} catch (ArchiveDownloadException $e) {
$view->httpsFail = $useHttps;
$view->error = $e->getMessage();
$messages = $e->getUpdateLogMessages();
} catch (UpdaterException $e) {
$view->error = $e->getMessage();
$messages = $e->getUpdateLogMessages();
}
$view->feedbackMessages = $messages;
$this->addCustomLogoInfo($view);
$result = $view->render();
Filesystem::deleteAllCacheOnUpdate();
return $result;
}
public function oneClickUpdatePartTwo()
{
if (!SettingsPiwik::isAutoUpdateEnabled()) {
throw new Exception('Auto updater is disabled');
}
Json::sendHeaderJSON();
$task = "Couldn't update Marketplace plugins.";
$nonce = Common::getRequestVar('nonce', '', 'string');
if (empty($nonce)) {
return json_encode(['No token. ' . $task]);
}
$value = Option::get('NonceOneClickUpdatePartTwo');
if (empty($value)) {
return json_encode(['Invalid token. ' . $task]);
}
$value = json_decode($value, true);
if (empty($value['nonce'])
|| empty($value['ttl'])
|| time() > (int) $value['ttl']
|| $nonce !== $value['nonce']) {
return json_encode(['Invalid nonce or nonce expired. ' . $task]);
}
try {
$messages = $this->updater->oneClickUpdatePartTwo();
} catch (UpdaterException $e) {
$messages = $e->getUpdateLogMessages();
$messages[] = $e->getMessage();
} catch (Exception $e) {
$messages = [$e->getMessage()];
}
return json_encode($messages);
}
public function oneClickResults()
{
if (!SettingsPiwik::isAutoUpdateEnabled()) {
throw new Exception('Auto updater is disabled');
}
$httpsFail = (bool) Common::getRequestVar('httpsFail', 0, 'int', $_POST);
$error = Common::getRequestVar('error', '', 'string', $_POST);
if ($httpsFail) {
$view = new View('@CoreUpdater/updateHttpsError');
$view->nonce = Nonce::getNonce('oneClickUpdate');
$view->error = $error;
} elseif ($error) {
$view = new View('@CoreUpdater/updateHttpError');
$view->error = $error;
} else {
$view = new View('@CoreUpdater/updateSuccess');
}
$messages = safe_unserialize(Common::unsanitizeInputValue(Common::getRequestVar('messages', '', 'string', $_POST)));
if (!is_array($messages)) {
$messages = array();
}
$view->feedbackMessages = $messages;
$this->addCustomLogoInfo($view);
$this->setBasicVariablesView($view);
return $view->render();
}
protected function redirectToDashboardWhenNoError(DbUpdater $updater)
{
if (count($updater->getSqlQueriesToExecute()) == 1
&& !$this->coreError
&& empty($this->warningMessages)
&& empty($this->errorMessages)
&& empty($this->deactivatedPlugins)
) {
Piwik::redirectToModule('CoreHome');
}
}
private function checkNewVersionIsAvailableOrDie()
{
if (!$this->updater->isNewVersionAvailable()) {
throw new Exception(Piwik::translate('CoreUpdater_ExceptionAlreadyLatestVersion', Version::VERSION));
}
}
public function index()
{
try {
return $this->runUpdaterAndExit();
} catch(NoUpdatesFoundException $e) {
Piwik::redirectToModule('CoreHome');
}
}
public function runUpdaterAndExit($doDryRun = null)
{
if (!SettingsPiwik::isAutoUpdateEnabled()) {
throw new Exception('Auto updater is disabled');
}
$updater = new DbUpdater();
$componentsWithUpdateFile = $updater->getComponentUpdates();
if (empty($componentsWithUpdateFile)) {
throw new NoUpdatesFoundException("Everything is already up to date.");
}
SettingsServer::setMaxExecutionTime(0);
$welcomeTemplate = '@CoreUpdater/runUpdaterAndExit_welcome';
$doneTemplate = '@CoreUpdater/runUpdaterAndExit_done';
$viewWelcome = new View($welcomeTemplate);
$this->addCustomLogoInfo($viewWelcome);
$this->setBasicVariablesView($viewWelcome);
$viewDone = new View($doneTemplate);
$this->addCustomLogoInfo($viewDone);
$this->setBasicVariablesView($viewDone);
$doExecuteUpdates = Common::getRequestVar('updateCorePlugins', 0, 'integer') == 1;
if (is_null($doDryRun)) {
$doDryRun = !$doExecuteUpdates;
}
if ($doDryRun) {
$migrations = $updater->getSqlQueriesToExecute();
$queryCount = count($migrations);
$migrations = $this->groupMigrations($migrations);
$viewWelcome->migrations = $migrations;
$viewWelcome->queryCount = $queryCount;
$viewWelcome->isMajor = $updater->hasMajorDbUpdate();
$this->doWelcomeUpdates($viewWelcome, $componentsWithUpdateFile);
return $viewWelcome->render();
}
// Web
if ($doExecuteUpdates) {
$this->warningMessages = array();
$this->doExecuteUpdates($viewDone, $updater, $componentsWithUpdateFile);
$this->redirectToDashboardWhenNoError($updater);
return $viewDone->render();
}
exit;
}
private function groupMigrations($migrations)
{
$result = [];
$group = null;
foreach ($migrations as $migration) {
$type = $migration instanceof DbMigration ? 'sql' : 'command';
if ($group === null
|| $type != $group['type']
) {
$group = [
'type' => $type,
'migrations' => [],
];
$result[] = $group;
}
$result[count($result) - 1]['migrations'][] = $migration;
}
return $result;
}
private function doWelcomeUpdates($view, $componentsWithUpdateFile)
{
$view->new_piwik_version = Version::VERSION;
$view->commandUpgradePiwik = "php " . Filesystem::getPathToPiwikRoot() . "/console core:update";
$instanceId = SettingsPiwik::getPiwikInstanceId();
if ($instanceId) {
$view->commandUpgradePiwik .= ' --matomo-domain="' . $instanceId . '"';
}
$pluginNamesToUpdate = array();
$dimensionsToUpdate = array();
$coreToUpdate = false;
// handle case of existing database with no tables
if (!DbHelper::isInstalled()) {
$this->errorMessages[] = Piwik::translate('CoreUpdater_EmptyDatabaseError', Config::getInstance()->database['dbname']);
$this->coreError = true;
$currentVersion = 'N/A';
} else {
$this->errorMessages = array();
try {
$currentVersion = Option::get('version_core');
} catch (Exception $e) {
$currentVersion = '<= 0.2.9';
}
foreach ($componentsWithUpdateFile as $name => $filenames) {
if ($name == 'core') {
$coreToUpdate = true;
} elseif (0 === strpos($name, 'log_')) {
$dimensionsToUpdate[] = $name;
} else {
$pluginNamesToUpdate[] = $name;
}
}
}
// check file integrity
[$success, $messages] = FileIntegrity::getFileIntegrityInformation();
if (!$success) {
$this->warningMessages[] = Piwik::translate('General_FileIntegrityWarning');
}
if (count($messages) > 0) {
$this->warningMessages = array_merge($this->warningMessages, $messages);
}
Filesystem::deleteAllCacheOnUpdate();
sort($dimensionsToUpdate);
$view->coreError = $this->coreError;
$view->warningMessages = $this->warningMessages;
$view->errorMessages = $this->errorMessages;
$view->current_piwik_version = $currentVersion;
$view->pluginNamesToUpdate = $pluginNamesToUpdate;
$view->dimensionsToUpdate = $dimensionsToUpdate;
$view->coreToUpdate = $coreToUpdate;
}
private function doExecuteUpdates($view, DbUpdater $updater, $componentsWithUpdateFile)
{
$result = $updater->updateComponents($componentsWithUpdateFile);
$this->coreError = $result['coreError'];
$this->warningMessages = $result['warnings'];
$this->errorMessages = $result['errors'];
$this->deactivatedPlugins = $result['deactivatedPlugins'];
$view->coreError = $this->coreError;
$view->warningMessages = $this->warningMessages;
$view->errorMessages = $this->errorMessages;
$view->deactivatedPlugins = $this->deactivatedPlugins;
}
private function getIncompatiblePlugins($piwikVersion)
{
return PluginManager::getInstance()->getIncompatiblePlugins($piwikVersion);
}
}