2020-11-10 04:49:18 +08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
use MailSo\Log\Enumerations\Type;
|
|
|
|
use MailSo\Log\Logger;
|
|
|
|
use RainLoop\Model\Account;
|
|
|
|
use RainLoop\Model\Identity;
|
|
|
|
use RainLoop\Providers\Identities\IIdentities;
|
|
|
|
|
|
|
|
class LdapIdentities implements IIdentities
|
|
|
|
{
|
2020-11-10 17:19:07 +08:00
|
|
|
/** @var resource */
|
|
|
|
private $ldap;
|
|
|
|
|
|
|
|
/** @var bool */
|
|
|
|
private $ldapAvailable = true;
|
|
|
|
/** @var bool */
|
|
|
|
private $ldapConnected = false;
|
|
|
|
/** @var bool */
|
|
|
|
private $ldapBound = false;
|
|
|
|
|
|
|
|
/** @var LdapConfig */
|
|
|
|
private $config;
|
|
|
|
|
|
|
|
/** @var Logger */
|
|
|
|
private $logger;
|
|
|
|
|
|
|
|
private const LOG_KEY = "Ldap";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* LdapIdentities constructor.
|
|
|
|
*
|
|
|
|
* @param LdapConfig $config
|
|
|
|
* @param Logger $logger
|
|
|
|
*/
|
|
|
|
public function __construct(LdapConfig $config, Logger $logger)
|
|
|
|
{
|
|
|
|
$this->config = $config;
|
|
|
|
$this->logger = $logger;
|
|
|
|
|
|
|
|
// Check if LDAP is available
|
|
|
|
if (!extension_loaded('ldap') || !function_exists('ldap_connect')) {
|
|
|
|
$this->ldapAvailable = false;
|
2021-08-28 05:49:03 +08:00
|
|
|
$logger->Write("The LDAP extension is not available!", Type::WARNING, self::LOG_KEY);
|
2020-11-10 17:19:07 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->Connect();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritDoc
|
|
|
|
*/
|
|
|
|
public function GetIdentities(Account $account): array
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
$this->EnsureBound();
|
|
|
|
} catch (LdapException $e) {
|
|
|
|
return []; // exceptions are only thrown from the handleerror function that does logging already
|
|
|
|
}
|
|
|
|
|
|
|
|
$identities = [];
|
|
|
|
|
|
|
|
// Try and get identity information
|
|
|
|
$username = @ldap_escape($account->Email(), "", LDAP_ESCAPE_FILTER);
|
|
|
|
|
|
|
|
try {
|
|
|
|
$userResults = $this->FindLdapResults(
|
|
|
|
$this->config->user_field_search,
|
|
|
|
$username,
|
|
|
|
$this->config->user_base,
|
|
|
|
$this->config->user_objectclass,
|
|
|
|
$this->config->user_field_name,
|
|
|
|
$this->config->user_field_mail
|
|
|
|
);
|
|
|
|
} catch (LdapException $e) {
|
|
|
|
return []; // exceptions are only thrown from the handleerror function that does logging already
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count($userResults) < 1) {
|
|
|
|
$this->logger->Write("Could not find user $username", Type::NOTICE, self::LOG_KEY);
|
|
|
|
return [];
|
|
|
|
} else if (count($userResults) > 1) {
|
|
|
|
$this->logger->Write("Found multiple matches for user $username", Type::WARNING, self::LOG_KEY);
|
|
|
|
}
|
|
|
|
|
|
|
|
$userResult = $userResults[0];
|
|
|
|
|
|
|
|
foreach ($userResult->emails as $email) {
|
|
|
|
$identity = new Identity($email, $email);
|
|
|
|
$identity->SetName($userResult->name);
|
|
|
|
|
|
|
|
if ($email === $account->Email())
|
|
|
|
$identity->SetId(""); // primary identity
|
|
|
|
|
|
|
|
$identities[] = $identity;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$this->config->group_get)
|
|
|
|
return $identities;
|
|
|
|
|
|
|
|
try {
|
|
|
|
$groupResults = $this->FindLdapResults(
|
|
|
|
$this->config->group_field_member,
|
|
|
|
$userResult->dn,
|
|
|
|
$this->config->group_base,
|
|
|
|
$this->config->group_objectclass,
|
|
|
|
$this->config->group_field_name,
|
|
|
|
$this->config->group_field_mail
|
|
|
|
);
|
|
|
|
} catch (LdapException $e) {
|
|
|
|
return []; // exceptions are only thrown from the handleerror function that does logging already
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($groupResults as $group) {
|
|
|
|
foreach ($group->emails as $email) {
|
|
|
|
$name = $this->config->group_sender_format;
|
|
|
|
$name = str_replace("#USER#", $userResult->name, $name);
|
|
|
|
$name = str_replace("#GROUP#", $group->name, $name);
|
|
|
|
|
|
|
|
$identity = new Identity($email, $email);
|
|
|
|
$identity->SetName($name);
|
|
|
|
$identity->SetBcc($email);
|
|
|
|
|
|
|
|
$identities[] = $identity;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $identities;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritDoc
|
|
|
|
* @throws \RainLoop\Exceptions\Exception
|
|
|
|
*/
|
2020-11-13 16:56:40 +08:00
|
|
|
public function SetIdentities(Account $account, array $identities): void
|
2020-11-10 17:19:07 +08:00
|
|
|
{
|
|
|
|
throw new \RainLoop\Exceptions\Exception("Ldap identities provider does not support storage");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritDoc
|
|
|
|
*/
|
|
|
|
public function SupportsStore(): bool
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritDoc
|
|
|
|
*/
|
|
|
|
public function Name(): string
|
|
|
|
{
|
|
|
|
return "Ldap";
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @throws LdapException */
|
|
|
|
private function EnsureConnected(): void
|
|
|
|
{
|
|
|
|
if ($this->ldapConnected) return;
|
|
|
|
|
|
|
|
$res = $this->Connect();
|
|
|
|
if (!$res)
|
|
|
|
$this->HandleLdapError("Connect");
|
|
|
|
}
|
|
|
|
|
|
|
|
private function Connect(): bool
|
|
|
|
{
|
|
|
|
// Set up connection
|
|
|
|
$ldap = @ldap_connect($this->config->server);
|
|
|
|
if ($ldap === false) {
|
|
|
|
$this->ldapAvailable = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set protocol version
|
|
|
|
$option = @ldap_set_option($ldap, LDAP_OPT_PROTOCOL_VERSION, $this->config->protocol);
|
|
|
|
if (!$option) {
|
|
|
|
$this->ldapAvailable = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->ldap = $ldap;
|
|
|
|
$this->ldapConnected = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @throws LdapException */
|
|
|
|
private function EnsureBound(): void
|
|
|
|
{
|
|
|
|
if ($this->ldapBound) return;
|
|
|
|
$this->EnsureConnected();
|
|
|
|
|
|
|
|
$res = $this->Bind();
|
|
|
|
if (!$res)
|
|
|
|
$this->HandleLdapError("Bind");
|
|
|
|
}
|
|
|
|
|
|
|
|
private function Bind(): bool
|
|
|
|
{
|
|
|
|
// Bind to LDAP here
|
|
|
|
$bindResult = @ldap_bind($this->ldap, $this->config->bind_user, $this->config->bind_password);
|
|
|
|
if (!$bindResult) {
|
|
|
|
$this->ldapAvailable = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->ldapBound = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $op
|
|
|
|
* @throws LdapException
|
|
|
|
*/
|
|
|
|
private function HandleLdapError(string $op = ""): void
|
|
|
|
{
|
|
|
|
// Obtain LDAP error and write logs
|
|
|
|
$errorNo = @ldap_errno($this->ldap);
|
|
|
|
$errorMsg = @ldap_error($this->ldap);
|
|
|
|
|
|
|
|
$message = empty($op) ? "LDAP Error: {$errorMsg} ({$errorNo})" : "LDAP Error during {$op}: {$errorMsg} ({$errorNo})";
|
|
|
|
$this->logger->Write($message, Type::ERROR, self::LOG_KEY);
|
|
|
|
throw new LdapException($message, $errorNo);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $searchField
|
|
|
|
* @param string $searchValue
|
|
|
|
* @param string $searchBase
|
|
|
|
* @param string $objectClass
|
|
|
|
* @param string $nameField
|
|
|
|
* @param string $mailField
|
|
|
|
* @return LdapResult[]
|
|
|
|
* @throws LdapException
|
|
|
|
*/
|
|
|
|
private function FindLdapResults(string $searchField, string $searchValue, string $searchBase, string $objectClass, string $nameField, string $mailField): array
|
|
|
|
{
|
|
|
|
$this->EnsureBound();
|
|
|
|
|
|
|
|
$nameField = strtolower($nameField);
|
|
|
|
$mailField = strtolower($mailField);
|
|
|
|
|
|
|
|
$filter = "(&(objectclass=$objectClass)($searchField=$searchValue))";
|
|
|
|
$ldapResult = @ldap_search($this->ldap, $searchBase, $filter, ['dn', $mailField, $nameField]);
|
|
|
|
if (!$ldapResult) {
|
|
|
|
$this->HandleLdapError("Fetch $objectClass");
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
$entries = @ldap_get_entries($this->ldap, $ldapResult);
|
|
|
|
if (!$entries) {
|
|
|
|
$this->HandleLdapError("Fetch $objectClass");
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
$results = [];
|
|
|
|
for ($i = 0; $i < $entries["count"]; $i++) {
|
|
|
|
$entry = $entries[$i];
|
|
|
|
|
|
|
|
$result = new LdapResult();
|
|
|
|
$result->dn = $entry["dn"];
|
|
|
|
$result->name = $this->LdapGetAttribute($entry, $nameField, true, true);
|
|
|
|
$result->emails = $this->LdapGetAttribute($entry, $mailField, false, false);
|
|
|
|
|
|
|
|
$results[] = $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $results;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array $entry
|
|
|
|
* @param string $attribute
|
|
|
|
* @param bool $single
|
|
|
|
* @param bool $required
|
|
|
|
* @return string|string[]
|
|
|
|
*/
|
|
|
|
private function LdapGetAttribute(array $entry, string $attribute, bool $single = true, bool $required = false)
|
|
|
|
{
|
|
|
|
if (!isset($entry[$attribute])) {
|
|
|
|
if ($required)
|
|
|
|
$this->logger->Write("Attribute $attribute not found on object {$entry['dn']} while required", Type::NOTICE, self::LOG_KEY);
|
|
|
|
|
|
|
|
return $single ? "" : [];
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($single) {
|
|
|
|
if ($entry[$attribute]["count"] > 1)
|
|
|
|
$this->logger->Write("Attribute $attribute is multivalues while only a single value is expected", Type::NOTICE, self::LOG_KEY);
|
|
|
|
|
|
|
|
return $entry[$attribute][0];
|
|
|
|
}
|
|
|
|
|
|
|
|
$result = $entry[$attribute];
|
|
|
|
unset($result["count"]);
|
|
|
|
return array_values($result);
|
|
|
|
}
|
2020-11-10 04:49:18 +08:00
|
|
|
}
|
|
|
|
|
2020-11-10 17:19:07 +08:00
|
|
|
class LdapResult
|
|
|
|
{
|
|
|
|
/** @var string */
|
|
|
|
public $dn;
|
2020-11-10 04:49:18 +08:00
|
|
|
|
2020-11-10 17:19:07 +08:00
|
|
|
/** @var string */
|
|
|
|
public $name;
|
2020-11-10 04:49:18 +08:00
|
|
|
|
2020-11-10 17:19:07 +08:00
|
|
|
/** @var string[] */
|
|
|
|
public $emails;
|
2020-11-13 16:56:40 +08:00
|
|
|
}
|