2005-06-19 21:00:58 +02:00
|
|
|
<?php
|
2009-07-15 21:31:25 +02:00
|
|
|
/**
|
|
|
|
* eGroupWare - SyncML based on Horde 3
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* Using the PEAR Log class (which need to be installed!)
|
|
|
|
*
|
|
|
|
* @link http://www.egroupware.org
|
|
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
|
|
* @package api
|
|
|
|
* @subpackage horde
|
|
|
|
* @author Anthony Mills <amills@pyramid6.com>
|
|
|
|
* @author Joerg Lehrke <jlehrke@noc.de>
|
|
|
|
* @copyright (c) The Horde Project (http://www.horde.org/)
|
|
|
|
* @version $Id$
|
|
|
|
*/
|
2005-06-19 21:00:58 +02:00
|
|
|
include_once 'Horde/SyncML/State.php';
|
|
|
|
include_once 'Horde/SyncML/Command.php';
|
|
|
|
include_once 'Horde/SyncML/Command/Sync/SyncElement.php';
|
|
|
|
include_once 'Horde/SyncML/Sync/TwoWaySync.php';
|
|
|
|
include_once 'Horde/SyncML/Sync/SlowSync.php';
|
2006-01-16 10:49:51 +01:00
|
|
|
include_once 'Horde/SyncML/Sync/OneWayFromServerSync.php';
|
2009-07-15 21:31:25 +02:00
|
|
|
include_once 'Horde/SyncML/Sync/OneWayFromClientSync.php';
|
2006-10-04 20:16:58 +02:00
|
|
|
include_once 'Horde/SyncML/Sync/RefreshFromServerSync.php';
|
2009-07-15 21:31:25 +02:00
|
|
|
include_once 'Horde/SyncML/Sync/RefreshFromClientSync.php';
|
2005-06-19 21:00:58 +02:00
|
|
|
|
2009-07-15 21:31:25 +02:00
|
|
|
class Horde_SyncML_Command_Sync extends Horde_SyncML_Command {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Name of the command.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
var $_cmdName = 'Sync';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Source database of the <Sync> command.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
var $_sourceURI;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Target database of the <Sync> command.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
var $_targetURI;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Optional parameter for the Target.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
var $_targetURIParameters;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SyncML_SyncElement object for the currently parsed sync command.
|
|
|
|
*
|
|
|
|
* @var SyncML_SyncElement
|
|
|
|
*/
|
|
|
|
var $_curItem;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* List of all SyncML_SyncElement objects that have parsed.
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
var $_syncElements = array();
|
|
|
|
|
|
|
|
function output($currentCmdID, &$output)
|
|
|
|
{
|
|
|
|
$state = &$_SESSION['SyncML.state'];
|
|
|
|
|
|
|
|
Horde::logMessage('SyncML: $this->_targetURI = ' . $this->_targetURI, __FILE__, __LINE__, PEAR_LOG_DEBUG);
|
|
|
|
|
|
|
|
$status = new Horde_SyncML_Command_Status(RESPONSE_OK, 'Sync');
|
|
|
|
|
|
|
|
$status->setCmdRef($this->_cmdID);
|
|
|
|
|
|
|
|
if ($this->_targetURI != null) {
|
|
|
|
$status->setTargetRef((isset($this->_targetURIParameters) ? $this->_targetURI.'?/'.$this->_targetURIParameters : $this->_targetURI));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->_sourceURI != null) {
|
|
|
|
$status->setSourceRef($this->_sourceURI);
|
|
|
|
}
|
|
|
|
|
|
|
|
$currentCmdID = $status->output($currentCmdID, $output);
|
|
|
|
|
2009-09-20 21:45:03 +02:00
|
|
|
if ($this->_targetURI != "configuration" && // Fix Funambol issue
|
|
|
|
($sync = &$state->getSync($this->_targetURI))) {
|
2009-07-15 21:31:25 +02:00
|
|
|
$currentCmdID = $sync->startSync($currentCmdID, $output);
|
|
|
|
|
|
|
|
foreach ($this->_syncElements as $element) {
|
|
|
|
$currentCmdID = $sync->nextSyncCommand($currentCmdID, $element, $output);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $currentCmdID;
|
|
|
|
}
|
2006-03-21 14:49:13 +01:00
|
|
|
|
2005-06-19 21:00:58 +02:00
|
|
|
|
|
|
|
function startElement($uri, $element, $attrs)
|
|
|
|
{
|
|
|
|
parent::startElement($uri, $element, $attrs);
|
|
|
|
|
2009-07-15 21:31:25 +02:00
|
|
|
switch (count($this->_stack)) {
|
2010-05-20 12:48:45 +02:00
|
|
|
case 2:
|
|
|
|
if ($element == 'Replace' ||
|
|
|
|
$element == 'Add' ||
|
|
|
|
$element == 'Delete') {
|
|
|
|
Horde::logMessage("SyncML: sync element $element found", __FILE__, __LINE__, PEAR_LOG_DEBUG);
|
|
|
|
$this->_curItem = &Horde_SyncML_Command_Sync_SyncElement::factory($element);
|
|
|
|
}
|
|
|
|
break;
|
2005-06-19 21:00:58 +02:00
|
|
|
}
|
2009-07-15 21:31:25 +02:00
|
|
|
if (isset($this->_curItem)) {
|
|
|
|
$this->_curItem->startElement($uri, $element, $attrs);
|
2005-06-19 21:00:58 +02:00
|
|
|
}
|
|
|
|
}
|
2009-07-15 21:31:25 +02:00
|
|
|
|
|
|
|
|
2005-06-19 21:00:58 +02:00
|
|
|
// We create a seperate Sync Element for the Sync Data sent
|
|
|
|
// from the Server to the client as we want to process the
|
|
|
|
// client sync information before.
|
|
|
|
|
|
|
|
function syncToClient($currentCmdID, &$output)
|
|
|
|
{
|
2009-12-01 11:14:23 +01:00
|
|
|
Horde::logMessage('SyncML: starting sync to client',
|
|
|
|
__FILE__, __LINE__, PEAR_LOG_DEBUG);
|
2005-06-19 21:00:58 +02:00
|
|
|
|
2009-12-01 11:14:23 +01:00
|
|
|
$state = &$_SESSION['SyncML.state'];
|
2005-06-19 21:00:58 +02:00
|
|
|
|
2009-12-01 11:14:23 +01:00
|
|
|
if ($state->getSyncStatus() >= CLIENT_SYNC_FINNISHED
|
|
|
|
&& $state->getSyncStatus() < SERVER_SYNC_FINNISHED)
|
|
|
|
{
|
|
|
|
$deviceInfo = $state->getClientDeviceInfo();
|
|
|
|
if (($targets = $state->getTargets())) {
|
|
|
|
foreach ($targets as $target)
|
2006-01-14 09:29:23 +01:00
|
|
|
{
|
2009-12-01 11:14:23 +01:00
|
|
|
$sync = &$state->getSync($target);
|
|
|
|
Horde::logMessage('SyncML[' . session_id() . ']: sync alerttype ' .
|
|
|
|
$sync->_syncType . ' found for target ' . $target,
|
|
|
|
__FILE__, __LINE__, PEAR_LOG_DEBUG);
|
|
|
|
if ($sync->_syncType == ALERT_ONE_WAY_FROM_CLIENT ||
|
|
|
|
$sync->_syncType == ALERT_REFRESH_FROM_CLIENT) {
|
|
|
|
Horde::logMessage('SyncML[' . session_id() .
|
|
|
|
']: From client Sync, no sync of ' . $target .
|
|
|
|
' to client', __FILE__, __LINE__, PEAR_LOG_DEBUG);
|
|
|
|
$state->clearSync($target);
|
|
|
|
|
|
|
|
} elseif ($state->getSyncStatus() >= CLIENT_SYNC_ACKNOWLEDGED) {
|
|
|
|
|
|
|
|
Horde::logMessage("SyncML: starting sync to client $target",
|
|
|
|
__FILE__, __LINE__, PEAR_LOG_DEBUG);
|
|
|
|
$attrs = array();
|
|
|
|
|
|
|
|
$state->setSyncStatus(SERVER_SYNC_DATA_PENDING);
|
|
|
|
|
|
|
|
$output->startElement($state->getURI(), 'Sync', $attrs);
|
|
|
|
$output->startElement($state->getURI(), 'CmdID', $attrs);
|
|
|
|
$output->characters($currentCmdID);
|
|
|
|
$currentCmdID++;
|
|
|
|
$output->endElement($state->getURI(), 'CmdID');
|
|
|
|
|
|
|
|
$output->startElement($state->getURI(), 'Target', $attrs);
|
|
|
|
$output->startElement($state->getURI(), 'LocURI', $attrs);
|
|
|
|
$chars = $sync->_sourceLocURI;
|
|
|
|
$output->characters($chars);
|
|
|
|
$output->endElement($state->getURI(), 'LocURI');
|
|
|
|
$output->endElement($state->getURI(), 'Target');
|
|
|
|
|
|
|
|
$output->startElement($state->getURI(), 'Source', $attrs);
|
|
|
|
$output->startElement($state->getURI(), 'LocURI', $attrs);
|
|
|
|
$chars = (isset($sync->_targetLocURIParameters) ? $sync->_targetLocURI.'?/'.$sync->_targetLocURIParameters : $sync->_targetLocURI);
|
|
|
|
$output->characters($chars);
|
|
|
|
$output->endElement($state->getURI(), 'LocURI');
|
|
|
|
$output->endElement($state->getURI(), 'Source');
|
|
|
|
|
|
|
|
if(!$sync->_syncDataLoaded)
|
|
|
|
{
|
|
|
|
$numberOfItems = $sync->loadData();
|
|
|
|
if($deviceInfo['supportNumberOfChanges'])
|
|
|
|
{
|
|
|
|
$output->startElement($state->getURI(), 'NumberOfChanges', $attrs);
|
|
|
|
$output->characters($numberOfItems);
|
|
|
|
$output->endElement($state->getURI(), 'NumberOfChanges');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$currentCmdID = $sync->endSync($currentCmdID, $output);
|
|
|
|
|
|
|
|
$output->endElement($state->getURI(), 'Sync');
|
|
|
|
|
|
|
|
if (isset($state->curSyncItem) ||
|
|
|
|
$state->getNumberOfElements() === false) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Horde::logMessage("SyncML: Waiting for client ACKNOWLEDGE for $target",
|
|
|
|
__FILE__, __LINE__, PEAR_LOG_DEBUG);
|
|
|
|
}
|
2006-01-14 09:29:23 +01:00
|
|
|
}
|
2009-12-01 11:14:23 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// no syncs left
|
|
|
|
if ($state->getTargets() === false &&
|
|
|
|
!isset($state->curSyncItem)) {
|
|
|
|
$state->setSyncStatus(SERVER_SYNC_FINNISHED);
|
|
|
|
}
|
|
|
|
Horde::logMessage('SyncML: syncStatus(syncToClient) = ' .
|
|
|
|
$state->getSyncStatus(), __FILE__, __LINE__, PEAR_LOG_DEBUG);
|
|
|
|
}
|
|
|
|
return $currentCmdID;
|
2005-06-19 21:00:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function endElement($uri, $element)
|
|
|
|
{
|
2009-07-15 21:31:25 +02:00
|
|
|
if (isset($this->_curItem)) {
|
|
|
|
$this->_curItem->endElement($uri, $element);
|
2005-06-19 21:00:58 +02:00
|
|
|
}
|
|
|
|
|
2009-07-15 21:31:25 +02:00
|
|
|
switch (count($this->_stack)) {
|
2005-06-19 21:00:58 +02:00
|
|
|
case 2:
|
2009-07-15 21:31:25 +02:00
|
|
|
if ($element == 'Replace' ||
|
|
|
|
$element == 'Add' ||
|
|
|
|
$element == 'Delete') {
|
|
|
|
$this->_syncElements[] = &$this->_curItem;
|
|
|
|
unset($this->_curItem);
|
2005-06-19 21:00:58 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 3:
|
2009-07-15 21:31:25 +02:00
|
|
|
if ($element == 'LocURI' && !isset($this->_curItem)) {
|
|
|
|
if ($this->_stack[1] == 'Source') {
|
2005-06-19 21:00:58 +02:00
|
|
|
$this->_sourceURI = trim($this->_chars);
|
2009-07-15 21:31:25 +02:00
|
|
|
} elseif ($this->_stack[1] == 'Target') {
|
2005-06-19 21:00:58 +02:00
|
|
|
$targetURIData = explode('?/',trim($this->_chars));
|
|
|
|
|
2009-07-15 21:31:25 +02:00
|
|
|
$this->_targetURI = $targetURIData[0];
|
|
|
|
|
|
|
|
if (isset($targetURIData[1])) {
|
|
|
|
$this->_targetURIParameters = $targetURIData[1];
|
|
|
|
}
|
2005-06-19 21:00:58 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
parent::endElement($uri, $element);
|
2009-07-15 21:31:25 +02:00
|
|
|
|
2005-06-19 21:00:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function characters($str)
|
|
|
|
{
|
2009-07-15 21:31:25 +02:00
|
|
|
if (isset($this->_curItem)) {
|
|
|
|
$this->_curItem->characters($str);
|
2005-06-19 21:00:58 +02:00
|
|
|
} else {
|
|
|
|
if (isset($this->_chars)) {
|
2009-07-15 21:31:25 +02:00
|
|
|
$this->_chars .= $str;
|
2005-06-19 21:00:58 +02:00
|
|
|
} else {
|
|
|
|
$this->_chars = $str;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|