279 строки
7.5 KiB
PHP
279 строки
7.5 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
/**
|
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
|
* @author Thomas Müller <thomas.mueller@tmit.eu>
|
|
*
|
|
* Mail
|
|
*
|
|
* This code is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License, version 3,
|
|
* as published by the Free Software Foundation.
|
|
*
|
|
* 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, version 3,
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>
|
|
*
|
|
*/
|
|
|
|
namespace OCA\Mail\Service;
|
|
|
|
use OCP\Contacts\IManager;
|
|
use OCP\IConfig;
|
|
use OCP\IGroupManager;
|
|
use OCP\IUserManager;
|
|
|
|
class ContactsIntegration {
|
|
/** @var IManager */
|
|
private $contactsManager;
|
|
|
|
/** @var IGroupManager */
|
|
private $groupManager;
|
|
|
|
/** @var IUserManager */
|
|
private $userManager;
|
|
|
|
/** @var IConfig */
|
|
private $config;
|
|
|
|
public function __construct(IManager $contactsManager,
|
|
IGroupManager $groupManager,
|
|
IUserManager $userManager,
|
|
IConfig $config) {
|
|
$this->contactsManager = $contactsManager;
|
|
$this->groupManager = $groupManager;
|
|
$this->userManager = $userManager;
|
|
$this->config = $config;
|
|
}
|
|
|
|
/**
|
|
* Extracts all matching contacts with email address and name
|
|
*
|
|
* @param string $userId
|
|
* @param string $term
|
|
* @return array
|
|
*/
|
|
public function getMatchingRecipient(string $userId, string $term): array {
|
|
if (!$this->contactsManager->isEnabled()) {
|
|
return [];
|
|
}
|
|
|
|
// If 'Allow username autocompletion in share dialog' is disabled in the admin sharing settings, then we must not
|
|
// auto-complete system users
|
|
$shareeEnumeration = $this->config->getAppValue('core', 'shareapi_allow_share_dialog_user_enumeration', 'no') === 'yes';
|
|
$shareeEnumerationInGroupOnly = $this->config->getAppValue('core', 'shareapi_restrict_user_enumeration_to_group', 'no') === 'yes';
|
|
$shareeEnumerationFullMatch = $this->config->getAppValue('core', 'shareapi_restrict_user_enumeration_full_match', 'yes') === 'yes';
|
|
$shareeEnumerationFullMatchUserId = $shareeEnumerationFullMatch && $this->config->getAppValue('core', 'shareapi_restrict_user_enumeration_full_match_userid', 'yes') === 'yes';
|
|
$shareeEnumerationFullMatchEmail = $shareeEnumerationFullMatch && $this->config->getAppValue('core', 'shareapi_restrict_user_enumeration_full_match_email', 'yes') === 'yes';
|
|
|
|
$result = $this->contactsManager->search($term, ['UID', 'FN', 'EMAIL'], ['enumeration' => $shareeEnumeration]);
|
|
if (empty($result)) {
|
|
return [];
|
|
}
|
|
$receivers = [];
|
|
|
|
if ($shareeEnumeration && $shareeEnumerationInGroupOnly) {
|
|
$user = $this->userManager->get($userId);
|
|
if ($user === null) {
|
|
return [];
|
|
}
|
|
$userGroups = $this->groupManager->getUserGroupIds($user);
|
|
}
|
|
|
|
foreach ($result as $r) {
|
|
$isSystemUser = isset($r['isLocalSystemBook']) && $r['isLocalSystemBook'];
|
|
$isInSameGroup = false;
|
|
if ($isSystemUser && $shareeEnumerationInGroupOnly) {
|
|
foreach ($userGroups as $userGroup) {
|
|
if ($this->groupManager->isInGroup($r['UID'], $userGroup)) {
|
|
$isInSameGroup = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!$shareeEnumerationFullMatch && !$isInSameGroup) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
$id = $r['UID'];
|
|
$fn = $r['FN'] ?? null;
|
|
if (!isset($r['EMAIL'])) {
|
|
continue;
|
|
}
|
|
$email = $r['EMAIL'];
|
|
if (!is_array($email)) {
|
|
$email = [$email];
|
|
}
|
|
$photo = isset($r['PHOTO']) ? $this->getPhotoUri($r['PHOTO']) : null;
|
|
|
|
// loop through all email addresses of this contact
|
|
foreach ($email as $e) {
|
|
if ($e === '') {
|
|
continue;
|
|
}
|
|
$lowerTerm = strtolower($term);
|
|
if (!$isSystemUser || $isInSameGroup || ($lowerTerm !== '' && (
|
|
($shareeEnumerationFullMatch && !empty($fn) && $lowerTerm === strtolower($fn)) ||
|
|
($shareeEnumerationFullMatchUserId && $lowerTerm === strtolower($id)) ||
|
|
($shareeEnumerationFullMatchEmail && $lowerTerm === strtolower($e))))) {
|
|
$receivers[] = [
|
|
'id' => $id,
|
|
// Show full name if possible or fall back to email
|
|
'label' => (empty($fn) ? $e : "$fn ($e)"),
|
|
'email' => $e,
|
|
'photo' => $photo,
|
|
];
|
|
}
|
|
}
|
|
}
|
|
|
|
return $receivers;
|
|
}
|
|
|
|
/**
|
|
* @param string $email
|
|
*
|
|
* @return false|null|string
|
|
*/
|
|
public function getPhoto(string $email) {
|
|
$result = $this->contactsManager->search($email, ['EMAIL']);
|
|
foreach ($result as $contact) {
|
|
if (!isset($contact['PHOTO']) || empty($contact['PHOTO'])) {
|
|
continue;
|
|
}
|
|
return $this->getPhotoUri($contact['PHOTO']);
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @return false|null|string
|
|
*/
|
|
private function getPhotoUri(string $raw) {
|
|
$uriPrefix = 'VALUE=uri:';
|
|
if (substr($raw, 0, strlen($uriPrefix)) === $uriPrefix) {
|
|
return substr($raw, strpos($raw, 'http'));
|
|
} else {
|
|
// ignore contacts >= 1.0 binary images
|
|
// TODO: fix
|
|
return null;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds a new email to an existing Contact
|
|
*
|
|
* @param string $uid
|
|
* @param string $mailAddr
|
|
* @param string $type
|
|
* @return array|null
|
|
*/
|
|
public function addEmailToContact(string $uid, string $mailAddr, string $type = 'HOME') {
|
|
if (!$this->contactsManager->isEnabled()) {
|
|
return null;
|
|
}
|
|
|
|
$result = $this->contactsManager->search($uid, ['UID'], ['types' => true, 'limit' => 1]);
|
|
|
|
if (count($result) !== 1) {
|
|
return null; // no match
|
|
}
|
|
|
|
$newEntry = [
|
|
'type' => $type,
|
|
'value' => $mailAddr
|
|
];
|
|
|
|
$match = $result[0];
|
|
$email = $match['EMAIL'] ?? [];
|
|
if (!empty($email) && !is_array($email[0])) {
|
|
$email = [$email];
|
|
}
|
|
$email[] = $newEntry;
|
|
$match['EMAIL'] = $email;
|
|
|
|
$updatedContact = $this->contactsManager->createOrUpdate($match, $match['addressbook-key']);
|
|
return $updatedContact;
|
|
}
|
|
|
|
/**
|
|
* Adds a new contact with the specified email to an addressbook
|
|
*
|
|
* @param string $uid
|
|
* @param string $mailAddr
|
|
* @param string $addressbook
|
|
* @return array|null
|
|
*/
|
|
public function newContact(string $name, string $mailAddr, string $type = 'HOME', string $addressbook = null) {
|
|
if (!$this->contactsManager->isEnabled()) {
|
|
return null;
|
|
}
|
|
|
|
if (!isset($addressbook)) {
|
|
$addressbook = key($this->contactsManager->getUserAddressBooks());
|
|
}
|
|
|
|
$contact = [
|
|
'FN' => $name,
|
|
'EMAIL' => [
|
|
[
|
|
'type' => $type,
|
|
'value' => $mailAddr
|
|
]
|
|
]
|
|
];
|
|
$createdContact = $this->contactsManager->createOrUpdate($contact, $addressbook);
|
|
return $createdContact;
|
|
}
|
|
|
|
/**
|
|
* @param string[] $fields
|
|
*/
|
|
private function doSearch(string $term, array $fields, bool $strictSearch): array {
|
|
$allowSystemUsers = $this->config->getAppValue('core', 'shareapi_allow_share_dialog_user_enumeration', 'no') === 'yes';
|
|
|
|
$result = $this->contactsManager->search($term, $fields, [
|
|
'strict_search' => $strictSearch
|
|
]);
|
|
$matches = [];
|
|
foreach ($result as $r) {
|
|
if (!$allowSystemUsers && isset($r['isLocalSystemBook']) && $r['isLocalSystemBook']) {
|
|
continue;
|
|
}
|
|
$id = $r['UID'];
|
|
$fn = $r['FN'];
|
|
$matches[] = [
|
|
'id' => $id,
|
|
'label' => $fn,
|
|
];
|
|
}
|
|
return $matches;
|
|
}
|
|
|
|
/**
|
|
* Extracts all Contacts with the specified mail address
|
|
*
|
|
* @param string $mailAddr
|
|
* @return array
|
|
*/
|
|
public function getContactsWithMail(string $mailAddr) {
|
|
return $this->doSearch($mailAddr, ['EMAIL'], true);
|
|
}
|
|
|
|
/**
|
|
* Extracts all Contacts with the specified name
|
|
*
|
|
* @param string $mailAddr
|
|
* @return array
|
|
*/
|
|
public function getContactsWithName(string $name) {
|
|
return $this->doSearch($name, ['FN'], false);
|
|
}
|
|
}
|