* * @author Arthur Schiwon * @author Christoph Wurst * @author Joas Schilling * @author John Molakvoæ * @author Julius Härtl * * @license GNU AGPL version 3 or any later version * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . * */ namespace OC\Collaboration\Collaborators; use OCP\Collaboration\Collaborators\ISearchPlugin; use OCP\Collaboration\Collaborators\ISearchResult; use OCP\Collaboration\Collaborators\SearchResultType; use OCP\Contacts\IManager; use OCP\Federation\ICloudIdManager; use OCP\IConfig; use OCP\IUserManager; use OCP\IUserSession; use OCP\Share\IShare; class RemotePlugin implements ISearchPlugin { protected bool $shareeEnumeration; private string $userId; public function __construct( private IManager $contactsManager, private ICloudIdManager $cloudIdManager, private IConfig $config, private IUserManager $userManager, IUserSession $userSession, ) { $this->userId = $userSession->getUser()?->getUID() ?? ''; $this->shareeEnumeration = $this->config->getAppValue('core', 'shareapi_allow_share_dialog_user_enumeration', 'yes') === 'yes'; } public function search($search, $limit, $offset, ISearchResult $searchResult): bool { $result = ['wide' => [], 'exact' => []]; $resultType = new SearchResultType('remotes'); // Search in contacts $addressBookContacts = $this->contactsManager->search($search, ['CLOUD', 'FN'], [ 'limit' => $limit, 'offset' => $offset, 'enumeration' => false, 'fullmatch' => false, ]); foreach ($addressBookContacts as $contact) { if (isset($contact['isLocalSystemBook'])) { continue; } if (isset($contact['CLOUD'])) { $cloudIds = $contact['CLOUD']; if (is_string($cloudIds)) { $cloudIds = [$cloudIds]; } $lowerSearch = strtolower($search); foreach ($cloudIds as $cloudId) { $cloudIdType = ''; if (\is_array($cloudId)) { $cloudIdData = $cloudId; $cloudId = $cloudIdData['value']; $cloudIdType = $cloudIdData['type']; } try { [$remoteUser, $serverUrl] = $this->splitUserRemote($cloudId); } catch (\InvalidArgumentException $e) { continue; } $localUser = $this->userManager->get($remoteUser); /** * Add local share if remote cloud id matches a local user ones */ if ($localUser !== null && $remoteUser !== $this->userId && $cloudId === $localUser->getCloudId()) { $result['wide'][] = [ 'label' => $contact['FN'], 'uuid' => $contact['UID'], 'value' => [ 'shareType' => IShare::TYPE_USER, 'shareWith' => $remoteUser ], 'shareWithDisplayNameUnique' => $contact['EMAIL'] !== null && $contact['EMAIL'] !== '' ? $contact['EMAIL'] : $contact['UID'], ]; } if (strtolower($contact['FN']) === $lowerSearch || strtolower($cloudId) === $lowerSearch) { if (strtolower($cloudId) === $lowerSearch) { $searchResult->markExactIdMatch($resultType); } $result['exact'][] = [ 'label' => $contact['FN'] . " ($cloudId)", 'uuid' => $contact['UID'], 'name' => $contact['FN'], 'type' => $cloudIdType, 'value' => [ 'shareType' => IShare::TYPE_REMOTE, 'shareWith' => $cloudId, 'server' => $serverUrl, ], ]; } else { $result['wide'][] = [ 'label' => $contact['FN'] . " ($cloudId)", 'uuid' => $contact['UID'], 'name' => $contact['FN'], 'type' => $cloudIdType, 'value' => [ 'shareType' => IShare::TYPE_REMOTE, 'shareWith' => $cloudId, 'server' => $serverUrl, ], ]; } } } } if (!$this->shareeEnumeration) { $result['wide'] = []; } else { $result['wide'] = array_slice($result['wide'], $offset, $limit); } /** * Add generic share with remote item for valid cloud ids that are not users of the local instance */ if (!$searchResult->hasExactIdMatch($resultType) && $this->cloudIdManager->isValidCloudId($search) && $offset === 0) { try { [$remoteUser, $serverUrl] = $this->splitUserRemote($search); $localUser = $this->userManager->get($remoteUser); if ($localUser === null || $search !== $localUser->getCloudId()) { $result['exact'][] = [ 'label' => $remoteUser . " ($serverUrl)", 'uuid' => $remoteUser, 'name' => $remoteUser, 'value' => [ 'shareType' => IShare::TYPE_REMOTE, 'shareWith' => $search, 'server' => $serverUrl, ], ]; } } catch (\InvalidArgumentException $e) { } } $searchResult->addResultSet($resultType, $result['wide'], $result['exact']); return true; } /** * split user and remote from federated cloud id * * @param string $address federated share address * @return array [user, remoteURL] * @throws \InvalidArgumentException */ public function splitUserRemote(string $address): array { try { $cloudId = $this->cloudIdManager->resolveCloudId($address); return [$cloudId->getUser(), $cloudId->getRemote()]; } catch (\InvalidArgumentException $e) { throw new \InvalidArgumentException('Invalid Federated Cloud ID', 0, $e); } } }