From efde9a7511e824caffec0da5b2c445b04dffa0bd Mon Sep 17 00:00:00 2001 From: Morris Jobke Date: Thu, 7 Jan 2021 16:26:51 +0100 Subject: [PATCH] Add setup check to verify that the used DB version is still supported in the next major release Signed-off-by: Morris Jobke --- .../composer/composer/autoload_classmap.php | 1 + .../composer/composer/autoload_static.php | 1 + .../lib/Controller/CheckSetupController.php | 9 +- .../lib/SetupChecks/SupportedDatabase.php | 104 ++++++++++++++++++ .../SetupChecks/SupportedDatabaseTest.php | 42 +++++++ core/js/setupchecks.js | 1 + 6 files changed, 157 insertions(+), 1 deletion(-) create mode 100644 apps/settings/lib/SetupChecks/SupportedDatabase.php create mode 100644 apps/settings/tests/SetupChecks/SupportedDatabaseTest.php diff --git a/apps/settings/composer/composer/autoload_classmap.php b/apps/settings/composer/composer/autoload_classmap.php index 916b6f8c55aad..bf15d877fe09e 100644 --- a/apps/settings/composer/composer/autoload_classmap.php +++ b/apps/settings/composer/composer/autoload_classmap.php @@ -61,4 +61,5 @@ 'OCA\\Settings\\SetupChecks\\LegacySSEKeyFormat' => $baseDir . '/../lib/SetupChecks/LegacySSEKeyFormat.php', 'OCA\\Settings\\SetupChecks\\PhpDefaultCharset' => $baseDir . '/../lib/SetupChecks/PhpDefaultCharset.php', 'OCA\\Settings\\SetupChecks\\PhpOutputBuffering' => $baseDir . '/../lib/SetupChecks/PhpOutputBuffering.php', + 'OCA\\Settings\\SetupChecks\\SupportedDatabase' => $baseDir . '/../lib/SetupChecks/SupportedDatabase.php', ); diff --git a/apps/settings/composer/composer/autoload_static.php b/apps/settings/composer/composer/autoload_static.php index 661fc4bd84d59..87635f63586dd 100644 --- a/apps/settings/composer/composer/autoload_static.php +++ b/apps/settings/composer/composer/autoload_static.php @@ -76,6 +76,7 @@ class ComposerStaticInitSettings 'OCA\\Settings\\SetupChecks\\LegacySSEKeyFormat' => __DIR__ . '/..' . '/../lib/SetupChecks/LegacySSEKeyFormat.php', 'OCA\\Settings\\SetupChecks\\PhpDefaultCharset' => __DIR__ . '/..' . '/../lib/SetupChecks/PhpDefaultCharset.php', 'OCA\\Settings\\SetupChecks\\PhpOutputBuffering' => __DIR__ . '/..' . '/../lib/SetupChecks/PhpOutputBuffering.php', + 'OCA\\Settings\\SetupChecks\\SupportedDatabase' => __DIR__ . '/..' . '/../lib/SetupChecks/SupportedDatabase.php', ); public static function getInitializer(ClassLoader $loader) diff --git a/apps/settings/lib/Controller/CheckSetupController.php b/apps/settings/lib/Controller/CheckSetupController.php index 5b004d753e21a..36148398c36ac 100644 --- a/apps/settings/lib/Controller/CheckSetupController.php +++ b/apps/settings/lib/Controller/CheckSetupController.php @@ -62,6 +62,7 @@ use OCA\Settings\SetupChecks\LegacySSEKeyFormat; use OCA\Settings\SetupChecks\PhpDefaultCharset; use OCA\Settings\SetupChecks\PhpOutputBuffering; +use OCA\Settings\SetupChecks\SupportedDatabase; use OCP\AppFramework\Controller; use OCP\AppFramework\Http\DataDisplayResponse; use OCP\AppFramework\Http\DataResponse; @@ -106,6 +107,8 @@ class CheckSetupController extends Controller { private $secureRandom; /** @var IniGetWrapper */ private $iniGetWrapper; + /** @var IDBConnection */ + private $connection; public function __construct($AppName, IRequest $request, @@ -121,7 +124,8 @@ public function __construct($AppName, IDateTimeFormatter $dateTimeFormatter, MemoryInfo $memoryInfo, ISecureRandom $secureRandom, - IniGetWrapper $iniGetWrapper) { + IniGetWrapper $iniGetWrapper, + IDBConnection $connection) { parent::__construct($AppName, $request); $this->config = $config; $this->clientService = $clientService; @@ -136,6 +140,7 @@ public function __construct($AppName, $this->memoryInfo = $memoryInfo; $this->secureRandom = $secureRandom; $this->iniGetWrapper = $iniGetWrapper; + $this->connection = $connection; } /** @@ -713,6 +718,7 @@ public function check() { $phpOutputBuffering = new PhpOutputBuffering(); $legacySSEKeyFormat = new LegacySSEKeyFormat($this->l10n, $this->config, $this->urlGenerator); $checkUserCertificates = new CheckUserCertificates($this->l10n, $this->config, $this->urlGenerator); + $supportedDatabases = new SupportedDatabase($this->l10n, $this->connection); return new DataResponse( [ @@ -759,6 +765,7 @@ public function check() { LegacySSEKeyFormat::class => ['pass' => $legacySSEKeyFormat->run(), 'description' => $legacySSEKeyFormat->description(), 'severity' => $legacySSEKeyFormat->severity(), 'linkToDocumentation' => $legacySSEKeyFormat->linkToDocumentation()], CheckUserCertificates::class => ['pass' => $checkUserCertificates->run(), 'description' => $checkUserCertificates->description(), 'severity' => $checkUserCertificates->severity(), 'elements' => $checkUserCertificates->elements()], 'isDefaultPhoneRegionSet' => $this->config->getSystemValueString('default_phone_region', '') !== '', + SupportedDatabase::class => ['pass' => $supportedDatabases->run(), 'description' => $supportedDatabases->description(), 'severity' => $supportedDatabases->severity()], ] ); } diff --git a/apps/settings/lib/SetupChecks/SupportedDatabase.php b/apps/settings/lib/SetupChecks/SupportedDatabase.php new file mode 100644 index 0000000000000..891ba34424be1 --- /dev/null +++ b/apps/settings/lib/SetupChecks/SupportedDatabase.php @@ -0,0 +1,104 @@ + + * + * @author Morris Jobke + * + * @license GNU AGPL version 3 or any later version + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * 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 + * along with this program. If not, see . + * + */ + +namespace OCA\Settings\SetupChecks; + +use Doctrine\DBAL\Platforms\MariaDb1027Platform; +use Doctrine\DBAL\Platforms\MySQL57Platform; +use Doctrine\DBAL\Platforms\MySQL80Platform; +use Doctrine\DBAL\Platforms\MySQLPlatform; +use Doctrine\DBAL\Platforms\OraclePlatform; +use Doctrine\DBAL\Platforms\PostgreSQL100Platform; +use Doctrine\DBAL\Platforms\PostgreSQL94Platform; +use Doctrine\DBAL\Platforms\SqlitePlatform; +use OCP\IDBConnection; +use OCP\IL10N; + +class SupportedDatabase { + /** @var IL10N */ + private $l10n; + /** @var IDBConnection */ + private $connection; + + private $description = ''; + + public function __construct(IL10N $l10n, IDBConnection $connection) { + $this->l10n = $l10n; + $this->connection = $connection; + + $this->check(); + } + + public function check() { + switch (get_class($this->connection->getDatabasePlatform())) { + case MySQL80Platform::class: # extends MySQL57Platform + case MySQL57Platform::class: # extends MySQLPlatform + case MariaDb1027Platform::class: # extends MySQLPlatform + case MySQLPlatform::class: + $result = $this->connection->prepare('SHOW VARIABLES LIKE "version";'); + $row = $result->execute()->fetchOne(); + $version = strtolower($row['Value']); + + if (strpos($version, 'mariadb') !== false) { + if (version_compare($version, '10.4', '<')) { + $this->description = $this->l10n->t('MariaDB version "%s" is used. Nextcloud 21 will no longer support this version and requires MariaDB 10.4 or higher.', $row['Value']); + return; + } + } else { + if (version_compare($version, '8', '<')) { + $this->description = $this->l10n->t('MySQL version "%s" is used. Nextcloud 21 will no longer support this version and requires MariaDB 8 or higher.', $row['Value']); + return; + } + } + break; + case SqlitePlatform::class: + break; + case PostgreSQL100Platform::class: # extends PostgreSQL94Platform + case PostgreSQL94Platform::class: + $result = $this->connection->prepare('SHOW server_version;'); + $row = $result->execute()->fetchOne(); + if (version_compare($row['server_version'], '9.6', '<')) { + $this->description = $this->l10n->t('PostgreSQL version "%s" is used. Nextcloud 21 will no longer support this version and requires PostgreSQL 9.6 or higher.', $row['server_version']); + return; + } + break; + case OraclePlatform::class: + break; + } + } + + public function description(): string { + return $this->description; + } + + public function severity(): string { + return 'info'; + } + + public function run(): bool { + return $this->description === ''; + } +} diff --git a/apps/settings/tests/SetupChecks/SupportedDatabaseTest.php b/apps/settings/tests/SetupChecks/SupportedDatabaseTest.php new file mode 100644 index 0000000000000..de709572261e9 --- /dev/null +++ b/apps/settings/tests/SetupChecks/SupportedDatabaseTest.php @@ -0,0 +1,42 @@ + + * + * @author Morris Jobke + * + * @license GNU AGPL version 3 or any later version + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as + * published by the Free Software Foundation, either version 3 of the + * License, or (at your option) any later version. + * + * 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 + * along with this program. If not, see . + * + */ + +namespace OCA\Settings\Tests; + +use OCA\Settings\SetupChecks\SupportedDatabase; +use OCP\IL10N; +use Test\TestCase; + +/** + * @group DB + */ +class SupportedDatabaseTest extends TestCase { + public function testPass(): void { + $l10n = $this->getMockBuilder(IL10N::class)->getMock(); + $check = new SupportedDatabase($l10n, \OC::$server->getDatabaseConnection()); + $this->assertTrue($check->run()); + } +} diff --git a/core/js/setupchecks.js b/core/js/setupchecks.js index 9d3f1ddc50894..c483454cb9113 100644 --- a/core/js/setupchecks.js +++ b/core/js/setupchecks.js @@ -513,6 +513,7 @@ OC.SetupChecks.addGenericSetupCheck(data, 'OCA\\Settings\\SetupChecks\\PhpOutputBuffering', messages) OC.SetupChecks.addGenericSetupCheck(data, 'OCA\\Settings\\SetupChecks\\LegacySSEKeyFormat', messages) OC.SetupChecks.addGenericSetupCheck(data, 'OCA\\Settings\\SetupChecks\\CheckUserCertificates', messages) + OC.SetupChecks.addGenericSetupCheck(data, 'OCA\\Settings\\SetupChecks\\SupportedDatabase', messages) } else { messages.push({