1
0
mirror of https://github.com/owncloudarchive/contacts.git synced 2024-12-10 21:24:33 +01:00
OwncloudContactsOfficial/lib/backend/ldap.php

714 lines
20 KiB
PHP
Raw Normal View History

<?php
/**
* ownCloud - Base class for Contacts backends
*
2013-05-09 14:07:27 +02:00
* @author Nicolas Mora
* @copyright 2013 Nicolas Mora (mail@babelouest.org)
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
* License as published by the Free Software Foundation; either
* version 3 of the License, or any later version.
*
* This library 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 library. If not, see <http://www.gnu.org/licenses/>.
*
*/
namespace OCA\Contacts\Backend;
use OCA\Contacts\Contact;
use OCA\Contacts\VObject\VCard;
use Sabre\VObject\Reader;
2013-05-09 14:07:27 +02:00
use OCA\Contacts\Connector\LdapConnector;
/**
* Subclass this class for Cantacts backends
*/
class Ldap extends AbstractBackend {
/**
* The name of the backend.
* @var string
*/
2013-05-09 14:07:27 +02:00
public $name='ldap';
static private $preparedQueries = array();
2013-05-09 14:07:27 +02:00
private $ldapConnection = null;
private $connector = null;
/**
2013-08-04 04:40:52 +02:00
* @brief validates and sets the ldap parameters
* @param $ldapParams array containing the parameters
* return boolean
*/
2013-05-09 14:07:27 +02:00
public function setLdapParams($aid) {
2013-08-04 04:40:52 +02:00
$tmp = $this->getPreferences($aid);
if ($tmp != false) {
$this->ldapParams = $tmp;
2013-05-09 14:07:27 +02:00
$this->connector = new LdapConnector($this->ldapParams['ldap_vcard_connector']);
return true;
} else {
return false;
}
}
2013-08-04 04:40:52 +02:00
2013-05-09 14:07:27 +02:00
/**
* @brief creates the ldap connection, then binds it according to the parameters previously given
* @return boolean connexion status
*/
public function ldapCreateAndBindConnection() {
if (!self::ldapIsConnected() && $this->ldapParams != null) {
// ldap connect
$this->ldapConnection = ldap_connect($this->ldapParams['ldapurl']);
ldap_set_option($this->ldapConnection, LDAP_OPT_REFERRALS, 0);
ldap_set_option($this->ldapConnection, LDAP_OPT_PROTOCOL_VERSION, 3);
if ($this->ldapConnection) {
// ldap bind
if ($this->ldapParams['ldapanonymous'] == 1) {
$ldapbind = ldap_bind($this->ldapConnection);
} else {
2013-08-04 04:40:52 +02:00
$ldapbind = ldap_bind($this->ldapConnection, $this->ldapParams['ldapuser'], base64_decode($this->ldapParams['ldappass']));
2013-05-09 14:07:27 +02:00
}
return $ldapbind;
}
return false;
}
return self::ldapIsConnected();
}
/**
* @brief close the current connection
* @return boolean closing success
*/
public function ldapCloseConnection() {
if (self::ldapIsConnected()) {
ldap_unbind($this->ldapConnection);
$this->ldapConnection = null;
return true;
}
}
/**
*
*/
public function ldapIsConnected() {
return ($this->ldapConnection != null);
}
/**
* @brief search a list in ldap server
2013-05-09 14:07:27 +02:00
* @param $ldapbasedn the base dn
* @param $bindsearch the search filter
* @param $entries the ldap entries to reach
* @param $start the starting point
* @param $num the number of entries to return
* @return array|false
*/
public function ldapFindMultiple($ldapbasedn, $bindsearch, $entriesName, $start=null, $num=null) {
if (($entriesName != null) && self::ldapCreateAndBindConnection() && $ldapbasedn != null && $bindsearch != null) {
2013-05-09 14:07:27 +02:00
if ($start==null) {
$start=0;
}
if ($num==null) {
$num=PHP_INT_MAX;
}
$pageSize = isset($this->ldapParams['ldappagesize'])?$this->ldapParams['ldappagesize']:"20";
$cookie = '';
$entries = array();
$cpt=0;
\OC_Log::write('contacts_ldap', __METHOD__." - search what $ldapbasedn, $bindsearch $pageSize", \OC_Log::DEBUG);
do {
ldap_control_paged_result($this->ldapConnection, $pageSize, true, $cookie);
$ldap_results = ldap_search ($this->ldapConnection, $ldapbasedn, $bindsearch, $entriesName);
$LdapEntries = ldap_get_entries ($this->ldapConnection, $ldap_results);
for ($i=0; $i<$LdapEntries['count']; $i++) {
$entries[] = $LdapEntries[$i];
}
ldap_control_paged_result_response($this->ldapConnection, $ldap_results, $cookie);
$cpt++;
} while($cookie !== null && $cookie != '' && $cpt < 10);
$entries['count'] = count($entries);
return $entries;
self::ldapCloseConnection();
}
return false;
}
/**
* @brief search one contact in ldap server
* @param $ldapbasedn the base dn
* @param $bindsearch the search filter
* @param $entries the ldap entries to reach
* @param $start the starting point
* @param $num the number of entries to return
* @return array|false
*/
public function ldapFindOne($ldapbasedn, $bindsearch, $entries, $start=null, $num=null) {
if (($entries != null) && self::ldapCreateAndBindConnection() && $ldapbasedn != null && $bindsearch != null) {
if ($start==null) {
$start=0;
}
if ($num==null) {
$num=PHP_INT_MAX;
}
2014-02-23 22:33:50 +01:00
\OC_Log::write('contacts_ldap', __METHOD__." - search $ldapbasedn, $bindsearch ", \OC_Log::DEBUG);
2013-06-02 21:50:57 +02:00
$ldap_results = @ldap_search ($this->ldapConnection, $ldapbasedn, $bindsearch, $entries);
2013-05-09 14:07:27 +02:00
if ($ldap_results) {
$entries = ldap_get_entries ($this->ldapConnection, $ldap_results);
2013-06-02 21:50:57 +02:00
if ($entries['count'] > 0) {
return $entries[0];
} else {
return false;
}
2013-05-09 14:07:27 +02:00
}
self::ldapCloseConnection();
2013-05-09 14:07:27 +02:00
}
return false;
}
2013-05-09 14:07:27 +02:00
/**
* @brief adds a new ldap entry
* @param $ldapDN the new DN (must be unique)
* @param $ldapValues the ldif values
* @return boolean insert status
*/
public function ldapAdd($ldapDN, $ldapValues) {
if (self::ldapIsConnected()) {
return @ldap_add($this->ldapConnection, $ldapDN, $ldapValues);
}
return false;
}
/**
* @brief modify a ldap entry
* @param $ldapDN the DN (must exists)
* @param $ldapValues the ldif values
* @return boolean modify status
*/
public function ldapUpdate($ldapDN, $ldapValues) {
if (self::ldapIsConnected()) {
$result = @ldap_modify($this->ldapConnection, $ldapDN, $ldapValues);
if (!$result) {
self::ldapDelete($ldapDN);
return self::ldapAdd($ldapDN, $ldapValues);
}
return true;
}
return false;
}
/**
* @brief delete a ldap entry
* @param $ldapDN the DN (must exists)
* @return boolean delete status
*/
public function ldapDelete($ldapDN) {
if (self::ldapIsConnected()) {
ldap_delete($this->ldapConnection, $ldapDN);
return true;
2013-05-09 14:07:27 +02:00
}
return false;
2013-05-09 14:07:27 +02:00
}
/**
* Sets up the backend
*
* @param string $cardsTableName
*/
public function __construct(
$userid = null,
$addressBooksTableName = '*PREFIX*contacts_ldap_addressbooks'
) {
$this->userid = $userid ? $userid : \OCP\User::getUser();
$this->addressbooks = array();
}
/**
2013-05-09 14:07:27 +02:00
* Returns the list of active addressbooks for a specific user.
2013-08-04 04:40:52 +02:00
*
2013-05-09 14:07:27 +02:00
* @param string $userid
* @return array
*/
public function getAddressBooksForUser(array $options = array()) {
2014-02-07 21:55:55 +01:00
$addressbookidList = $this->getAddressbookList();
$this->addressbooks = array();
foreach($addressbookidList as $addressbookid) {
$this->addressbooks[] = self::getAddressBook($addressbookid);
2013-05-09 14:07:27 +02:00
}
2014-02-07 21:55:55 +01:00
return $this->addressbooks;
2013-08-04 04:40:52 +02:00
}
/**
* Get an addressbook's properties
*
* The returned array MUST contain 'displayname' and an integer 'permissions'
* value using there ownCloud CRUDS constants (which MUST be at least
* \OCP\PERMISSION_READ).
* Currently the only ones supported are 'displayname' and
* 'description', but backends can implement additional.
*
* @param string $addressbookid
* @return array $properties
*/
public function getAddressBook($addressbookid, array $options = array()) {
//\OC_Log::write('contacts', __METHOD__.' id: '
// . $addressbookid, \OC_Log::DEBUG);
if($this->addressbooks && isset($this->addressbooks[$addressbookid])) {
//print(__METHOD__ . ' ' . __LINE__ .' addressBookInfo: ' . print_r($this->addressbooks[$addressbookid], true));
return $this->addressbooks[$addressbookid];
}
// Hmm, not found. Lets query the db.
2013-08-04 04:40:52 +02:00
$preferences = self::getPreferences($addressbookid);
if ($preferences != false) {
$current = array();
2014-02-07 21:55:55 +01:00
$current['id'] = (string)$addressbookid;
$current['displayname'] = (string)$preferences['displayname'];
$current['description'] = (string)$preferences['description'];
2013-08-04 04:40:52 +02:00
$current['owner'] = $this->userid;
2014-02-07 21:55:55 +01:00
$current['uri'] = (string)$preferences['uri'];
2013-08-04 04:40:52 +02:00
$current['permissions'] = \OCP\PERMISSION_ALL;
$current['lastmodified'] = self::lastModifiedAddressBook($addressbookid);
2013-08-04 04:40:52 +02:00
return $current;
} else {
return array();
}
}
/**
* Test if the address book exists
* @return bool
*/
public function hasAddressBook($addressbookid) {
if($this->addressbooks && isset($this->addressbooks[$addressbookid])) {
return true;
}
return count($this->getAddressBook($addressbookid)) > 0;
}
/**
* Updates an addressbook's properties
*
* The $properties array contains the changes to be made.
*
* Currently the only ones supported are 'displayname' and
* 'description', but backends can implement additional.
*
* @param string $addressbookid
* @param array $properties
* @return bool
*/
public function updateAddressBook($addressbookid, array $properties, array $options = array()) {
2013-08-04 04:40:52 +02:00
// TODO: use backend settings
return true;
}
/**
* Creates a new address book
*
* Currently the only ones supported are 'displayname' and
* 'description', but backends can implement additional.
* 'displayname' MUST be present.
*
* @param array $properties
* @return string|false The ID if the newly created AddressBook or false on error.
*/
public function createAddressBook(array $properties, array $options = array()) {
2013-08-04 04:40:52 +02:00
// TODO: use backend settings
}
/**
* Deletes an entire addressbook and all its contents
*
* @param string $addressbookid
* @return bool
*/
public function deleteAddressBook($addressbookid, array $options = array()) {
$addressbook = self::getAddressBook($addressbookid);
2013-08-04 04:40:52 +02:00
// TODO: use backend settings
return true;
}
/**
* @brief Get the last modification time for an address book.
*
* Must return a UNIX time stamp or null if the backend
* doesn't support it.
*
* TODO: Implement default methods get/set for backends that
* don't support.
* @param string $addressbookid
* @returns int | null
*/
public function lastModifiedAddressBook($addressbookid, array $options = array()) {
2013-08-04 04:40:52 +02:00
return null;
}
/**
* Returns all contacts for a specific addressbook id.
*
* The returned array MUST contain the unique ID of the contact mapped to 'id', a
* displayname mapped to 'displayname' and an integer 'permissions' value using there
* ownCloud CRUDS constants (which MUST be at least \OCP\PERMISSION_READ), and SHOULD
* contain the properties of the contact formatted as a vCard 3.0
* https://tools.ietf.org/html/rfc2426 mapped to 'carddata' or as an
* \OCA\Contacts\VObject\VCard object mapped to 'vcard'.
*
* Example:
*
* array(
* 0 => array('id' => '4e111fef5df', 'permissions' => 1, 'displayname' => 'John Q. Public', 'vcard' => $object),
* 1 => array('id' => 'bbcca2d1535', 'permissions' => 32, 'displayname' => 'Jane Doe', 'carddata' => $data)
* );
*
* For contacts that contain loads of data, the 'carddata' or 'vcard' MAY be omitted
* as it can be fetched later.
*
* TODO: Some sort of ETag?
*
* @param string $addressbookid
* @param bool $omitdata Don't fetch the entire carddata or vcard.
* @return array
*/
public function getContacts($addressbookid, array $options = array()) {
$cards = array();
2013-05-09 14:07:27 +02:00
$vcards = array();
if(is_array($addressbookid) && count($addressbookid)) {
$id_array = $addressbookid;
} elseif(is_int($addressbookid) || is_string($addressbookid)) {
$id_array = array($addressbookid);
} else {
\OC_Log::write('contacts_ldap', __METHOD__.'. Addressbook id(s) argument is empty: '. print_r($id, true), \OC_Log::DEBUG);
return false;
}
foreach ($id_array as $one_id) {
2013-05-09 14:07:27 +02:00
if (self::setLdapParams($one_id)) {
//OCP\Util::writeLog('contacts_ldap', __METHOD__.' Connector OK', \OC_Log::DEBUG);
$info = self::ldapFindMultiple(
$this->ldapParams['ldapbasednsearch'],
2014-02-07 21:55:55 +01:00
$this->ldapParams['ldapfilter'],
$this->connector->getLdapEntries(),
isset($options['offset']) ? $options['offset'] : null,
isset($options['limit']) ? $options['limit'] : null
);
2013-05-09 14:07:27 +02:00
for ($i=0; $i<$info["count"]; $i++) {
$a_card = $this->connector->ldapToVCard($info[$i]);
$cards[] = self::getSabreFormatCard($addressbookid, $a_card);
}
//OCP\Util::writeLog('contacts_ldap', __METHOD__.' counts '.count($cards), \OC_Log::DEBUG);
}
}
return $cards;
}
/**
* Returns a specfic contact.
*
* Same as getContacts except that either 'carddata' or 'vcard' is mandatory.
*
* @param string $addressbookid
* @param mixed $id
* @return array|bool
*/
public function getContact($addressbookid, $ids, array $options = array()) {
2013-05-09 14:07:27 +02:00
if (!is_array($ids)) {
$a_ids = array($ids);
} else {
$a_ids = $ids;
}
$cards = array();
$toReturn = false;
self::setLdapParams($addressbookid);
if (self::setLdapParams($addressbookid)) {
foreach ($a_ids as $id) {
$cid = str_replace(".vcf", "", $id);
if (ldap_explode_dn(base64_decode($cid),0) == false) {
$ldifEntry = $this->connector->getLdifEntry("X-URI", null);
$filter = "";
if (isset($ldifEntry[0]['unassigned'])) {
$filter = $this->connector->getUnassignedVCardProperty() . "=X-URI:" . $cid ."*";
} else {
$filter = $ldifEntry[0]['name'] . "=" . $cid ."*";
}
$card = self::ldapFindOne($this->ldapParams['ldapbasednsearch'],
2013-05-09 14:07:27 +02:00
$filter,
$this->connector->getLdapEntries());
} else {
$card = self::ldapFindOne(base64_decode($cid),
2013-05-09 14:07:27 +02:00
'objectClass=*',
$this->connector->getLdapEntries());
}
}
if ($card != null) {
return self::getSabreFormatCard($addressbookid, $this->connector->ldapToVCard($card));
2013-05-09 14:07:27 +02:00
}
}
return false;
}
/**
* @brief construct a vcard in Sabre format
* @param integer $aid Addressbook Id
* @param OC_VObject $card VCard
* @return array
*/
public static function getSabreFormatCard($aid, $vcard) {
/*
* array return format :
* array( 'id' => 'bbcca2d1535',
* 'permissions' => 32,
* 'displayname' => 'Jane Doe',
* 'carddata' => $data)
*/
$FN = (string)$vcard->FN;
$UID = (string)$vcard->UID;
$REV = (string)$vcard->REV;
2013-05-09 14:07:27 +02:00
if (isset($vcard->{'X-URI'})) {
$URI = (string)$vcard->{'X-URI'};
} else if (isset($vcard->UID)) {
$URI = (string)$vcard->UID.'.vcf';
} else {
$URI = (string)$vcard->{'X-LDAP-DN'};
}
return array('id' => $UID,
2013-05-09 14:07:27 +02:00
'permissions' => \OCP\PERMISSION_READ,
'displayname' => $FN,
'carddata' => $vcard->serialize(),
'uri' => $URI,
'lastmodified' => $REV);
}
/**
* Creates a new contact
*
* @param string $addressbookid
* @param VCard|string $carddata
* @return string|bool The identifier for the new contact or false on error.
*/
public function createContact($addressbookid, $contact, array $options = array()) {
2014-02-23 22:33:50 +01:00
$backtrace = debug_backtrace();
$trace=array();
foreach ($backtrace as $elt) {
foreach ($elt as $key => $line) {
if ($key == "file" || $key == "line") {
$trace[] = $line;
}
}
}
//error_log("stay added ".print_r($trace,1));
$uri = isset($options['uri']) ? $options['uri'] : null;
2014-02-23 22:33:50 +01:00
// 2014/02/13 Sometimes, a card is created without a name (I don't like that)...
if (!isset($contact->N)) {
$generated = "gruik".rand(0, 65535);
$contact->N = $generated;
$contact->FN = $generated;
error_log("Generated name: $generated");
}
if(!$contact instanceof VCard) {
try {
$contact = Reader::read($contact);
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.', exception: '.$e->getMessage(), \OCP\Util::ERROR);
return false;
}
}
2014-02-23 22:33:50 +01:00
error_log("adding ".$contact->serialize());
try {
$contact->validate(VCard::REPAIR|VCard::UPGRADE);
} catch (\Exception $e) {
OCP\Util::writeLog('contacts', __METHOD__ . ' ' .
'Error validating vcard: ' . $e->getMessage(), \OCP\Util::ERROR);
return false;
}
2013-05-09 14:07:27 +02:00
self::setLdapParams($addressbookid);
self::ldapCreateAndBindConnection();
$newDN = $this->connector->getLdapId() . "=" . $contact->FN . "," . $this->ldapParams['ldapbasednmodify'];
$contact->{'X-LDAP-DN'} = base64_encode($newDN);
2013-05-09 14:07:27 +02:00
if ($uri!=null) {
$contact->{'X-URI'} = $uri;
2014-02-23 22:33:50 +01:00
} else {
$contact->{'X-URI'} = $contact->{'UID'}.".vcf";
2013-05-09 14:07:27 +02:00
}
$ldifEntries = $this->connector->VCardToLdap($contact);
2013-05-09 14:07:27 +02:00
// Inserts the new card
$cardId = self::ldapAdd($newDN, $ldifEntries);
if ($cardId) {
self::ldapCloseConnection();
return $contact->UID;
2013-05-09 14:07:27 +02:00
} else {
self::ldapCloseConnection();
return false;
}
}
/**
* Updates a contact
*
* @param string $addressbookid
* @param mixed $id
* @param string $carddata
* @return bool
*/
public function updateContact($addressbookid, $id, $carddata, array $options = array()) {
2014-02-23 22:33:50 +01:00
$backtrace = debug_backtrace();
$trace=array();
foreach ($backtrace as $elt) {
foreach ($elt as $key => $line) {
if ($key == "file" || $key == "line") {
$trace[] = $line;
}
}
}
//error_log("stay modified ".print_r($trace,1));
2014-02-10 21:16:43 +01:00
if(!$carddata instanceof VCard) {
try {
2014-02-23 22:33:50 +01:00
$vcard = \Sabre\VObject\Reader::read($carddata);
2014-02-10 21:16:43 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.', exception: '.$e->getMessage(), \OCP\Util::ERROR);
return false;
}
} else {
$vcard = $carddata;
}
2013-05-09 14:07:27 +02:00
2014-02-23 22:33:50 +01:00
//error_log("updating ".$vcard->serialize());
2013-05-09 14:07:27 +02:00
if (!is_array($id)) {
$a_ids = array($id);
} else {
$a_ids = $id;
}
self::setLdapParams($addressbookid);
self::ldapCreateAndBindConnection();
$ldifEntries = $this->connector->VCardToLdap($vcard);
foreach ($a_ids as $cid) {
// Never EVER modify an Ldap:dn nor a VCard:UID
if (isset($vcard->{'X-LDAP-DN'})) {
$dn = base64_decode($vcard->{'X-LDAP-DN'});
} else {
// A little bit complicated but hopefully, we won't often go into this
$tmpCard = self::getContact($addressbookid, array($cid));
$tmpVCard = \Sabre\VObject\Reader::read($tmpCard['carddata']);
$dn = base64_decode($tmpVCard->{'X-LDAP-DN'});
}
// Updates the existing card
$result = self::ldapUpdate($dn, $ldifEntries);
}
self::ldapCloseConnection();
return $result;
}
/**
* Deletes a contact
*
* @param string $addressbookid
* @param mixed $id
* @return bool
*/
public function deleteContact($addressbookid, $id, array $options = array()) {
2014-02-23 22:33:50 +01:00
$backtrace = debug_backtrace();
$trace=array();
foreach ($backtrace as $elt) {
foreach ($elt as $key => $line) {
if ($key == "file" || $key == "line") {
$trace[] = $line;
}
}
}
//error_log("stay dead ".print_r($trace, 1));
2013-05-09 14:07:27 +02:00
self::setLdapParams($addressbookid);
self::ldapCreateAndBindConnection();
2014-02-23 22:33:50 +01:00
$card=null;
if (is_array($id)) {
$card = self::getContact($addressbookid, $id);
} else {
$card = self::getContact($addressbookid, array($id));
}
if ($card) {
$vcard = \Sabre\VObject\Reader::read($card['carddata']);
$decodedId = base64_decode($vcard->{'X-LDAP-DN'});
// Deletes the existing card
$result = self::ldapDelete($decodedId);
self::ldapCloseConnection();
return $result;
} else {
return false;
}
}
/**
* @brief Get the last modification time for a contact.
*
* Must return a UNIX time stamp or null if the backend
* doesn't support it.
*
* @param string $addressbookid
* @param mixed $id
* @returns int | null
*/
public function lastModifiedContact($addressbookid, $id) {
2013-08-04 04:40:52 +02:00
$contact = getContact($addressbookid, $id);
if ($contact != null) {
return $contact['lastmodified'];
} else {
return null;
}
}
2013-05-09 14:07:27 +02:00
2014-02-10 21:16:43 +01:00
// Please remove this
public function debug_string_backtrace() {
ob_start();
debug_print_backtrace();
$trace = ob_get_contents();
ob_end_clean();
// Remove first item from backtrace as it's this function which
// is redundant.
$trace = preg_replace ('/^#0\s+' . __FUNCTION__ . "[^\n]*\n/", '', $trace, 1);
// Renumber backtrace items.
$trace = preg_replace ('/^#(\d+)/me', '\'#\' . ($1 - 1)', $trace);
return $trace;
}
}