2017-08-31 23:47:02 +03:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* @copyright Copyright (c) 2017 Arthur Schiwon <blizzz@arthur-schiwon.de>
|
|
|
|
*
|
|
|
|
* @author Arthur Schiwon <blizzz@arthur-schiwon.de>
|
2020-04-29 12:57:22 +03:00
|
|
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
2019-12-03 21:57:53 +03:00
|
|
|
* @author Joas Schilling <coding@schilljs.com>
|
|
|
|
* @author John Molakvoæ (skjnldsv) <skjnldsv@protonmail.com>
|
|
|
|
* @author Julius Härtl <jus@bitgrid.net>
|
2017-08-31 23:47:02 +03:00
|
|
|
*
|
|
|
|
* @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
|
2019-12-03 21:57:53 +03:00
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2017-08-31 23:47:02 +03:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace OC\Collaboration\Collaborators;
|
|
|
|
|
|
|
|
use OCP\Collaboration\Collaborators\ISearchPlugin;
|
|
|
|
use OCP\Collaboration\Collaborators\ISearchResult;
|
2017-09-06 17:09:29 +03:00
|
|
|
use OCP\Collaboration\Collaborators\SearchResultType;
|
2017-08-31 23:47:02 +03:00
|
|
|
use OCP\Contacts\IManager;
|
|
|
|
use OCP\Federation\ICloudIdManager;
|
|
|
|
use OCP\IConfig;
|
2018-10-17 22:20:15 +03:00
|
|
|
use OCP\IUserManager;
|
2018-10-30 13:41:56 +03:00
|
|
|
use OCP\IUserSession;
|
2020-06-24 17:49:16 +03:00
|
|
|
use OCP\Share\IShare;
|
2017-08-31 23:47:02 +03:00
|
|
|
|
|
|
|
class RemotePlugin implements ISearchPlugin {
|
|
|
|
protected $shareeEnumeration;
|
|
|
|
|
|
|
|
/** @var IManager */
|
|
|
|
private $contactsManager;
|
|
|
|
/** @var ICloudIdManager */
|
|
|
|
private $cloudIdManager;
|
|
|
|
/** @var IConfig */
|
|
|
|
private $config;
|
2018-10-17 22:20:15 +03:00
|
|
|
/** @var IUserManager */
|
|
|
|
private $userManager;
|
2018-10-29 19:00:09 +03:00
|
|
|
/** @var string */
|
2018-10-30 13:41:56 +03:00
|
|
|
private $userId = '';
|
2017-08-31 23:47:02 +03:00
|
|
|
|
2018-10-30 13:41:56 +03:00
|
|
|
public function __construct(IManager $contactsManager, ICloudIdManager $cloudIdManager, IConfig $config, IUserManager $userManager, IUserSession $userSession) {
|
2017-08-31 23:47:02 +03:00
|
|
|
$this->contactsManager = $contactsManager;
|
|
|
|
$this->cloudIdManager = $cloudIdManager;
|
|
|
|
$this->config = $config;
|
2018-10-17 22:20:15 +03:00
|
|
|
$this->userManager = $userManager;
|
2018-10-30 13:41:56 +03:00
|
|
|
$user = $userSession->getUser();
|
|
|
|
if ($user !== null) {
|
|
|
|
$this->userId = $user->getUID();
|
|
|
|
}
|
2017-08-31 23:47:02 +03:00
|
|
|
$this->shareeEnumeration = $this->config->getAppValue('core', 'shareapi_allow_share_dialog_user_enumeration', 'yes') === 'yes';
|
|
|
|
}
|
|
|
|
|
|
|
|
public function search($search, $limit, $offset, ISearchResult $searchResult) {
|
|
|
|
$result = ['wide' => [], 'exact' => []];
|
2017-09-06 17:09:29 +03:00
|
|
|
$resultType = new SearchResultType('remotes');
|
2017-08-31 23:47:02 +03:00
|
|
|
|
|
|
|
// Search in contacts
|
2020-01-10 18:06:44 +03:00
|
|
|
$addressBookContacts = $this->contactsManager->search($search, ['CLOUD', 'FN'], ['limit' => $limit, 'offset' => $offset]);
|
2017-08-31 23:47:02 +03:00
|
|
|
foreach ($addressBookContacts as $contact) {
|
|
|
|
if (isset($contact['isLocalSystemBook'])) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (isset($contact['CLOUD'])) {
|
|
|
|
$cloudIds = $contact['CLOUD'];
|
2018-10-29 19:00:09 +03:00
|
|
|
if (is_string($cloudIds)) {
|
2017-08-31 23:47:02 +03:00
|
|
|
$cloudIds = [$cloudIds];
|
|
|
|
}
|
|
|
|
$lowerSearch = strtolower($search);
|
2018-10-29 19:00:09 +03:00
|
|
|
foreach ($cloudIds as $cloudId) {
|
|
|
|
$cloudIdType = '';
|
|
|
|
if (\is_array($cloudId)) {
|
|
|
|
$cloudIdData = $cloudId;
|
|
|
|
$cloudId = $cloudIdData['value'];
|
|
|
|
$cloudIdType = $cloudIdData['type'];
|
|
|
|
}
|
2017-08-31 23:47:02 +03:00
|
|
|
try {
|
2021-01-12 12:15:48 +03:00
|
|
|
[$remoteUser, $serverUrl] = $this->splitUserRemote($cloudId);
|
2017-08-31 23:47:02 +03:00
|
|
|
} catch (\InvalidArgumentException $e) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-10-17 22:20:15 +03:00
|
|
|
$localUser = $this->userManager->get($remoteUser);
|
2018-10-23 10:24:04 +03:00
|
|
|
/**
|
|
|
|
* Add local share if remote cloud id matches a local user ones
|
|
|
|
*/
|
2020-04-09 17:07:47 +03:00
|
|
|
if ($localUser !== null && $remoteUser !== $this->userId && $cloudId === $localUser->getCloudId()) {
|
2018-10-23 10:24:04 +03:00
|
|
|
$result['wide'][] = [
|
|
|
|
'label' => $contact['FN'],
|
|
|
|
'uuid' => $contact['UID'],
|
|
|
|
'value' => [
|
2020-06-24 17:49:16 +03:00
|
|
|
'shareType' => IShare::TYPE_USER,
|
2018-10-23 10:24:04 +03:00
|
|
|
'shareWith' => $remoteUser
|
2020-10-07 13:31:30 +03:00
|
|
|
],
|
|
|
|
'shareWithDisplayNameUnique' => $contact['EMAIL'] !== null && $contact['EMAIL'] !== '' ? $contact['EMAIL'] : $contact['UID'],
|
2018-10-23 10:24:04 +03:00
|
|
|
];
|
2018-10-17 22:20:15 +03:00
|
|
|
}
|
|
|
|
|
2017-08-31 23:47:02 +03:00
|
|
|
if (strtolower($contact['FN']) === $lowerSearch || strtolower($cloudId) === $lowerSearch) {
|
|
|
|
if (strtolower($cloudId) === $lowerSearch) {
|
2017-09-12 23:38:36 +03:00
|
|
|
$searchResult->markExactIdMatch($resultType);
|
2017-08-31 23:47:02 +03:00
|
|
|
}
|
|
|
|
$result['exact'][] = [
|
|
|
|
'label' => $contact['FN'] . " ($cloudId)",
|
2018-10-17 22:57:22 +03:00
|
|
|
'uuid' => $contact['UID'],
|
2018-10-26 14:46:47 +03:00
|
|
|
'name' => $contact['FN'],
|
2018-10-29 19:00:09 +03:00
|
|
|
'type' => $cloudIdType,
|
2017-08-31 23:47:02 +03:00
|
|
|
'value' => [
|
2020-06-24 17:49:16 +03:00
|
|
|
'shareType' => IShare::TYPE_REMOTE,
|
2017-08-31 23:47:02 +03:00
|
|
|
'shareWith' => $cloudId,
|
|
|
|
'server' => $serverUrl,
|
|
|
|
],
|
|
|
|
];
|
|
|
|
} else {
|
|
|
|
$result['wide'][] = [
|
|
|
|
'label' => $contact['FN'] . " ($cloudId)",
|
2018-10-17 22:57:22 +03:00
|
|
|
'uuid' => $contact['UID'],
|
2018-10-26 14:46:47 +03:00
|
|
|
'name' => $contact['FN'],
|
2018-10-29 19:00:09 +03:00
|
|
|
'type' => $cloudIdType,
|
2017-08-31 23:47:02 +03:00
|
|
|
'value' => [
|
2020-06-24 17:49:16 +03:00
|
|
|
'shareType' => IShare::TYPE_REMOTE,
|
2017-08-31 23:47:02 +03:00
|
|
|
'shareWith' => $cloudId,
|
|
|
|
'server' => $serverUrl,
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$this->shareeEnumeration) {
|
|
|
|
$result['wide'] = [];
|
2018-02-06 20:11:44 +03:00
|
|
|
} else {
|
|
|
|
$result['wide'] = array_slice($result['wide'], $offset, $limit);
|
2017-08-31 23:47:02 +03:00
|
|
|
}
|
|
|
|
|
2018-10-17 22:21:39 +03:00
|
|
|
/**
|
|
|
|
* Add generic share with remote item for valid cloud ids that are not users of the local instance
|
|
|
|
*/
|
2017-09-06 17:09:29 +03:00
|
|
|
if (!$searchResult->hasExactIdMatch($resultType) && $this->cloudIdManager->isValidCloudId($search) && $offset === 0) {
|
2018-10-17 22:21:39 +03:00
|
|
|
try {
|
2021-01-12 12:15:48 +03:00
|
|
|
[$remoteUser, $serverUrl] = $this->splitUserRemote($search);
|
2018-10-17 22:21:39 +03:00
|
|
|
$localUser = $this->userManager->get($remoteUser);
|
|
|
|
if ($localUser === null || $search !== $localUser->getCloudId()) {
|
|
|
|
$result['exact'][] = [
|
2019-05-23 18:03:04 +03:00
|
|
|
'label' => $remoteUser . " ($serverUrl)",
|
|
|
|
'uuid' => $remoteUser,
|
|
|
|
'name' => $remoteUser,
|
2018-10-17 22:21:39 +03:00
|
|
|
'value' => [
|
2020-06-24 17:49:16 +03:00
|
|
|
'shareType' => IShare::TYPE_REMOTE,
|
2018-10-17 22:21:39 +03:00
|
|
|
'shareWith' => $search,
|
2019-05-23 18:03:04 +03:00
|
|
|
'server' => $serverUrl,
|
2018-10-17 22:21:39 +03:00
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
} catch (\InvalidArgumentException $e) {
|
|
|
|
}
|
2017-08-31 23:47:02 +03:00
|
|
|
}
|
|
|
|
|
2017-09-06 17:09:29 +03:00
|
|
|
$searchResult->addResultSet($resultType, $result['wide'], $result['exact']);
|
2017-08-31 23:47:02 +03:00
|
|
|
|
2017-09-12 23:38:36 +03:00
|
|
|
return true;
|
2017-08-31 23:47:02 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* split user and remote from federated cloud id
|
|
|
|
*
|
|
|
|
* @param string $address federated share address
|
|
|
|
* @return array [user, remoteURL]
|
|
|
|
* @throws \InvalidArgumentException
|
|
|
|
*/
|
|
|
|
public function splitUserRemote($address) {
|
|
|
|
try {
|
|
|
|
$cloudId = $this->cloudIdManager->resolveCloudId($address);
|
|
|
|
return [$cloudId->getUser(), $cloudId->getRemote()];
|
|
|
|
} catch (\InvalidArgumentException $e) {
|
|
|
|
throw new \InvalidArgumentException('Invalid Federated Cloud ID', 0, $e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|