Files
sysPass/lib/SP/Services/Upgrade/UpgradeDatabaseService.php
Rubén D e4e6e04c57 * [MOD] Updated copyright
Signed-off-by: Rubén D <nuxsmin@syspass.org>
2020-12-07 19:42:41 +01:00

229 lines
6.5 KiB
PHP

<?php
/*
* sysPass
*
* @author nuxsmin
* @link https://syspass.org
* @copyright 2012-2020, Rubén Domínguez nuxsmin@$syspass.org
*
* This file is part of sysPass.
*
* sysPass is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* sysPass 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with sysPass. If not, see <http://www.gnu.org/licenses/>.
*/
namespace SP\Services\Upgrade;
use Exception;
use SP\Config\ConfigData;
use SP\Core\Events\Event;
use SP\Core\Events\EventMessage;
use SP\Providers\Log\FileLogHandler;
use SP\Services\Service;
use SP\Storage\Database\Database;
use SP\Storage\Database\MySQLFileParser;
use SP\Storage\File\FileException;
use SP\Storage\File\FileHandler;
use SP\Util\VersionUtil;
/**
* Class UpgradeDatabaseService
*
* @package SP\Services\Upgrade
*/
final class UpgradeDatabaseService extends Service implements UpgradeInterface
{
/**
* @var array Versiones actualizables
*/
const UPGRADES = [
'300.18010101',
'300.18072302',
'300.18072501',
'300.18083001',
'300.18083002',
'300.18091101',
'300.18092401',
'300.18093001',
'300.18111801',
'300.18111901',
'310.19012201',
'310.19042701'
];
/**
* @var Database
*/
protected $db;
/**
* Check if it needs to be upgraded
*
* @param $version
*
* @return bool
*/
public static function needsUpgrade($version)
{
return empty($version) || VersionUtil::checkVersion($version, self::UPGRADES);
}
/**
* Inicia el proceso de actualización de la BBDD.
*
* @param int $version con la versión de la BBDD actual
* @param ConfigData $configData
*
* @return bool
* @throws FileException
* @throws UpgradeException
*/
public function upgrade($version, ConfigData $configData)
{
$this->eventDispatcher->notifyEvent('upgrade.db.start',
new Event($this, EventMessage::factory()
->addDescription(__u('Update DB')))
);
foreach (self::UPGRADES as $upgradeVersion) {
if (VersionUtil::checkVersion($version, $upgradeVersion)) {
if ($this->applyPreUpgrade($upgradeVersion) === false) {
throw new UpgradeException(
__u('Error while applying an auxiliary update'),
UpgradeException::CRITICAL,
__u('Please, check the event log for more details')
);
}
if ($this->applyUpgrade($upgradeVersion) === false) {
throw new UpgradeException(
__u('Error while updating the database'),
UpgradeException::CRITICAL,
__u('Please, check the event log for more details')
);
}
logger('DB Upgrade: ' . $upgradeVersion);
$configData->setDatabaseVersion($upgradeVersion);
$this->config->saveConfig($configData, false);
}
}
$this->eventDispatcher->notifyEvent('upgrade.db.end',
new Event($this, EventMessage::factory()
->addDescription(__u('Update DB')))
);
return true;
}
/**
* Aplicar actualizaciones auxiliares antes de actualizar la BBDD
*
* @param $version
*
* @return bool
*/
private function applyPreUpgrade($version)
{
return true;
}
/**
* Actualiza la BBDD según la versión.
*
* @param int $version con la versión a actualizar
*
* @returns bool
* @return bool
* @throws UpgradeException
*/
private function applyUpgrade($version)
{
$queries = $this->getQueriesFromFile($version);
if (count($queries) === 0) {
logger(__('Update file does not contain data'), 'ERROR');
throw new UpgradeException(__u('Update file does not contain data'), UpgradeException::ERROR, $version);
}
foreach ($queries as $query) {
try {
$this->eventDispatcher->notifyEvent('upgrade.db.process',
new Event($this, EventMessage::factory()
->addDetail(__u('Version'), $version))
);
// Direct PDO handling
$this->db->getDbHandler()
->getConnection()
->exec($query);
} catch (Exception $e) {
processException($e);
logger('SQL: ' . $query);
$this->eventDispatcher->notifyEvent('exception',
new Event($this, EventMessage::factory()
->addDescription(__u('Error while updating the database'))
->addDetail('ERROR', sprintf('%s (%s)', $e->getMessage(), $e->getCode())))
);
throw new UpgradeException(__u('Error while updating the database'));
}
}
$this->eventDispatcher->notifyEvent('upgrade.db.process',
new Event($this, EventMessage::factory()
->addDescription(__u('Database updating was completed successfully.')))
);
return true;
}
/**
* Obtener las consultas de actualización desde un archivo
*
* @param $filename
*
* @return array|bool
* @throws UpgradeException
*/
private function getQueriesFromFile($filename)
{
$fileName = SQL_PATH . DIRECTORY_SEPARATOR . str_replace('.', '', $filename) . '.sql';
try {
$parser = new MySQLFileParser(new FileHandler($fileName));
return $parser->parse('$$');
} catch (FileException $e) {
processException($e);
throw new UpgradeException($e->getMessage());
}
}
/**
* initialize
*/
protected function initialize()
{
$this->eventDispatcher->attach($this->dic->get(FileLogHandler::class));
$this->db = $this->dic->get(Database::class);
}
}