diff --git a/src/app/DataMigrator/DAV.php b/src/app/DataMigrator/DAV.php index bdfd290b..0b1394d0 100644 --- a/src/app/DataMigrator/DAV.php +++ b/src/app/DataMigrator/DAV.php @@ -1,411 +1,420 @@ username . ($account->loginas ? "**{$account->loginas}" : ''); $baseUri = rtrim($account->uri, '/'); $baseUri = preg_replace('|^dav|', 'http', $baseUri); $this->client = new DAVClient($username, $account->password, $baseUri); $this->engine = $engine; $this->account = $account; } /** * Check user credentials. * * @throws \Exception */ public function authenticate(): void { try { $this->client->options(); } catch (\Exception $e) { throw new \Exception("Invalid DAV credentials or server."); } } /** * Create an item in a folder. * * @param Item $item Item to import * * @throws \Exception */ public function createItem(Item $item): void { - if ($item->existing) { - $href = $item->existing; + $is_file = false; + $href = $item->existing ?: null; + + if (strlen($item->content)) { + $content = $item->content; } else { - $href = $this->getFolderPath($item->folder) . '/' . pathinfo($item->filename, PATHINFO_BASENAME); + $content = $item->filename; + $is_file = true; + } + + if (empty($href)) { + $href = $this->getFolderPath($item->folder) . '/' . basename($item->filename); } - $object = new DAVOpaque($item->filename, true); + $object = new DAVOpaque($content, $is_file); $object->href = $href; switch ($item->folder->type) { case Engine::TYPE_EVENT: case Engine::TYPE_TASK: $object->contentType = 'text/calendar; charset=utf-8'; break; case Engine::TYPE_CONTACT: $object->contentType = 'text/vcard; charset=utf-8'; break; } if ($this->client->create($object) === false) { throw new \Exception("Failed to save DAV object at {$href}"); } } /** * Create a folder. * * @param Folder $folder Folder data * * @throws \Exception on error */ public function createFolder(Folder $folder): void { $dav_type = $this->type2DAV($folder->type); $folders = $this->client->listFolders($dav_type); if ($folders === false) { throw new \Exception("Failed to list folders on the DAV server"); } // Note: iRony flattens the list by modifying the folder name // This is not going to work with Cyrus DAV, but anyway folder // hierarchies support is not full in Kolab 4. foreach ($folders as $dav_folder) { if (str_replace(' » ', '/', $dav_folder->name) === $folder->fullname) { // do nothing, folder already exists return; } } $home = $this->client->getHome($dav_type); $folder_id = Utils::uuidStr(); $collection_type = $dav_type == DAVClient::TYPE_VCARD ? 'addressbook' : 'calendar'; // We create all folders on the top-level $dav_folder = new DAVFolder(); $dav_folder->name = $folder->fullname; $dav_folder->href = rtrim($home, '/') . '/' . $folder_id; $dav_folder->components = [$dav_type]; $dav_folder->types = ['collection', $collection_type]; if ($this->client->folderCreate($dav_folder) === false) { throw new \Exception("Failed to create a DAV folder {$dav_folder->href}"); } } /** * Fetching an item */ public function fetchItem(Item $item): void { - // Save the item content to a file - $location = $item->folder->location; - - if (!file_exists($location)) { - mkdir($location, 0740, true); - } - - $location .= '/' . basename($item->id); - $result = $this->client->getObjects(dirname($item->id), $this->type2DAV($item->folder->type), [$item->id]); if ($result === false) { throw new \Exception("Failed to fetch DAV item for {$item->id}"); } // TODO: Do any content changes, e.g. organizer/attendee email migration - if (file_put_contents($location, (string) $result[0]) === false) { - throw new \Exception("Failed to write to {$location}"); - } + $content = (string) $result[0]; + + if (strlen($content) > Engine::MAX_ITEM_SIZE) { + // Save the item content to a file + $location = $item->folder->tempFileLocation(basename($item->id)); + + if (file_put_contents($location, $content) === false) { + throw new \Exception("Failed to write to {$location}"); + } - $item->filename = $location; + $item->filename = $location; + } else { + $item->content = $content; + $item->filename = basename($item->id); + } } /** * Fetch a list of folder items */ public function fetchItemList(Folder $folder, $callback, ImporterInterface $importer): void { // Get existing messages' headers from the destination mailbox $existing = $importer->getItems($folder); $dav_type = $this->type2DAV($folder->type); $location = $this->getFolderPath($folder); $search = new DAVSearch($dav_type, false); // TODO: We request only properties relevant to incremental migration, // i.e. to find that something exists and its last update time. // Some servers (iRony) do ignore that and return full VCARD/VEVENT/VTODO // content, if there's many objects we'll have a memory limit issue. // Also, this list should be controlled by the exporter. $search->dataProperties = ['UID', 'REV', 'DTSTAMP']; $set = new ItemSet(); $result = $this->client->search( $location, $search, function ($item) use (&$set, $dav_type, $folder, $existing, $callback) { // Skip an item that exists and did not change $exists = null; if (!empty($existing[$item->uid])) { $exists = $existing[$item->uid]['href']; switch ($dav_type) { case DAVClient::TYPE_VCARD: if ($existing[$item->uid]['rev'] == $item->rev) { return null; } break; case DAVClient::TYPE_VEVENT: case DAVClient::TYPE_VTODO: if ($existing[$item->uid]['dtstamp'] == (string) $item->dtstamp) { return null; } break; } } $set->items[] = Item::fromArray([ 'id' => $item->href, 'folder' => $folder, 'existing' => $exists, ]); if (count($set->items) == self::CHUNK_SIZE) { $callback($set); $set = new ItemSet(); } return null; } ); if (count($set->items)) { $callback($set); } if ($result === false) { throw new \Exception("Failed to get items from a DAV folder {$location}"); } // TODO: Delete items that do not exist anymore? } /** * Get a list of items, limited to their essential propeties * used in incremental migration. * * @param Folder $folder Folder data * * @throws \Exception on error */ public function getItems(Folder $folder): array { $dav_type = $this->type2DAV($folder->type); $location = $this->getFolderPath($folder); $search = new DAVSearch($dav_type); // TODO: We request only properties relevant to incremental migration, // i.e. to find that something exists and its last update time. // Some servers (iRony) do ignore that and return full VCARD/VEVENT/VTODO // content, if there's many objects we'll have a memory limit issue. // Also, this list should be controlled by the exporter. $search->dataProperties = ['UID', 'X-MS-ID', 'REV', 'DTSTAMP']; $items = $this->client->search( $location, $search, function ($item) use ($dav_type) { // Slim down the result to properties we might need $object = [ 'href' => $item->href, 'uid' => $item->uid, ]; if (!empty($item->custom['X-MS-ID'])) { $object['x-ms-id'] = $item->custom['X-MS-ID']; } switch ($dav_type) { case DAVClient::TYPE_VCARD: $object['rev'] = $item->rev; break; case DAVClient::TYPE_VEVENT: case DAVClient::TYPE_VTODO: $object['dtstamp'] = (string) $item->dtstamp; break; } return [$item->uid, $object]; } ); if ($items === false) { throw new \Exception("Failed to get items from a DAV folder {$location}"); } return $items; } /** * Get folders hierarchy */ public function getFolders($types = []): array { $result = []; foreach (['VEVENT', 'VTODO', 'VCARD'] as $component) { $type = $this->typeFromDAV($component); // Skip folder types we do not support (need) if (!empty($types) && !in_array($type, $types)) { continue; } $folders = $this->client->listFolders($component); foreach ($folders as $folder) { // Skip other users/shared folders if ($this->shouldSkip($folder)) { continue; } $result[$folder->href] = Folder::fromArray([ 'fullname' => str_replace(' » ', '/', $folder->name), 'href' => $folder->href, 'type' => $type, ]); } } return $result; } /** * Get folder relative URI */ protected function getFolderPath(Folder $folder): string { $cache_key = $folder->type . '!' . $folder->fullname; if (isset($this->folderPaths[$cache_key])) { return $this->folderPaths[$cache_key]; } $folders = $this->client->listFolders($this->type2DAV($folder->type)); if ($folders === false) { throw new \Exception("Failed to list folders on the DAV server"); } // Note: iRony flattens the list by modifying the folder name // This is not going to work with Cyrus DAV, but anyway folder // hierarchies support is not full in Kolab 4. foreach ($folders as $dav_folder) { if (str_replace(' » ', '/', $dav_folder->name) === $folder->fullname) { return $this->folderPaths[$cache_key] = rtrim($dav_folder->href, '/'); } } throw new \Exception("Folder not found: {$folder->fullname}"); } /** * Map Kolab type into DAV object type */ protected static function type2DAV(string $type): string { switch ($type) { case Engine::TYPE_EVENT: return DAVClient::TYPE_VEVENT; case Engine::TYPE_TASK: return DAVClient::TYPE_VTODO; case Engine::TYPE_CONTACT: case Engine::TYPE_GROUP: return DAVClient::TYPE_VCARD; default: throw new \Exception("Cannot map type '{$type}' to DAV"); } } /** * Map DAV object type into Kolab type */ protected static function typeFromDAV(string $type): string { switch ($type) { case DAVClient::TYPE_VEVENT: return Engine::TYPE_EVENT; case DAVClient::TYPE_VTODO: return Engine::TYPE_TASK; case DAVClient::TYPE_VCARD: // TODO what about groups return Engine::TYPE_CONTACT; default: throw new \Exception("Cannot map type '{$type}' from DAV"); } } /** * Check if the folder should not be migrated */ private function shouldSkip($folder): bool { // When dealing with iRony DAV other user folders names have distinct names // there's no other way to recognize them than by the name pattern. // ;et's hope that users do not have personal folders with names starting with a bracket. if (preg_match('~\(.*\) .*~', $folder->name)) { return true; } if (str_starts_with($folder->name, 'shared » ')) { return true; } // TODO: Cyrus DAV shared folders return false; } } diff --git a/src/app/DataMigrator/EWS.php b/src/app/DataMigrator/EWS.php index a4a41f4b..f22ab4d9 100644 --- a/src/app/DataMigrator/EWS.php +++ b/src/app/DataMigrator/EWS.php @@ -1,470 +1,515 @@ Engine::TYPE_EVENT, EWS\Contact::FOLDER_TYPE => Engine::TYPE_CONTACT, EWS\Task::FOLDER_TYPE => Engine::TYPE_TASK, ]; /** @var Account Account to operate on */ protected $account; /** @var Engine Data migrator engine */ protected $engine; /** * Object constructor */ public function __construct(Account $account, Engine $engine) { $this->account = $account; $this->engine = $engine; } /** * Server autodiscovery */ public static function autodiscover(string $user, string $password): ?string { // You should never run the Autodiscover more than once. // It can make between 1 and 5 calls before giving up, or before finding your server, // depending on how many different attempts it needs to make. // TODO: Autodiscovery may fail with an exception thrown. Handle this nicely. // TODO: Looks like this autodiscovery also does not work w/Basic Auth? $api = API\ExchangeAutodiscover::getAPI($user, $password); $server = $api->getClient()->getServer(); $version = $api->getClient()->getVersion(); return sprintf('ews://%s:%s@%s', urlencode($user), urlencode($password), $server); } /** * Authenticate to EWS (initialize the EWS client) */ public function authenticate(): void { if (!empty($this->account->params['client_id'])) { $this->api = $this->authenticateWithOAuth2( $this->account->host, $this->account->username, $this->account->params['client_id'], $this->account->params['client_secret'], $this->account->params['tenant_id'] ); } else { // Note: This initializes the client, but not yet connects to the server // TODO: To know that the credentials work we'll have to do some API call. $this->api = $this->authenticateWithPassword( $this->account->host, $this->account->username, $this->account->password, $this->account->loginas ); } } /** * Autodiscover the server and authenticate the user */ protected function authenticateWithPassword(string $server, string $user, string $password, string $loginas = null) { // Note: Since 2023-01-01 EWS at Office365 requires OAuth2, no way back to basic auth. \Log::debug("[EWS] Using basic authentication on $server..."); $options = []; if ($loginas) { $options['impersonation'] = $loginas; } $this->engine->setOption('ews', [ 'options' => $options, 'server' => $server, ]); return API::withUsernameAndPassword($server, $user, $password, $this->apiOptions($options)); } /** * Authenticate with a token (Office365) */ protected function authenticateWithToken(string $server, string $user, string $token, $expires_at = null) { \Log::debug("[EWS] Using token authentication on $server..."); $options = ['impersonation' => $user]; $this->engine->setOption('ews', [ 'options' => $options, 'server' => $server, 'token' => $token, 'expires_at' => $expires_at, ]); return API::withCallbackToken($server, $token, $this->apiOptions($options)); } /** * Authenticate with OAuth2 (Office365) - get the token */ protected function authenticateWithOAuth2( string $server, string $user, string $client_id, string $client_secret, string $tenant_id ) { // See https://github.com/Garethp/php-ews/blob/master/examples/basic/authenticatingWithOAuth.php // See https://github.com/Garethp/php-ews/issues/236#issuecomment-1292521527 // To register OAuth2 app goto https://entra.microsoft.com > Applications > App registrations \Log::debug("[EWS] Fetching OAuth2 token from $server..."); $scope = 'https://outlook.office365.com/.default'; $token_uri = "https://login.microsoftonline.com/{$tenant_id}/oauth2/v2.0/token"; // $authUri = "https://login.microsoftonline.com/{$tenant_id}/oauth2/authorize"; $response = Http::asForm() ->timeout(5) ->post($token_uri, [ 'client_id' => $client_id, 'client_secret' => $client_secret, 'scope' => $scope, 'grant_type' => 'client_credentials', ]) ->throwUnlessStatus(200); $token = $response->json('access_token'); // Note: Office365 default token expiration time is ~1h, $expires_in = $response->json('expires_in'); $expires_at = now()->addSeconds($expires_in)->toDateTimeString(); return $this->authenticateWithToken($server, $user, $token, $expires_at); } /** * Get folders hierarchy */ public function getFolders($types = []): array { - // Get full folders hierarchy + if (empty($types)) { + $types = array_values($this->type_map); + } + + // Create FolderClass filter + $search = new Type\OrType(); + foreach ($types as $type) { + $type = array_search($type, $this->type_map); + $search->addContains(Type\Contains::buildFromArray([ + 'FieldURI' => [ + Type\FieldURI::buildFromArray(['FieldURI' => 'folder:FolderClass']), + ], + 'Constant' => Type\ConstantValueType::buildFromArray([ + 'Value' => $type, + ]), + 'ContainmentComparison' => 'Exact', + 'ContainmentMode' => 'FullString', + ])); + } + + // Get full folders hierarchy (filtered by folder class) + // Use of the filter reduces the response size by excluding system folders $options = [ 'Traversal' => 'Deep', + 'Restriction' => ['Or' => $search], ]; $folders = $this->api->getChildrenFolders('root', $options); $result = []; foreach ($folders as $folder) { $class = $folder->getFolderClass(); $type = $this->type_map[$class] ?? null; // Skip folder types we do not support (need) if (empty($type) || (!empty($types) && !in_array($type, $types))) { continue; } - // Note: Folder names are localized + // Note: Folder names are localized, even INBOX $name = $fullname = $folder->getDisplayName(); $id = $folder->getFolderId()->getId(); $parentId = $folder->getParentFolderId()->getId(); // Create folder name with full path if ($parentId && !empty($result[$parentId])) { $fullname = $result[$parentId]->fullname . '/' . $name; } // Top-level folder, check if it's a special folder we should ignore // FIXME: Is there a better way to distinguish user folders from system ones? if ( in_array($fullname, $this->folder_exceptions) || strpos($fullname, 'OwaFV15.1All') === 0 ) { continue; } $result[$id] = Folder::fromArray([ 'id' => $folder->getFolderId()->toArray(true), 'total' => $folder->getTotalCount(), 'class' => $class, 'type' => $this->type_map[$class] ?? null, 'name' => $name, 'fullname' => $fullname, ]); } return $result; } /** * Fetch a list of folder items */ public function fetchItemList(Folder $folder, $callback, Interface\ImporterInterface $importer): void { // Job processing - initialize environment $this->initEnv($this->engine->queue); // The folder is empty, we can stop here if (empty($folder->total)) { // TODO: Delete all existing items? return; } // Get items already imported // TODO: This might be slow and/or memory expensive, we should consider // whether storing list of imported items in some cache wouldn't be a better // solution. Of course, cache would not get changes in the destination account. $existing = $importer->getItems($folder); // Create X-MS-ID index for easier search in existing items // Note: For some objects we could use UID (events), but for some we don't have UID in Exchange. // Also because fetching extra properties here is problematic, we use X-MS-ID. $existingIndex = []; array_walk( $existing, function (&$item, $idx) use (&$existingIndex) { if (!empty($item['x-ms-id'])) { [$id, $changeKey] = explode('!', $item['x-ms-id']); $item['changeKey'] = $changeKey; $existingIndex[$id] = $idx; unset($item['x-ms-id']); } } ); $request = [ // Exchange's maximum is 1000 'IndexedPageItemView' => ['MaxEntriesReturned' => 100, 'Offset' => 0, 'BasePoint' => 'Beginning'], 'ParentFolderIds' => $folder->id, 'Traversal' => 'Shallow', 'ItemShape' => [ 'BaseShape' => 'IdOnly', 'AdditionalProperties' => [ - 'FieldURI' => ['FieldURI' => 'item:ItemClass'], + 'FieldURI' => [ + ['FieldURI' => 'item:ItemClass'], + // ['FieldURI' => 'item:Size'], + ], ], ], ]; $request = Type::buildFromArray($request); // Note: It is not possible to get mimeContent with FindItem request // That's why we first get the list of object identifiers and // then call GetItem on each separately. // TODO: It might be feasible to get all properties for object types // for which we don't use MimeContent, for better performance. // Request first page $response = $this->api->getClient()->FindItem($request); + $set = new ItemSet(); + // @phpstan-ignore-next-line - foreach ($response as $item) { + foreach ($response->getItems() as $item) { if ($item = $this->toItem($item, $folder, $existing, $existingIndex)) { - $callback($item); + $set->items[] = $item; + if (count($set->items) == self::CHUNK_SIZE) { + $callback($set); + $set = new ItemSet(); + } } } // Request other pages until we got all while (!$response->isIncludesLastItemInRange()) { // @phpstan-ignore-next-line $response = $this->api->getNextPage($response); - foreach ($response as $item) { + foreach ($response->getItems() as $item) { if ($item = $this->toItem($item, $folder, $existing, $existingIndex)) { - $callback($item); + $set->items[] = $item; + if (count($set->items) == self::CHUNK_SIZE) { + $callback($set); + $set = new ItemSet(); + } } } } + if (count($set->items)) { + $callback($set); + } + // TODO: Delete items that do not exist anymore? } /** * Fetching an item */ public function fetchItem(Item $item): void { // Job processing - initialize environment $this->initEnv($this->engine->queue); if ($driver = EWS\Item::factory($this, $item)) { - $item->filename = $driver->fetchItem($item); + $driver->processItem($item); + return; } - if (empty($item->filename)) { - throw new \Exception("Failed to fetch an item from EWS"); - } + throw new \Exception("Failed to fetch an item from EWS"); } /** * Get the source account */ public function getSourceAccount(): Account { return $this->engine->source; } /** * Get the destination account */ public function getDestinationAccount(): Account { return $this->engine->destination; } /** * Synchronize specified object */ protected function toItem(Type $item, Folder $folder, $existing, $existingIndex): ?Item { $id = $item->getItemId()->toArray(); $exists = null; // Detect an existing item, skip if nothing changed if (isset($existingIndex[$id['Id']])) { $idx = $existingIndex[$id['Id']]; if ($existing[$idx]['changeKey'] == $id['ChangeKey']) { return null; } $exists = $existing[$idx]['href']; } - $item = Item::fromArray([ - 'id' => $id, + if (!EWS\Item::isValidItem($item)) { + return null; + } + + return Item::fromArray([ + 'id' => $id['Id'], 'class' => $item->getItemClass(), 'folder' => $folder, 'existing' => $exists, ]); - - // TODO: We don't need to instantiate Item at this point, instead - // implement EWS\Item::validateClass() method - if ($driver = EWS\Item::factory($this, $item)) { - return $item; - } - - return null; } /** * Set common API options */ protected function apiOptions(array $options): array { if (empty($options['version'])) { $options['version'] = API\ExchangeWebServices::VERSION_2013; } // In debug mode record all responses if (\config('app.debug')) { $options['httpPlayback'] = [ 'mode' => 'record', 'recordLocation' => \storage_path('ews'), ]; } // Options for testing foreach (['httpClient', 'httpPlayback'] as $opt) { if (($val = $this->engine->getOption($opt)) !== null) { $options[$opt] = $val; } } return $options; } /** * Initialize environment for job execution * * @param Queue $queue Queue */ protected function initEnv(Queue $queue): void { $ews = $queue->data['options']['ews']; if (!empty($ews['token'])) { // TODO: Refresh the token if needed $this->api = API::withCallbackToken( $ews['server'], $ews['token'], $this->apiOptions($ews['options']) ); } else { $this->api = API::withUsernameAndPassword( $ews['server'], $this->account->username, $this->account->password, $this->apiOptions($ews['options']) ); } } } diff --git a/src/app/DataMigrator/EWS/Appointment.php b/src/app/DataMigrator/EWS/Appointment.php index f1fbed42..6295e237 100644 --- a/src/app/DataMigrator/EWS/Appointment.php +++ b/src/app/DataMigrator/EWS/Appointment.php @@ -1,102 +1,101 @@ 'calendar:UID']; + $request['ItemShape']['AdditionalProperties']['FieldURI'][] = ['FieldURI' => 'calendar:UID']; return $request; } /** * Process event object */ - protected function processItem(Type $item) + protected function convertItem(Type $item) { // Initialize $this->itemId (for some unit tests) $this->getUID($item); // Decode MIME content $ical = base64_decode((string) $item->getMimeContent()); $itemId = implode("\r\n ", str_split($this->itemId, 75 - strlen('X-MS-ID:'))); $ical = preg_replace('/\r\nBEGIN:VEVENT\r\n/', "\r\nBEGIN:VEVENT\r\nX-MS-ID:{$itemId}\r\n", $ical, 1); // TODO: replace source email with destination email address in ORGANIZER/ATTENDEE // Inject attachment bodies into the iCalendar content // Calendar event attachments are exported as: // ATTACH:CID:81490FBA13A3DC2BF071B894C96B44BA51BEAAED@eurprd05.prod.outlook.com if ($item->getHasAttachments()) { // FIXME: I've tried hard and no matter what ContentId property is always empty // This means we can't match the CID from iCalendar with the attachment. // That's why we'll just remove all ATTACH:CID:... occurrences // and inject attachments to the main event $ical = preg_replace('/\r\nATTACH:CID:[^\r]+\r\n(\r\n [^\r\n]*)?/', '', $ical); foreach ((array) $item->getAttachments()->getFileAttachment() as $attachment) { $_attachment = $this->getAttachment($attachment); $ctype = $_attachment->getContentType(); $body = $_attachment->getContent(); // It looks like Exchange may have an issue with plain text files. // We'll skip empty files if (!strlen($body)) { continue; } // FIXME: This is imo inconsistence on php-ews side that MimeContent // is base64 encoded, but Content isn't // TODO: We should not do it in memory to not exceed the memory limit $body = base64_encode($body); $body = rtrim(chunk_split($body, 74, "\r\n "), ' '); // Inject the attachment at the end of the first VEVENT block // TODO: We should not do it in memory to not exceed the memory limit $append = "ATTACH;VALUE=BINARY;ENCODING=BASE64;FMTTYPE={$ctype}:\r\n {$body}"; $pos = strpos($ical, "\r\nEND:VEVENT"); $ical = substr_replace($ical, $append, $pos + 2, 0); } } return $ical; } /** * Get Item UID (Generate a new one if needed) */ protected function getUID(Type $item): string { // Only appointments have UID property $this->uid = $item->getUID(); // This also sets $this->itemId; return parent::getUID($item); } } diff --git a/src/app/DataMigrator/EWS/Contact.php b/src/app/DataMigrator/EWS/Contact.php index 2df6fe8c..2cbe6d21 100644 --- a/src/app/DataMigrator/EWS/Contact.php +++ b/src/app/DataMigrator/EWS/Contact.php @@ -1,86 +1,86 @@ getMimeContent()); // Remove empty properties that EWS is exporting $vcard = preg_replace('|\n[^:]+:;*\r|', '', $vcard); // Inject UID (and Exchange item ID) to the vCard $uid = $this->getUID($item); $itemId = implode("\r\n ", str_split($this->itemId, 75 - strlen('X-MS-ID:'))); // TODO: Use DAV\Vcard instead of string matching and replacement $vcard = str_replace("BEGIN:VCARD", "BEGIN:VCARD\r\nUID:{$uid}\r\nX-MS-ID:{$itemId}", $vcard); // Note: Looks like PHOTO property is exported properly, so we // don't have to handle attachments as we do for calendar items // TODO: Use vCard v4 for anniversary and spouse? Roundcube works with what's below // Spouse: X-MS-SPOUSE;TYPE=N:Partner Name if (preg_match('/(X-MS-SPOUSE[;:][^\r\n]+)/', $vcard, $matches)) { $spouse = preg_replace('/^[^:]+:/', '', $matches[1]); $vcard = str_replace($matches[1], "X-SPOUSE:{$spouse}", $vcard); } - // TODO: X-MS-ANNIVERSARY;VALUE=DATE:2020-11-12 + // Anniversary: X-MS-ANNIVERSARY;VALUE=DATE:2020-11-12 if (preg_match('/(X-MS-ANNIVERSARY[;:][^\r\n]+)/', $vcard, $matches)) { $date = preg_replace('/^[^:]+:/', '', $matches[1]); $vcard = str_replace($matches[1], "X-ANNIVERSARY:{$date}", $vcard); } // Exchange 2010 for some reason do not include email addresses in the vCard if (!preg_match('/\nEMAIL[^:]*:[^\r\n]+/', $vcard) && ($emailEntries = $item->getEmailAddresses())) { $emails = []; // Note that the Entry property is either an array (multiple addresses) // or an object (single address). Not a great API design. if (!is_array($emailEntries->Entry)) { $emailEntries->Entry = [$emailEntries->Entry]; } foreach ($emailEntries->Entry as $email) { $emails[] = 'EMAIL;TYPE=internet:' . strval($email); } if ($emails) { $vcard = str_replace("BEGIN:VCARD\r\n", "BEGIN:VCARD\r\n" . implode("\r\n", $emails) . "\r\n", $vcard); } } return $vcard; } } diff --git a/src/app/DataMigrator/EWS/DistList.php b/src/app/DataMigrator/EWS/DistList.php index 671cf77b..83ef5ef4 100644 --- a/src/app/DataMigrator/EWS/DistList.php +++ b/src/app/DataMigrator/EWS/DistList.php @@ -1,77 +1,90 @@ 'item:Body']; + + return $request; + } + /** * Convert distribution list object to vCard */ - protected function processItem(Type $item) + protected function convertItem(Type $item) { // Groups (Distribution Lists) are not exported in vCard format, they use eml $data = [ 'UID' => [$this->getUID($item)], 'KIND' => ['group'], 'FN' => [$item->getDisplayName()], 'REV' => [$item->getLastModifiedTime(), ['VALUE' => 'DATE-TIME']], 'X-MS-ID' => [$this->itemId], ]; $vcard = "BEGIN:VCARD\r\nVERSION:4.0\r\nPRODID:Kolab EWS Data Migrator\r\n"; foreach ($data as $key => $prop) { $vcard .= $this->formatProp($key, $prop[0], isset($prop[1]) ? $prop[1] : []); } - // TODO: The group description property in Exchange is not available via EWS XML, - // at least not at outlook.office.com (Exchange 2010). It is available in the - // MimeContent which is in email message format. However, Kolab Webclient does not - // make any use of the NOTE property for contact groups. - // Process list members if ($members = $item->getMembers()) { // The Member property is either array (multiple members) or Type\MemberType // object (a group with just a one member). if (!is_array($members->Member)) { $members->Member = [$members->Member]; } foreach ($members->Member as $member) { $mailbox = $member->getMailbox(); $mailto = $mailbox->getEmailAddress(); $name = $mailbox->getName(); $id = $mailbox->getItemId(); // "mailto:" members are not fully supported by Kolab Webclient. // For members that are contacts (have ItemId specified) we use urn:uuid: // syntax that has good support. if ($id) { $contactUID = sha1($id->toArray()['Id']); $vcard .= $this->formatProp('MEMBER', "urn:uuid:{$contactUID}"); } elseif ($mailto) { if ($name && $name != $mailto) { $mailto = urlencode(sprintf('"%s" <%s>', addcslashes($name, '"'), $mailto)); } $vcard .= $this->formatProp('MEMBER', "mailto:{$mailto}"); } } } + // Note: Kolab Webclient does not make any use of the NOTE property for contact groups + if ($body = (string) $item->getBody()) { + $vcard .= $this->formatProp('NOTE', $body); + } + $vcard .= "END:VCARD\r\n"; return $vcard; } } diff --git a/src/app/DataMigrator/EWS/Item.php b/src/app/DataMigrator/EWS/Item.php index 0ba08c5f..50c6dbd4 100644 --- a/src/app/DataMigrator/EWS/Item.php +++ b/src/app/DataMigrator/EWS/Item.php @@ -1,176 +1,193 @@ driver = $driver; $this->folder = $folder; } /** * Factory method. * Returns object suitable to handle specified item type. */ public static function factory(EWS $driver, ItemInterface $item) { $item_class = str_replace('IPM.', '', $item->class); $item_class = "\App\DataMigrator\EWS\\{$item_class}"; if (class_exists($item_class)) { return new $item_class($driver, $item->folder); } } /** - * Fetch the specified object and put into a file + * Validate that specified EWS Item is of supported type */ - public function fetchItem(ItemInterface $item) + public static function isValidItem(Type $item): bool { - $itemId = $item->id; + $item_class = str_replace('IPM.', '', $item->getItemClass()); + $item_class = "\App\DataMigrator\EWS\\{$item_class}"; + + return class_exists($item_class); + } + + /** + * Process an item (fetch data and convert it) + */ + public function processItem(ItemInterface $item): void + { + $itemId = ['Id' => $item->id]; + + \Log::debug("[EWS] Fetching item {$item->id}..."); // Fetch the item $ewsItem = $this->driver->api->getItem($itemId, $this->getItemRequest()); $uid = $this->getUID($ewsItem); \Log::debug("[EWS] Saving item {$uid}..."); // Apply type-specific format converters - $content = $this->processItem($ewsItem); + $content = $this->convertItem($ewsItem); if (!is_string($content)) { - return; + throw new \Exception("Failed to fetch EWS item {$this->itemId}"); } - $uid = preg_replace('/[^a-zA-Z0-9_:@-]/', '', $uid); - - $location = $this->folder->location; + $filename = $uid . '.' . $this->fileExtension(); - if (!file_exists($location)) { - mkdir($location, 0740, true); - } - - $location .= '/' . $uid . '.' . $this->fileExtension(); + if (strlen($content) > Engine::MAX_ITEM_SIZE) { + $location = $this->folder->tempFileLocation($filename); - file_put_contents($location, $content); + if (file_put_contents($location, $content) === false) { + throw new \Exception("Failed to write to file at {$location}"); + } - return $location; + $item->filename = $location; + } else { + $item->content = $content; + $item->filename = $filename; + } } /** * Item conversion code */ - abstract protected function processItem(Type $item); + abstract protected function convertItem(Type $item); /** * Get GetItem request parameters */ - protected function getItemRequest(): array + protected static function getItemRequest(): array { $request = [ 'ItemShape' => [ // Reqest default set of properties 'BaseShape' => 'Default', // Additional properties, e.g. LastModifiedTime - // FIXME: How to add multiple properties here? 'AdditionalProperties' => [ - 'FieldURI' => ['FieldURI' => 'item:LastModifiedTime'], - ] + 'FieldURI' => [ + ['FieldURI' => 'item:LastModifiedTime'], + ], + ], ] ]; return $request; } /** * Fetch attachment object from Exchange */ protected function getAttachment(Type\FileAttachmentType $attachment) { $request = [ 'AttachmentIds' => [ $attachment->getAttachmentId()->toXmlObject() ], 'AttachmentShape' => [ 'IncludeMimeContent' => true, ] ]; return $this->driver->api->getClient()->GetAttachment($request); } /** * Get Item UID (Generate a new one if needed) */ protected function getUID(Type $item): string { $itemId = $item->getItemId()->toArray(); if ($this->uid === null) { // Tasks, contacts, distlists do not have an UID. We have to generate one // and inject it into the output file. // We'll use the ItemId (excluding the ChangeKey part) as a base for the UID, // this way we can e.g. get distlist members references working. $this->uid = sha1($itemId['Id']); // $this->uid = \App\Utils::uuidStr(); } $this->itemId = implode('!', $itemId); return $this->uid; } /** * Filename extension for cached file in-processing */ protected function fileExtension(): string { return constant(static::class . '::FILE_EXT') ?: 'txt'; } /** * VCard/iCal property formatting */ protected function formatProp($name, $value, array $params = []): string { $cal = new \Sabre\VObject\Component\VCalendar(); $prop = new \Sabre\VObject\Property\Text($cal, $name, $value, $params); $value = $prop->serialize(); // Revert escaping for some props if ($name == 'RRULE') { $value = str_replace("\\", '', $value); } return $value; } } diff --git a/src/app/DataMigrator/EWS/Note.php b/src/app/DataMigrator/EWS/Note.php index b8120b8d..abb1607b 100644 --- a/src/app/DataMigrator/EWS/Note.php +++ b/src/app/DataMigrator/EWS/Note.php @@ -1,41 +1,41 @@ getMimeContent()); return $email; } } diff --git a/src/app/DataMigrator/EWS/Task.php b/src/app/DataMigrator/EWS/Task.php index 76dc6b4a..e7f46e37 100644 --- a/src/app/DataMigrator/EWS/Task.php +++ b/src/app/DataMigrator/EWS/Task.php @@ -1,319 +1,319 @@ [$this->getUID($item)], 'DTSTAMP' => [$this->formatDate($item->getLastModifiedTime()), ['VALUE' => 'DATE-TIME']], 'CREATED' => [$this->formatDate($item->getDateTimeCreated()), ['VALUE' => 'DATE-TIME']], 'SEQUENCE' => [intval($item->getChangeCount())], 'SUMMARY' => [$item->getSubject()], 'DESCRIPTION' => [(string) $item->getBody()], 'PERCENT-COMPLETE' => [intval($item->getPercentComplete())], 'X-MS-ID' => [$this->itemId], ]; if ($dueDate = $item->getDueDate()) { $data['DUE'] = [$this->formatDate($dueDate), ['VALUE' => 'DATE-TIME']]; } if ($startDate = $item->getStartDate()) { $data['DTSTART'] = [$this->formatDate($startDate), ['VALUE' => 'DATE-TIME']]; } if ($status = $item->getStatus()) { $status = strtoupper($status); $status_map = [ 'COMPLETED' => 'COMPLETED', 'INPROGRESS' => 'IN-PROGRESS', 'DEFERRED' => 'X-DEFERRED', 'NOTSTARTED' => 'X-NOTSTARTED', 'WAITINGONOTHERS' => 'X-WAITINGFOROTHERS', ]; if (isset($status_map[$status])) { $data['STATUS'] = [$status_map[$status]]; } } if (($categories = $item->getCategories()) && $categories->String) { $data['CATEGORIES'] = [$categories->String]; } if ($sensitivity = $item->getSensitivity()) { $sensitivity_map = [ 'CONFIDENTIAL' => 'CONFIDENTIAL', 'NORMAL' => 'PUBLIC', 'PERSONAL' => 'PUBLIC', 'PRIVATE' => 'PRIVATE', ]; $data['CLASS'] = [$sensitivity_map[strtoupper($sensitivity)] ?? 'PUBLIC']; } if ($importance = $item->getImportance()) { $importance_map = [ 'HIGH' => '9', 'NORMAL' => '5', 'LOW' => '1', ]; $data['PRIORITY'] = [$importance_map[strtoupper($importance)] ?? '0']; } $this->setTaskOrganizer($data, $item); $this->setTaskRecurrence($data, $item); $ical = "BEGIN:VCALENDAR\r\nMETHOD:PUBLISH\r\nVERSION:2.0\r\nPRODID:Kolab EWS Data Migrator\r\nBEGIN:VTODO\r\n"; foreach ($data as $key => $prop) { $ical .= $this->formatProp($key, $prop[0], isset($prop[1]) ? $prop[1] : []); } // Attachments if ($item->getHasAttachments()) { foreach ((array) $item->getAttachments()->getFileAttachment() as $attachment) { $_attachment = $this->getAttachment($attachment); $ctype = $_attachment->getContentType(); $body = $_attachment->getContent(); // It looks like Exchange may have an issue with plain text files. // We'll skip empty files if (!strlen($body)) { continue; } // FIXME: This is imo inconsistence on php-ews side that MimeContent // is base64 encoded, but Content isn't // TODO: We should not do it in memory to not exceed the memory limit $body = base64_encode($body); $body = rtrim(chunk_split($body, 74, "\r\n "), ' '); // Inject the attachment at the end of the VTODO block // TODO: We should not do it in memory to not exceed the memory limit $ical .= "ATTACH;VALUE=BINARY;ENCODING=BASE64;FMTTYPE={$ctype}:\r\n {$body}"; } } $ical .= $this->getVAlarm($item); $ical .= "END:VTODO\r\n"; $ical .= "END:VCALENDAR\r\n"; return $ical; } /** * Set task organizer/attendee */ protected function setTaskOrganizer(array &$data, Type $task) { // FIXME: Looks like the owner might be an email address or just a full user name $owner = $task->getOwner(); $source = $this->driver->getSourceAccount(); $destination = $this->driver->getDestinationAccount(); if (strpos($owner, '@') && $owner != $source->email) { // Task owned by another person $data['ORGANIZER'] = ["mailto:{$owner}"]; // FIXME: Because attendees are not specified in EWS, assume the user is an attendee if ($destination->email) { $params = ['ROLE' => 'REQ-PARTICIPANT', 'CUTYPE' => 'INDIVIDUAL']; $data['ATTENDEE'] = ["mailto:{$destination->email}", $params]; } return; } // Otherwise it must be owned by the user if ($destination->email) { $data['ORGANIZER'] = ["mailto:{$destination->email}"]; } } /** * Set task recurrence rule */ protected function setTaskRecurrence(array &$data, Type $task) { if (empty($task->getIsRecurring()) || empty($task->getRecurrence())) { return; } $r = $task->getRecurrence(); $rrule = []; if ($recurrence = $r->getDailyRecurrence()) { $rrule['FREQ'] = 'DAILY'; $rrule['INTERVAL'] = $recurrence->getInterval() ?: 1; } elseif ($recurrence = $r->getWeeklyRecurrence()) { $rrule['FREQ'] = 'WEEKLY'; $rrule['INTERVAL'] = $recurrence->getInterval() ?: 1; $rrule['BYDAY'] = $this->mapDays($recurrence->getDaysOfWeek()); $rrule['WKST'] = $this->mapDays($recurrence->getFirstDayOfWeek()); } elseif ($recurrence = $r->getAbsoluteMonthlyRecurrence()) { $rrule['FREQ'] = 'MONTHLY'; $rrule['INTERVAL'] = $recurrence->getInterval() ?: 1; $rrule['BYMONTHDAY'] = $recurrence->getDayOfMonth(); } elseif ($recurrence = $r->getRelativeMonthlyRecurrence()) { $rrule['FREQ'] = 'MONTHLY'; $rrule['INTERVAL'] = $recurrence->getInterval() ?: 1; $rrule['BYDAY'] = $this->mapDays($recurrence->getDaysOfWeek(), $recurrence->getDayOfWeekIndex()); } elseif ($recurrence = $r->getAbsoluteYearlyRecurrence()) { $rrule['FREQ'] = 'YEARLY'; $rrule['BYMONTH'] = $this->mapMonths($recurrence->getMonth()); $rrule['BYMONTHDAY'] = $recurrence->getDayOfMonth(); } elseif ($recurrence = $r->getRelativeYearlyRecurrence()) { $rrule['FREQ'] = 'YEARLY'; $rrule['BYMONTH'] = $this->mapMonths($recurrence->getMonth()); $rrule['BYDAY'] = $this->mapDays($recurrence->getDaysOfWeek(), $recurrence->getDayOfWeekIndex()); } else { // There might be *Regeneration rules that we don't support \Log::debug("[EWS] Unsupported Recurrence property value. Ignored."); } if (!empty($rrule)) { if ($recurrence = $r->getNumberedRecurrence()) { $rrule['COUNT'] = $recurrence->getNumberOfOccurrences(); } elseif ($recurrence = $r->getEndDateRecurrence()) { $rrule['UNTIL'] = $this->formatDate($recurrence->getEndDate()); } $rrule = array_filter($rrule); $rrule = trim(array_reduce( array_keys($rrule), function ($carry, $key) use ($rrule) { return $carry . ';' . $key . '=' . $rrule[$key]; } ), ';'); $data['RRULE'] = [$rrule]; } } /** * Get VALARM block for the task Reminder */ protected function getVAlarm(Type $task): string { // FIXME: To me it looks like ReminderMinutesBeforeStart property is not used $date = $this->formatDate($task->getReminderDueBy()); if (empty($task->getReminderIsSet()) || empty($date)) { return ''; } return "BEGIN:VALARM\r\n" . "ACTION:DISPLAY\r\nTRIGGER;VALUE=DATE-TIME:{$date}\r\n" . "END:VALARM\r\n"; } /** * Convert EWS representation of recurrence days to iCal */ protected function mapDays(string $days, string $index = ''): string { if (preg_match('/(Day|Weekday|WeekendDay)/', $days)) { // not supported return ''; } $days_map = [ 'Sunday' => 'SU', 'Monday' => 'MO', 'Tuesday' => 'TU', 'Wednesday' => 'WE', 'Thursday' => 'TH', 'Friday' => 'FR', 'Saturday' => 'SA', ]; $index_map = [ 'First' => 1, 'Second' => 2, 'Third' => 3, 'Fourth' => 4, 'Last' => -1, ]; $days = explode(' ', $days); $days = array_map( function ($day) use ($days_map, $index_map, $index) { return ($index ? $index_map[$index] : '') . $days_map[$day]; }, $days ); return implode(',', $days); } /** * Convert EWS representation of recurrence month to iCal */ protected function mapMonths(string $months): string { $months_map = ['January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', 'September', 'October', 'November', 'December']; $months = explode(' ', $months); $months = array_map( function ($month) use ($months_map) { return array_search($month, $months_map) + 1; }, $months ); return implode(',', $months); } /** * Format EWS date-time into a iCalendar date-time */ protected function formatDate($datetime) { if (empty($datetime)) { return null; } return str_replace(['Z', '-', ':'], '', $datetime); } } diff --git a/src/app/DataMigrator/Engine.php b/src/app/DataMigrator/Engine.php index c238d307..b9f606fa 100644 --- a/src/app/DataMigrator/Engine.php +++ b/src/app/DataMigrator/Engine.php @@ -1,347 +1,361 @@ source = $source; $this->destination = $destination; $this->options = $options; // Create a unique identifier for the migration request $queue_id = md5(strval($source) . strval($destination) . ($options['type'] ?? '')); // TODO: When running in 'sync' mode we shouldn't create a queue at all // If queue exists, we'll display the progress only if ($queue = Queue::find($queue_id)) { // If queue contains no jobs, assume invalid // TODO: An better API to manage (reset) queues if (!$queue->jobs_started || !empty($options['force'])) { $queue->delete(); } else { while (true) { $this->debug(sprintf("Progress [%d of %d]\n", $queue->jobs_finished, $queue->jobs_started)); if ($queue->jobs_started == $queue->jobs_finished) { break; } sleep(1); $queue->refresh(); } return; } } // Initialize the source $this->exporter = $this->initDriver($source, ExporterInterface::class); $this->exporter->authenticate(); // Initialize the destination $this->importer = $this->initDriver($destination, ImporterInterface::class); $this->importer->authenticate(); // Create a queue $this->createQueue($queue_id); - // We'll store output in storage/ tree + // We'll store temp files in storage/ tree $location = storage_path('export/') . $source->email; if (!file_exists($location)) { mkdir($location, 0740, true); } - $types = preg_split('/\s*,\s*/', strtolower($options['type'] ?? '')); + $types = empty($options['type']) ? [] : preg_split('/\s*,\s*/', strtolower($options['type'])); $this->debug("Fetching folders hierarchy..."); $folders = $this->exporter->getFolders($types); $count = 0; $async = empty($options['sync']); foreach ($folders as $folder) { $this->debug("Processing folder {$folder->fullname}..."); $folder->queueId = $queue_id; $folder->location = $location; if ($async) { // Dispatch the job (for async execution) Jobs\FolderJob::dispatch($folder); $count++; } else { $this->processFolder($folder); } } if ($count) { $this->queue->bumpJobsStarted($count); } if ($async) { $this->debug(sprintf('Done. %d %s created in queue: %s.', $count, Str::plural('job', $count), $queue_id)); } else { $this->debug(sprintf('Done (queue: %s).', $queue_id)); } } /** * Processing of a folder synchronization */ public function processFolder(Folder $folder): void { // Job processing - initialize environment if (!$this->queue) { $this->envFromQueue($folder->queueId); } // Create the folder on the destination server $this->importer->createFolder($folder); $count = 0; $async = empty($this->options['sync']); // Fetch items from the source $this->exporter->fetchItemList( $folder, function ($item_or_set) use (&$count, $async) { if ($async) { // Dispatch the job (for async execution) if ($item_or_set instanceof ItemSet) { Jobs\ItemSetJob::dispatch($item_or_set); } else { Jobs\ItemJob::dispatch($item_or_set); } $count++; } else { if ($item_or_set instanceof ItemSet) { $this->processItemSet($item_or_set); } else { $this->processItem($item_or_set); } } }, $this->importer ); if ($count) { $this->queue->bumpJobsStarted($count); } if ($async) { $this->queue->bumpJobsFinished(); } } /** * Processing of item synchronization */ public function processItem(Item $item): void { // Job processing - initialize environment if (!$this->queue) { $this->envFromQueue($item->folder->queueId); } $this->exporter->fetchItem($item); $this->importer->createItem($item); - if (!empty($item->filename)) { - unlink($item->filename); + if (!empty($item->filename) && str_starts_with($item->filename, storage_path('export/'))) { + @unlink($item->filename); } if (empty($this->options['sync'])) { $this->queue->bumpJobsFinished(); } } /** * Processing of item-set synchronization */ public function processItemSet(ItemSet $set): void { // Job processing - initialize environment if (!$this->queue) { $this->envFromQueue($set->items[0]->folder->queueId); } - // TODO: Some exporters, e.g. DAV, might optimize fetching multiple items in one go, - // we'll need a new API to do that - - foreach ($set->items as $item) { - $this->exporter->fetchItem($item); + $importItem = function (Item $item) { $this->importer->createItem($item); - if (!empty($item->filename)) { - unlink($item->filename); + if (!empty($item->filename) && str_starts_with($item->filename, storage_path('export/'))) { + @unlink($item->filename); + } + }; + + // Some exporters, e.g. DAV, might optimize fetching multiple items in one go + if ($this->exporter instanceof FetchItemSetInterface) { + $this->exporter->fetchItemSet($set, $importItem); + } else { + foreach ($set->items as $item) { + $this->exporter->fetchItem($item); + $importItem($item); } } // TODO: We should probably also track number of items migrated if (empty($this->options['sync'])) { $this->queue->bumpJobsFinished(); } } /** * Print progress/debug information */ public function debug($line) { if (!empty($this->options['stdout'])) { $output = new \Symfony\Component\Console\Output\ConsoleOutput(); $output->writeln("$line"); } else { \Log::debug("[DataMigrator] $line"); } } /** * Get migration option value. */ public function getOption(string $name) { return $this->options[$name] ?? null; } /** * Set migration queue option. Use this if you need to pass * some data between queue processes. */ public function setOption(string $name, $value): void { $this->options[$name] = $value; if ($this->queue) { $this->queue->data = $this->queueData(); $this->queue->save(); } } /** * Create a queue for the request * * @param string $queue_id Unique queue identifier */ protected function createQueue(string $queue_id): void { $this->queue = new Queue(); $this->queue->id = $queue_id; $this->queue->data = $this->queueData(); $this->queue->save(); } /** * Prepare queue data */ protected function queueData() { $options = $this->options; unset($options['stdout']); // jobs aren't in stdout anymore // TODO: data should be encrypted return [ 'source' => (string) $this->source, 'destination' => (string) $this->destination, 'options' => $options, ]; } /** * Initialize environment for job execution * * @param string $queueId Queue identifier */ protected function envFromQueue(string $queueId): void { $this->queue = Queue::findOrFail($queueId); $this->source = new Account($this->queue->data['source']); $this->destination = new Account($this->queue->data['destination']); $this->options = $this->queue->data['options']; $this->importer = $this->initDriver($this->destination, ImporterInterface::class); $this->exporter = $this->initDriver($this->source, ExporterInterface::class); } /** * Initialize (and select) migration driver */ protected function initDriver(Account $account, string $interface) { switch ($account->scheme) { case 'ews': $driver = new EWS($account, $this); break; case 'dav': case 'davs': $driver = new DAV($account, $this); break; case 'imap': case 'imaps': case 'tls': case 'ssl': $driver = new IMAP($account, $this); break; + case 'test': + $driver = new Test($account, $this); + break; + default: throw new \Exception("Failed to init driver for '{$account->scheme}'"); } // Make sure driver is used in the direction it supports if (!is_a($driver, $interface)) { throw new \Exception(sprintf( "'%s' driver does not implement %s", class_basename($driver), class_basename($interface) )); } return $driver; } } diff --git a/src/app/DataMigrator/IMAP.php b/src/app/DataMigrator/IMAP.php index 14d105b2..34b85e04 100644 --- a/src/app/DataMigrator/IMAP.php +++ b/src/app/DataMigrator/IMAP.php @@ -1,437 +1,450 @@ account = $account; $this->engine = $engine; // TODO: Move this to self::authenticate()? $config = self::getConfig($account->username, $account->password, $account->uri); $this->imap = self::initIMAP($config); } /** * Authenticate */ public function authenticate(): void { } /** * Create a folder. * * @param Folder $folder Folder data * * @throws \Exception on error */ public function createFolder(Folder $folder): void { if ($folder->type != 'mail') { throw new \Exception("IMAP does not support folder of type {$folder->type}"); } if ($folder->fullname == 'INBOX') { // INBOX always exists return; } if (!$this->imap->createFolder($folder->fullname)) { \Log::warning("Failed to create the folder: {$this->imap->error}"); if (str_contains($this->imap->error, "Mailbox already exists")) { // Not an error } else { throw new \Exception("Failed to create an IMAP folder {$folder->fullname}"); } } // TODO: Migrate folder subscription state } /** * Create an item in a folder. * * @param Item $item Item to import * * @throws \Exception */ public function createItem(Item $item): void { $mailbox = $item->folder->fullname; - if ($item->filename) { + if (strlen($item->content)) { + $result = $this->imap->append( + $mailbox, + $item->content, + $item->data['flags'], + $item->data['internaldate'], + true + ); + + if ($result === false) { + throw new \Exception("Failed to append IMAP message into {$mailbox}"); + } + } elseif ($item->filename) { $result = $this->imap->appendFromFile( $mailbox, $item->filename, null, $item->data['flags'], $item->data['internaldate'], true ); if ($result === false) { throw new \Exception("Failed to append IMAP message into {$mailbox}"); } } // When updating an existing email message we have to... if ($item->existing) { if (!empty($result)) { // Remove the old one $this->imap->flag($mailbox, $item->existing['uid'], 'DELETED'); $this->imap->expunge($mailbox, $item->existing['uid']); } else { // Update flags foreach ($item->existing['flags'] as $flag) { if (!in_array($flag, $item->data['flags'])) { $this->imap->unflag($mailbox, $item->existing['uid'], $flag); } } foreach ($item->data['flags'] as $flag) { if (!in_array($flag, $item->existing['flags'])) { $this->imap->flag($mailbox, $item->existing['uid'], $flag); } } } } } /** * Fetching an item */ public function fetchItem(Item $item): void { [$uid, $messageId] = explode(':', $item->id, 2); $mailbox = $item->folder->fullname; // Get message flags $header = $this->imap->fetchHeader($mailbox, (int) $uid, true, false, ['FLAGS']); if ($header === false) { throw new \Exception("Failed to get IMAP message headers for {$mailbox}/{$uid}"); } // Remove flags that we can't append (e.g. RECENT) $flags = $this->filterImapFlags(array_keys($header->flags)); // If message already exists in the destination account we should update only flags // and be done with it. On the other hand for Drafts it's not unusual to get completely // different body for the same Message-ID. Same can happen not only in Drafts, I suppose. // So, we compare size and INTERNALDATE timestamp. if ( !$item->existing || $header->timestamp != $item->existing['timestamp'] || $header->size != $item->existing['size'] ) { - // Save the message content to a file - $location = $item->folder->location; - - if (!file_exists($location)) { - mkdir($location, 0740, true); - } + // Handle message content in memory (up to 20MB), bigger messages will use a temp file + if ($header->size > Engine::MAX_ITEM_SIZE) { + // Save the message content to a file + $location = $item->folder->tempFileLocation($uid . '.eml'); - // TODO: What if parent folder not yet exists? - $location .= '/' . $uid . '.eml'; + $fp = fopen($location, 'w'); - // TODO: We should consider streaming the message, it should be possible - // with append() and handlePartBody(), but I don't know if anyone tried that. - - $fp = fopen($location, 'w'); + if (!$fp) { + throw new \Exception("Failed to open 'php://temp' stream"); + } - if (!$fp) { - throw new \Exception("Failed to write to {$location}"); + $result = $this->imap->handlePartBody($mailbox, $uid, true, '', null, null, $fp); + } else { + $result = $this->imap->handlePartBody($mailbox, $uid, true); } - $result = $this->imap->handlePartBody($mailbox, $uid, true, '', null, null, $fp); - if ($result === false) { - fclose($fp); + if (!empty($fp)) { + fclose($fp); + } + throw new \Exception("Failed to fetch IMAP message for {$mailbox}/{$uid}"); } - $item->filename = $location; - - fclose($fp); + if (!empty($fp) && !empty($location)) { + $item->filename = $location; + fclose($fp); + } else { + $item->content = $result; + } } $item->data = [ 'flags' => $flags, 'internaldate' => $header->internaldate, ]; } /** * Fetch a list of folder items */ public function fetchItemList(Folder $folder, $callback, ImporterInterface $importer): void { // Get existing messages' headers from the destination mailbox $existing = $importer->getItems($folder); $mailbox = $folder->fullname; // TODO: We should probably first use SEARCH/SORT to skip messages marked as \Deleted // It would also allow us to get headers in chunks 200 messages at a time, or so. // TODO: fetchHeaders() fetches too many headers, we should slim-down, here we need // only UID FLAGS INTERNALDATE BODY.PEEK[HEADER.FIELDS (DATE FROM MESSAGE-ID)] $messages = $this->imap->fetchHeaders($mailbox, '1:*', true, false, ['Message-Id']); if ($messages === false) { throw new \Exception("Failed to get all IMAP message headers for {$mailbox}"); } if (empty($messages)) { \Log::debug("Nothing to migrate for {$mailbox}"); return; } $set = new ItemSet(); foreach ($messages as $message) { // If Message-Id header does not exist create it based on internaldate/From/Date $id = $this->getMessageId($message, $mailbox); // Skip message that exists and did not change $exists = null; if (isset($existing[$id])) { $flags = $this->filterImapFlags(array_keys($message->flags)); if ( $flags == $existing[$id]['flags'] && $message->timestamp == $existing[$id]['timestamp'] && $message->size == $existing[$id]['size'] ) { continue; } $exists = $existing[$id]; } $set->items[] = Item::fromArray([ 'id' => $message->uid . ':' . $id, 'folder' => $folder, 'existing' => $exists, ]); if (count($set->items) == self::CHUNK_SIZE) { $callback($set); $set = new ItemSet(); } } if (count($set->items)) { $callback($set); } // TODO: Delete messages that do not exist anymore? } /** * Get folders hierarchy */ public function getFolders($types = []): array { $folders = $this->imap->listMailboxes('', ''); if ($folders === false) { throw new \Exception("Failed to get list of IMAP folders"); } // TODO: Migrate folder subscription state $result = []; foreach ($folders as $folder) { if ($this->shouldSkip($folder)) { \Log::debug("Skipping folder {$folder}."); continue; } $result[] = Folder::fromArray([ 'fullname' => $folder, 'type' => 'mail' ]); } return $result; } /** * Get a list of folder items, limited to their essential propeties * used in incremental migration to skip unchanged items. */ public function getItems(Folder $folder): array { $mailbox = $folder->fullname; // TODO: We should probably first use SEARCH/SORT to skip messages marked as \Deleted // TODO: fetchHeaders() fetches too many headers, we should slim-down, here we need // only UID FLAGS INTERNALDATE BODY.PEEK[HEADER.FIELDS (DATE FROM MESSAGE-ID)] $messages = $this->imap->fetchHeaders($mailbox, '1:*', true, false, ['Message-Id']); if ($messages === false) { throw new \Exception("Failed to get IMAP message headers in {$mailbox}"); } $result = []; foreach ($messages as $message) { // Remove flags that we can't append (e.g. RECENT) $flags = $this->filterImapFlags(array_keys($message->flags)); // Generate message ID if the header does not exist $id = $this->getMessageId($message, $mailbox); $result[$id] = [ 'uid' => $message->uid, 'flags' => $flags, 'size' => $message->size, 'timestamp' => $message->timestamp, ]; } return $result; } /** * Initialize IMAP connection and authenticate the user */ private static function initIMAP(array $config, string $login_as = null): \rcube_imap_generic { $imap = new \rcube_imap_generic(); if (\config('app.debug')) { $imap->setDebug(true, 'App\Backends\IMAP::logDebug'); } if ($login_as) { $config['options']['auth_cid'] = $config['user']; $config['options']['auth_pw'] = $config['password']; $config['options']['auth_type'] = 'PLAIN'; $config['user'] = $login_as; } $imap->connect($config['host'], $config['user'], $config['password'], $config['options']); if (!$imap->connected()) { $message = sprintf("Login failed for %s against %s. %s", $config['user'], $config['host'], $imap->error); \Log::error($message); throw new \Exception("Connection to IMAP failed"); } return $imap; } /** * Get IMAP configuration */ private static function getConfig($user, $password, $uri): array { $uri = \parse_url($uri); $default_port = 143; $ssl_mode = null; if (isset($uri['scheme'])) { if (preg_match('/^(ssl|imaps)/', $uri['scheme'])) { $default_port = 993; $ssl_mode = 'ssl'; } elseif ($uri['scheme'] === 'tls') { $ssl_mode = 'tls'; } } $config = [ 'host' => $uri['host'], 'user' => $user, 'password' => $password, 'options' => [ 'port' => !empty($uri['port']) ? $uri['port'] : $default_port, 'ssl_mode' => $ssl_mode, 'socket_options' => [ 'ssl' => [ // TODO: These configuration options make sense for "local" Kolab IMAP, // but when connecting to external one we might want to just disable // cert validation, or make it optional via Account URI parameters 'verify_peer' => \config('imap.verify_peer'), 'verify_peer_name' => \config('imap.verify_peer'), 'verify_host' => \config('imap.verify_host') ], ], ], ]; return $config; } /** * Limit IMAP flags to these that can be migrated */ private function filterImapFlags($flags) { // TODO: Support custom flags migration return array_filter( $flags, function ($flag) { return isset($this->imap->flags[$flag]); } ); } /** * Check if the folder should not be migrated */ private function shouldSkip($folder): bool { // TODO: This should probably use NAMESPACE information if (preg_match('~(Shared Folders|Other Users)/.*~', $folder)) { return true; } return false; } /** * Return Message-Id, generate unique identifier if Message-Id does not exist */ private function getMessageId($message, $folder): string { if (!empty($message->messageID)) { return $message->messageID; } return md5($folder . $message->from . ($message->date ?: $message->timestamp)); } } diff --git a/src/app/DataMigrator/Interface/FetchItemSetInterface.php b/src/app/DataMigrator/Interface/FetchItemSetInterface.php new file mode 100644 index 00000000..6137bcb9 --- /dev/null +++ b/src/app/DataMigrator/Interface/FetchItemSetInterface.php @@ -0,0 +1,14 @@ + $value) { $obj->{$key} = $value; } return $obj; } + + /** + * Returns location of a temp file for an Item content + */ + public function tempFileLocation(string $filename): string + { + $filename = preg_replace('/[^a-zA-Z0-9_:@.-]/', '', $filename); + + $location = $this->location; + + // TODO: What if parent folder not yet exists? + if (!file_exists($location)) { + mkdir($location, 0740, true); + } + + $location .= '/' . $filename; + + return $location; + } } diff --git a/src/app/DataMigrator/Interface/Item.php b/src/app/DataMigrator/Interface/Item.php index 061c494a..e7995324 100644 --- a/src/app/DataMigrator/Interface/Item.php +++ b/src/app/DataMigrator/Interface/Item.php @@ -1,44 +1,50 @@ $value) { $obj->{$key} = $value; } return $obj; } } diff --git a/src/app/DataMigrator/Interface/ItemSet.php b/src/app/DataMigrator/Interface/ItemSet.php index a3b5c5ac..001975ac 100644 --- a/src/app/DataMigrator/Interface/ItemSet.php +++ b/src/app/DataMigrator/Interface/ItemSet.php @@ -1,28 +1,45 @@ Items list */ public $items = []; - // TODO: Every item has a $folder property, this makes the set - // needlesly big when serialized. We should probably store $folder - // once with the set and remove it from an item on serialize - // and back in unserialize. - /** * Create an ItemSet instance */ public static function set(array $items = []): ItemSet { $obj = new self(); $obj->items = $items; return $obj; } + + public function serialize(): ?string + { + // Every item has a Folder property, this makes the set + // needlesly big when serialized. Make the size more compact. + $folder = count($this->items) ? $this->items[0]->folder : null; + + foreach ($this->items as $item) { + $item->folder = null; + } + + return serialize([$folder, $this->items]); + } + + public function unserialize(string $data): void + { + [$folder, $this->items] = unserialize($data); + + foreach ($this->items as $item) { + $item->folder = $folder; + } + } } diff --git a/src/app/DataMigrator/Test.php b/src/app/DataMigrator/Test.php new file mode 100644 index 00000000..6fedbbee --- /dev/null +++ b/src/app/DataMigrator/Test.php @@ -0,0 +1,153 @@ +engine = $engine; + $this->account = $account; + } + + public static function init($folders) + { + self::$folders = $folders; + + self::$fetchedItems = []; + self::$createdItems = []; + self::$createdFolders = []; + } + + /** + * Check user credentials. + * + * @throws \Exception + */ + public function authenticate(): void + { + } + + /** + * Create an item in a folder. + * + * @param Item $item Item to import + * + * @throws \Exception + */ + public function createItem(Item $item): void + { + self::$createdItems[] = $item; + } + + /** + * Create a folder. + * + * @param Folder $folder Folder data + * + * @throws \Exception on error + */ + public function createFolder(Folder $folder): void + { + self::$createdFolders[] = $folder; + } + + /** + * Fetching an item + */ + public function fetchItem(Item $item): void + { + $item->content = 'content'; + $item->filename = 'test.eml'; + + self::$fetchedItems[] = $item; + } + + /** + * Fetch a list of folder items + */ + public function fetchItemList(Folder $folder, $callback, ImporterInterface $importer): void + { + // Get existing messages' headers from the destination mailbox + $existing = $importer->getItems($folder); + + $set = new ItemSet(); + + foreach ((self::$folders[$folder->id]['items'] ?? []) as $itemId => $item) { + $exists = null; // TODO + + $item['id'] = $itemId; + $item['folder'] = $folder; + $item['existing'] = $exists; + + $set->items[] = Item::fromArray($item); + + if (count($set->items) == self::CHUNK_SIZE) { + $callback($set); + $set = new ItemSet(); + } + } + + if (count($set->items)) { + $callback($set); + } + } + + /** + * Get a list of items, limited to their essential propeties + * used in incremental migration. + * + * @param Folder $folder Folder data + * + * @throws \Exception on error + */ + public function getItems(Folder $folder): array + { + return self::$folders[$folder->id]['existing_items'] ?? []; + } + + /** + * Get folders hierarchy + */ + public function getFolders($types = []): array + { + $result = []; + + foreach (self::$folders as $folderId => $folder) { + // Skip folder types we do not support (need) + if (!empty($types) && !in_array($folder['type'], $types)) { + continue; + } + + $folder['id'] = $folderId; + $folder['total'] = count($folder['items']); + + $result[] = Folder::fromArray($folder); + } + + return $result; + } +} diff --git a/src/tests/Feature/DataMigrator/EngineTest.php b/src/tests/Feature/DataMigrator/EngineTest.php index 1b8440a2..4d323116 100644 --- a/src/tests/Feature/DataMigrator/EngineTest.php +++ b/src/tests/Feature/DataMigrator/EngineTest.php @@ -1,47 +1,167 @@ [ + 'type' => Engine::TYPE_MAIL, + 'name' => 'Inbox', + 'fullname' => 'Inbox', + 'items' => [ + 'm1' => [], + 'm2' => [], + 'm3' => [], + 'm4' => [], + ], + 'existing_items' => [ + ], + ], + 'Contacts' => [ + 'type' => Engine::TYPE_CONTACT, + 'name' => 'Contacts', + 'fullname' => 'Contacts', + 'items' => [ + 'c1' => [], + 'c2' => [], + ], + 'existing_items' => [ + ], + ], + 'Calendar' => [ + 'type' => Engine::TYPE_EVENT, + 'name' => 'Calendar', + 'fullname' => 'Calendar', + 'items' => [ + 'e1' => [], + 'e2' => [], + ], + 'existing_items' => [ + ], + ], + ]; + /** * {@inheritDoc} */ public function setUp(): void { parent::setUp(); MigratorQueue::truncate(); } /** * {@inheritDoc} */ public function tearDown(): void { MigratorQueue::truncate(); parent::tearDown(); } /** * Test asynchronous migration */ public function testAsyncMigration(): void { + $source = new Account('test://test%40domain.tld:test@test'); + $destination = new Account('test://test%40kolab.org:test@test'); + $engine = new Engine(); + + Test::init($this->data); + Queue::fake(); + + // Migration initial run + $engine->migrate($source, $destination, []); + + $queue = MigratorQueue::first(); + + Queue::assertPushed(FolderJob::class, 3); + Queue::assertPushed( + FolderJob::class, + function ($job) use ($queue) { + $folder = TestCase::getObjectProperty($job, 'folder'); + return $folder->id === 'Inbox' && $folder->queueId = $queue->id; + } + ); + Queue::assertPushed( + FolderJob::class, + function ($job) use ($queue) { + $folder = TestCase::getObjectProperty($job, 'folder'); + return $folder->id === 'Contacts' && $folder->queueId = $queue->id; + } + ); + Queue::assertPushed( + FolderJob::class, + function ($job) use ($queue) { + $folder = TestCase::getObjectProperty($job, 'folder'); + return $folder->id === 'Calendar' && $folder->queueId = $queue->id; + } + ); + + $this->assertCount(0, Test::$createdFolders); + $this->assertSame(3, $queue->jobs_started); + $this->assertSame(0, $queue->jobs_finished); + $this->assertSame([], $queue->data['options']); + // TODO: Assert Source and destination in the queue + // TODO: Test 'force' option, test executing with an existing queue + // TODO: Test jobs execution $this->markTestIncomplete(); } /** * Test synchronous migration */ public function testSyncMigration(): void { - $this->markTestIncomplete(); + $source = new Account('test://test%40domain.tld:test@test'); + $destination = new Account('test://test%40kolab.org:test@test'); + $engine = new Engine(); + + Test::init($this->data); + Queue::fake(); + + $engine->migrate($source, $destination, ['sync' => true]); + + $queue = MigratorQueue::first(); + + Queue::assertNothingPushed(); + + $this->assertSame(0, $queue->jobs_started); + $this->assertSame(0, $queue->jobs_finished); + $this->assertSame(['sync' => true], $queue->data['options']); + + $this->assertCount(3, Test::$createdFolders); + $this->assertCount(8, Test::$createdItems); + $this->assertCount(8, Test::$fetchedItems); + + Test::init($this->data); + + // Test 'type' argument + $engine->migrate($source, $destination, ['sync' => true, 'type' => 'contact,event']); + + $queue = MigratorQueue::whereNot('id', $queue->id)->first(); + + Queue::assertNothingPushed(); + + $this->assertSame(0, $queue->jobs_started); + $this->assertSame(0, $queue->jobs_finished); + $this->assertSame(['sync' => true, 'type' => 'contact,event'], $queue->data['options']); + + $this->assertCount(2, Test::$createdFolders); + $this->assertCount(4, Test::$createdItems); + $this->assertCount(4, Test::$fetchedItems); } } diff --git a/src/tests/Feature/DomainTest.php b/src/tests/Feature/DomainTest.php index c83437e8..80bf8209 100644 --- a/src/tests/Feature/DomainTest.php +++ b/src/tests/Feature/DomainTest.php @@ -1,461 +1,458 @@ domains as $domain) { $this->deleteTestDomain($domain); } $this->deleteTestUser('user@gmail.com'); } /** * {@inheritDoc} */ public function tearDown(): void { foreach ($this->domains as $domain) { $this->deleteTestDomain($domain); } $this->deleteTestUser('user@gmail.com'); parent::tearDown(); } /** * Tests for Domain::assignPackage() */ public function testAssignPackage(): void { $user = $this->getTestUser('user@gmail.com'); $domain = $this->getTestDomain('gmail.com', [ 'status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_EXTERNAL, ]); $package = \App\Package::withObjectTenantContext($user)->where('title', 'domain-hosting')->first(); $wallet = $user->wallets()->first(); $domain->assignPackage($package, $user); $this->assertCount(1, $entitlements = $wallet->entitlements()->get()); $this->assertSame(0, $entitlements[0]->cost); // Assert that units_free might not work as we intended to // The second domain is still free, but it should cost 100. $domain = $this->getTestDomain('public-active.com', [ 'status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_EXTERNAL, ]); $domain->assignPackage($package, $user); $this->assertCount(2, $entitlements = $wallet->entitlements()->get()); $this->assertSame(0, $entitlements[0]->cost); $this->assertSame(0, $entitlements[1]->cost); // Make assigning domain that is already assigned is not possible $this->expectException(\Exception::class); $domain->assignPackage($package, $user); } /** * Test domain create/creating observer */ public function testCreate(): void { Queue::fake(); $domain = Domain::create([ 'namespace' => 'GMAIL.COM', 'status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_EXTERNAL, ]); $result = Domain::where('namespace', 'gmail.com')->first(); $this->assertSame('gmail.com', $result->namespace); $this->assertSame($domain->id, $result->id); $this->assertSame($domain->type, $result->type); $this->assertSame(Domain::STATUS_NEW, $result->status); } /** * Test domain creating jobs */ public function testCreateJobs(): void { // Fake the queue, assert that no jobs were pushed... Queue::fake(); Queue::assertNothingPushed(); $domain = Domain::create([ 'namespace' => 'gmail.com', 'status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_EXTERNAL, ]); Queue::assertPushed(\App\Jobs\Domain\CreateJob::class, 1); Queue::assertPushed( \App\Jobs\Domain\CreateJob::class, function ($job) use ($domain) { $domainId = TestCase::getObjectProperty($job, 'domainId'); $domainNamespace = TestCase::getObjectProperty($job, 'domainNamespace'); return $domainId === $domain->id && $domainNamespace === $domain->namespace; } ); - - $job = new \App\Jobs\Domain\CreateJob($domain->id); - $job->handle(); } /** * Tests getPublicDomains() method */ public function testGetPublicDomains(): void { $public_domains = Domain::getPublicDomains(); $this->assertNotContains('public-active.com', $public_domains); $queue = Queue::fake(); $domain = Domain::create([ 'namespace' => 'public-active.com', 'status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_EXTERNAL, ]); // External domains should not be returned $public_domains = Domain::getPublicDomains(); $this->assertNotContains('public-active.com', $public_domains); $domain->type = Domain::TYPE_PUBLIC; $domain->save(); $public_domains = Domain::getPublicDomains(); $this->assertContains('public-active.com', $public_domains); // Domains of other tenants should not be returned $tenant = Tenant::whereNotIn('id', [\config('app.tenant_id')])->first(); $domain->tenant_id = $tenant->id; $domain->save(); $public_domains = Domain::getPublicDomains(); $this->assertNotContains('public-active.com', $public_domains); } /** * Test domain (ownership) confirmation * * @group dns */ public function testConfirm(): void { /* DNS records for positive and negative tests - kolab.org: ci-success-cname A 212.103.80.148 ci-success-cname MX 10 mx01.kolabnow.com. ci-success-cname TXT "v=spf1 mx -all" kolab-verify.ci-success-cname CNAME 2b719cfa4e1033b1e1e132977ed4fe3e.ci-success-cname ci-failure-cname A 212.103.80.148 ci-failure-cname MX 10 mx01.kolabnow.com. kolab-verify.ci-failure-cname CNAME 2b719cfa4e1033b1e1e132977ed4fe3e.ci-failure-cname ci-success-txt A 212.103.80.148 ci-success-txt MX 10 mx01.kolabnow.com. ci-success-txt TXT "v=spf1 mx -all" ci-success-txt TXT "kolab-verify=de5d04ababb52d52e2519a2f16d11422" ci-failure-txt A 212.103.80.148 ci-failure-txt MX 10 mx01.kolabnow.com. kolab-verify.ci-failure-txt TXT "kolab-verify=de5d04ababb52d52e2519a2f16d11422" ci-failure-none A 212.103.80.148 ci-failure-none MX 10 mx01.kolabnow.com. */ $queue = Queue::fake(); $domain_props = ['status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_EXTERNAL]; $domain = $this->getTestDomain('ci-failure-none.kolab.org', $domain_props); $this->assertTrue($domain->confirm() === false); $this->assertFalse($domain->isConfirmed()); $domain = $this->getTestDomain('ci-failure-txt.kolab.org', $domain_props); $this->assertTrue($domain->confirm() === false); $this->assertFalse($domain->isConfirmed()); $domain = $this->getTestDomain('ci-failure-cname.kolab.org', $domain_props); $this->assertTrue($domain->confirm() === false); $this->assertFalse($domain->isConfirmed()); $domain = $this->getTestDomain('ci-success-txt.kolab.org', $domain_props); $this->assertTrue($domain->confirm()); $this->assertTrue($domain->isConfirmed()); $domain = $this->getTestDomain('ci-success-cname.kolab.org', $domain_props); $this->assertTrue($domain->confirm()); $this->assertTrue($domain->isConfirmed()); } /** * Test domain deletion */ public function testDelete(): void { Queue::fake(); $domain = $this->getTestDomain('gmail.com', [ 'status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_PUBLIC, ]); $domain->delete(); $this->assertTrue($domain->fresh()->trashed()); $this->assertFalse($domain->fresh()->isDeleted()); // Delete the domain for real $job = new \App\Jobs\Domain\DeleteJob($domain->id); $job->handle(); $this->assertTrue(Domain::withTrashed()->where('id', $domain->id)->first()->isDeleted()); $domain->forceDelete(); $this->assertCount(0, Domain::withTrashed()->where('id', $domain->id)->get()); } /** * Test eventlog on domain deletion */ public function testDeleteAndEventLog(): void { Queue::fake(); $domain = $this->getTestDomain('gmail.com', [ 'status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_PUBLIC, ]); EventLog::createFor($domain, EventLog::TYPE_SUSPENDED, 'test'); $domain->delete(); $this->assertCount(1, EventLog::where('object_id', $domain->id)->where('object_type', Domain::class)->get()); $domain->forceDelete(); $this->assertCount(0, EventLog::where('object_id', $domain->id)->where('object_type', Domain::class)->get()); } /** * Test isEmpty() method */ public function testIsEmpty(): void { Queue::fake(); $this->deleteTestUser('user@gmail.com'); $this->deleteTestGroup('group@gmail.com'); $this->deleteTestResource('resource@gmail.com'); $this->deleteTestSharedFolder('folder@gmail.com'); // Empty domain $domain = $this->getTestDomain('gmail.com', [ 'status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_EXTERNAL, ]); $this->assertTrue($domain->isEmpty()); $this->getTestUser('user@gmail.com'); $this->assertFalse($domain->isEmpty()); $this->deleteTestUser('user@gmail.com'); $this->assertTrue($domain->isEmpty()); $this->getTestGroup('group@gmail.com'); $this->assertFalse($domain->isEmpty()); $this->deleteTestGroup('group@gmail.com'); $this->assertTrue($domain->isEmpty()); $this->getTestResource('resource@gmail.com'); $this->assertFalse($domain->isEmpty()); $this->deleteTestResource('resource@gmail.com'); $this->getTestSharedFolder('folder@gmail.com'); $this->assertFalse($domain->isEmpty()); $this->deleteTestSharedFolder('folder@gmail.com'); // TODO: Test with an existing alias, but not other objects in a domain // Empty public domain $domain = Domain::where('namespace', 'libertymail.net')->first(); $this->assertFalse($domain->isEmpty()); } /** * Test domain restoring */ public function testRestore(): void { Queue::fake(); $domain = $this->getTestDomain('gmail.com', [ 'status' => Domain::STATUS_NEW | Domain::STATUS_SUSPENDED | Domain::STATUS_LDAP_READY | Domain::STATUS_CONFIRMED, 'type' => Domain::TYPE_PUBLIC, ]); $user = $this->getTestUser('user@gmail.com'); $sku = \App\Sku::where('title', 'domain-hosting')->first(); $now = \Carbon\Carbon::now(); // Assign two entitlements to the domain, so we can assert that only the // ones deleted last will be restored $ent1 = \App\Entitlement::create([ 'wallet_id' => $user->wallets->first()->id, 'sku_id' => $sku->id, 'cost' => 0, 'entitleable_id' => $domain->id, 'entitleable_type' => Domain::class, ]); $ent2 = \App\Entitlement::create([ 'wallet_id' => $user->wallets->first()->id, 'sku_id' => $sku->id, 'cost' => 0, 'entitleable_id' => $domain->id, 'entitleable_type' => Domain::class, ]); $domain->delete(); $this->assertTrue($domain->fresh()->trashed()); $this->assertFalse($domain->fresh()->isDeleted()); $this->assertTrue($ent1->fresh()->trashed()); $this->assertTrue($ent2->fresh()->trashed()); // Backdate some properties \App\Entitlement::withTrashed()->where('id', $ent2->id)->update(['deleted_at' => $now->subMinutes(2)]); \App\Entitlement::withTrashed()->where('id', $ent1->id)->update(['updated_at' => $now->subMinutes(10)]); Queue::fake(); $domain->restore(); $domain->refresh(); $this->assertFalse($domain->trashed()); $this->assertFalse($domain->isDeleted()); $this->assertFalse($domain->isSuspended()); $this->assertFalse($domain->isLdapReady()); $this->assertFalse($domain->isActive()); $this->assertFalse($domain->isConfirmed()); $this->assertTrue($domain->isNew()); // Assert entitlements $this->assertTrue($ent2->fresh()->trashed()); $this->assertFalse($ent1->fresh()->trashed()); $this->assertTrue($ent1->updated_at->greaterThan(\Carbon\Carbon::now()->subSeconds(5))); // We expect only one CreateJob and one UpdateJob // Because how Illuminate/Database/Eloquent/SoftDeletes::restore() method // is implemented we cannot skip the UpdateJob in any way. // I don't want to overwrite this method, the extra job shouldn't do any harm. $this->assertCount(2, Queue::pushedJobs()); // @phpstan-ignore-line Queue::assertPushed(\App\Jobs\Domain\UpdateJob::class, 1); Queue::assertPushed(\App\Jobs\Domain\CreateJob::class, 1); Queue::assertPushed( \App\Jobs\Domain\CreateJob::class, function ($job) use ($domain) { return $domain->id === TestCase::getObjectProperty($job, 'domainId'); } ); } /** * Tests for Domain::walletOwner() (from EntitleableTrait) */ public function testWalletOwner(): void { $domain = $this->getTestDomain('kolab.org'); $john = $this->getTestUser('john@kolab.org'); $this->assertSame($john->id, $domain->walletOwner()->id); // A domain without an owner $domain = $this->getTestDomain('gmail.com', [ 'status' => Domain::STATUS_NEW | Domain::STATUS_SUSPENDED | Domain::STATUS_LDAP_READY | Domain::STATUS_CONFIRMED, 'type' => Domain::TYPE_PUBLIC, ]); $this->assertSame(null, $domain->walletOwner()); } /** * Test domain verifying */ public function testVerify(): void { Queue::fake(); // A domain with DNS records $domain = $this->getTestDomain('gmail.com', [ 'status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_PUBLIC, ]); $this->assertTrue($domain->verify()); $this->assertTrue($domain->isVerified()); // A domain without DNS records $domain = $this->getTestDomain('public-active.com', [ 'status' => Domain::STATUS_NEW, 'type' => Domain::TYPE_PUBLIC, ]); $this->assertFalse($domain->verify()); $this->assertFalse($domain->isVerified()); } } diff --git a/src/tests/Unit/DataMigrator/EWS/AppointmentTest.php b/src/tests/Unit/DataMigrator/EWS/AppointmentTest.php index 8e11b99b..89cf961e 100644 --- a/src/tests/Unit/DataMigrator/EWS/AppointmentTest.php +++ b/src/tests/Unit/DataMigrator/EWS/AppointmentTest.php @@ -1,113 +1,113 @@ 'test']); $appointment = new EWS\Appointment($ews, $folder); $ical = file_get_contents(__DIR__ . '/../../../data/ews/event/1.ics'); $ical = preg_replace('/\r?\n/', "\r\n", $ical); // FIXME: I haven't found a way to convert xml content into a Type instance // therefore we create it "manually", but it would be better to have both // vcard and xml in a single data file that we could just get content from. $item = Type::buildFromArray([ 'MimeContent' => base64_encode($ical), 'ItemId' => new Type\ItemIdType( 'AAMkAGEzOGRlODRiLTBkN2ItNDgwZS04ZDJmLTM5NDEyY2Q0NGQ0OABGAAAAAAC9tlDYSlG2TaxWBr' . 'A1OzWtBwAs2ajhknXlRYN/pbC8JqblAAAAAAEOAAAs2ajhknXlRYN/pbC8JqblAAJnrWkBAAA=', 'EQAAABYAAAAs2ajhknXlRYN/pbC8JqblAAJnqlKm', ), 'UID' => '1F3C13D7E99642A75ABE23D50487B454-8FE68B2E68E1B348', 'Subject' => 'test subject', 'HasAttachments' => false, 'IsAssociated' => false, 'Start' => '2023-11-21T11:00:00Z', 'End' => '2023-11-21T11:30:00Z', 'LegacyFreeBusyStatus' => 'Tentative', 'CalendarItemType' => 'Single', 'Organizer' => [ 'Mailbox' => [ 'Name' => 'Aleksander Machniak', 'EmailAddress' => 'test@kolab.org', 'RoutingType' => 'SMTP', 'MailboxType' => 'Contact', ], ], 'RequiredAttendees' => (object) [ 'Attendee' => [ Type\AttendeeType::buildFromArray([ 'Mailbox' => [ 'Name' => 'Aleksander Machniak', 'EmailAddress' => 'test@kolab.org', 'RoutingType' => 'SMTP', 'MailboxType' => 'Contact', ], 'ResponseType' => 'Unknown', ]), Type\AttendeeType::buildFromArray([ 'Mailbox' => [ 'Name' => 'Alec Machniak', 'EmailAddress' => 'test@outlook.com', 'RoutingType' => 'SMTP', 'MailboxType' => 'Mailbox', ], 'ResponseType' => 'Unknown', ]), ], ], ]); // Convert the Exchange item into iCalendar - $ical = $this->invokeMethod($appointment, 'processItem', [$item]); + $ical = $this->invokeMethod($appointment, 'convertItem', [$item]); // Parse the iCalendar output $event = new Vevent(); $this->invokeMethod($event, 'fromIcal', [$ical]); $msId = implode('!', $item->getItemId()->toArray()); $this->assertSame($msId, $event->custom['X-MS-ID']); $this->assertSame($item->getUID(), $event->uid); $this->assertSame('test description', $event->description); $this->assertSame('test subject', $event->summary); $this->assertSame('CONFIRMED', $event->status); $this->assertSame('PUBLIC', $event->class); $this->assertSame('Microsoft Exchange Server 2010', $event->prodid); $this->assertSame('2023-11-20T14:50:05+00:00', $event->dtstamp->getDateTime()->format('c')); $this->assertSame('2023-11-21T12:00:00+01:00', $event->dtstart->getDateTime()->format('c')); $this->assertSame('2023-11-21T12:30:00+01:00', $event->dtend->getDateTime()->format('c')); // Organizer/attendees $this->assertSame('test@kolab.org', $event->organizer['email']); $this->assertSame('Aleksander Machniak', $event->organizer['cn']); $this->assertSame('ORGANIZER', $event->organizer['role']); $this->assertSame('ACCEPTED', $event->organizer['partstat']); $this->assertSame(false, $event->organizer['rsvp']); $this->assertCount(1, $event->attendees); $this->assertSame('alec@outlook.com', $event->attendees[0]['email']); $this->assertSame('Alec Machniak', $event->attendees[0]['cn']); $this->assertSame('REQ-PARTICIPANT', $event->attendees[0]['role']); $this->assertSame('NEEDS-ACTION', $event->attendees[0]['partstat']); $this->assertSame(true, $event->attendees[0]['rsvp']); } } diff --git a/src/tests/Unit/DataMigrator/EWS/ContactTest.php b/src/tests/Unit/DataMigrator/EWS/ContactTest.php index 87a4be69..ac4cb767 100644 --- a/src/tests/Unit/DataMigrator/EWS/ContactTest.php +++ b/src/tests/Unit/DataMigrator/EWS/ContactTest.php @@ -1,141 +1,141 @@ 'test']); $contact = new EWS\Contact($ews, $folder); $vcard = file_get_contents(__DIR__ . '/../../../data/ews/contact/1.vcf'); $vcard = preg_replace('/\r?\n/', "\r\n", $vcard); // FIXME: I haven't found a way to convert xml content into a Type instance // therefore we create it "manually", but it would be better to have both // vcard and xml in a single data file that we could just get content from. $item = Type::buildFromArray([ 'MimeContent' => base64_encode($vcard), 'ItemId' => new Type\ItemIdType( 'AAMkAGEzOGRlODRiLTBkN2ItNDgwZS04ZDJmLTM5NDEyY2Q0NGQ0OABGAAAAAAC9tlDYSlG2TaxWBr' . 'A1OzWtBwAs2ajhknXlRYN/pbC8JqblAAAAAAEOAAAs2ajhknXlRYN/pbC8JqblAAJnrWkBAAA=', 'EQAAABYAAAAs2ajhknXlRYN/pbC8JqblAAJnqlKm', ), 'HasAttachments' => false, 'LastModifiedTime' => '2024-07-15T11:17:39,701Z', 'DisplayName' => 'Nowy Nazwisko', 'GivenName' => 'Nowy', 'Surname' => 'Nazwisko', 'EmailAddresses' => (object) [ 'Entry' => [ Type\EmailAddressDictionaryEntryType::buildFromArray([ 'Key' => 'EmailAddress1', 'Name' => 'test1@outlook.com', 'RoutingType' => 'SMTP', 'MailboxType' => 'Contact', '_value' => 'christian1@outlook.com', ]), Type\EmailAddressDictionaryEntryType::buildFromArray([ 'Key' => 'EmailAddress2', 'Name' => 'test2@outlook.com', 'RoutingType' => 'SMTP', 'MailboxType' => 'Contact', '_value' => 'test2@outlook.com', ]), ], ], /* ContactPicture.jpg image/jpeg 2081 2024-07-15T11:17:38 false true category en-US Mr Nowy Bartosz Nazwisko Jr. Nowy Nazwisko alec Company Testowa Warsaw mazowickie Poland 00-001 home123456 1234556679200 2014-10-11T11:59:00Z IT Developer Office Location 2020-11-12T11:59:00Z true */ ]); // Convert the Exchange item into vCard - $vcard = $this->invokeMethod($contact, 'processItem', [$item]); + $vcard = $this->invokeMethod($contact, 'convertItem', [$item]); // Parse the vCard $contact = new Vcard(); $this->invokeMethod($contact, 'fromVcard', [$vcard]); $this->assertMatchesRegularExpression('/^[a-f0-9]{40}$/', $contact->uid); $this->assertSame('PUBLIC', $contact->class); $this->assertSame('Nowy Nazwisko', $contact->fn); $this->assertSame(null, $contact->kind); $this->assertSame('Microsoft Exchange', $contact->prodid); $this->assertSame('2024-07-15T11:17:39,701Z', $contact->rev); $this->assertSame('Notatki do kontaktu', $contact->note); // EWS Properties with special handling $msId = implode('!', $item->getItemId()->toArray()); $this->assertSame($msId, $contact->custom['X-MS-ID']); $this->assertSame('Partner Name', $contact->custom['X-SPOUSE']); $this->assertSame('2020-11-12', $contact->custom['X-ANNIVERSARY']); $this->assertCount(2, $contact->email); $this->assertSame('internet', $contact->email[0]['type']); $this->assertSame('christian1@outlook.com', $contact->email[0]['email']); $this->assertSame('internet', $contact->email[1]['type']); $this->assertSame('test2@outlook.com', $contact->email[1]['email']); } } diff --git a/src/tests/Unit/DataMigrator/EWS/DistListTest.php b/src/tests/Unit/DataMigrator/EWS/DistListTest.php index e9e198a9..5665e38f 100644 --- a/src/tests/Unit/DataMigrator/EWS/DistListTest.php +++ b/src/tests/Unit/DataMigrator/EWS/DistListTest.php @@ -1,90 +1,96 @@ 'test']); $distlist = new EWS\DistList($ews, $folder); // FIXME: I haven't found a way to convert xml content into a Type instance // therefore we create it "manually", but it would be better to have both // vcard and xml in a single data file that we could just get content from. $item = Type::buildFromArray([ 'ItemId' => new Type\ItemIdType( 'AAMkAGEzOGRlODRiLTBkN2ItNDgwZS04ZDJmLTM5NDEyY2Q0NGQ0OABGAAAAAAC9tlDYSlG2TaxWBr' . 'A1OzWtBwAs2ajhknXlRYN/pbC8JqblAAAAAAEOAAAs2ajhknXlRYN/pbC8JqblAAJnrWkBAAA=', 'EQAAABYAAAAs2ajhknXlRYN/pbC8JqblAAJnqlKm', ), 'Subject' => 'subject list', 'LastModifiedTime' => '2024-06-27T13:44:32Z', 'DisplayName' => 'Lista', 'FileAs' => 'lista', + 'Body' => [ + 'BodyType' => 'Text', + 'IsTruncated' => false, + '_value' => 'distlist body', + ], 'Members' => (object) [ 'Member' => [ Type\MemberType::buildFromArray([ 'Key' => 'AAAAAIErH6S+oxAZnW4A3QEPVAIAAAGAYQBsAGUAYwBAAGEAbABlAGMALgBw' . 'AGwAAABTAE0AVABQAAAAYQBsAGUAYwBAAGEAbABlAGMALgBwAGwAAAA=', 'Mailbox' => Type\Mailbox::buildFromArray([ 'Name' => 'Alec', 'EmailAddress' => 'alec@kolab.org', 'RoutingType' => 'SMTP', 'MailboxType' => 'OneOff', ]), 'Status' => 'Normal', ]), Type\MemberType::buildFromArray([ 'Key' => 'AAAAAIErH6S+oxAZnW4A3QEPVAIAAAGAYQBsAGUAYwBAAGEAbABlAGMALgBw' . 'AGwAAABTAE0AVABQAAAAYQBsAGUAYwBAAGEAbABlAGMALgBwAGwAAAB=', 'Mailbox' => Type\Mailbox::buildFromArray([ 'Name' => 'Christian', 'EmailAddress' => 'christian@kolab.org', 'RoutingType' => 'SMTP', 'MailboxType' => 'OneOff', 'ItemId' => new Type\ItemIdType('AAA', 'BBB'), ]), 'Status' => 'Normal', ]), ], ], ]); // Convert the Exchange item into vCard - $vcard = $this->invokeMethod($distlist, 'processItem', [$item]); + $vcard = $this->invokeMethod($distlist, 'convertItem', [$item]); // Parse the vCard $distlist = new Vcard(); $this->invokeMethod($distlist, 'fromVcard', [$vcard]); $msId = implode('!', $item->getItemId()->toArray()); $this->assertSame(['X-MS-ID' => $msId], $distlist->custom); $this->assertMatchesRegularExpression('/^[a-f0-9]{40}$/', $distlist->uid); $this->assertSame('group', $distlist->kind); $this->assertSame('Lista', $distlist->fn); + $this->assertSame('distlist body', $distlist->note); $this->assertSame('Kolab EWS Data Migrator', $distlist->prodid); $this->assertSame('2024-06-27T13:44:32Z', $distlist->rev); $members = [ 'mailto:%22Alec%22+%3Calec%40kolab.org%3E', 'urn:uuid:' . sha1('AAA'), ]; $this->assertSame($members, $distlist->member); } } diff --git a/src/tests/Unit/DataMigrator/EWS/TaskTest.php b/src/tests/Unit/DataMigrator/EWS/TaskTest.php index 43b72474..4d3d120a 100644 --- a/src/tests/Unit/DataMigrator/EWS/TaskTest.php +++ b/src/tests/Unit/DataMigrator/EWS/TaskTest.php @@ -1,161 +1,161 @@ source = $source; $engine->destination = $destination; $ews = new EWS($source, $engine); $folder = Folder::fromArray(['id' => 'test']); $task = new EWS\Task($ews, $folder); // FIXME: I haven't found a way to convert xml content into a Type instance // therefore we create it "manually", but it would be better to have it in XML. $html = '' . '
task notes
'; $item = Type\TaskType::buildFromArray([ 'ItemId' => new Type\ItemIdType( 'AAMkAGEzOGRlODRiLTBkN2ItNDgwZS04ZDJmLTM5NDEyY2Q0NGQ0OABGAAAAAAC9tlDYSlG2TaxWBr' . 'A1OzWtBwAs2ajhknXlRYN/pbC8JqblAAAAAAEOAAAs2ajhknXlRYN/pbC8JqblAAJnrWkBAAA=', 'EQAAABYAAAAs2ajhknXlRYN/pbC8JqblAAJnqlKm', ), 'ItemClass' => 'IPM.Task', 'Subject' => 'Nowe zadanie', 'LastModifiedTime' => '2024-06-27T13:44:32Z', 'Sensitivity' => 'Private', // TODO: Looks like EWS has Body->IsTruncated property, but is it relevant? 'Body' => new Type\BodyType($html, 'HTML'), 'Importance' => 'High', 'DateTimeCreated' => '2024-06-27T08:58:05Z', 'ReminderDueBy' => '2024-07-17T07:00:00Z', 'ReminderIsSet' => true, 'ReminderNextTime' => '2024-07-17T07:00:00Z', 'ReminderMinutesBeforeStart' => '0', 'DueDate' => '2024-06-26T22:00:00Z', 'IsComplete' => false, 'IsRecurring' => true, 'Owner' => 'Alec Machniak', 'PercentComplete' => '10', 'Recurrence' => Type\TaskRecurrenceType::buildFromArray([ 'WeeklyRecurrence' => [ 'Interval' => '1', 'DaysOfWeek' => 'Thursday', 'FirstDayOfWeek' => 'Sunday', ], 'NoEndRecurrence' => [ 'StartDate' => '2024-06-27Z', ], ]), 'Status' => 'NotStarted', 'ChangeCount' => '2', /* testdisk.log application/octet-stream 299368b3-06e4-42df-959e-d428046f55e6 249 2024-07-16T12:13:58 false false 2024-06-27T08:58:05Z 3041 Kategoria Niebieski false false false false false 2024-06-27T08:58:05Z true en-US false false false true true true true Alec Machniak 2024-07-16T12:14:38Z false NotFlagged AQEAAAAAAAESAQAAAmjiC08AAAAA 1 Not Started */ ]); // Convert the Exchange item into iCalendar - $ical = $this->invokeMethod($task, 'processItem', [$item]); + $ical = $this->invokeMethod($task, 'convertItem', [$item]); // Parse the iCalendar output $task = new Vtodo(); $this->invokeMethod($task, 'fromIcal', [$ical]); $msId = implode('!', $item->getItemId()->toArray()); $this->assertSame($msId, $task->custom['X-MS-ID']); $this->assertMatchesRegularExpression('/^[a-f0-9]{40}$/', $task->uid); $this->assertSame('Nowe zadanie', $task->summary); $this->assertSame($html, $task->description); $this->assertSame('Kolab EWS Data Migrator', $task->prodid); $this->assertSame('2', $task->sequence); $this->assertSame('9', $task->priority); $this->assertSame('PRIVATE', $task->class); $this->assertSame(10, $task->percentComplete); $this->assertSame('X-NOTSTARTED', $task->status); $this->assertSame('2024-06-27T13:44:32+00:00', $task->dtstamp->getDateTime()->format('c')); $this->assertSame('2024-06-27T08:58:05+00:00', $task->created->getDateTime()->format('c')); $this->assertSame('2024-06-26T22:00:00+00:00', $task->due->getDateTime()->format('c')); $this->assertSame('test@kolab.org', $task->organizer['email']); $this->assertSame('WEEKLY', $task->rrule['freq']); $this->assertSame('1', $task->rrule['interval']); $this->assertSame('TH', $task->rrule['byday']); $this->assertSame('SU', $task->rrule['wkst']); $this->assertCount(1, $task->valarms); $this->assertCount(2, $task->valarms[0]); $this->assertSame('DISPLAY', $task->valarms[0]['action']); $this->assertSame('2024-07-17T07:00:00+00:00', $task->valarms[0]['trigger']->format('c')); } /** * Test processing Recurrence property */ - public function testProcessItemRecurrence(): void + public function testConvertItemRecurrence(): void { $this->markTestIncomplete(); } }