forked from extern/egroupware
don't use the note to compare contacts anymore. They can be problematic regarding line breaks and such.
This commit is contained in:
parent
44b0e47999
commit
57601366dd
537
addressbook/inc/class.vcaladdressbook.inc.php
Normal file
537
addressbook/inc/class.vcaladdressbook.inc.php
Normal file
@ -0,0 +1,537 @@
|
||||
<?php
|
||||
/**************************************************************************\
|
||||
* eGroupWare - iCalendar Parser *
|
||||
* http://www.egroupware.org *
|
||||
* Written by Lars Kneschke <lkneschke@egroupware.org> *
|
||||
* -------------------------------------------- *
|
||||
* This program is free software; you can redistribute it and/or modify it *
|
||||
* under the terms of the GNU General Public License as published by the *
|
||||
* Free Software Foundation; either version 2 of the License. *
|
||||
\**************************************************************************/
|
||||
|
||||
/* $Id$ */
|
||||
|
||||
require_once EGW_SERVER_ROOT.'/addressbook/inc/class.boaddressbook.inc.php';
|
||||
require_once EGW_SERVER_ROOT.'/phpgwapi/inc/horde/Horde/iCalendar.php';
|
||||
|
||||
class vcaladdressbook extends boaddressbook
|
||||
{
|
||||
#function vcaladdressbook()
|
||||
#{
|
||||
# $this->boaddressbook();
|
||||
#}
|
||||
|
||||
/**
|
||||
* @return int contact id
|
||||
* @param string $_vcard the vcard
|
||||
* @param int $_abID the internal addressbook id
|
||||
* @param int $_vcardProfile profile id for mapping from vcard values to egw addressbook
|
||||
* @desc import a vard into addressbook
|
||||
*/
|
||||
function addVCard($_vcard, $_abID)
|
||||
{
|
||||
if(!$contact = $this->vcardtoegw($_vcard)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if($_abID > 0)
|
||||
{
|
||||
// update entry
|
||||
$contact['ab_id'] = $_abID;
|
||||
return $this->update_entry($contact);
|
||||
}
|
||||
else
|
||||
{
|
||||
// add entry
|
||||
return $this->add_entry($contact);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* return a vcard
|
||||
*
|
||||
* @param int $_id the id of the contact
|
||||
* @param int $_vcardProfile profile id for mapping from vcard values to egw addressbook
|
||||
* @return string containing the vcard
|
||||
*/
|
||||
function getVCard($_id)
|
||||
{
|
||||
require_once(EGW_SERVER_ROOT.'/phpgwapi/inc/horde/Horde/iCalendar/vcard.php');
|
||||
|
||||
$vCard =& new Horde_iCalendar_vcard;
|
||||
|
||||
if(!is_array($this->supportedFields))
|
||||
{
|
||||
$this->setSupportedFields();
|
||||
}
|
||||
|
||||
foreach($this->supportedFields as $databaseFields)
|
||||
{
|
||||
foreach($databaseFields as $databaseField)
|
||||
{
|
||||
if(!empty($databaseField))
|
||||
{
|
||||
$fields[] = $databaseField;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#_debug_array($fields);
|
||||
|
||||
if($this->check_perms($_id,EGW_ACL_READ))
|
||||
{
|
||||
//$data = array('id' => $_id, 'fields' => $fields);
|
||||
$entry = $this->so->read_entry($_id,$fields);
|
||||
$entry = $this->strip_html($entry);
|
||||
if($this->xmlrpc)
|
||||
{
|
||||
$entry = $this->data2xmlrpc($entry);
|
||||
}
|
||||
#_debug_array($entry);
|
||||
$sysCharSet = $GLOBALS['egw']->translation->charset();
|
||||
|
||||
foreach($this->supportedFields as $vcardField => $databaseFields)
|
||||
{
|
||||
$options = array();
|
||||
$value = '';
|
||||
foreach($databaseFields as $databaseField)
|
||||
{
|
||||
$tempVal = ';';
|
||||
if(!empty($databaseField))
|
||||
{
|
||||
#$tempVal = trim('value').';';
|
||||
$tempVal = trim($entry[0][$databaseField]).';';
|
||||
}
|
||||
$value .= $tempVal;
|
||||
}
|
||||
// remove the last ;
|
||||
$value = substr($value, 0, -1);
|
||||
|
||||
switch($vcardField)
|
||||
{
|
||||
// TODO handle multiple categories
|
||||
case 'CATEGORIES':
|
||||
$catData = ExecMethod('phpgwapi.categories.return_single',$value);
|
||||
$value = $catData[0]['name'];
|
||||
break;
|
||||
case 'CLASS':
|
||||
$value = ($value == 'private' ? 'PRIVATE' : 'PUBLIC');
|
||||
break;
|
||||
case 'BDAY':
|
||||
if(!empty($value))
|
||||
{
|
||||
$dateParts = explode('/',$value);
|
||||
$value = sprintf('%04d%02d%02dT000000Z',$dateParts[2],$dateParts[0],$dateParts[1]);
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
$value = $GLOBALS['egw']->translation->convert($value,$sysCharSet,'utf-8');
|
||||
|
||||
// don't add the entry if it contains only ';'
|
||||
if(strlen(str_replace(';','',$value)) != 0)
|
||||
{
|
||||
$vCard->setAttribute($vcardField, $value);
|
||||
}
|
||||
if(preg_match('/([\000-\012\015\016\020-\037\075])/',$value))
|
||||
{
|
||||
$options['ENCODING'] = 'QUOTED-PRINTABLE';
|
||||
}
|
||||
if(preg_match('/([\177-\377])/',$value))
|
||||
{
|
||||
$options['CHARSET'] = 'UTF-8';
|
||||
}
|
||||
$vCard->setParameter($vcardField, $options);
|
||||
}
|
||||
|
||||
$result = $vCard->exportvCalendar();
|
||||
|
||||
return $result;
|
||||
}
|
||||
|
||||
if($this->xmlrpc)
|
||||
{
|
||||
$GLOBALS['server']->xmlrpc_error($GLOBALS['xmlrpcerr']['no_access'],$GLOBALS['xmlrpcstr']['no_access']);
|
||||
}
|
||||
return False;
|
||||
}
|
||||
|
||||
function search($_vcard) {
|
||||
if(!$contact = $this->vcardtoegw($_vcard)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// they can be problematic
|
||||
unset($contact['note']);
|
||||
|
||||
if($foundContacts = $this->read_entries(array('query' => $contact))) {
|
||||
return $foundContacts[0][id];
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
function setSupportedFields($_productManufacturer='file', $_productName='')
|
||||
{
|
||||
$defaultFields[0] = array(
|
||||
'ADR' => array('','','adr_one_street','adr_one_locality','adr_one_region',
|
||||
'adr_one_postalcode','adr_one_countryname'),
|
||||
'CATEGORIES' => array('cat_id'),
|
||||
'CLASS' => array('access'),
|
||||
'EMAIL' => array('email'),
|
||||
'N' => array('n_family','n_given','','',''),
|
||||
'NOTE' => array('note'),
|
||||
'ORG' => array('org_name',''),
|
||||
'TEL;CELL' => array('tel_cell'),
|
||||
'TEL;FAX' => array('tel_fax'),
|
||||
'TEL;HOME' => array('tel_home'),
|
||||
'TEL;WORK' => array('tel_work'),
|
||||
'TITLE' => array('title'),
|
||||
);
|
||||
|
||||
$defaultFields[1] = array(
|
||||
'ADR;WORK' => array('','','adr_one_street','adr_one_locality','adr_one_region',
|
||||
'adr_one_postalcode','adr_one_countryname'),
|
||||
'ADR;HOME' => array('','','adr_two_street','adr_two_locality','adr_two_region',
|
||||
'adr_two_postalcode','adr_two_countryname'),
|
||||
'BDAY' => array('bday'),
|
||||
'CATEGORIES' => array('cat_id'),
|
||||
'EMAIL;INTERNET;WORK' => array('email'),
|
||||
'EMAIL;INTERNET;HOME' => array('email_home'),
|
||||
'N' => array('n_family','n_given','n_middle','n_prefix','n_suffix'),
|
||||
'NOTE' => array('note'),
|
||||
'ORG' => array('org_name','org_unit'),
|
||||
'TEL;CELL;WORK' => array('tel_cell'),
|
||||
'TEL;FAX;WORK' => array('tel_fax'),
|
||||
'TEL;HOME' => array('tel_home'),
|
||||
'TEL;PAGER;WORK' => array('tel_pager'),
|
||||
'TEL;WORK' => array('tel_work'),
|
||||
'TITLE' => array('title'),
|
||||
'URL;WORK' => array('url'),
|
||||
);
|
||||
|
||||
$defaultFields[2] = array(
|
||||
'ADR;HOME' => array('','','adr_one_street','adr_one_locality','adr_one_region',
|
||||
'adr_one_postalcode','adr_one_countryname'),
|
||||
'BDAY' => array('bday'),
|
||||
'CATEGORIES' => array('cat_id'),
|
||||
'CLASS' => array('access'),
|
||||
'EMAIL' => array('email'),
|
||||
'N' => array('n_family','n_given','','',''),
|
||||
'NOTE' => array('note'),
|
||||
'ORG' => array('org_name',''),
|
||||
'TEL;CELL;WORK' => array('tel_cell'),
|
||||
'TEL;FAX;WORK' => array('tel_fax'),
|
||||
'TEL;HOME' => array('tel_home'),
|
||||
'TEL;WORK' => array('tel_work'),
|
||||
'TITLE' => array('title'),
|
||||
'URL' => array('url'),
|
||||
);
|
||||
|
||||
$defaultFields[3] = array(
|
||||
'ADR;WORK' => array('','','adr_one_street','adr_one_locality','adr_one_region',
|
||||
'adr_one_postalcode','adr_one_countryname'),
|
||||
'ADR;HOME' => array('','','adr_two_street','adr_two_locality','adr_two_region',
|
||||
'adr_two_postalcode','adr_two_countryname'),
|
||||
'BDAY' => array('bday'),
|
||||
'EMAIL;INTERNET;WORK' => array('email'),
|
||||
'EMAIL;INTERNET;HOME' => array('email_home'),
|
||||
'N' => array('n_family','n_given','','',''),
|
||||
'NOTE' => array('note'),
|
||||
'ORG' => array('org_name','org_unit'),
|
||||
'TEL;CELL;WORK' => array('tel_cell'),
|
||||
'TEL;FAX;WORK' => array('tel_fax'),
|
||||
'TEL;HOME' => array('tel_home'),
|
||||
'TEL;PAGER;WORK' => array('tel_pager'),
|
||||
'TEL;WORK' => array('tel_work'),
|
||||
'TITLE' => array('title'),
|
||||
'URL;WORK' => array('url'),
|
||||
);
|
||||
|
||||
switch(strtolower($_productManufacturer))
|
||||
{
|
||||
case 'nexthaus corporation':
|
||||
switch(strtolower($_productName))
|
||||
{
|
||||
case 'syncje outlook edition':
|
||||
default:
|
||||
$this->supportedFields = $defaultFields[1];
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
// multisync does not provide anymore information then the manufacturer
|
||||
// we suppose multisync with evolution
|
||||
case 'the multisync project':
|
||||
switch(strtolower($_productName))
|
||||
{
|
||||
default:
|
||||
$this->supportedFields = $defaultFields[0];
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
case 'siemens':
|
||||
switch(strtolower($_productName))
|
||||
{
|
||||
case 'sx1':
|
||||
default:
|
||||
$this->supportedFields = $defaultFields[3];
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
case 'sonyericsson':
|
||||
switch(strtolower($_productName))
|
||||
{
|
||||
case 'd750i':
|
||||
default:
|
||||
$this->supportedFields = $defaultFields[2];
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
case 'synthesis ag':
|
||||
switch(strtolower($_productName))
|
||||
{
|
||||
default:
|
||||
$this->supportedFields = $defaultFields[0];
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
case 'file': // used outside of SyncML, eg. by the calendar itself ==> all possible fields
|
||||
$this->supportedFields = $defaultFields[1];
|
||||
break;
|
||||
|
||||
// the fallback for SyncML
|
||||
default:
|
||||
error_log("Client not found: $_productManufacturer $_productName");
|
||||
$this->supportedFields = $defaultFields[0];
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
function vcardtoegw($_vcard) {
|
||||
if(!is_array($this->supportedFields))
|
||||
{
|
||||
$this->setSupportedFields();
|
||||
}
|
||||
|
||||
$this->supportedFields[0] = array(
|
||||
);
|
||||
|
||||
require_once(EGW_SERVER_ROOT.'/phpgwapi/inc/horde/Horde/iCalendar.php');
|
||||
|
||||
$vCard = Horde_iCalendar::newComponent('vcard', $container);
|
||||
|
||||
if(!$vCard->parsevCalendar($_vcard,'VCARD'))
|
||||
{
|
||||
return False;
|
||||
}
|
||||
$vcardValues = $vCard->getAllAttributes();
|
||||
|
||||
#print "<pre>$_vcard</pre>";
|
||||
|
||||
#_debug_array($vcardValues);
|
||||
|
||||
foreach($vcardValues as $key => $vcardRow)
|
||||
{
|
||||
$rowName = $vcardRow['name'];
|
||||
$mailtype = ';INTERNET';
|
||||
$tempVal = ';WORK';
|
||||
|
||||
if(isset($vcardRow['params']['INTERNET']))
|
||||
{
|
||||
$rowName .= ";INTERNET";
|
||||
}
|
||||
if(isset($vcardRow['params']['CELL']))
|
||||
{
|
||||
$rowName .= ';CELL';
|
||||
}
|
||||
if(isset($vcardRow['params']['FAX']))
|
||||
{
|
||||
$rowName .= ';FAX';
|
||||
}
|
||||
if(isset($vcardRow['params']['PAGER']))
|
||||
{
|
||||
$rowName .= ';PAGER';
|
||||
}
|
||||
if(isset($vcardRow['params']['WORK']))
|
||||
{
|
||||
$rowName .= ';WORK';
|
||||
}
|
||||
if(isset($vcardRow['params']['HOME']))
|
||||
{
|
||||
$rowName .= ';HOME';
|
||||
}
|
||||
|
||||
$rowNames[$rowName] = $key;
|
||||
}
|
||||
|
||||
#error_log('rowNames: '.print_r($rowNames,true));
|
||||
|
||||
// now we have all rowNames the vcard provides
|
||||
// we just need to map to the right addressbook fieldnames
|
||||
// we need also to take care about ADR for example. we do not
|
||||
// support this. We support only ADR;WORK or ADR;HOME
|
||||
|
||||
foreach($rowNames as $rowName => $vcardKey)
|
||||
{
|
||||
switch($rowName)
|
||||
{
|
||||
case 'ADR':
|
||||
case 'TEL':
|
||||
case 'TEL;FAX':
|
||||
case 'TEL;CELL':
|
||||
case 'TEL;PAGER':
|
||||
if(!isset($rowNames[$rowName. ';WORK']))
|
||||
{
|
||||
$finalRowNames[$rowName. ';WORK'] = $vcardKey;
|
||||
}
|
||||
break;
|
||||
case 'EMAIL':
|
||||
case 'EMAIL;WORK':
|
||||
if(!isset($rowNames['EMAIL;INTERNET;WORK']))
|
||||
{
|
||||
$finalRowNames['EMAIL;INTERNET;WORK'] = $vcardKey;
|
||||
}
|
||||
break;
|
||||
case 'EMAIL;HOME':
|
||||
if(!isset($rowNames['EMAIL;INTERNET;HOME']))
|
||||
{
|
||||
$finalRowNames['EMAIL;INTERNET;HOME'] = $vcardKey;
|
||||
}
|
||||
break;
|
||||
|
||||
case 'CATEGORIES':
|
||||
#cat_id = 7,8
|
||||
$vcardData['category'] = array();
|
||||
if ($attributes['value'])
|
||||
{
|
||||
if (!is_object($this->cat))
|
||||
{
|
||||
if (!is_object($GLOBALS['egw']->categories))
|
||||
{
|
||||
$GLOBALS['egw']->categories =& CreateObject('phpgwapi.categories',$this->owner,'addressbook');
|
||||
}
|
||||
$this->cat =& $GLOBALS['egw']->categories;
|
||||
}
|
||||
foreach(explode(',',$attributes['value']) as $cat_name)
|
||||
{
|
||||
if (!($cat_id = $this->cat->name2id($cat_name)))
|
||||
{
|
||||
$cat_id = $this->cat->add( array('name' => $cat_name,'descr' => $cat_name ));
|
||||
}
|
||||
$vcardData['category'][] = $cat_id;
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
case 'VERSION':
|
||||
break;
|
||||
default:
|
||||
$finalRowNames[$rowName] = $vcardKey;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
#_debug_array($finalRowNames);
|
||||
$contact = array();
|
||||
|
||||
foreach($finalRowNames as $key => $vcardKey)
|
||||
{
|
||||
if(isset($this->supportedFields[$key]))
|
||||
{
|
||||
$fieldNames = $this->supportedFields[$key];
|
||||
foreach($fieldNames as $fieldKey => $fieldName)
|
||||
{
|
||||
if(!empty($fieldName))
|
||||
{
|
||||
switch($fieldName)
|
||||
{
|
||||
case 'access':
|
||||
if($vcardValues[$vcardKey]['values'][$fieldKey] == 'PRIVATE')
|
||||
{
|
||||
$contact[$fieldName] = 'private';
|
||||
}
|
||||
else
|
||||
{
|
||||
$contact[$fieldName] = 'public';
|
||||
}
|
||||
break;
|
||||
case 'cat_id':
|
||||
if (!is_object($this->cat))
|
||||
{
|
||||
if (!is_object($GLOBALS['egw']->categories))
|
||||
{
|
||||
$GLOBALS['egw']->categories =& CreateObject('phpgwapi.categories',$GLOBALS['egw_info']['user']['account_id'],'addressbook');
|
||||
}
|
||||
$this->cat =& $GLOBALS['egw']->categories;
|
||||
}
|
||||
foreach(explode(',',$vcardValues[$vcardKey]['values'][$fieldKey]) as $cat_name)
|
||||
{
|
||||
if (!($cat_id = $this->cat->name2id($cat_name)))
|
||||
{
|
||||
$cat_id = $this->cat->add( array('name' => $cat_name,'descr' => $cat_name ));
|
||||
}
|
||||
$contact[$fieldName] = $cat_id;
|
||||
}
|
||||
break;
|
||||
default:
|
||||
$contact[$fieldName] = $vcardValues[$vcardKey]['values'][$fieldKey];
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#return true;
|
||||
|
||||
/* _debug_array($contact);exit; */
|
||||
$contact['fn'] = trim($contact['n_given'].' '.$contact['n_family']);
|
||||
if(!$contact['tel_work'])
|
||||
{
|
||||
$contact['tel_work'] = '';
|
||||
}
|
||||
if(!$contact['tel_home'])
|
||||
{
|
||||
$contact['tel_home'] = '';
|
||||
}
|
||||
if(!$contact['tel_voice'])
|
||||
{
|
||||
$contact['tel_voice'] = '';
|
||||
}
|
||||
if(!$contact['tel_fax'])
|
||||
{
|
||||
$contact['tel_fax'] = '';
|
||||
}
|
||||
if(!$contact['tel_msg'])
|
||||
{
|
||||
$contact['tel_msg'] = '';
|
||||
}
|
||||
if(!$contact['tel_cell'])
|
||||
{
|
||||
$contact['tel_cell'] = '';
|
||||
}
|
||||
if(!$contact['tel_pager'])
|
||||
{
|
||||
$contact['tel_pager'] = '';
|
||||
}
|
||||
if(!$contact['tel_car'])
|
||||
{
|
||||
$contact['tel_car'] = '';
|
||||
}
|
||||
if(!$contact['tel_isdn'])
|
||||
{
|
||||
$contact['tel_isdn'] = '';
|
||||
}
|
||||
if(!$contact['tel_video'])
|
||||
{
|
||||
$contact['tel_video'] = '';
|
||||
}
|
||||
|
||||
return $contact;
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user