1
0
mirror of https://github.com/owncloudarchive/contacts.git synced 2025-01-22 11:52:18 +01:00

929 lines
26 KiB
PHP
Raw Normal View History

<?php
/**
* ownCloud - Database backend for Contacts
*
* @author Thomas Tanghus
2014-01-26 00:40:22 +01:00
* @copyright 2013-2014 Thomas Tanghus (thomas@tanghus.net)
*
* 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;
2013-11-28 14:15:17 +01:00
use OCA\Contacts\Contact,
OCA\Contacts\VObject\VCard,
OCA\Contacts\Utils\Properties,
Sabre\VObject\Reader;
/**
2014-03-08 17:37:05 +01:00
* Subclass this class for Contacts backends.
*/
class Database extends AbstractBackend {
public $name = 'local';
static private $preparedQueries = array();
2013-03-12 09:15:40 +01:00
2014-03-08 17:37:05 +01:00
/**
* The cached address books.
* @var array[]
*/
public $addressbooks;
/**
* The table that holds the address books.
* @var string
*/
public $addressBooksTableName;
/**
* The table that holds the contact vCards.
* @var string
*/
public $cardsTableName;
/**
* The table that holds the indexed vCard properties.
* @var string
*/
public $indexTableName;
/**
* Sets up the backend
*
* @param string $addressBooksTableName
* @param string $cardsTableName
*/
public function __construct(
$userid = null,
$options = array(
'addressBooksTableName' => '*PREFIX*contacts_addressbooks',
'cardsTableName' => '*PREFIX*contacts_cards',
'indexTableName' => '*PREFIX*contacts_cards_properties'
)
) {
$this->userid = $userid ? $userid : \OCP\User::getUser();
$this->addressBooksTableName = $options['addressBooksTableName'];
$this->cardsTableName = $options['cardsTableName'];
$this->indexTableName = $options['indexTableName'];
$this->addressbooks = array();
}
/**
* Returns the list of addressbooks for a specific user.
*
* @see AbstractBackend::getAddressBooksForUser
* @return array
*/
public function getAddressBooksForUser(array $options = array()) {
try {
2014-03-08 16:22:51 +01:00
if (!isset(self::$preparedQueries['addressbooksforuser'])) {
2013-03-30 04:22:30 +01:00
$sql = 'SELECT `id`, `displayname`, `description`, `ctag` AS `lastmodified`, `userid` AS `owner`, `uri` FROM `'
. $this->addressBooksTableName
. '` WHERE `userid` = ? ORDER BY `displayname`';
self::$preparedQueries['addressbooksforuser'] = \OCP\DB::prepare($sql);
}
2014-03-08 16:22:51 +01:00
$result = self::$preparedQueries['addressbooksforuser']->execute(array($this->userid));
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: ' . \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return $this->addressbooks;
}
2014-03-08 16:22:51 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.' exception: ' . $e->getMessage(), \OCP\Util::ERROR);
return $this->addressbooks;
}
2014-03-08 16:22:51 +01:00
while ($row = $result->fetchRow()) {
$row['permissions'] = \OCP\PERMISSION_ALL;
$this->addressbooks[$row['id']] = $row;
}
2014-03-08 16:22:51 +01:00
return $this->addressbooks;
}
public function getAddressBook($addressbookid, array $options = array()) {
2013-03-12 09:15:40 +01:00
//\OCP\Util::writeLog('contacts', __METHOD__.' id: '
// . $addressbookid, \OCP\Util::DEBUG);
2014-03-08 16:22:51 +01:00
if ($this->addressbooks && isset($this->addressbooks[$addressbookid])) {
//print(__METHOD__ . ' ' . __LINE__ .' addressBookInfo: ' . print_r($this->addressbooks[$addressbookid], true));
return $this->addressbooks[$addressbookid];
2013-03-12 09:15:40 +01:00
}
2014-03-08 16:22:51 +01:00
// Hmm, not found. Lets query the db.
2013-03-12 09:15:40 +01:00
try {
$query = 'SELECT `id`, `displayname`, `description`, `userid` AS `owner`, `ctag` AS `lastmodified`, `uri` FROM `'
. $this->addressBooksTableName
. '` WHERE `id` = ? AND `userid` = ?';
2014-03-08 16:22:51 +01:00
if (!isset(self::$preparedQueries['getaddressbook'])) {
self::$preparedQueries['getaddressbook'] = \OCP\DB::prepare($query);
}
2014-03-08 16:22:51 +01:00
$result = self::$preparedQueries['getaddressbook']->execute(array($addressbookid, $this->userid));
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: '
2013-03-12 09:15:40 +01:00
. \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
2013-05-26 22:15:52 +02:00
return null;
}
2014-03-08 16:22:51 +01:00
$row = $result->fetchRow();
2014-03-08 16:22:51 +01:00
if (!$row) {
return null;
2013-03-12 09:15:40 +01:00
}
2014-03-08 17:37:05 +01:00
$row['permissions'] = \OCP\PERMISSION_ALL;
$row['backend'] = $this->name;
$this->addressbooks[$addressbookid] = $row;
return $row;
2014-03-08 16:22:51 +01:00
2013-03-12 09:15:40 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.' exception: '
. $e->getMessage(), \OCP\Util::ERROR);
2013-05-26 22:15:52 +02:00
return null;
2013-03-12 09:15:40 +01:00
}
2014-03-08 16:22:51 +01:00
2013-03-12 09:15:40 +01:00
}
public function hasAddressBook($addressbookid, array $options = array()) {
2014-03-08 16:22:51 +01:00
// First check if it's already cached
2014-03-08 16:22:51 +01:00
if ($this->addressbooks && isset($this->addressbooks[$addressbookid])) {
return true;
2013-03-12 09:15:40 +01:00
}
2014-03-08 16:22:51 +01:00
return count($this->getAddressBook($addressbookid)) > 0;
2013-03-12 09:15:40 +01:00
}
/**
* Updates an addressbook's properties
*
* @param string $addressbookid
* @param array $changes
* @return bool
*/
public function updateAddressBook($addressbookid, array $changes, array $options = array()) {
2014-03-08 16:22:51 +01:00
if (count($changes) === 0) {
return false;
}
$query = 'UPDATE `' . $this->addressBooksTableName . '` SET ';
$updates = array();
2014-03-08 16:22:51 +01:00
if (isset($changes['displayname'])) {
$query .= '`displayname` = ?, ';
$updates[] = $changes['displayname'];
2014-03-08 16:22:51 +01:00
if ($this->addressbooks && isset($this->addressbooks[$addressbookid])) {
$this->addressbooks[$addressbookid]['displayname'] = $changes['displayname'];
}
2014-03-08 16:22:51 +01:00
}
2014-03-08 16:22:51 +01:00
if (isset($changes['description'])) {
$query .= '`description` = ?, ';
$updates[] = $changes['description'];
2014-03-08 16:22:51 +01:00
if ($this->addressbooks && isset($this->addressbooks[$addressbookid])) {
$this->addressbooks[$addressbookid]['description'] = $changes['description'];
}
2014-03-08 16:22:51 +01:00
}
2013-04-27 02:20:12 +02:00
$query .= '`ctag` = ? + 1 WHERE `id` = ?';
$updates[] = time();
$updates[] = $addressbookid;
try {
2014-03-08 16:22:51 +01:00
$stmt = \OCP\DB::prepare($query);
$result = $stmt->execute($updates);
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts',
__METHOD__. 'DB error: '
2013-09-16 02:24:08 +02:00
. \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return false;
}
2014-03-08 16:22:51 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts',
__METHOD__ . ', exception: '
. $e->getMessage(), \OCP\Util::ERROR);
return false;
}
return true;
}
/**
* Creates a new address book
*
* Supported properties are 'displayname', 'description' and 'uri'.
* 'uri' is supported to allow to add from CardDAV requests, and MUST
* be used for the 'uri' database field if present.
* 'displayname' MUST be present.
*
* @param array $properties
* @param array $options - Optional (backend specific options)
* @return string|false The ID if the newly created AddressBook or false on error.
*/
public function createAddressBook(array $properties, array $options = array()) {
2014-03-08 16:22:51 +01:00
if (count($properties) === 0 || !isset($properties['displayname'])) {
return false;
}
$query = 'INSERT INTO `' . $this->addressBooksTableName . '` '
. '(`userid`,`displayname`,`uri`,`description`,`ctag`) VALUES(?,?,?,?,?)';
$updates = array($this->userid, $properties['displayname']);
$updates[] = isset($properties['uri'])
? $properties['uri']
: $this->createAddressBookURI($properties['displayname']);
$updates[] = isset($properties['description']) ? $properties['description'] : '';
$ctag = time();
$updates[] = $ctag;
try {
2014-03-08 16:22:51 +01:00
if (!isset(self::$preparedQueries['createaddressbook'])) {
self::$preparedQueries['createaddressbook'] = \OCP\DB::prepare($query);
}
2014-03-08 16:22:51 +01:00
$result = self::$preparedQueries['createaddressbook']->execute($updates);
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: ' . \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return false;
}
2014-03-08 16:22:51 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__ . ', exception: ' . $e->getMessage(), \OCP\Util::ERROR);
return false;
}
$newid = \OCP\DB::insertid($this->addressBooksTableName);
2014-03-08 16:22:51 +01:00
if ($this->addressbooks) {
$updates['id'] = $newid;
$updates['ctag'] = $ctag;
$updates['lastmodified'] = $ctag;
$updates['permissions'] = \OCP\PERMISSION_ALL;
$this->addressbooks[$newid] = $updates;
}
2014-03-08 16:22:51 +01:00
return $newid;
}
/**
* Deletes an entire addressbook and all its contents
*
2013-05-25 01:35:08 +02:00
* NOTE: For efficience this method bypasses the cleanup hooks and deletes
* property indexes and category/group relations by itself.
*
* @param string $addressbookid
* @param array $options - Optional (backend specific options)
* @return bool
*/
public function deleteAddressBook($addressbookid, array $options = array()) {
// Get all contact ids for this address book
$ids = array();
$result = null;
$stmt = \OCP\DB::prepare('SELECT `id` FROM `' . $this->cardsTableName . '`'
. ' WHERE `addressbookid` = ?');
2014-03-08 16:22:51 +01:00
try {
2014-03-08 16:22:51 +01:00
$result = $stmt->execute(array($addressbookid));
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: '
2013-09-16 02:24:08 +02:00
. \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return false;
}
2014-03-08 16:22:51 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.
', exception: ' . $e->getMessage(), \OCP\Util::ERROR);
return false;
}
2014-03-08 16:22:51 +01:00
if (!is_null($result)) {
while ($id = $result->fetchOne()) {
$ids[] = $id;
}
2014-03-08 16:22:51 +01:00
}
2013-09-16 02:24:08 +02:00
\OCP\Util::emitHook('OCA\Contacts', 'pre_deleteAddressBook',
array('addressbookid' => $addressbookid, 'contactids' => $ids)
);
2013-05-25 01:35:08 +02:00
// Delete contacts in address book.
2014-03-08 16:22:51 +01:00
if (!isset(self::$preparedQueries['deleteaddressbookcontacts'])) {
self::$preparedQueries['deleteaddressbookcontacts'] =
\OCP\DB::prepare('DELETE FROM `' . $this->cardsTableName
. '` WHERE `addressbookid` = ?');
}
2014-03-08 16:22:51 +01:00
try {
self::$preparedQueries['deleteaddressbookcontacts']
->execute(array($addressbookid));
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.
', exception: ' . $e->getMessage(), \OCP\Util::ERROR);
return false;
}
2013-05-25 01:35:08 +02:00
// Delete the address book.
2014-03-08 16:22:51 +01:00
if (!isset(self::$preparedQueries['deleteaddressbook'])) {
self::$preparedQueries['deleteaddressbook'] =
\OCP\DB::prepare('DELETE FROM `'
. $this->addressBooksTableName . '` WHERE `id` = ?');
}
2014-03-08 16:22:51 +01:00
try {
self::$preparedQueries['deleteaddressbook']
->execute(array($addressbookid));
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.
', exception: ' . $e->getMessage(), \OCP\Util::ERROR);
return false;
}
2014-03-08 16:22:51 +01:00
if ($this->addressbooks && isset($this->addressbooks[$addressbookid])) {
unset($this->addressbooks[$addressbookid]);
}
return true;
}
/**
* @brief Updates ctag for addressbook
* @param integer $id
* @return boolean
*/
2013-11-07 14:24:55 +01:00
public function setModifiedAddressBook($id) {
$query = 'UPDATE `' . $this->addressBooksTableName
2013-04-27 02:20:12 +02:00
. '` SET `ctag` = ? + 1 WHERE `id` = ?';
2014-03-08 16:22:51 +01:00
if (!isset(self::$preparedQueries['touchaddressbook'])) {
self::$preparedQueries['touchaddressbook'] = \OCP\DB::prepare($query);
}
2014-03-08 16:22:51 +01:00
2013-04-27 02:20:12 +02:00
$ctag = time();
self::$preparedQueries['touchaddressbook']->execute(array($ctag, $id));
return true;
}
2013-03-16 15:59:23 +01:00
public function lastModifiedAddressBook($addressbookid) {
2014-03-08 16:22:51 +01:00
if ($this->addressbooks && isset($this->addressbooks[$addressbookid])) {
return $this->addressbooks[$addressbookid]['lastmodified'];
}
2014-03-08 16:22:51 +01:00
$addressBook = $this->getAddressBook($addressbookid);
return $addressBook ? $addressBook['lastmodified'] : null;
}
/**
* Returns the number of contacts in a specific address book.
*
* @param string $addressbookid
* @param bool $omitdata Don't fetch the entire carddata or vcard.
* @return array
*/
public function numContacts($addressbookid) {
2014-03-08 16:22:51 +01:00
$query = 'SELECT COUNT(*) AS `count` FROM `' . $this->cardsTableName . '` WHERE '
. '`addressbookid` = ?';
2014-03-08 16:22:51 +01:00
if (!isset(self::$preparedQueries['count'])) {
self::$preparedQueries['count'] = \OCP\DB::prepare($query);
}
2014-03-08 16:22:51 +01:00
$result = self::$preparedQueries['count']->execute(array($addressbookid));
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: ' . \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return null;
}
2014-03-08 16:22:51 +01:00
return (int)$result->fetchOne();
}
/**
* Returns all contacts for a specific addressbook id.
*
* @param string $addressbookid
* @param array $options - Optional (backend specific options)
* @param bool $omitdata Don't fetch the entire carddata or vcard.
* @return array
*/
public function getContacts($addressbookid, array $options = array() ) {
//\OCP\Util::writeLog('contacts', __METHOD__.' addressbookid: ' . $addressbookid, \OCP\Util::DEBUG);
$cards = array();
try {
$omitdata = isset($options['omitdata']) ? $options['omitdata'] : false;
2013-03-16 15:59:23 +01:00
$qfields = $omitdata ? '`id`, `fullname` AS `displayname`' : '*';
$query = 'SELECT ' . $qfields . ' FROM `' . $this->cardsTableName
. '` WHERE `addressbookid` = ? ORDER BY `fullname`';
$stmt = \OCP\DB::prepare(
$query,
isset($options['limit']) ? $options['limit'] : null,
isset($options['offset']) ? $options['offset'] : null
);
2014-03-08 16:22:51 +01:00
$result = $stmt->execute(array($addressbookid));
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: ' . \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return $cards;
}
2014-03-08 16:22:51 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.', exception: '.$e->getMessage(), \OCP\Util::ERROR);
return $cards;
}
2014-03-08 16:22:51 +01:00
if (!is_null($result)) {
while ($row = $result->fetchRow()) {
$row['permissions'] = \OCP\PERMISSION_ALL;
$cards[] = $row;
}
2014-03-08 16:22:51 +01:00
}
return $cards;
}
/**
* Returns a specific contact.
*
2013-03-12 09:15:40 +01:00
* The $id for Database and Shared backends can be an array containing
* either 'id' or 'uri' to be able to play seamlessly with the
* CardDAV backend.
2013-03-25 17:10:21 +01:00
* FIXME: $addressbookid isn't used in the query, so there's no access control.
* OTOH the groups backend - OC_VCategories - doesn't no about parent collections
* only object IDs. Hmm.
* I could make a hack and add an optional, not documented 'nostrict' argument
* so it doesn't look for addressbookid.
*
* @param string $addressbookid
* @param mixed $id Contact ID
* @param array $options - Optional (backend specific options)
* @return array|null
*/
public function getContact($addressbookid, $id, array $options = array()) {
//\OCP\Util::writeLog('contacts', __METHOD__.' identifier: ' . $addressbookid . ' ' . $id['uri'], \OCP\Util::DEBUG);
2013-03-12 09:15:40 +01:00
$noCollection = isset($options['noCollection']) ? $options['noCollection'] : false;
2014-03-08 17:37:05 +01:00
$whereQuery = '`id` = ?';
2014-03-08 16:22:51 +01:00
if (is_array($id)) {
2014-03-08 17:37:05 +01:00
$whereQuery = '';
2014-03-08 16:22:51 +01:00
if (isset($id['id'])) {
2013-03-12 09:15:40 +01:00
$id = $id['id'];
2014-03-08 16:22:51 +01:00
} elseif (isset($id['uri'])) {
2014-03-08 17:37:05 +01:00
$whereQuery = '`uri` = ?';
2013-03-12 09:15:40 +01:00
$id = $id['uri'];
} else {
throw new \Exception(
__METHOD__ . ' If second argument is an array, either \'id\' or \'uri\' has to be set.'
);
}
}
$ids = array($id);
2013-03-25 17:10:21 +01:00
2014-03-08 16:22:51 +01:00
if (!$noCollection) {
2014-03-08 17:37:05 +01:00
$whereQuery .= ' AND `addressbookid` = ?';
2013-03-25 17:10:21 +01:00
$ids[] = $addressbookid;
}
try {
2014-03-08 17:37:05 +01:00
$query = 'SELECT `id`, `uri`, `carddata`, `lastmodified`, `addressbookid` AS `parent`, `fullname` AS `displayname` FROM `'
. $this->cardsTableName . '` WHERE ' . $whereQuery;
$stmt = \OCP\DB::prepare($query);
2013-03-25 17:10:21 +01:00
$result = $stmt->execute($ids);
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: ' . \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return null;
}
2014-03-08 16:22:51 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.', exception: '.$e->getMessage(), \OCP\Util::ERROR);
\OCP\Util::writeLog('contacts', __METHOD__.', id: '. $id, \OCP\Util::DEBUG);
return null;
}
$row = $result->fetchRow();
2014-03-08 16:22:51 +01:00
if (!$row) {
\OCP\Util::writeLog('contacts', __METHOD__.', Not found, id: '. $id, \OCP\Util::DEBUG);
return null;
}
2014-03-08 16:22:51 +01:00
$row['permissions'] = \OCP\PERMISSION_ALL;
return $row;
}
public function hasContact($addressbookid, $id) {
return $this->getContact($addressbookid, $id) !== false;
}
/**
* Creates a new contact
*
2013-03-12 09:15:40 +01:00
* In the Database and Shared backends contact be either a Contact object or a string
* with carddata to be able to play seamlessly with the CardDAV backend.
* If this method is called by the CardDAV backend, the carddata is already validated.
2013-03-17 21:31:46 +01:00
* NOTE: It's assumed that this method is called either from the CardDAV backend, the
* import script, or from the ownCloud web UI in which case either the uri parameter is
* set, or the contact has a UID. If neither is set, it will fail.
2013-03-12 09:15:40 +01:00
*
* @param string $addressbookid
* @param VCard|string $contact
* @param array $options - Optional (backend specific options)
* @return string|bool The identifier for the new contact or false on error.
*/
public function createContact($addressbookid, $contact, array $options = array()) {
2013-10-03 18:09:19 +02:00
$qname = 'createcontact';
$uri = isset($options['uri']) ? $options['uri'] : null;
2014-03-08 16:22:51 +01:00
if (!$contact instanceof VCard) {
2013-03-12 09:15:40 +01:00
try {
$contact = Reader::read($contact);
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.', exception: '.$e->getMessage(), \OCP\Util::ERROR);
return false;
}
}
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-11-28 14:15:17 +01:00
$uri = is_null($uri) ? $this->uniqueURI($addressbookid, $contact->UID . '.vcf') : $uri;
$now = new \DateTime;
$contact->REV = $now->format(\DateTime::W3C);
$appinfo = \OCP\App::getAppInfo('contacts');
$appversion = \OCP\App::getAppVersion('contacts');
$prodid = '-//ownCloud//NONSGML '.$appinfo['name'].' '.$appversion.'//EN';
2013-03-29 16:42:33 +01:00
$contact->PRODID = $prodid;
2014-03-08 16:22:51 +01:00
if (!isset(self::$preparedQueries[$qname])) {
2013-10-03 18:09:19 +02:00
self::$preparedQueries[$qname] = \OCP\DB::prepare('INSERT INTO `'
. $this->cardsTableName
. '` (`addressbookid`,`fullname`,`carddata`,`uri`,`lastmodified`) VALUES(?,?,?,?,?)' );
}
2014-03-08 16:22:51 +01:00
try {
2013-10-03 18:09:19 +02:00
$result = self::$preparedQueries[$qname]
->execute(
array(
$addressbookid,
(string)$contact->FN,
$contact->serialize(),
$uri,
time()
)
);
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: ' . \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return false;
}
2014-03-08 16:22:51 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.', exception: '.$e->getMessage(), \OCP\Util::ERROR);
return false;
}
$newid = \OCP\DB::insertid($this->cardsTableName);
2013-11-07 14:24:55 +01:00
$this->setModifiedAddressBook($addressbookid);
2013-09-16 02:24:08 +02:00
\OCP\Util::emitHook('OCA\Contacts', 'post_createContact',
array('id' => $newid, 'parent' => $addressbookid, 'backend' => $this->name, 'contact' => $contact)
);
return (string)$newid;
}
/**
* Updates a contact
*
* @param string $addressbookid
2013-03-12 09:15:40 +01:00
* @param mixed $id Contact ID
* @param VCard|string $contact
* @param array $options - Optional (backend specific options)
2013-03-12 09:15:40 +01:00
* @see getContact
* @return bool
*/
public function updateContact($addressbookid, $id, $contact, array $options = array()) {
$noCollection = isset($options['noCollection']) ? $options['noCollection'] : false;
$isBatch = isset($options['isBatch']) ? $options['isBatch'] : false;
2013-10-03 18:09:19 +02:00
$qname = 'updatecontact';
$updateRevision = true;
$isCardDAV = false;
2014-03-08 16:22:51 +01:00
if (!$contact instanceof VCard) {
2013-03-12 09:15:40 +01:00
try {
$contact = Reader::read($contact);
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.', exception: '.$e->getMessage(), \OCP\Util::ERROR);
return false;
}
}
2013-10-03 18:09:19 +02:00
2014-03-08 16:22:51 +01:00
if (is_array($id)) {
2013-03-12 09:15:40 +01:00
$where_query = '';
2014-03-08 16:22:51 +01:00
if (isset($id['id'])) {
2013-03-12 09:15:40 +01:00
$id = $id['id'];
2014-03-08 16:22:51 +01:00
} elseif (isset($id['uri'])) {
$updateRevision = false;
$isCardDAV = true;
2013-10-03 18:09:19 +02:00
$id = $this->getIdFromUri($id['uri']);
2014-03-08 16:22:51 +01:00
if (is_null($id)) {
2013-10-03 18:09:19 +02:00
\OCP\Util::writeLog('contacts', __METHOD__ . ' Couldn\'t find contact', \OCP\Util::ERROR);
return false;
}
2014-03-08 16:22:51 +01:00
2013-03-12 09:15:40 +01:00
} else {
throw new \Exception(
2013-03-12 09:15:40 +01:00
__METHOD__ . ' If second argument is an array, either \'id\' or \'uri\' has to be set.'
);
}
}
2013-03-25 17:10:21 +01:00
2014-03-08 16:22:51 +01:00
if ($updateRevision || !isset($contact->REV)) {
$now = new \DateTime;
$contact->REV = $now->format(\DateTime::W3C);
}
$data = $contact->serialize();
2013-03-25 17:10:21 +01:00
2014-03-08 16:22:51 +01:00
if ($noCollection) {
2013-09-03 14:07:20 +02:00
$me = $this->getContact(null, $id, $options);
$addressbookid = $me['parent'];
}
2013-10-03 18:09:19 +02:00
$updates = array($contact->FN, $data, time(), $id, $addressbookid);
2013-03-25 17:10:21 +01:00
$query = 'UPDATE `' . $this->cardsTableName
2013-10-03 18:09:19 +02:00
. '` SET `fullname` = ?,`carddata` = ?, `lastmodified` = ? WHERE `id` = ? AND `addressbookid` = ?';
2014-03-08 16:22:51 +01:00
if (!isset(self::$preparedQueries[$qname])) {
self::$preparedQueries[$qname] = \OCP\DB::prepare($query);
}
2014-03-08 16:22:51 +01:00
try {
2014-03-08 16:22:51 +01:00
2013-03-25 17:10:21 +01:00
$result = self::$preparedQueries[$qname]->execute($updates);
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: ' . \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return false;
}
2014-03-08 16:22:51 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.', exception: '
. $e->getMessage(), \OCP\Util::ERROR);
\OCP\Util::writeLog('contacts', __METHOD__.', id' . $id, \OCP\Util::DEBUG);
return false;
}
2013-11-07 14:24:55 +01:00
$this->setModifiedAddressBook($addressbookid);
2014-03-08 16:22:51 +01:00
if (!$isBatch) {
2013-09-16 02:24:08 +02:00
\OCP\Util::emitHook('OCA\Contacts', 'post_updateContact',
array(
'backend' => $this->name,
'addressBookId' => $addressbookid,
'contactId' => $id,
'contact' => $contact,
'carddav' => $isCardDAV
)
);
2014-03-08 16:22:51 +01:00
}
2014-03-08 16:22:51 +01:00
return true;
}
/**
* Deletes a contact
*
* @param string $addressbookid
* @param string $id
* @param array $options - Optional (backend specific options)
2013-03-12 09:15:40 +01:00
* @see getContact
* @return bool
*/
public function deleteContact($addressbookid, $id, array $options = array()) {
// TODO: pass the uri in $options instead.
2013-10-03 18:09:19 +02:00
$qname = 'deletecontact';
$noCollection = isset($options['noCollection']) ? $options['noCollection'] : false;
$isBatch = isset($options['isBatch']) ? $options['isBatch'] : false;
2014-03-08 16:22:51 +01:00
if (is_array($id)) {
if (isset($id['id'])) {
2013-03-12 09:15:40 +01:00
$id = $id['id'];
2014-03-08 16:22:51 +01:00
} elseif (isset($id['uri'])) {
2013-10-03 18:09:19 +02:00
$id = $this->getIdFromUri($id['uri']);
2014-03-08 16:22:51 +01:00
if (is_null($id)) {
2013-10-03 18:09:19 +02:00
\OCP\Util::writeLog('contacts', __METHOD__ . ' Couldn\'t find contact', \OCP\Util::ERROR);
return false;
}
2014-03-08 16:22:51 +01:00
2013-03-12 09:15:40 +01:00
} else {
throw new \Exception(
2013-03-12 09:15:40 +01:00
__METHOD__ . ' If second argument is an array, either \'id\' or \'uri\' has to be set.'
);
}
}
2014-03-08 16:22:51 +01:00
if (!$isBatch) {
2013-09-16 02:24:08 +02:00
\OCP\Util::emitHook('OCA\Contacts', 'pre_deleteContact',
array('id' => $id)
);
}
2013-10-03 18:09:19 +02:00
2014-03-08 16:22:51 +01:00
if ($noCollection) {
2013-10-03 18:09:19 +02:00
$me = $this->getContact(null, $id, $options);
$addressbookid = $me['parent'];
}
2014-03-08 16:22:51 +01:00
if (!isset(self::$preparedQueries[$qname])) {
self::$preparedQueries[$qname] = \OCP\DB::prepare('DELETE FROM `'
. $this->cardsTableName
2013-10-03 18:09:19 +02:00
. '` WHERE `id` = ? AND `addressbookid` = ?');
}
2014-03-08 16:22:51 +01:00
2013-10-03 18:09:19 +02:00
\OCP\Util::writeLog('contacts', __METHOD__ . ' updates: ' . $id . '/' . $addressbookid, \OCP\Util::DEBUG);
try {
2014-03-08 16:22:51 +01:00
$result = self::$preparedQueries[$qname]->execute(array($id, $addressbookid));
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: '
. \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return false;
}
2014-03-08 16:22:51 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__.
', exception: ' . $e->getMessage(), \OCP\Util::ERROR);
\OCP\Util::writeLog('contacts', __METHOD__.', id: '
. $id, \OCP\Util::DEBUG);
return false;
}
2014-03-08 16:22:51 +01:00
2013-11-07 14:24:55 +01:00
$this->setModifiedAddressBook($addressbookid);
return true;
}
2013-03-15 17:02:33 +01:00
/**
* @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) {
2014-03-08 16:22:51 +01:00
2013-03-15 17:02:33 +01:00
$contact = $this->getContact($addressbookid, $id);
2013-03-16 15:59:23 +01:00
return ($contact ? $contact['lastmodified'] : null);
2014-03-08 16:22:51 +01:00
2013-03-15 17:02:33 +01:00
}
2013-10-03 18:09:19 +02:00
/**
* @brief Get the contact id from the uri.
*
* @param mixed $id
* @returns int | null
*/
public function getIdFromUri($uri) {
2014-03-08 16:22:51 +01:00
2013-10-03 18:09:19 +02:00
$query = 'SELECT `id` FROM `'. $this->cardsTableName . '` WHERE `uri` = ?';
$stmt = \OCP\DB::prepare($query);
$result = $stmt->execute(array($uri));
2014-03-08 16:22:51 +01:00
2013-10-03 18:09:19 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: ' . \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return null;
}
2014-03-08 16:22:51 +01:00
$one = $result->fetchOne();
2014-03-08 16:22:51 +01:00
if (!$one) {
2013-10-03 18:09:19 +02:00
\OCP\Util::writeLog('contacts', __METHOD__.', Not found, uri: '. $uri, \OCP\Util::DEBUG);
return null;
}
return $one;
2013-10-03 18:09:19 +02:00
}
private function createAddressBookURI($displayname, $userid = null) {
2014-03-08 16:22:51 +01:00
$userid = $userid ? $userid : \OCP\User::getUser();
$name = str_replace(' ', '_', strtolower($displayname));
2014-03-08 16:22:51 +01:00
try {
$stmt = \OCP\DB::prepare('SELECT `uri` FROM `' . $this->addressBooksTableName . '` WHERE `userid` = ? ');
$result = $stmt->execute(array($userid));
2014-03-08 16:22:51 +01:00
2013-09-16 02:24:08 +02:00
if (\OCP\DB::isError($result)) {
\OCP\Util::writeLog('contacts', __METHOD__. 'DB error: ' . \OC_DB::getErrorMessage($result), \OCP\Util::ERROR);
return $name;
}
2014-03-08 16:22:51 +01:00
} catch(\Exception $e) {
\OCP\Util::writeLog('contacts', __METHOD__ . ' exception: ' . $e->getMessage(), \OCP\Util::ERROR);
return $name;
}
$uris = array();
2014-03-08 16:22:51 +01:00
while ($row = $result->fetchRow()) {
$uris[] = $row['uri'];
}
$newname = $name;
2014-03-08 16:22:51 +01:00
$i = 1
;
while (in_array($newname, $uris)) {
$newname = $name.$i;
$i = $i + 1;
}
return $newname;
}
2013-11-28 14:15:17 +01:00
/**
* @brief Checks if a contact with the same URI already exist in the address book.
* @param string $addressBookId Address book ID.
* @param string $uri
* @returns string Unique URI
*/
protected function uniqueURI($addressBookId, $uri) {
2014-02-03 14:52:53 +01:00
$stmt = \OCP\DB::prepare( 'SELECT COUNT(*) AS `count` FROM `' . $this->cardsTableName . '` WHERE `addressbookid` = ? AND `uri` = ?' );
2013-11-28 14:15:17 +01:00
$result = $stmt->execute(array($addressBookId, $uri));
$result = $result->fetchRow();
2013-11-28 14:15:17 +01:00
2014-03-08 16:22:51 +01:00
if (is_array($result) && count($result) > 0 && $result['count'] > 0) {
while (true) {
2013-11-28 14:15:17 +01:00
$uri = Properties::generateUID() . '.vcf';
$result = $stmt->execute(array($addressBookId, $uri));
2014-03-08 16:22:51 +01:00
if (is_array($result) && count($result) > 0 && $result['count'] > 0) {
2013-11-28 14:15:17 +01:00
continue;
} else {
return $uri;
}
2014-03-08 16:22:51 +01:00
2013-11-28 14:15:17 +01:00
}
}
return $uri;
2013-11-28 14:15:17 +01:00
}
}