mirror of
https://github.com/Yetangitu/ampache
synced 2025-10-03 09:49:30 +02:00

move it from Auth::ldap_auth to LDAP::auth add support for other protocol versions add support for StartTLS add support for custom name field => if (givenName = John) and (sn = Doe), "givenName sn" gives "John Doe" add support for custom attribute name for group's member attribute clean code
221 lines
7.6 KiB
PHP
221 lines
7.6 KiB
PHP
<?php
|
|
/**
|
|
*
|
|
* LICENSE: GNU Affero General Public License, version 3 (AGPLv3)
|
|
* Copyright 2001 - 2015 Ampache.org
|
|
*
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
* This class defines custom LDAP exceptions that will be used in the
|
|
* main LDAP class.
|
|
*/
|
|
class LDAPException extends Exception
|
|
{
|
|
/**
|
|
* A LDAPException may be constructed thanks to a message, or an error
|
|
* code. If the given argument is an integer, the exception will be
|
|
* produced with message:
|
|
*
|
|
* LDAP error: [errno] errmsg
|
|
*
|
|
* Otherwise, the provided message will be used.
|
|
*
|
|
* @param mixed $message
|
|
*/
|
|
public function __construct ($message)
|
|
{
|
|
if (is_int (message)) {
|
|
$message = 'LDAP error: [' . $message . '] ' . ldap_err2str($message);
|
|
}
|
|
|
|
debug_event('LDAP', 'Exception: ' . $message, 6);
|
|
parent::__construct ($message);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
* This class handles all the contacts with a LDAP server
|
|
*/
|
|
class LDAP
|
|
{
|
|
/**
|
|
* Constructor
|
|
*
|
|
* This should never be called
|
|
*/
|
|
public function __construct()
|
|
{
|
|
debug_event('LDAP', '__construct has been called. This should not happen', 2);
|
|
}
|
|
|
|
|
|
/**
|
|
* ldap_auth
|
|
*
|
|
* This handles authentication against a LDAP server.
|
|
*
|
|
* @param string $username
|
|
* @param string $password
|
|
* @return array
|
|
*/
|
|
public static function auth ($username, $password)
|
|
{
|
|
try {
|
|
/* Connect to the LDAP
|
|
Note: This does not open a connection. It checks whether
|
|
the given parameters are plausibe and can be used to open a
|
|
connection as soon as one is needed. */
|
|
|
|
if (! $url = AmpConfig::get('ldap_url')) {
|
|
throw new LDAPException('Required configuration value missing: ldap_url');
|
|
}
|
|
|
|
if (! $link = ldap_connect ($url)) {
|
|
throw new LDAPException('Could not connect to ' . $url);
|
|
}
|
|
|
|
/* Set the LDAP protocol version (default: 3) */
|
|
|
|
$protocol_version = AmpConfig::get('ldap_protocol_version', 3);
|
|
if (! ldap_set_option ($link, LDAP_OPT_PROTOCOL_VERSION, $protocol_version)) {
|
|
throw new LDAPException('Could not set option PROTOCOL_VERSION to ' . $protocol_version);
|
|
}
|
|
|
|
/* Use StartTLS if asked */
|
|
|
|
if (AmpConfig::get('ldap_start_tls', "false") != "false") {
|
|
if (! ldap_start_tls ($link)) {
|
|
throw new LDAPException('Could not use StartTLS');
|
|
}
|
|
}
|
|
|
|
/* Connect to the LDAP using the given username and password.
|
|
If these parameters do not exist, an anonymous connection
|
|
will be used */
|
|
|
|
$ampache_username = AmpConfig::get('ldap_username');
|
|
$ampache_password = AmpConfig::get('ldap_password');
|
|
|
|
if (! ldap_bind ($link, $ampache_username, $ampache_password)) {
|
|
throw new LDAPException('Could not bind to server using username `'
|
|
. $ampache_username . '` and password `'
|
|
. $ampache_password . '`');
|
|
}
|
|
|
|
/* Search for the user with given base_dn, filter, objectclass and username */
|
|
|
|
if (! $filter = AmpConfig::get('ldap_filter')) {
|
|
throw new LDAPException('Required configuration value missing: ldap_filter');
|
|
}
|
|
|
|
if (strpos ($filter, '%v') !== false) {
|
|
$filter = str_replace('%v', $username, $filter);
|
|
} else {
|
|
$filter = "($filter=$username)"; // Backward compatibility
|
|
}
|
|
|
|
if (! $objectclass = AmpConfig::get('ldap_objectclass')) {
|
|
throw new LDAPException('Required configuration value missing: ldap_objectclass');
|
|
}
|
|
|
|
$search = "(&(objectclass=$objectclass)$filter)";
|
|
debug_event ('LDAP', 'search: ' . $search, 5);
|
|
|
|
if (! $base_dn = AmpConfig::get('ldap_search_dn')) {
|
|
throw new LDAPException('Required configuration value missing: ldap_search_dn');
|
|
}
|
|
|
|
if (! $result = ldap_search ($link, $base_dn, $search)) {
|
|
throw new LDAPException(ldap_errno($link));
|
|
}
|
|
|
|
/* Bind with the user's DN and the password */
|
|
|
|
if (! $user_entry = ldap_first_entry ($link, $result)) {
|
|
throw new LDAPException('Empty search result');
|
|
}
|
|
|
|
if (! $user_dn = ldap_get_dn ($link, $user_entry)) {
|
|
throw new LDAPException(ldap_errno($link));
|
|
}
|
|
|
|
$user_entry = ldap_get_entries ($link, $result) [0];
|
|
|
|
if (! ldap_bind ($link, $user_dn, $password)) {
|
|
throw new LDAPException('Wrong password');
|
|
}
|
|
|
|
/* Test if the user is in the required group (optional) */
|
|
|
|
if ($group_dn = AmpConfig::get('ldap_require_group')) {
|
|
$member_attribute = AmpConfig::get('ldap_member_attribute', 'member');
|
|
|
|
if (! $group_result = ldap_read ($link, $group_dn, 'objectClass=*', [$member_attribute])) {
|
|
throw new LDAPException("Could not read member attribute `$member_attribute`"
|
|
. " for group `$group_dn`");
|
|
}
|
|
|
|
if (! $group_infos = ldap_get_entries ($link, $group_result) [0]) {
|
|
throw new LDAPException('Empty group search result');
|
|
}
|
|
|
|
if (! in_array ($username, $group_infos[$member_attribute])) {
|
|
throw new LDAPException("`$username` is not member of the group `$group_dn`");
|
|
}
|
|
}
|
|
|
|
/* Obtain name and email field. Reconstruct name field to allow
|
|
custom things like "givenName sn" */
|
|
|
|
$name_field = AmpConfig::get('ldap_name_field', 'cn');
|
|
$name_fields = explode(' ', $name_field);
|
|
$names = array_map (function ($name_field) use ($user_entry) {
|
|
return $user_entry[$name_field][0]; }, $name_fields);
|
|
$name = trim(implode(' ', $names));
|
|
|
|
$email_field = AmpConfig::get('ldap_email_field', 'mail');
|
|
$email = $user_entry[$email_field][0];
|
|
|
|
$return_value = [
|
|
'success' => true,
|
|
'type' => 'ldap',
|
|
'username' => $username,
|
|
'name' => $name,
|
|
'email' => $email
|
|
];
|
|
} catch (LDAPException $e) {
|
|
$message = $e->getMessage();
|
|
|
|
debug_event('LDAP', 'Error during authentication: ' . $message, 3);
|
|
|
|
$return_value = [
|
|
'success' => false,
|
|
'error' => $message
|
|
];
|
|
}
|
|
|
|
ldap_unbind ($link);
|
|
|
|
debug_event('LDAP', 'Return value of authentication: ' . json_encode($return_value), 6);
|
|
|
|
return $return_value;
|
|
}
|
|
}
|