Skip to content

Commit

Permalink
feat(upgrade): migration attributes
Browse files Browse the repository at this point in the history
Signed-off-by: Maxence Lange <maxence@artificial-owl.com>
  • Loading branch information
ArtificialOwl committed Aug 2, 2024
1 parent 7b32dd5 commit e027825
Show file tree
Hide file tree
Showing 21 changed files with 1,274 additions and 0 deletions.
102 changes: 102 additions & 0 deletions core/Command/Db/Migrations/PreviewCommand.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,102 @@
<?php

declare(strict_types=1);

/**
* SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
* SPDX-License-Identifier: AGPL-3.0-or-later
*/
namespace OC\Core\Command\Db\Migrations;

use OC\Migration\MetadataManager;
use OC\Updater\ReleaseMetadata;
use OCP\Migration\Attributes\MigrationAttribute;
use Symfony\Component\Console\Command\Command;
use Symfony\Component\Console\Helper\Table;
use Symfony\Component\Console\Helper\TableCell;
use Symfony\Component\Console\Helper\TableCellStyle;
use Symfony\Component\Console\Helper\TableSeparator;
use Symfony\Component\Console\Input\InputArgument;
use Symfony\Component\Console\Input\InputInterface;
use Symfony\Component\Console\Output\OutputInterface;

/**
* @since 30.0.0
*/
class PreviewCommand extends Command {
private bool $initiated = false;
public function __construct(
private readonly MetadataManager $metadataManager,

Check failure on line 29 in core/Command/Db/Migrations/PreviewCommand.php

View workflow job for this annotation

GitHub Actions / static-code-analysis

InvalidDocblock

core/Command/Db/Migrations/PreviewCommand.php:29:3: InvalidDocblock: Param1 of OC\Core\Command\Db\Migrations\PreviewCommand::__construct has invalid syntax (see https://psalm.dev/008)

Check failure on line 29 in core/Command/Db/Migrations/PreviewCommand.php

View workflow job for this annotation

GitHub Actions / static-code-analysis

ParseError

core/Command/Db/Migrations/PreviewCommand.php:29:20: ParseError: Syntax error, unexpected T_STRING, expecting T_VARIABLE on line 29 (see https://psalm.dev/173)

Check failure

Code scanning / Psalm

InvalidDocblock Error

Param1 of OC\Core\Command\Db\Migrations\PreviewCommand::__construct has invalid syntax

Check failure

Code scanning / Psalm

ParseError Error

Syntax error, unexpected T_STRING, expecting T_VARIABLE on line 29
private readonly ReleaseMetadata $releaseMetadata,
) {
parent::__construct();
}

protected function configure(): void {
$this
->setName('migrations:preview')
->setDescription('Get preview of available DB migrations in case of initiating an upgrade')
->addArgument('version', InputArgument::REQUIRED, 'The destination version number');

parent::configure();
}

public function execute(InputInterface $input, OutputInterface $output): int {
$version = $input->getArgument('version');
if (filter_var($version, FILTER_VALIDATE_URL)) {
$metadata = $this->releaseMetadata->downloadMetadata($version);

Check failure on line 47 in core/Command/Db/Migrations/PreviewCommand.php

View workflow job for this annotation

GitHub Actions / static-code-analysis

UndefinedThisPropertyFetch

core/Command/Db/Migrations/PreviewCommand.php:47:16: UndefinedThisPropertyFetch: Instance property OC\Core\Command\Db\Migrations\PreviewCommand::$releaseMetadata is not defined (see https://psalm.dev/041)

Check failure

Code scanning / Psalm

UndefinedThisPropertyFetch Error

Instance property OC\Core\Command\Db\Migrations\PreviewCommand::$releaseMetadata is not defined
} elseif (str_starts_with($version, '/')) {
$metadata = json_decode(file_get_contents($version), true, flags: JSON_THROW_ON_ERROR);
} else {
$metadata = $this->releaseMetadata->getMetadata($version);

Check failure on line 51 in core/Command/Db/Migrations/PreviewCommand.php

View workflow job for this annotation

GitHub Actions / static-code-analysis

UndefinedThisPropertyFetch

core/Command/Db/Migrations/PreviewCommand.php:51:16: UndefinedThisPropertyFetch: Instance property OC\Core\Command\Db\Migrations\PreviewCommand::$releaseMetadata is not defined (see https://psalm.dev/041)

Check failure

Code scanning / Psalm

UndefinedThisPropertyFetch Error

Instance property OC\Core\Command\Db\Migrations\PreviewCommand::$releaseMetadata is not defined
}

$parsed = $this->metadataManager->getMigrationsAttributesFromReleaseMetadata($metadata['migrations'] ?? [], true);

Check failure on line 54 in core/Command/Db/Migrations/PreviewCommand.php

View workflow job for this annotation

GitHub Actions / static-code-analysis

UndefinedThisPropertyFetch

core/Command/Db/Migrations/PreviewCommand.php:54:13: UndefinedThisPropertyFetch: Instance property OC\Core\Command\Db\Migrations\PreviewCommand::$metadataManager is not defined (see https://psalm.dev/041)

Check failure

Code scanning / Psalm

UndefinedThisPropertyFetch Error

Instance property OC\Core\Command\Db\Migrations\PreviewCommand::$metadataManager is not defined

$table = new Table($output);
$this->displayMigrations($table, 'core', $parsed['core'] ?? []);
foreach ($parsed['apps'] as $appId => $migrations) {
if (!empty($migrations)) {
$this->displayMigrations($table, $appId, $migrations);
}
}
$table->render();

return 0;
}

private function displayMigrations(Table $table, string $appId, array $data): void {
if (empty($data)) {
return;
}

if ($this->initiated) {
$table->addRow(new TableSeparator());
}
$this->initiated = true;

$table->addRow(
[
new TableCell(
$appId,
[
'colspan' => 2,
'style' => new TableCellStyle(['cellFormat' => '<info>%s</info>'])
]
)
]
)->addRow(new TableSeparator());

/** @var MigrationAttribute[] $attributes */
foreach($data as $migration => $attributes) {
$attributesStr = [];
foreach($attributes as $attribute) {
$definition = '<info>' . $attribute->definition() . "</info>";
$definition .= empty($attribute->getDescription()) ? '' : "\n " . $attribute->getDescription();
$definition .= empty($attribute->getNotes()) ? '' : "\n <comment>" . implode("</comment>\n <comment>", $attribute->getNotes()) . '</comment>';
$attributesStr[] = $definition;
}
$table->addRow([$migration, implode("\n", $attributesStr)]);
}
}
}
1 change: 1 addition & 0 deletions core/register_command.php
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,7 @@
$application->add(Server::get(Command\Db\AddMissingIndices::class));
$application->add(Server::get(Command\Db\AddMissingPrimaryKeys::class));

$application->add(Server::get(Command\Db\Migrations\PreviewCommand::class));
if ($config->getSystemValueBool('debug', false)) {
$application->add(Server::get(Command\Db\Migrations\StatusCommand::class));
$application->add(Server::get(Command\Db\Migrations\MigrateCommand::class));
Expand Down
17 changes: 17 additions & 0 deletions lib/private/Migration/Exceptions/AttributeException.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
<?php

declare(strict_types=1);

/**
* SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
* SPDX-License-Identifier: AGPL-3.0-or-later
*/
namespace OC\Migration\Exceptions;

use Exception;

/**
* @since 30.0.0
*/
class AttributeException extends Exception {
}
156 changes: 156 additions & 0 deletions lib/private/Migration/MetadataManager.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,156 @@
<?php

declare(strict_types=1);

/**
* SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
* SPDX-License-Identifier: AGPL-3.0-only
*/
namespace OC\Migration;

use OC\DB\Connection;
use OC\DB\MigrationService;
use OC\Migration\Exceptions\AttributeException;
use OCP\App\IAppManager;
use OCP\Migration\Attributes\GenericMigrationAttribute;
use OCP\Migration\Attributes\MigrationAttribute;
use Psr\Log\LoggerInterface;
use ReflectionClass;

/**
* Helps managing DB Migrations' Metadata
*
* @since 30.0.0
*/
class MetadataManager {
public function __construct(
private readonly IAppManager $appManager,

Check failure on line 27 in lib/private/Migration/MetadataManager.php

View workflow job for this annotation

GitHub Actions / static-code-analysis

InvalidDocblock

lib/private/Migration/MetadataManager.php:27:3: InvalidDocblock: Param1 of OC\Migration\MetadataManager::__construct has invalid syntax (see https://psalm.dev/008)

Check failure on line 27 in lib/private/Migration/MetadataManager.php

View workflow job for this annotation

GitHub Actions / static-code-analysis

ParseError

lib/private/Migration/MetadataManager.php:27:20: ParseError: Syntax error, unexpected T_STRING, expecting T_VARIABLE on line 27 (see https://psalm.dev/173)

Check failure

Code scanning / Psalm

InvalidDocblock Error

Param1 of OC\Migration\MetadataManager::__construct has invalid syntax

Check failure

Code scanning / Psalm

ParseError Error

Syntax error, unexpected T_STRING, expecting T_VARIABLE on line 27
private readonly Connection $connection,
private readonly LoggerInterface $logger,
) {
}

/**
* We get all migrations from an app (or 'core'), and
* for each migration files we extract its attributes
*
* @param string $appId
*
* @return array<string, MigrationAttribute[]>
* @since 30.0.0
*/
public function extractMigrationAttributes(string $appId): array {
$ms = new MigrationService($appId, $this->connection);

Check failure on line 43 in lib/private/Migration/MetadataManager.php

View workflow job for this annotation

GitHub Actions / static-code-analysis

UndefinedThisPropertyFetch

lib/private/Migration/MetadataManager.php:43:38: UndefinedThisPropertyFetch: Instance property OC\Migration\MetadataManager::$connection is not defined (see https://psalm.dev/041)

Check failure

Code scanning / Psalm

UndefinedThisPropertyFetch Error

Instance property OC\Migration\MetadataManager::$connection is not defined

$metadata = [];
foreach($ms->getAvailableVersions() as $version) {
$metadata[$version] = [];
$class = new ReflectionClass($ms->createInstance($version));

Check failure on line 48 in lib/private/Migration/MetadataManager.php

View workflow job for this annotation

GitHub Actions / static-code-analysis

InaccessibleMethod

lib/private/Migration/MetadataManager.php:48:38: InaccessibleMethod: Cannot access protected method OC\DB\MigrationService::createInstance from context OC\Migration\MetadataManager (see https://psalm.dev/003)

Check failure

Code scanning / Psalm

InaccessibleMethod Error

Cannot access protected method OC\DB\MigrationService::createInstance from context OC\Migration\MetadataManager
$attributes = $class->getAttributes();
foreach ($attributes as $attribute) {
$item = $attribute->newInstance();
if ($item instanceof MigrationAttribute) {
$metadata[$version][] = $item;
}
}
}

return $metadata;
}

/**
* convert direct data from release metadata into a list of Migrations' Attribute
*
* @param array<array-key, array<array-key, array>> $metadata
* @param bool $filterKnownMigrations ignore metadata already done in local instance
*
* @return array{apps: array<array-key, array<string, MigrationAttribute[]>>, core: array<string, MigrationAttribute[]>}
* @since 30.0.0
*/
public function getMigrationsAttributesFromReleaseMetadata(
array $metadata,
bool $filterKnownMigrations = false
): array {
$appsAttributes = [];
foreach (array_keys($metadata['apps']) as $appId) {
if ($filterKnownMigrations && !$this->appManager->isInstalled($appId)) {

Check failure on line 76 in lib/private/Migration/MetadataManager.php

View workflow job for this annotation

GitHub Actions / static-code-analysis

UndefinedThisPropertyFetch

lib/private/Migration/MetadataManager.php:76:35: UndefinedThisPropertyFetch: Instance property OC\Migration\MetadataManager::$appManager is not defined (see https://psalm.dev/041)

Check failure

Code scanning / Psalm

UndefinedThisPropertyFetch Error

Instance property OC\Migration\MetadataManager::$appManager is not defined
continue; // if not interested and app is not installed
}

$done = ($filterKnownMigrations) ? $this->getKnownMigrations($appId) : [];
$appsAttributes[$appId] = $this->parseMigrations($metadata['apps'][$appId] ?? [], $done);
}

$done = ($filterKnownMigrations) ? $this->getKnownMigrations('core') : [];
return [
'core' => $this->parseMigrations($metadata['core'] ?? [], $done),
'apps' => $appsAttributes
];
}

/**
* convert raw data to a list of MigrationAttribute
*
* @param array $migrations
* @param array $ignoreMigrations
*
* @return array<string, MigrationAttribute[]>
*/
private function parseMigrations(array $migrations, array $ignoreMigrations = []): array {
$parsed = [];
foreach (array_keys($migrations) as $entry) {
if (in_array($entry, $ignoreMigrations)) {
continue;
}

$parsed[$entry] = [];
foreach ($migrations[$entry] as $item) {
try {
$parsed[$entry][] = $this->createAttribute($item);
} catch (AttributeException $e) {
$this->logger->warning('exception while trying to create attribute', ['exception' => $e, 'item' => json_encode($item)]);

Check failure

Code scanning / Psalm

UndefinedThisPropertyFetch Error

Instance property OC\Migration\MetadataManager::$logger is not defined
$parsed[$entry][] = new GenericMigrationAttribute($item);

Check failure

Code scanning / Psalm

TooManyArguments Error

Too many arguments for OCP\Migration\Attributes\GenericMigrationAttribute::__construct - expecting 0 but saw 1
}
}
}

return $parsed;
}

/**
* returns migrations already done
*
* @param string $appId
*
* @return array
* @throws \Exception
*/
private function getKnownMigrations(string $appId): array {
$ms = new MigrationService($appId, $this->connection);

Check failure

Code scanning / Psalm

UndefinedThisPropertyFetch Error

Instance property OC\Migration\MetadataManager::$connection is not defined
return $ms->getMigratedVersions();
}

/**
* generate (deserialize) a MigrationAttribute from a serialized version
*
* @param array $item
*
* @return MigrationAttribute
* @throws AttributeException
*/
private function createAttribute(array $item): MigrationAttribute {
$class = $item['class'] ?? '';
$namespace = 'OCP\Migration\Attributes\\';
if (!str_starts_with($class, $namespace)
|| !ctype_alpha(substr($class, strlen($namespace)))) {
throw new AttributeException('class name does not looks valid');
}

try {
$attribute = new $class($item['table'] ?? '');
return $attribute->import($item);
} catch (\Error) {
throw new AttributeException('cannot import Attribute');
}
}
}
17 changes: 17 additions & 0 deletions lib/private/Updater/Exceptions/ReleaseMetadataException.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
<?php

declare(strict_types=1);

/**
* SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
* SPDX-License-Identifier: AGPL-3.0-or-later
*/
namespace OC\Updater\Exceptions;

use Exception;

/**
* @since 30.0.0
*/
class ReleaseMetadataException extends Exception {
}
79 changes: 79 additions & 0 deletions lib/private/Updater/ReleaseMetadata.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
<?php

declare(strict_types=1);

/**
* SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
* SPDX-License-Identifier: AGPL-3.0-or-later
*/
namespace OC\Updater;

use Exception;
use JsonException;
use OC\Updater\Exceptions\ReleaseMetadataException;
use OCP\Http\Client\IClientService;

/** retrieve releases metadata from official servers
*
* @since 30.0.0
*/
class ReleaseMetadata {
public function __construct(
private readonly IClientService $clientService,

Check failure

Code scanning / Psalm

InvalidDocblock Error

Param1 of OC\Updater\ReleaseMetadata::__construct has invalid syntax

Check failure

Code scanning / Psalm

ParseError Error

Syntax error, unexpected T_STRING, expecting T_VARIABLE on line 22
) {
}

/**
* returns metadata based on release version
*
* - version is a stable release, metadata is downloaded from official releases folder
* - version is not a table release, metadata is downloaded from official prereleases folder
* - version is a major version (30, 31, 32, ...), latest metadata are downloaded
*
* @param string $version
*
* @return array
* @throws ReleaseMetadataException
* @since 30.0.0
*/
public function getMetadata(string $version): array {
if (!str_contains($version, '.')) {
$url = 'https://download.nextcloud.com/server/releases/latest-' . $version . '.metadata';
} else {
[,,$minor] = explode('.', $version);
if (ctype_digit($minor)) {
$url = 'https://download.nextcloud.com/server/releases/nextcloud-' . $version . '.metadata';
} else {
$url = 'https://download.nextcloud.com/server/prereleases/nextcloud-' . $version . '.metadata';
}
}
return $this->downloadMetadata($url);
}

/**
* download Metadata from a link
*
* @param string $url
*
* @return array
* @throws ReleaseMetadataException
* @since 30.0.0
*/
public function downloadMetadata(string $url): array {
$client = $this->clientService->newClient();

Check failure

Code scanning / Psalm

UndefinedThisPropertyFetch Error

Instance property OC\Updater\ReleaseMetadata::$clientService is not defined
try {
$response = $client->get($url, [
'timeout' => 10,
'connect_timeout' => 10
]);
} catch (Exception $e) {
throw new ReleaseMetadataException('could not reach metadata at ' . $url, previous: $e);
}

try {
return json_decode($response->getBody(), true, flags: JSON_THROW_ON_ERROR);
} catch (JsonException) {
throw new ReleaseMetadataException('remote document is not valid');
}
}
}
Loading

0 comments on commit e027825

Please sign in to comment.