Skip to content
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 6 additions & 1 deletion lib/UserMigration/MailAccountMigrator.php
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
use OCA\Mail\UserMigration\Service\AccountMigrationService;
use OCA\Mail\UserMigration\Service\AppConfigMigrationService;
use OCA\Mail\UserMigration\Service\InternalAddressesMigrationService;
use OCA\Mail\UserMigration\Service\QuickActionsMigrationService;
use OCA\Mail\UserMigration\Service\SMIMEMigrationService;
use OCA\Mail\UserMigration\Service\TagsMigrationService;
use OCA\Mail\UserMigration\Service\TextBlocksMigrationService;
Expand Down Expand Up @@ -43,6 +44,7 @@ public function __construct(
private readonly TextBlocksMigrationService $textBlocksMigrationService,
private readonly TagsMigrationService $tagsMigrationService,
private readonly SMIMEMigrationService $sMimeMigrationService,
private readonly QuickActionsMigrationService $quickActionsMigrationService,
) {
}

Expand All @@ -60,6 +62,7 @@ public function export(IUser $user,
$this->tagsMigrationService->exportTags($user, $exportDestination, $output);
$this->sMimeMigrationService->exportCertificates($user, $exportDestination, $output);
$this->accountMigrationService->exportAccounts($user, $exportDestination, $output);
$this->quickActionsMigrationService->exportQuickActions($user, $exportDestination, $output);
}

#[\Override]
Expand All @@ -73,8 +76,9 @@ public function import(IUser $user, IImportSource $importSource, OutputInterface
$this->trustedSendersMigrationService->importTrustedSenders($user, $importSource, $output);
$this->textBlocksMigrationService->importTextBlocks($user, $importSource, $output);
$newCertificateIds = $this->sMimeMigrationService->importCertificates($user, $importSource, $output);
$newAccountIds = $this->accountMigrationService->importAccounts($user, $importSource, $output, $newCertificateIds);
$newAccountAndMailboxIds = $this->accountMigrationService->importAccounts($user, $importSource, $output, $newCertificateIds);
$newTagIds = $this->tagsMigrationService->importTags($user, $importSource, $output);
$this->quickActionsMigrationService->importQuickActions($user, $importSource, $output, $newAccountAndMailboxIds, $newTagIds);

$this->accountMigrationService->scheduleBackgroundJobs($user, $output);
}
Expand All @@ -92,6 +96,7 @@ public function import(IUser $user, IImportSource $importSource, OutputInterface
private function deleteExistingData(IUser $user, OutputInterface $output): void {
$output->writeln($this->l10n->t("Deleting existing mail data for user {$user->getUID()}"), OutputInterface::VERBOSITY_VERBOSE);

$this->quickActionsMigrationService->deleteAllQuickActions($user, $output);
$this->accountMigrationService->deleteAllAccounts($user, $output);
$this->appConfigMigrationService->deleteAppConfiguration($user, $output);
$this->internalAddressesMigrationService->removeInternalAddresses($user, $output);
Expand Down
181 changes: 181 additions & 0 deletions lib/UserMigration/Service/QuickActionsMigrationService.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,181 @@
<?php

declare(strict_types=1);

/**
* SPDX-FileCopyrightText: 2026 Nextcloud GmbH and Nextcloud contributors
* SPDX-License-Identifier: AGPL-3.0-or-later
*/

namespace OCA\Mail\UserMigration\Service;

use JsonException;
use OCA\Mail\Service\QuickActionsService;
use OCA\Mail\UserMigration\MailAccountMigrator;
use OCP\DB\Exception;
use OCP\IL10N;
use OCP\IUser;
use OCP\UserMigration\IExportDestination;
use OCP\UserMigration\IImportSource;
use OCP\UserMigration\UserMigrationException;
use Symfony\Component\Console\Output\OutputInterface;

class QuickActionsMigrationService {
public const QUICK_ACTIONS_FILE = MailAccountMigrator::EXPORT_ROOT . '/quick_actions.json';

public function __construct(
private readonly QuickActionsService $quickActionsService,
private readonly IL10N $l10n,
) {
}

/**
* Export all quick actions the user defined across
* their accounts.
*
* @param IUser $user
* @param IExportDestination $exportDestination
* @param OutputInterface $output
* @throws UserMigrationException
*/
public function exportQuickActions(IUser $user, IExportDestination $exportDestination, OutputInterface $output): void {
$output->writeln(
$this->l10n->t('Exporting quick actions for user %s', [$user->getUID()]),
OutputInterface::VERBOSITY_VERBOSE
);

$quickActions = $this->quickActionsService->findAll($user->getUID());

try {
$exportDestination->addFileContents(self::QUICK_ACTIONS_FILE, json_encode($quickActions, JSON_THROW_ON_ERROR));
} catch (JsonException|UserMigrationException $exception) {
throw new UserMigrationException(
"Failed to export quick actions for user {$user->getUID()}",
previous: $exception
);
}
}

/**
* Import all quick actions the user defined across
* their accounts.
*
* @throws UserMigrationException
* @throws Exception
* @throws JsonException
* @throws \OCA\Mail\Exception\ServiceException
*/
public function importQuickActions(IUser $user, IImportSource $importSource, OutputInterface $output, array $accountAndMailboxMapping, array $tagMapping): void {
$output->writeln(
$this->l10n->t('Importing quick actions for user %s', [$user->getUID()]),
OutputInterface::VERBOSITY_VERBOSE
);

$quickActions = json_decode($importSource->getFileContents(self::QUICK_ACTIONS_FILE), true);
$this->validateQuickActions($quickActions);

foreach ($quickActions as $quickAction) {
$createdQuickAction = $this->quickActionsService->create($quickAction['name'], $accountAndMailboxMapping['accounts'][$quickAction['accountId']]);

foreach ($quickAction['actionSteps'] as $actionStep) {
$this->quickActionsService->createActionStep($actionStep['name'], $actionStep['order'], $createdQuickAction->getId(), $tagMapping[$actionStep['tagId']] ?? null, $accountAndMailboxMapping['mailboxes'][$actionStep['mailboxId']] ?? null);
}
}
}

public function deleteAllQuickActions(IUser $user, OutputInterface $output): void {
$output->writeln(
$this->l10n->t('Deleting all quick actions for user %s', [$user->getUID()]),
OutputInterface::VERBOSITY_VERBOSE
);

$this->quickActionsService->deleteAll($user->getUID());

Check failure on line 92 in lib/UserMigration/Service/QuickActionsMigrationService.php

View workflow job for this annotation

GitHub Actions / static-psalm-analysis dev-master

UndefinedMethod

lib/UserMigration/Service/QuickActionsMigrationService.php:92:31: UndefinedMethod: Method OCA\Mail\Service\QuickActionsService::deleteAll does not exist (see https://psalm.dev/022)

Check failure on line 92 in lib/UserMigration/Service/QuickActionsMigrationService.php

View workflow job for this annotation

GitHub Actions / static-psalm-analysis dev-stable32

UndefinedMethod

lib/UserMigration/Service/QuickActionsMigrationService.php:92:31: UndefinedMethod: Method OCA\Mail\Service\QuickActionsService::deleteAll does not exist (see https://psalm.dev/022)

Check failure on line 92 in lib/UserMigration/Service/QuickActionsMigrationService.php

View workflow job for this annotation

GitHub Actions / static-psalm-analysis dev-stable33

UndefinedMethod

lib/UserMigration/Service/QuickActionsMigrationService.php:92:31: UndefinedMethod: Method OCA\Mail\Service\QuickActionsService::deleteAll does not exist (see https://psalm.dev/022)
}

/**
* Validate the parsed quick actions to ensure they
* have the expected structure and types.
*
* @throws UserMigrationException
*/
private function validateQuickActions(mixed $quickActions): void {
$quickActionsArrayIsValid = is_array($quickActions) && array_is_list($quickActions);
if (!$quickActionsArrayIsValid) {
throw new UserMigrationException('Invalid quick actions export structure');
}

foreach ($quickActions as $quickAction) {
$quickActionArrayIsValid = is_array($quickAction);

$idIsValid = $quickActionArrayIsValid
&& array_key_exists('id', $quickAction)
&& is_int($quickAction['id']);

$nameIsValid = $quickActionArrayIsValid
&& array_key_exists('name', $quickAction)
&& is_string($quickAction['name']);

$orderIsValid = $quickActionArrayIsValid
&& array_key_exists('accountId', $quickAction)
&& is_int($quickAction['accountId']);

$actionStepsArrayIsValid = $quickActionArrayIsValid
&& array_key_exists('actionSteps', $quickAction)
&& is_array($quickAction['actionSteps'])
&& array_is_list($quickAction['actionSteps'])
&& $this->validateQuickSteps($quickAction['actionSteps']);

if (
!$idIsValid
|| !$nameIsValid
|| !$orderIsValid
|| !$actionStepsArrayIsValid
) {
throw new UserMigrationException('Invalid quick action entry');
}
}
}

private function validateQuickSteps(mixed $quickSteps): bool {
$quickStepsArrayIsValid = true;

foreach ($quickSteps as $actionStep) {
$actionStepArrayIsValid = is_array($actionStep);

$idIsValid = $actionStepArrayIsValid
&& array_key_exists('id', $actionStep)
&& is_int($actionStep['id']);

$nameIsValid = $actionStepArrayIsValid
&& array_key_exists('name', $actionStep)
&& is_string($actionStep['name']);

$orderIsValid = $actionStepArrayIsValid
&& array_key_exists('order', $actionStep)
&& is_int($actionStep['order']);

$actionIdIsValid = $actionStepArrayIsValid
&& array_key_exists('actionId', $actionStep)
&& is_int($actionStep['actionId']);

$tagIdIsValid = $actionStepArrayIsValid
&& array_key_exists('tagId', $actionStep)
&& (is_int($actionStep['tagId']) || is_null($actionStep['tagId']));

$mailboxIdIsValid = $actionStepArrayIsValid
&& array_key_exists('mailboxId', $actionStep)
&& (is_int($actionStep['mailboxId']) || is_null($actionStep['mailboxId']));

$actionStepIsValid = $idIsValid
&& $nameIsValid
&& $orderIsValid
&& $actionIdIsValid
&& $tagIdIsValid
&& $mailboxIdIsValid;

$quickStepsArrayIsValid = $quickStepsArrayIsValid && $actionStepIsValid;
}

return $quickStepsArrayIsValid;
}
}
Loading
Loading