2015-10-30 15:05:25 +00:00
< ? php
/**
2016-01-12 14:02:16 +00:00
* @ author Arthur Schiwon < blizzz @ owncloud . com >
* @ author Björn Schießle < schiessle @ owncloud . com >
* @ author Scrutinizer Auto - Fixer < auto - fixer @ scrutinizer - ci . com >
2015-10-30 15:05:25 +00:00
* @ author Thomas Müller < thomas . mueller @ tmit . eu >
*
2016-01-12 14:02:16 +00:00
* @ copyright Copyright ( c ) 2016 , ownCloud , Inc .
2015-10-30 15:05:25 +00:00
* @ license AGPL - 3.0
*
* 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\DAV\CardDAV ;
2015-11-05 15:46:37 +00:00
use OCA\DAV\Connector\Sabre\Principal ;
2015-11-25 14:24:50 +00:00
use OCP\IDBConnection ;
2015-10-30 15:05:25 +00:00
use Sabre\CardDAV\Backend\BackendInterface ;
use Sabre\CardDAV\Backend\SyncSupport ;
use Sabre\CardDAV\Plugin ;
use Sabre\DAV\Exception\BadRequest ;
2015-12-14 10:48:31 +00:00
use Sabre\VObject\Component\VCard ;
2015-11-25 14:24:50 +00:00
use Sabre\VObject\Reader ;
2015-10-30 15:05:25 +00:00
class CardDavBackend implements BackendInterface , SyncSupport {
2015-11-05 15:46:37 +00:00
/** @var Principal */
private $principalBackend ;
2015-11-25 14:24:50 +00:00
/** @var string */
private $dbCardsTable = 'cards' ;
/** @var string */
private $dbCardsPropertiesTable = 'cards_properties' ;
/** @var IDBConnection */
private $db ;
/** @var array properties to index */
public static $indexProperties = array (
'BDAY' , 'UID' , 'N' , 'FN' , 'TITLE' , 'ROLE' , 'NOTE' , 'NICKNAME' ,
'ORG' , 'CATEGORIES' , 'EMAIL' , 'TEL' , 'IMPP' , 'ADR' , 'URL' , 'GEO' , 'CLOUD' );
2016-01-12 12:23:50 +00:00
const ACCESS_OWNER = 1 ;
const ACCESS_READ_WRITE = 2 ;
const ACCESS_READ = 3 ;
2015-11-25 14:24:50 +00:00
/**
* CardDavBackend constructor .
*
* @ param IDBConnection $db
* @ param Principal $principalBackend
*/
2015-12-04 12:38:32 +00:00
public function __construct ( IDBConnection $db , Principal $principalBackend ) {
2015-10-30 15:05:25 +00:00
$this -> db = $db ;
2015-11-05 15:46:37 +00:00
$this -> principalBackend = $principalBackend ;
2015-10-30 15:05:25 +00:00
}
/**
2016-01-13 10:55:50 +00:00
* Returns the list of address books for a specific user .
2015-10-30 15:05:25 +00:00
*
* Every addressbook should have the following properties :
* id - an arbitrary unique id
* uri - the 'basename' part of the url
* principaluri - Same as the passed parameter
*
* Any additional clark - notation property may be passed besides this . Some
* common ones are :
* { DAV : } displayname
* { urn : ietf : params : xml : ns : carddav } addressbook - description
* { http :// calendarserver . org / ns / } getctag
*
* @ param string $principalUri
* @ return array
*/
function getAddressBooksForUser ( $principalUri ) {
$query = $this -> db -> getQueryBuilder ();
$query -> select ([ 'id' , 'uri' , 'displayname' , 'principaluri' , 'description' , 'synctoken' ])
-> from ( 'addressbooks' )
-> where ( $query -> expr () -> eq ( 'principaluri' , $query -> createParameter ( 'principaluri' )))
-> setParameter ( 'principaluri' , $principalUri );
$addressBooks = [];
$result = $query -> execute ();
while ( $row = $result -> fetch ()) {
$addressBooks [] = [
'id' => $row [ 'id' ],
'uri' => $row [ 'uri' ],
'principaluri' => $row [ 'principaluri' ],
'{DAV:}displayname' => $row [ 'displayname' ],
'{' . Plugin :: NS_CARDDAV . '}addressbook-description' => $row [ 'description' ],
'{http://calendarserver.org/ns/}getctag' => $row [ 'synctoken' ],
'{http://sabredav.org/ns}sync-token' => $row [ 'synctoken' ] ? $row [ 'synctoken' ] : '0' ,
];
}
$result -> closeCursor ();
2015-11-05 15:46:37 +00:00
// query for shared calendars
2016-01-11 16:29:01 +00:00
$principals = $this -> principalBackend -> getGroupMembership ( $principalUri );
$principals [] = $principalUri ;
2015-11-05 15:46:37 +00:00
$query = $this -> db -> getQueryBuilder ();
2016-01-12 12:23:50 +00:00
$result = $query -> select ([ 'a.id' , 'a.uri' , 'a.displayname' , 'a.principaluri' , 'a.description' , 'a.synctoken' , 's.uri' , 's.access' ])
-> from ( 'dav_shares' , 's' )
2016-01-12 14:44:50 +00:00
-> join ( 's' , 'addressbooks' , 'a' , $query -> expr () -> eq ( 's.resourceid' , 'a.id' ))
2016-01-12 12:23:50 +00:00
-> where ( $query -> expr () -> in ( 's.principaluri' , $query -> createParameter ( 'principaluri' )))
-> andWhere ( $query -> expr () -> eq ( 's.type' , $query -> createParameter ( 'type' )))
2015-11-05 15:46:37 +00:00
-> setParameter ( 'type' , 'addressbook' )
2016-01-11 16:29:01 +00:00
-> setParameter ( 'principaluri' , $principals , \Doctrine\DBAL\Connection :: PARAM_STR_ARRAY )
2015-11-05 15:46:37 +00:00
-> execute ();
while ( $row = $result -> fetch ()) {
$addressBooks [] = [
'id' => $row [ 'id' ],
'uri' => $row [ 'uri' ],
2016-01-12 12:23:50 +00:00
'principaluri' => $principalUri ,
2015-11-05 15:46:37 +00:00
'{DAV:}displayname' => $row [ 'displayname' ],
'{' . Plugin :: NS_CARDDAV . '}addressbook-description' => $row [ 'description' ],
'{http://calendarserver.org/ns/}getctag' => $row [ 'synctoken' ],
'{http://sabredav.org/ns}sync-token' => $row [ 'synctoken' ] ? $row [ 'synctoken' ] : '0' ,
2016-01-12 12:23:50 +00:00
'{' . \OCA\DAV\CardDAV\Sharing\Plugin :: NS_OWNCLOUD . '}owner-principal' => $row [ 'principaluri' ],
'{' . \OCA\DAV\CardDAV\Sharing\Plugin :: NS_OWNCLOUD . '}read-only' => $row [ 'access' ] === self :: ACCESS_READ ,
2015-11-05 15:46:37 +00:00
];
}
$result -> closeCursor ();
2015-10-30 15:05:25 +00:00
return $addressBooks ;
}
2015-12-22 15:11:53 +00:00
/**
* @ param int $addressBookId
*/
public function getAddressBookById ( $addressBookId ) {
$query = $this -> db -> getQueryBuilder ();
$result = $query -> select ([ 'id' , 'uri' , 'displayname' , 'principaluri' , 'description' , 'synctoken' ])
-> from ( 'addressbooks' )
-> where ( $query -> expr () -> eq ( 'id' , $query -> createNamedParameter ( $addressBookId )))
-> execute ();
$row = $result -> fetch ();
$result -> closeCursor ();
if ( $row === false ) {
return null ;
}
return [
'id' => $row [ 'id' ],
'uri' => $row [ 'uri' ],
'principaluri' => $row [ 'principaluri' ],
'{DAV:}displayname' => $row [ 'displayname' ],
'{' . Plugin :: NS_CARDDAV . '}addressbook-description' => $row [ 'description' ],
'{http://calendarserver.org/ns/}getctag' => $row [ 'synctoken' ],
'{http://sabredav.org/ns}sync-token' => $row [ 'synctoken' ] ? $row [ 'synctoken' ] : '0' ,
];
}
/**
* @ param $addressBookUri
* @ return array | null
*/
public function getAddressBooksByUri ( $principal , $addressBookUri ) {
2015-11-05 15:46:37 +00:00
$query = $this -> db -> getQueryBuilder ();
$result = $query -> select ([ 'id' , 'uri' , 'displayname' , 'principaluri' , 'description' , 'synctoken' ])
-> from ( 'addressbooks' )
-> where ( $query -> expr () -> eq ( 'uri' , $query -> createNamedParameter ( $addressBookUri )))
2015-12-22 15:11:53 +00:00
-> andWhere ( $query -> expr () -> eq ( 'principaluri' , $query -> createNamedParameter ( $principal )))
2015-11-05 15:46:37 +00:00
-> setMaxResults ( 1 )
-> execute ();
$row = $result -> fetch ();
2015-12-02 08:20:58 +00:00
$result -> closeCursor ();
2015-11-24 22:53:27 +00:00
if ( $row === false ) {
2015-11-05 15:46:37 +00:00
return null ;
}
return [
'id' => $row [ 'id' ],
'uri' => $row [ 'uri' ],
'principaluri' => $row [ 'principaluri' ],
'{DAV:}displayname' => $row [ 'displayname' ],
'{' . Plugin :: NS_CARDDAV . '}addressbook-description' => $row [ 'description' ],
'{http://calendarserver.org/ns/}getctag' => $row [ 'synctoken' ],
'{http://sabredav.org/ns}sync-token' => $row [ 'synctoken' ] ? $row [ 'synctoken' ] : '0' ,
];
}
2015-10-30 15:05:25 +00:00
/**
* Updates properties for an address book .
*
* The list of mutations is stored in a Sabre\DAV\PropPatch object .
* To do the actual updates , you must tell this object which properties
* you ' re going to process with the handle () method .
*
* Calling the handle method is like telling the PropPatch object " I
* promise I can handle updating this property " .
*
2015-11-18 14:04:53 +00:00
* Read the PropPatch documentation for more info and examples .
2015-10-30 15:05:25 +00:00
*
* @ param string $addressBookId
* @ param \Sabre\DAV\PropPatch $propPatch
* @ return void
*/
function updateAddressBook ( $addressBookId , \Sabre\DAV\PropPatch $propPatch ) {
$supportedProperties = [
'{DAV:}displayname' ,
'{' . Plugin :: NS_CARDDAV . '}addressbook-description' ,
];
$propPatch -> handle ( $supportedProperties , function ( $mutations ) use ( $addressBookId ) {
$updates = [];
foreach ( $mutations as $property => $newValue ) {
switch ( $property ) {
case '{DAV:}displayname' :
$updates [ 'displayname' ] = $newValue ;
break ;
case '{' . Plugin :: NS_CARDDAV . '}addressbook-description' :
$updates [ 'description' ] = $newValue ;
break ;
}
}
$query = $this -> db -> getQueryBuilder ();
$query -> update ( 'addressbooks' );
foreach ( $updates as $key => $value ) {
$query -> set ( $key , $query -> createNamedParameter ( $value ));
}
$query -> where ( $query -> expr () -> eq ( 'id' , $query -> createNamedParameter ( $addressBookId )))
-> execute ();
$this -> addChange ( $addressBookId , " " , 2 );
return true ;
});
}
/**
* Creates a new address book
*
* @ param string $principalUri
* @ param string $url Just the 'basename' of the url .
* @ param array $properties
2015-12-22 15:11:53 +00:00
* @ return int
2015-11-10 10:37:07 +00:00
* @ throws BadRequest
2015-10-30 15:05:25 +00:00
*/
function createAddressBook ( $principalUri , $url , array $properties ) {
$values = [
'displayname' => null ,
'description' => null ,
'principaluri' => $principalUri ,
'uri' => $url ,
'synctoken' => 1
];
foreach ( $properties as $property => $newValue ) {
switch ( $property ) {
case '{DAV:}displayname' :
$values [ 'displayname' ] = $newValue ;
break ;
case '{' . Plugin :: NS_CARDDAV . '}addressbook-description' :
$values [ 'description' ] = $newValue ;
break ;
default :
throw new BadRequest ( 'Unknown property: ' . $property );
}
}
2015-11-10 10:37:07 +00:00
// Fallback to make sure the displayname is set. Some clients may refuse
// to work with addressbooks not having a displayname.
if ( is_null ( $values [ 'displayname' ])) {
$values [ 'displayname' ] = $url ;
}
2015-10-30 15:05:25 +00:00
$query = $this -> db -> getQueryBuilder ();
$query -> insert ( 'addressbooks' )
-> values ([
'uri' => $query -> createParameter ( 'uri' ),
'displayname' => $query -> createParameter ( 'displayname' ),
'description' => $query -> createParameter ( 'description' ),
'principaluri' => $query -> createParameter ( 'principaluri' ),
'synctoken' => $query -> createParameter ( 'synctoken' ),
])
-> setParameters ( $values )
-> execute ();
2015-12-22 15:11:53 +00:00
return $query -> getLastInsertId ();
2015-10-30 15:05:25 +00:00
}
/**
* Deletes an entire addressbook and all its contents
*
* @ param mixed $addressBookId
* @ return void
*/
function deleteAddressBook ( $addressBookId ) {
$query = $this -> db -> getQueryBuilder ();
$query -> delete ( 'cards' )
-> where ( $query -> expr () -> eq ( 'addressbookid' , $query -> createParameter ( 'addressbookid' )))
-> setParameter ( 'addressbookid' , $addressBookId )
-> execute ();
$query -> delete ( 'addressbookchanges' )
-> where ( $query -> expr () -> eq ( 'addressbookid' , $query -> createParameter ( 'addressbookid' )))
-> setParameter ( 'addressbookid' , $addressBookId )
-> execute ();
$query -> delete ( 'addressbooks' )
-> where ( $query -> expr () -> eq ( 'id' , $query -> createParameter ( 'id' )))
-> setParameter ( 'id' , $addressBookId )
-> execute ();
2015-11-05 15:46:37 +00:00
$query -> delete ( 'dav_shares' )
-> where ( $query -> expr () -> eq ( 'resourceid' , $query -> createNamedParameter ( $addressBookId )))
-> andWhere ( $query -> expr () -> eq ( 'type' , $query -> createNamedParameter ( 'addressbook' )))
-> execute ();
2015-11-25 14:24:50 +00:00
$query -> delete ( $this -> dbCardsPropertiesTable )
-> where ( $query -> expr () -> eq ( 'addressbookid' , $query -> createNamedParameter ( $addressBookId )))
-> execute ();
2015-10-30 15:05:25 +00:00
}
/**
* Returns all cards for a specific addressbook id .
*
* This method should return the following properties for each card :
* * carddata - raw vcard data
* * uri - Some unique url
* * lastmodified - A unix timestamp
*
* It ' s recommended to also return the following properties :
* * etag - A unique etag . This must change every time the card changes .
* * size - The size of the card in bytes .
*
* If these last two properties are provided , less time will be spent
* calculating them . If they are specified , you can also ommit carddata .
* This may speed up certain requests , especially with large cards .
*
* @ param mixed $addressBookId
* @ return array
*/
function getCards ( $addressBookId ) {
$query = $this -> db -> getQueryBuilder ();
$query -> select ([ 'id' , 'uri' , 'lastmodified' , 'etag' , 'size' , 'carddata' ])
-> from ( 'cards' )
-> where ( $query -> expr () -> eq ( 'addressbookid' , $query -> createNamedParameter ( $addressBookId )));
$cards = [];
$result = $query -> execute ();
while ( $row = $result -> fetch ()) {
$row [ 'etag' ] = '"' . $row [ 'etag' ] . '"' ;
$row [ 'carddata' ] = $this -> readBlob ( $row [ 'carddata' ]);
$cards [] = $row ;
}
$result -> closeCursor ();
return $cards ;
}
/**
* Returns a specfic card .
*
* The same set of properties must be returned as with getCards . The only
* exception is that 'carddata' is absolutely required .
*
* If the card does not exist , you must return false .
*
* @ param mixed $addressBookId
* @ param string $cardUri
* @ return array
*/
function getCard ( $addressBookId , $cardUri ) {
$query = $this -> db -> getQueryBuilder ();
$query -> select ([ 'id' , 'uri' , 'lastmodified' , 'etag' , 'size' , 'carddata' ])
-> from ( 'cards' )
-> where ( $query -> expr () -> eq ( 'addressbookid' , $query -> createNamedParameter ( $addressBookId )))
-> andWhere ( $query -> expr () -> eq ( 'uri' , $query -> createNamedParameter ( $cardUri )))
-> setMaxResults ( 1 );
$result = $query -> execute ();
$row = $result -> fetch ();
if ( ! $row ) {
return false ;
}
$row [ 'etag' ] = '"' . $row [ 'etag' ] . '"' ;
$row [ 'carddata' ] = $this -> readBlob ( $row [ 'carddata' ]);
return $row ;
}
/**
* Returns a list of cards .
*
* This method should work identical to getCard , but instead return all the
* cards in the list as an array .
*
* If the backend supports this , it may allow for some speed - ups .
*
* @ param mixed $addressBookId
2015-11-20 15:42:34 +00:00
* @ param string [] $uris
2015-10-30 15:05:25 +00:00
* @ return array
*/
function getMultipleCards ( $addressBookId , array $uris ) {
$query = $this -> db -> getQueryBuilder ();
$query -> select ([ 'id' , 'uri' , 'lastmodified' , 'etag' , 'size' , 'carddata' ])
-> from ( 'cards' )
-> where ( $query -> expr () -> eq ( 'addressbookid' , $query -> createNamedParameter ( $addressBookId )))
-> andWhere ( $query -> expr () -> in ( 'uri' , $query -> createParameter ( 'uri' )))
-> setParameter ( 'uri' , $uris , \Doctrine\DBAL\Connection :: PARAM_STR_ARRAY );
$cards = [];
$result = $query -> execute ();
while ( $row = $result -> fetch ()) {
$row [ 'etag' ] = '"' . $row [ 'etag' ] . '"' ;
$row [ 'carddata' ] = $this -> readBlob ( $row [ 'carddata' ]);
$cards [] = $row ;
}
$result -> closeCursor ();
return $cards ;
}
/**
* Creates a new card .
*
* The addressbook id will be passed as the first argument . This is the
* same id as it is returned from the getAddressBooksForUser method .
*
* The cardUri is a base uri , and doesn ' t include the full path . The
* cardData argument is the vcard body , and is passed as a string .
*
* It is possible to return an ETag from this method . This ETag is for the
* newly created resource , and must be enclosed with double quotes ( that
* is , the string itself must contain the double quotes ) .
*
* You should only return the ETag if you store the carddata as - is . If a
* subsequent GET request on the same card does not have the same body ,
* byte - by - byte and you did return an ETag here , clients tend to get
* confused .
*
* If you don ' t return an ETag , you can just return null .
*
* @ param mixed $addressBookId
* @ param string $cardUri
* @ param string $cardData
2015-11-20 15:42:34 +00:00
* @ return string
2015-10-30 15:05:25 +00:00
*/
function createCard ( $addressBookId , $cardUri , $cardData ) {
$etag = md5 ( $cardData );
$query = $this -> db -> getQueryBuilder ();
$query -> insert ( 'cards' )
-> values ([
2015-12-14 10:48:31 +00:00
'carddata' => $query -> createNamedParameter ( $cardData , \PDO :: PARAM_LOB ),
2015-10-30 15:05:25 +00:00
'uri' => $query -> createNamedParameter ( $cardUri ),
'lastmodified' => $query -> createNamedParameter ( time ()),
'addressbookid' => $query -> createNamedParameter ( $addressBookId ),
'size' => $query -> createNamedParameter ( strlen ( $cardData )),
'etag' => $query -> createNamedParameter ( $etag ),
])
-> execute ();
$this -> addChange ( $addressBookId , $cardUri , 1 );
2015-11-25 14:24:50 +00:00
$this -> updateProperties ( $addressBookId , $cardUri , $cardData );
2015-10-30 15:05:25 +00:00
return '"' . $etag . '"' ;
}
/**
* Updates a card .
*
* The addressbook id will be passed as the first argument . This is the
* same id as it is returned from the getAddressBooksForUser method .
*
* The cardUri is a base uri , and doesn ' t include the full path . The
* cardData argument is the vcard body , and is passed as a string .
*
* It is possible to return an ETag from this method . This ETag should
* match that of the updated resource , and must be enclosed with double
* quotes ( that is : the string itself must contain the actual quotes ) .
*
* You should only return the ETag if you store the carddata as - is . If a
* subsequent GET request on the same card does not have the same body ,
* byte - by - byte and you did return an ETag here , clients tend to get
* confused .
*
* If you don ' t return an ETag , you can just return null .
*
* @ param mixed $addressBookId
* @ param string $cardUri
* @ param string $cardData
2015-11-20 15:42:34 +00:00
* @ return string
2015-10-30 15:05:25 +00:00
*/
function updateCard ( $addressBookId , $cardUri , $cardData ) {
$etag = md5 ( $cardData );
$query = $this -> db -> getQueryBuilder ();
$query -> update ( 'cards' )
-> set ( 'carddata' , $query -> createNamedParameter ( $cardData , \PDO :: PARAM_LOB ))
-> set ( 'lastmodified' , $query -> createNamedParameter ( time ()))
-> set ( 'size' , $query -> createNamedParameter ( strlen ( $cardData )))
-> set ( 'etag' , $query -> createNamedParameter ( $etag ))
-> where ( $query -> expr () -> eq ( 'uri' , $query -> createNamedParameter ( $cardUri )))
-> andWhere ( $query -> expr () -> eq ( 'addressbookid' , $query -> createNamedParameter ( $addressBookId )))
-> execute ();
$this -> addChange ( $addressBookId , $cardUri , 2 );
2015-11-25 14:24:50 +00:00
$this -> updateProperties ( $addressBookId , $cardUri , $cardData );
2015-10-30 15:05:25 +00:00
return '"' . $etag . '"' ;
}
/**
* Deletes a card
*
* @ param mixed $addressBookId
* @ param string $cardUri
* @ return bool
*/
function deleteCard ( $addressBookId , $cardUri ) {
2016-01-21 11:19:28 +00:00
try {
$cardId = $this -> getCardId ( $cardUri );
} catch ( \InvalidArgumentException $e ) {
$cardId = null ;
}
2015-10-30 15:05:25 +00:00
$query = $this -> db -> getQueryBuilder ();
$ret = $query -> delete ( 'cards' )
-> where ( $query -> expr () -> eq ( 'addressbookid' , $query -> createNamedParameter ( $addressBookId )))
-> andWhere ( $query -> expr () -> eq ( 'uri' , $query -> createNamedParameter ( $cardUri )))
-> execute ();
$this -> addChange ( $addressBookId , $cardUri , 3 );
2015-11-25 14:24:50 +00:00
if ( $ret === 1 ) {
2016-01-21 11:19:28 +00:00
if ( $cardId !== null ) {
$this -> purgeProperties ( $addressBookId , $cardId );
}
2015-11-25 14:24:50 +00:00
return true ;
}
return false ;
2015-10-30 15:05:25 +00:00
}
/**
* The getChanges method returns all the changes that have happened , since
* the specified syncToken in the specified address book .
*
* This function should return an array , such as the following :
*
* [
* 'syncToken' => 'The current synctoken' ,
* 'added' => [
* 'new.txt' ,
* ],
* 'modified' => [
* 'modified.txt' ,
* ],
* 'deleted' => [
* 'foo.php.bak' ,
* 'old.txt'
* ]
* ];
*
* The returned syncToken property should reflect the * current * syncToken
* of the calendar , as reported in the { http :// sabredav . org / ns } sync - token
* property . This is needed here too , to ensure the operation is atomic .
*
* If the $syncToken argument is specified as null , this is an initial
* sync , and all members should be reported .
*
* The modified property is an array of nodenames that have changed since
* the last token .
*
* The deleted property is an array with nodenames , that have been deleted
* from collection .
*
* The $syncLevel argument is basically the 'depth' of the report . If it ' s
* 1 , you only have to report changes that happened only directly in
* immediate descendants . If it ' s 2 , it should also include changes from
* the nodes below the child collections . ( grandchildren )
*
* The $limit argument allows a client to specify how many results should
* be returned at most . If the limit is not specified , it should be treated
* as infinite .
*
* If the limit ( infinite or not ) is higher than you ' re willing to return ,
* you should throw a Sabre\DAV\Exception\TooMuchMatches () exception .
*
* If the syncToken is expired ( due to data cleanup ) or unknown , you must
* return null .
*
* The limit is 'suggestive' . You are free to ignore it .
*
* @ param string $addressBookId
* @ param string $syncToken
* @ param int $syncLevel
* @ param int $limit
* @ return array
*/
function getChangesForAddressBook ( $addressBookId , $syncToken , $syncLevel , $limit = null ) {
// Current synctoken
$stmt = $this -> db -> prepare ( 'SELECT `synctoken` FROM `*PREFIX*addressbooks` WHERE `id` = ?' );
$stmt -> execute ([ $addressBookId ]);
$currentToken = $stmt -> fetchColumn ( 0 );
if ( is_null ( $currentToken )) return null ;
$result = [
'syncToken' => $currentToken ,
'added' => [],
'modified' => [],
'deleted' => [],
];
if ( $syncToken ) {
$query = " SELECT `uri`, `operation` FROM `*PREFIX*addressbookchanges` WHERE `synctoken` >= ? AND `synctoken` < ? AND `addressbookid` = ? ORDER BY `synctoken` " ;
if ( $limit > 0 ) {
$query .= " `LIMIT` " . ( int ) $limit ;
}
// Fetching all changes
$stmt = $this -> db -> prepare ( $query );
$stmt -> execute ([ $syncToken , $currentToken , $addressBookId ]);
$changes = [];
// This loop ensures that any duplicates are overwritten, only the
// last change on a node is relevant.
while ( $row = $stmt -> fetch ( \PDO :: FETCH_ASSOC )) {
$changes [ $row [ 'uri' ]] = $row [ 'operation' ];
}
foreach ( $changes as $uri => $operation ) {
switch ( $operation ) {
case 1 :
$result [ 'added' ][] = $uri ;
break ;
case 2 :
$result [ 'modified' ][] = $uri ;
break ;
case 3 :
$result [ 'deleted' ][] = $uri ;
break ;
}
}
} else {
// No synctoken supplied, this is the initial sync.
$query = " SELECT `uri` FROM `*PREFIX*cards` WHERE `addressbookid` = ? " ;
$stmt = $this -> db -> prepare ( $query );
$stmt -> execute ([ $addressBookId ]);
$result [ 'added' ] = $stmt -> fetchAll ( \PDO :: FETCH_COLUMN );
}
return $result ;
}
/**
* Adds a change record to the addressbookchanges table .
*
* @ param mixed $addressBookId
* @ param string $objectUri
* @ param int $operation 1 = add , 2 = modify , 3 = delete
* @ return void
*/
protected function addChange ( $addressBookId , $objectUri , $operation ) {
$sql = 'INSERT INTO `*PREFIX*addressbookchanges`(`uri`, `synctoken`, `addressbookid`, `operation`) SELECT ?, `synctoken`, ?, ? FROM `*PREFIX*addressbooks` WHERE `id` = ?' ;
$stmt = $this -> db -> prepare ( $sql );
$stmt -> execute ([
$objectUri ,
$addressBookId ,
$operation ,
$addressBookId
]);
$stmt = $this -> db -> prepare ( 'UPDATE `*PREFIX*addressbooks` SET `synctoken` = `synctoken` + 1 WHERE `id` = ?' );
$stmt -> execute ([
$addressBookId
]);
}
private function readBlob ( $cardData ) {
if ( is_resource ( $cardData )) {
return stream_get_contents ( $cardData );
}
return $cardData ;
}
2015-11-18 14:04:53 +00:00
/**
2015-12-22 15:11:53 +00:00
* @ param AddressBook $book
2015-11-18 14:04:53 +00:00
* @ param string [] $add
* @ param string [] $remove
*/
2015-12-22 15:11:53 +00:00
public function updateShares ( $book , $add , $remove ) {
2015-11-05 15:46:37 +00:00
foreach ( $add as $element ) {
2015-12-22 15:11:53 +00:00
$this -> shareWith ( $book , $element );
2015-11-05 15:46:37 +00:00
}
foreach ( $remove as $element ) {
2015-12-22 15:11:53 +00:00
$this -> unshare ( $book -> getBookId (), $element );
2015-11-05 15:46:37 +00:00
}
}
2015-11-25 14:24:50 +00:00
/**
* search contact
*
* @ param int $addressBookId
* @ param string $pattern which should match within the $searchProperties
* @ param array $searchProperties defines the properties within the query pattern should match
* @ return array an array of contacts which are arrays of key - value - pairs
*/
public function search ( $addressBookId , $pattern , $searchProperties ) {
$query = $this -> db -> getQueryBuilder ();
2015-12-14 12:51:12 +00:00
$query2 = $this -> db -> getQueryBuilder ();
$query2 -> selectDistinct ( 'cp.cardid' ) -> from ( $this -> dbCardsPropertiesTable , 'cp' );
2015-11-25 14:24:50 +00:00
foreach ( $searchProperties as $property ) {
2015-12-14 12:51:12 +00:00
$query2 -> orWhere (
$query2 -> expr () -> andX (
$query2 -> expr () -> eq ( 'cp.name' , $query -> createNamedParameter ( $property )),
$query2 -> expr () -> like ( 'cp.value' , $query -> createNamedParameter ( '%' . $this -> db -> escapeLikeParameter ( $pattern ) . '%' ))
)
2015-11-25 14:24:50 +00:00
);
}
2015-12-14 12:51:12 +00:00
$query2 -> andWhere ( $query2 -> expr () -> eq ( 'cp.addressbookid' , $query -> createNamedParameter ( $addressBookId )));
$query -> select ( 'c.carddata' ) -> from ( $this -> dbCardsTable , 'c' )
-> where ( $query -> expr () -> in ( 'c.id' , $query -> createFunction ( $query2 -> getSQL ())));
$result = $query -> execute ();
2015-11-25 14:24:50 +00:00
$cards = $result -> fetchAll ();
2015-12-14 12:51:12 +00:00
2015-11-25 14:24:50 +00:00
$result -> closeCursor ();
return array_map ( function ( $array ) { return $this -> readBlob ( $array [ 'carddata' ]);}, $cards );
}
/**
* get URI from a given contact
*
* @ param int $id
* @ return string
*/
public function getCardUri ( $id ) {
$query = $this -> db -> getQueryBuilder ();
$query -> select ( 'uri' ) -> from ( $this -> dbCardsTable )
-> where ( $query -> expr () -> eq ( 'id' , $query -> createParameter ( 'id' )))
-> setParameter ( 'id' , $id );
$result = $query -> execute ();
$uri = $result -> fetch ();
$result -> closeCursor ();
if ( ! isset ( $uri [ 'uri' ])) {
throw new \InvalidArgumentException ( 'Card does not exists: ' . $id );
}
return $uri [ 'uri' ];
}
/**
* return contact with the given URI
*
* @ param string $uri
* @ returns array
*/
public function getContact ( $uri ) {
$result = [];
$query = $this -> db -> getQueryBuilder ();
$query -> select ( '*' ) -> from ( $this -> dbCardsTable )
-> where ( $query -> expr () -> eq ( 'uri' , $query -> createParameter ( 'uri' )))
-> setParameter ( 'uri' , $uri );
$queryResult = $query -> execute ();
$contact = $queryResult -> fetch ();
$queryResult -> closeCursor ();
if ( is_array ( $contact )) {
$result = $contact ;
}
return $result ;
}
2015-11-18 14:04:53 +00:00
/**
2015-12-22 15:11:53 +00:00
* @ param AddressBook $addressBook
2015-11-18 14:04:53 +00:00
* @ param string $element
*/
2015-12-22 15:11:53 +00:00
private function shareWith ( $addressBook , $element ) {
2015-11-05 15:46:37 +00:00
$user = $element [ 'href' ];
$parts = explode ( ':' , $user , 2 );
if ( $parts [ 0 ] !== 'principal' ) {
return ;
}
$p = $this -> principalBackend -> getPrincipalByPath ( $parts [ 1 ]);
if ( is_null ( $p )) {
return ;
}
2015-11-18 14:04:53 +00:00
// remove the share if it already exists
2015-12-22 15:11:53 +00:00
$this -> unshare ( $addressBook -> getBookId (), $element [ 'href' ]);
2016-01-12 12:40:00 +00:00
$access = self :: ACCESS_READ ;
if ( isset ( $element [ 'readOnly' ])) {
$access = $element [ 'readOnly' ] ? self :: ACCESS_READ : self :: ACCESS_READ_WRITE ;
}
2015-11-18 14:04:53 +00:00
2015-12-22 15:11:53 +00:00
$newUri = sha1 ( $addressBook -> getName () . $addressBook -> getOwner ());
2015-11-05 15:46:37 +00:00
$query = $this -> db -> getQueryBuilder ();
$query -> insert ( 'dav_shares' )
-> values ([
'principaluri' => $query -> createNamedParameter ( $parts [ 1 ]),
2016-01-12 12:23:50 +00:00
'uri' => $query -> createNamedParameter ( $newUri ),
2015-11-05 15:46:37 +00:00
'type' => $query -> createNamedParameter ( 'addressbook' ),
2016-01-12 12:40:00 +00:00
'access' => $query -> createNamedParameter ( $access ),
2015-12-22 15:11:53 +00:00
'resourceid' => $query -> createNamedParameter ( $addressBook -> getBookId ())
2015-11-05 15:46:37 +00:00
]);
$query -> execute ();
}
2015-11-18 14:04:53 +00:00
/**
2015-12-22 15:11:53 +00:00
* @ param int $addressBookId
2015-11-18 14:04:53 +00:00
* @ param string $element
*/
2015-12-22 15:11:53 +00:00
private function unshare ( $addressBookId , $element ) {
2015-12-14 15:14:35 +00:00
$parts = explode ( ':' , $element , 2 );
2015-11-05 15:46:37 +00:00
if ( $parts [ 0 ] !== 'principal' ) {
return ;
}
$p = $this -> principalBackend -> getPrincipalByPath ( $parts [ 1 ]);
if ( is_null ( $p )) {
return ;
}
$query = $this -> db -> getQueryBuilder ();
$query -> delete ( 'dav_shares' )
2015-12-22 15:11:53 +00:00
-> where ( $query -> expr () -> eq ( 'resourceid' , $query -> createNamedParameter ( $addressBookId )))
2015-11-05 15:46:37 +00:00
-> andWhere ( $query -> expr () -> eq ( 'type' , $query -> createNamedParameter ( 'addressbook' )))
-> andWhere ( $query -> expr () -> eq ( 'principaluri' , $query -> createNamedParameter ( $parts [ 1 ])))
;
$query -> execute ();
}
/**
2015-11-18 14:04:53 +00:00
* Returns the list of people whom this address book is shared with .
2015-11-05 15:46:37 +00:00
*
* Every element in this array should have the following properties :
* * href - Often a mailto : address
* * commonName - Optional , for example a first + last name
* * status - See the Sabre\CalDAV\SharingPlugin :: STATUS_ constants .
* * readOnly - boolean
* * summary - Optional , a description for the share
*
* @ return array
*/
2016-01-12 12:23:50 +00:00
public function getShares ( $addressBookId ) {
2015-11-05 15:46:37 +00:00
$query = $this -> db -> getQueryBuilder ();
$result = $query -> select ([ 'principaluri' , 'access' ])
-> from ( 'dav_shares' )
2016-01-12 12:23:50 +00:00
-> where ( $query -> expr () -> eq ( 'resourceid' , $query -> createNamedParameter ( $addressBookId )))
2015-11-05 15:46:37 +00:00
-> andWhere ( $query -> expr () -> eq ( 'type' , $query -> createNamedParameter ( 'addressbook' )))
-> execute ();
$shares = [];
while ( $row = $result -> fetch ()) {
$p = $this -> principalBackend -> getPrincipalByPath ( $row [ 'principaluri' ]);
$shares [] = [
'href' => " principal: ${ p['uri'] } " ,
'commonName' => isset ( $p [ '{DAV:}displayname' ]) ? $p [ '{DAV:}displayname' ] : '' ,
'status' => 1 ,
2016-01-12 12:23:50 +00:00
'readOnly' => ( $row [ 'access' ] === self :: ACCESS_READ ),
'{' . \OCA\DAV\CardDAV\Sharing\Plugin :: NS_OWNCLOUD . '}principal' => $p [ 'uri' ]
2015-11-05 15:46:37 +00:00
];
}
return $shares ;
}
2015-11-25 14:24:50 +00:00
/**
* update properties table
*
* @ param int $addressBookId
* @ param string $cardUri
* @ param string $vCardSerialized
*/
protected function updateProperties ( $addressBookId , $cardUri , $vCardSerialized ) {
$cardId = $this -> getCardId ( $cardUri );
$vCard = $this -> readCard ( $vCardSerialized );
$this -> purgeProperties ( $addressBookId , $cardId );
$query = $this -> db -> getQueryBuilder ();
$query -> insert ( $this -> dbCardsPropertiesTable )
-> values (
[
'addressbookid' => $query -> createNamedParameter ( $addressBookId ),
'cardid' => $query -> createNamedParameter ( $cardId ),
'name' => $query -> createParameter ( 'name' ),
'value' => $query -> createParameter ( 'value' ),
'preferred' => $query -> createParameter ( 'preferred' )
]
);
foreach ( $vCard -> children as $property ) {
if ( ! in_array ( $property -> name , self :: $indexProperties )) {
continue ;
}
$preferred = 0 ;
foreach ( $property -> parameters as $parameter ) {
if ( $parameter -> name == 'TYPE' && strtoupper ( $parameter -> getValue ()) == 'PREF' ) {
$preferred = 1 ;
break ;
}
}
$query -> setParameter ( 'name' , $property -> name );
$query -> setParameter ( 'value' , substr ( $property -> getValue (), 0 , 254 ));
$query -> setParameter ( 'preferred' , $preferred );
$query -> execute ();
}
}
/**
* read vCard data into a vCard object
*
* @ param string $cardData
* @ return VCard
*/
protected function readCard ( $cardData ) {
return Reader :: read ( $cardData );
}
/**
* delete all properties from a given card
*
* @ param int $addressBookId
* @ param int $cardId
*/
protected function purgeProperties ( $addressBookId , $cardId ) {
$query = $this -> db -> getQueryBuilder ();
$query -> delete ( $this -> dbCardsPropertiesTable )
-> where ( $query -> expr () -> eq ( 'cardid' , $query -> createNamedParameter ( $cardId )))
-> andWhere ( $query -> expr () -> eq ( 'addressbookid' , $query -> createNamedParameter ( $addressBookId )));
$query -> execute ();
}
/**
* get ID from a given contact
*
* @ param string $uri
* @ return int
*/
protected function getCardId ( $uri ) {
$query = $this -> db -> getQueryBuilder ();
$query -> select ( 'id' ) -> from ( $this -> dbCardsTable )
-> where ( $query -> expr () -> eq ( 'uri' , $query -> createNamedParameter ( $uri )));
$result = $query -> execute ();
$cardIds = $result -> fetch ();
$result -> closeCursor ();
if ( ! isset ( $cardIds [ 'id' ])) {
throw new \InvalidArgumentException ( 'Card does not exists: ' . $uri );
}
return ( int ) $cardIds [ 'id' ];
}
2016-01-12 12:23:50 +00:00
/**
2016-01-13 10:55:50 +00:00
* For shared address books the sharee is set in the ACL of the address book
2016-01-12 12:23:50 +00:00
* @ param $addressBookId
* @ param $acl
* @ return array
*/
public function applyShareAcl ( $addressBookId , $acl ) {
$shares = $this -> getShares ( $addressBookId );
foreach ( $shares as $share ) {
$acl [] = [
'privilege' => '{DAV:}read' ,
'principal' => $share [ '{' . \OCA\DAV\CardDAV\Sharing\Plugin :: NS_OWNCLOUD . '}principal' ],
'protected' => true ,
];
if ( ! $share [ 'readOnly' ]) {
$acl [] = [
'privilege' => '{DAV:}write' ,
'principal' => $share [ '{' . \OCA\DAV\CardDAV\Sharing\Plugin :: NS_OWNCLOUD . '}principal' ],
'protected' => true ,
];
}
}
return $acl ;
}
2015-10-30 15:05:25 +00:00
}