diff --git a/application/cmdbabstract.class.inc.php b/application/cmdbabstract.class.inc.php index a121a003ff..edaab59516 100644 --- a/application/cmdbabstract.class.inc.php +++ b/application/cmdbabstract.class.inc.php @@ -533,6 +533,11 @@ JS $sLabel = Dict::S('Tag:Synchronized'); $sSynchroTagId = 'synchro_icon-'.$this->GetKey(); $aTags[$sSynchroTagId] = ['title' => $sTip, 'css_classes' => 'ibo-object-details--tag--synchronized', 'decoration_classes' => 'fas fa-lock', 'label' => $sLabel]; + if (UserRights::IsActionAllowed('SynchroReplica', UR_ACTION_READ)) { + $sFilter = 'SELECT SynchroReplica WHERE dest_class=\''.get_class($this).'\' AND dest_id='.$this->GetKey(); + $sUrlSearchReplica = 'UI.php?operation=search&filter='.urlencode(json_encode([$sFilter, [], []])); + $oPage->add_ready_script("$('#$sSynchroTagId').on('click',function() {window.location = '$sUrlSearchReplica' });"); + } } } diff --git a/core/attributedef.class.inc.php b/core/attributedef.class.inc.php index 9fc7b01059..4921decff7 100644 --- a/core/attributedef.class.inc.php +++ b/core/attributedef.class.inc.php @@ -3082,13 +3082,17 @@ class AttributeObjectKey extends AttributeDBFieldVoid { return 0; } - if (MetaModel::IsValidObject($proposedValue)) - { + if (MetaModel::IsValidObject($proposedValue)) { return $proposedValue->GetKey(); } return (int)$proposedValue; } + + public function GetTargetClass($iType = EXTKEY_RELATIVE) + { + return ''; + } } /** diff --git a/synchro/replica.php b/synchro/replica.php index f991a9bf0f..6aafa59227 100644 --- a/synchro/replica.php +++ b/synchro/replica.php @@ -18,13 +18,14 @@ */ use Combodo\iTop\Application\WebPage\iTopWebPage; +use Combodo\iTop\Core\CMDBChange\CMDBChangeOrigin; require_once('../approot.inc.php'); require_once(APPROOT.'/application/application.inc.php'); require_once(APPROOT.'/application/startup.inc.php'); require_once(APPROOT.'/application/loginwebpage.class.inc.php'); -LoginWebPage::DoLogin(); +LoginWebPage::DoLogin(); $sOperation = utils::ReadParam('operation', 'menu'); $oAppContext = new ApplicationContext(); @@ -33,46 +34,163 @@ $oP = new iTopWebPage("iTop - Synchro Replicas"); // Main program $sOperation = utils::ReadParam('operation', 'details'); -try + +/** + * @param \DBObject|null $oReplica + * @param $this + * + * @return \SynchroLog + * @throws \ArchivedObjectException + * @throws \CoreCannotSaveObjectException + * @throws \CoreException + * @throws \CoreUnexpectedValue + * @throws \CoreWarning + * @throws \MySQLException + * @throws \OQLException + * @throws \SynchroExceptionNotStarted + */ +function Synchro($oReplica): SynchroLog { - switch($sOperation) - { + $oDataSource = MetaModel::GetObject('SynchroDataSource', $oReplica->Get('sync_source_id')); + + $oStatLog = new SynchroLog(); + $oStatLog->Set('sync_source_id', $oDataSource->GetKey()); + $oStatLog->Set('start_date', time()); + $oStatLog->Set('status', 'running'); + $oStatLog->AddTrace('Manual synchro'); + + // Get the list of SQL columns + $aAttCodesExpected = array(); + $aAttCodesToReconcile = array(); + $aAttCodesToUpdate = array(); + $sSelectAtt = 'SELECT SynchroAttribute WHERE sync_source_id = :source_id AND (update = 1 OR reconcile = 1)'; + $oSetAtt = new DBObjectSet(DBObjectSearch::FromOQL($sSelectAtt), array() /* order by*/, array('source_id' => $oDataSource->GetKey()) /* aArgs */); + while ($oSyncAtt = $oSetAtt->Fetch()) { + if ($oSyncAtt->Get('update')) { + $aAttCodesToUpdate[$oSyncAtt->Get('attcode')] = $oSyncAtt; + } + if ($oSyncAtt->Get('reconcile')) { + $aAttCodesToReconcile[$oSyncAtt->Get('attcode')] = $oSyncAtt; + } + $aAttCodesExpected[$oSyncAtt->Get('attcode')] = $oSyncAtt; + } + + // Get the list of attributes, determine reconciliation keys and update targets + // + if ($oDataSource->Get('reconciliation_policy') == 'use_attributes') { + $aReconciliationKeys = $aAttCodesToReconcile; + } elseif ($oDataSource->Get('reconciliation_policy') == 'use_primary_key') { + // Override the settings made at the attribute level ! + $aReconciliationKeys = array('primary_key' => null); + } + + if (count($aAttCodesToUpdate) == 0) { + $oStatLog->AddTrace('No attribute to update'); + throw new SynchroExceptionNotStarted('There is no attribute to update'); + } + if (count($aReconciliationKeys) == 0) { + $oStatLog->AddTrace('No attribute for reconciliation'); + throw new SynchroExceptionNotStarted('No attribute for reconciliation'); + } + + + $aAttributesToUpdate = array(); + foreach ($aAttCodesToUpdate as $sAttCode => $oSyncAtt) { + $oAttDef = MetaModel::GetAttributeDef($oDataSource->GetTargetClass(), $sAttCode); + if ($oAttDef->IsWritable()) { + $aAttributesToUpdate[$sAttCode] = $oSyncAtt; + } + } + // Create a change used for logging all the modifications/creations happening during the synchro + $oChange = MetaModel::NewObject('CMDBChange'); + $oChange->Set('date', time()); + $sUserString = CMDBChange::GetCurrentUserName(); + $oChange->Set('userinfo', $sUserString.' '.Dict::S('Core:SyncDataExchangeComment')); + $oChange->Set('origin', CMDBChangeOrigin::SYNCHRO_DATA_SOURCE); + $oChange->DBInsert(); + CMDBObject::SetCurrentChange($oChange); + + $oReplica->InitExtendedData($oDataSource); + + $oReplica->Synchro($oDataSource, $aReconciliationKeys, $aAttributesToUpdate, $oChange, $oStatLog); + $oReplica->DBUpdate(); + + return $oStatLog; +} + +try { + switch ($sOperation) { case 'details': - $iId = utils::ReadParam('id', null); - if ($iId == null) - { - throw new ApplicationException(Dict::Format('UI:Error:1ParametersMissing', 'id')); - } - $oReplica = MetaModel::GetObject('SynchroReplica', $iId); - $oReplica->DisplayDetails($oP); - break; - + $iId = utils::ReadParam('id', null); + if ($iId == null) { + throw new ApplicationException(Dict::Format('UI:Error:1ParametersMissing', 'id')); + } + $oReplica = MetaModel::GetObject('SynchroReplica', $iId); + $oReplica->DisplayDetails($oP); + break; + case 'oql': - $sOQL = utils::ReadParam('oql', null, false, 'raw_data'); - if ($sOQL == null) - { - throw new ApplicationException(Dict::Format('UI:Error:1ParametersMissing', 'oql')); - } - $oFilter = DBObjectSearch::FromOQL($sOQL); + $sOQL = utils::ReadParam('oql', null, false, 'raw_data'); + if ($sOQL == null) { + throw new ApplicationException(Dict::Format('UI:Error:1ParametersMissing', 'oql')); + } + $oFilter = DBObjectSearch::FromOQL($sOQL); $oBlock1 = new DisplayBlock($oFilter, 'search', false, array('menu' => false, 'table_id' => '1')); - $oBlock1->Display($oP, 0); - $oP->add('
'.MetaModel::GetClassIcon('SynchroReplica').Dict::S('Core:SynchroReplica:ListOfReplicas').'
'); - $iSourceId = utils::ReadParam('datasource', null); - if ($iSourceId != null) - { - $oSource = MetaModel::GetObject('SynchroDataSource', $iSourceId); - $oP->p(Dict::Format('Core:SynchroReplica:BackToDataSource', $oSource->GetHyperlink()).''); - } - $oBlock = new DisplayBlock($oFilter, 'list', false, array('menu'=>false)); - $oBlock->Display($oP, 1); - break; + $oBlock1->Display($oP, 0); + $oP->add(''.MetaModel::GetClassIcon('SynchroReplica').Dict::S('Core:SynchroReplica:ListOfReplicas').'
'); + $iSourceId = utils::ReadParam('datasource', null); + if ($iSourceId != null) { + $oSource = MetaModel::GetObject('SynchroDataSource', $iSourceId); + $oP->p(Dict::Format('Core:SynchroReplica:BackToDataSource', $oSource->GetHyperlink()).''); + } + $oBlock = new DisplayBlock($oFilter, 'list', false, array('menu' => false)); + $oBlock->Display($oP, 1); + break; case 'delete': case 'select_for_deletion': - // Redirect to the page that implements bulk delete - $sDelete = utils::GetAbsoluteUrlAppRoot().'pages/UI.php?'.$_SERVER['QUERY_STRING']; - header("Location: $sDelete"); - break; + // Redirect to the page that implements bulk delete + $sDelete = utils::GetAbsoluteUrlAppRoot().'pages/UI.php?'.$_SERVER['QUERY_STRING']; + header("Location: $sDelete"); + break; + + case 'unlinksynchro': + $iId = utils::ReadParam('id', null); + if ($iId == null) { + throw new ApplicationException(Dict::Format('UI:Error:1ParametersMissing', 'id')); + } + $oReplica = MetaModel::GetObject('SynchroReplica', $iId); + $oReplica->Set('dest_id', ''); + $oReplica->Set('status', 'new'); + $oReplica->DBWrite(); + + $oStatLog = Synchro($oReplica); + $oP->add(implode('