mirror of
https://github.com/Combodo/iTop.git
synced 2026-05-22 08:42:17 +02:00
Merge branch 'feature/uninstallation2' into feature/uninstallation
This commit is contained in:
@@ -21,16 +21,16 @@ use Combodo\iTop\DataFeatureRemoval\Service\DataFeatureRemoverExtensionService;
|
|||||||
use Combodo\iTop\Setup\FeatureRemoval\DryRemovalRuntimeEnvironment;
|
use Combodo\iTop\Setup\FeatureRemoval\DryRemovalRuntimeEnvironment;
|
||||||
use Combodo\iTop\Setup\FeatureRemoval\SetupAudit;
|
use Combodo\iTop\Setup\FeatureRemoval\SetupAudit;
|
||||||
use ContextTag;
|
use ContextTag;
|
||||||
|
use CoreException;
|
||||||
use Dict;
|
use Dict;
|
||||||
use Exception;
|
use Exception;
|
||||||
use IssueLog;
|
|
||||||
use MetaModel;
|
use MetaModel;
|
||||||
|
use MissingDependencyException;
|
||||||
use SetupUtils;
|
use SetupUtils;
|
||||||
use utils;
|
use utils;
|
||||||
|
|
||||||
class DataFeatureRemovalController extends Controller
|
class DataFeatureRemovalController extends Controller
|
||||||
{
|
{
|
||||||
private array $aRemovedExtensionsForCheck = [];
|
|
||||||
private ?array $aExtensionsToCheck = null;
|
private ?array $aExtensionsToCheck = null;
|
||||||
private bool $bForcedUninstallation = false;
|
private bool $bForcedUninstallation = false;
|
||||||
private array $aCountClassesToCleanup = [];
|
private array $aCountClassesToCleanup = [];
|
||||||
@@ -55,6 +55,7 @@ class DataFeatureRemovalController extends Controller
|
|||||||
$aParams['sSetupUrl'] = utils::GetAbsoluteUrlAppRoot().'setup';
|
$aParams['sSetupUrl'] = utils::GetAbsoluteUrlAppRoot().'setup';
|
||||||
$aParams['iCount'] = $this->iCount;
|
$aParams['iCount'] = $this->iCount;
|
||||||
|
|
||||||
|
Session::Set('bForceCompilation', true);
|
||||||
$this->AddLinkedStylesheet(utils::GetAbsoluteUrlModulesRoot().DataFeatureRemovalHelper::MODULE_NAME.'/assets/css/DataFeatureRemoval.css');
|
$this->AddLinkedStylesheet(utils::GetAbsoluteUrlModulesRoot().DataFeatureRemovalHelper::MODULE_NAME.'/assets/css/DataFeatureRemoval.css');
|
||||||
$this->AddLinkedScript(utils::GetAbsoluteUrlModulesRoot().DataFeatureRemovalHelper::MODULE_NAME.'/assets/js/DataFeatureRemoval.js');
|
$this->AddLinkedScript(utils::GetAbsoluteUrlModulesRoot().DataFeatureRemovalHelper::MODULE_NAME.'/assets/js/DataFeatureRemoval.js');
|
||||||
$this->DisplayPage($aParams);
|
$this->DisplayPage($aParams);
|
||||||
@@ -77,33 +78,6 @@ class DataFeatureRemovalController extends Controller
|
|||||||
$this->aAnalysisDataTable = $this->GetTableData('Analysis', $aColumns, $aData);
|
$this->aAnalysisDataTable = $this->GetTableData('Analysis', $aColumns, $aData);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function OperationAnalyze(): void
|
|
||||||
{
|
|
||||||
$iCount = $this->ReadExtensionsDiff();
|
|
||||||
|
|
||||||
$this->m_sOperation = 'Main';
|
|
||||||
try {
|
|
||||||
if ($iCount > 0) {
|
|
||||||
$this->Analyze();
|
|
||||||
}
|
|
||||||
$this->OperationMain();
|
|
||||||
} catch (Exception $e) {
|
|
||||||
IssueLog::Error(__METHOD__, null, ['stack' => $e->getTraceAsString(), 'exception' => $e->getMessage()]);
|
|
||||||
$this->OperationMain($e->getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private function Analyze(): void
|
|
||||||
{
|
|
||||||
//TODO : Run data audit with added extension too, not just removed ones
|
|
||||||
$this->Compile($this->aExtensionsToCheck['to_be_removed']);
|
|
||||||
$sSourceEnv = MetaModel::GetEnvironment();
|
|
||||||
$oSetupAudit = new SetupAudit($sSourceEnv);
|
|
||||||
$aGetRemovedClasses = $oSetupAudit->RunDataAudit();
|
|
||||||
IssueLog::Debug(__METHOD__, null, ['aGetRemovedClasses' => $aGetRemovedClasses]);
|
|
||||||
$this->aCountClassesToCleanup = $aGetRemovedClasses;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function OperationAnalysisResult(): void
|
public function OperationAnalysisResult(): void
|
||||||
{
|
{
|
||||||
$aParams = [];
|
$aParams = [];
|
||||||
@@ -133,26 +107,48 @@ class DataFeatureRemovalController extends Controller
|
|||||||
$aParams['aHiddenInputs'] = $aHiddenInputs;
|
$aParams['aHiddenInputs'] = $aHiddenInputs;
|
||||||
|
|
||||||
$aAddedExtensions = json_decode($aHiddenInputs['added_extensions'], true);
|
$aAddedExtensions = json_decode($aHiddenInputs['added_extensions'], true);
|
||||||
|
|
||||||
$aRemovedExtensions = json_decode($aHiddenInputs['removed_extensions'], true);
|
$aRemovedExtensions = json_decode($aHiddenInputs['removed_extensions'], true);
|
||||||
|
if (count($aRemovedExtensions) == 0) {
|
||||||
|
$this->ReadExtensionsDiff();
|
||||||
|
$aAddedExtensions = $this->aExtensionsToCheck['to_be_installed'];
|
||||||
|
$aHiddenInputs['added_extensions'] = utils::HtmlEntities(json_encode($aAddedExtensions));
|
||||||
|
$aRemovedExtensions = $this->aExtensionsToCheck['to_be_removed'];
|
||||||
|
$aHiddenInputs['removed_extensions'] = utils::HtmlEntities(json_encode($aRemovedExtensions));
|
||||||
|
}
|
||||||
|
|
||||||
|
$aRemoveExtensionCodes = array_keys($aRemovedExtensions);
|
||||||
|
|
||||||
$aParams['aAddedExtensions'] = $aAddedExtensions;
|
$aParams['aAddedExtensions'] = $aAddedExtensions;
|
||||||
$aParams['aRemovedExtensions'] = $aRemovedExtensions;
|
$aParams['aRemovedExtensions'] = $aRemovedExtensions;
|
||||||
|
|
||||||
IssueLog::Debug(__METHOD__.' Extensions given in parameter', null, [
|
DataFeatureRemovalLog::Debug(__METHOD__.' Extensions given in parameter', null, [
|
||||||
'added_extensions' => $aAddedExtensions,
|
'added_extensions' => $aAddedExtensions,
|
||||||
'removed_extensions' => $aRemovedExtensions]);
|
'removed_extensions' => $aRemovedExtensions]);
|
||||||
|
|
||||||
$this->Compile(array_keys($aRemovedExtensions), false);
|
$aParams['sTransactionId'] = utils::GetNewTransactionId();
|
||||||
|
$aParams['iColumnCount'] = $this->iColumnCount;
|
||||||
|
$aParams['aAvailableExtensions'] = $this->SplitArrayIntoColumns($this->GetExtensionsDiff($aAddedExtensions, $aRemovedExtensions), $this->iColumnCount);
|
||||||
|
|
||||||
|
$bForceCompilation = Session::Get('bForceCompilation', false);
|
||||||
|
try {
|
||||||
|
$this->Compile($aRemoveExtensionCodes, $bForceCompilation);
|
||||||
|
} catch (CoreException $e) {
|
||||||
|
$aParams['DataFeatureRemovalErrorMessage'] = $e->getHtmlDesc();
|
||||||
|
$this->DisplayPage($aParams, 'AnalysisResult');
|
||||||
|
return;
|
||||||
|
} catch (Exception $e) {
|
||||||
|
$aParams['DataFeatureRemovalErrorMessage'] = $e->getMessage();
|
||||||
|
$this->DisplayPage($aParams, 'AnalysisResult');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
$sSourceEnv = MetaModel::GetEnvironment();
|
$sSourceEnv = MetaModel::GetEnvironment();
|
||||||
$oSetupAudit = new SetupAudit($sSourceEnv);
|
$oSetupAudit = new SetupAudit($sSourceEnv);
|
||||||
$aGetRemovedClasses = array_keys($oSetupAudit->RunDataAudit());
|
$aGetRemovedClasses = array_keys($oSetupAudit->RunDataAudit());
|
||||||
IssueLog::Debug(__METHOD__, null, ['aGetRemovedClasses' => $aGetRemovedClasses]);
|
DataFeatureRemovalLog::Debug(__METHOD__, null, ['aGetRemovedClasses' => $aGetRemovedClasses]);
|
||||||
|
|
||||||
$aParams['sTransactionId'] = utils::GetNewTransactionId();
|
|
||||||
$aParams['aClasses'] = $aGetRemovedClasses;
|
$aParams['aClasses'] = $aGetRemovedClasses;
|
||||||
$aParams['iColumnCount'] = $this->iColumnCount;
|
|
||||||
$aParams['aAvailableExtensions'] = $this->SplitArrayIntoColumns($this->GetExtensionsDiff($aAddedExtensions, $aRemovedExtensions), $this->iColumnCount);
|
|
||||||
|
|
||||||
new ContextTag(ContextTag::TAG_SETUP);
|
new ContextTag(ContextTag::TAG_SETUP);
|
||||||
$aParams['sLaunchSetupUrl'] = utils::GetAbsoluteUrlAppRoot().'setup/wizard.php';
|
$aParams['sLaunchSetupUrl'] = utils::GetAbsoluteUrlAppRoot().'setup/wizard.php';
|
||||||
@@ -170,6 +166,13 @@ class DataFeatureRemovalController extends Controller
|
|||||||
$this->DisplayPage($aParams, 'AnalysisResult');
|
$this->DisplayPage($aParams, 'AnalysisResult');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array $aRemovedExtensions
|
||||||
|
* @param bool $bForceCompilation
|
||||||
|
* @return void
|
||||||
|
* @throws \ConfigException
|
||||||
|
* @throws \CoreException
|
||||||
|
*/
|
||||||
private function Compile(array $aRemovedExtensions, bool $bForceCompilation = true): void
|
private function Compile(array $aRemovedExtensions, bool $bForceCompilation = true): void
|
||||||
{
|
{
|
||||||
$sSourceEnv = MetaModel::GetEnvironment();
|
$sSourceEnv = MetaModel::GetEnvironment();
|
||||||
@@ -346,7 +349,7 @@ class DataFeatureRemovalController extends Controller
|
|||||||
}
|
}
|
||||||
|
|
||||||
$sTransactionId = utils::ReadPostedParam('transaction_id', null, utils::ENUM_SANITIZATION_FILTER_TRANSACTION_ID);
|
$sTransactionId = utils::ReadPostedParam('transaction_id', null, utils::ENUM_SANITIZATION_FILTER_TRANSACTION_ID);
|
||||||
IssueLog::Debug(__FUNCTION__.": Transaction [$sTransactionId]");
|
DataFeatureRemovalLog::Debug(__FUNCTION__.": Transaction [$sTransactionId]");
|
||||||
if (empty($sTransactionId) || !utils::IsTransactionValid($sTransactionId, false)) {
|
if (empty($sTransactionId) || !utils::IsTransactionValid($sTransactionId, false)) {
|
||||||
throw new DataFeatureRemovalException(Dict::S("iTopUpdate:Error:InvalidToken"));
|
throw new DataFeatureRemovalException(Dict::S("iTopUpdate:Error:InvalidToken"));
|
||||||
}
|
}
|
||||||
@@ -375,13 +378,13 @@ class DataFeatureRemovalController extends Controller
|
|||||||
|
|
||||||
if ($aExtensionData['installed'] && $aSelectedExtensionsFromUI[$sCode] !== 'on') {
|
if ($aExtensionData['installed'] && $aSelectedExtensionsFromUI[$sCode] !== 'on') {
|
||||||
$aExtensionData['extra_flags']['selected'] = false;
|
$aExtensionData['extra_flags']['selected'] = false;
|
||||||
$this->aExtensionsToCheck['to_be_removed'][] = $sCode;
|
$this->aExtensionsToCheck['to_be_removed'][$sCode] = $sCode;
|
||||||
if (!$aExtensionData['extra_flags']['uninstallable'] || $aExtensionData['extra_flags']['remote']) {
|
if (!$aExtensionData['extra_flags']['uninstallable'] || $aExtensionData['extra_flags']['remote']) {
|
||||||
$this->bForcedUninstallation = true;
|
$this->bForcedUninstallation = true;
|
||||||
}
|
}
|
||||||
} elseif (!$aExtensionData['installed'] && $aSelectedExtensionsFromUI[$sCode] === 'on') {
|
} elseif (!$aExtensionData['installed'] && $aSelectedExtensionsFromUI[$sCode] === 'on') {
|
||||||
$aExtensionData['extra_flags']['selected'] = true;
|
$aExtensionData['extra_flags']['selected'] = true;
|
||||||
$this->aExtensionsToCheck['to_be_installed'][] = $sCode;
|
$this->aExtensionsToCheck['to_be_installed'][$sCode] = $sCode;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return count($this->aExtensionsToCheck['to_be_installed']) + count($this->aExtensionsToCheck['to_be_removed']);
|
return count($this->aExtensionsToCheck['to_be_installed']) + count($this->aExtensionsToCheck['to_be_removed']);
|
||||||
|
|||||||
@@ -4,64 +4,87 @@
|
|||||||
|
|
||||||
|
|
||||||
{% UIPanel ForInformation { sTitle:'DataFeatureRemoval:Analysis:Title'|dict_s} %}
|
{% UIPanel ForInformation { sTitle:'DataFeatureRemoval:Analysis:Title'|dict_s} %}
|
||||||
{% UIPanel Neutral { sTitle:'DataFeatureRemoval:Features:Title'|dict_s, sSubTitle: '' } %}
|
{% if null != DataFeatureRemovalErrorMessage %}
|
||||||
{% UIMultiColumn Standard {} %}
|
<div id="feature_removal_error_msg_div" style="display:block">
|
||||||
{% for iColumnIndex in 0..iColumnCount-1 %}
|
{% UIAlert ForFailure { sTitle:'DataFeatureRemoval:Failure:Title'|dict_s, sId: 'feature_removal_error_msg', sContent:DataFeatureRemovalErrorMessage } %}
|
||||||
{% UIColumn Standard {} %}
|
{% EndUIAlert %}
|
||||||
{% for aExtension in aAvailableExtensions[iColumnIndex] %}
|
</div>
|
||||||
{% if aExtension['installed'] %}
|
|
||||||
{% UIExtensionDetails Installed { sCode : aExtension['code'], sLabel : aExtension['label'], sDescription : aExtension['description'], aMetaData : [aExtension['version'], aExtension['source']], aExtraFlags : aExtension['extra_flags']} %}{% EndUIExtensionDetails %}
|
|
||||||
{% else %}
|
|
||||||
{% UIExtensionDetails NotInstalled { sCode : aExtension['code'], sLabel : aExtension['label'], sDescription : aExtension['description'], aMetaData : [aExtension['version'], aExtension['source']], aExtraFlags : aExtension['extra_flags']} %}{% EndUIExtensionDetails %}
|
|
||||||
{% endif %}
|
|
||||||
{% endfor %}
|
|
||||||
{% EndUIColumn %}
|
|
||||||
{% endfor %}
|
|
||||||
{% EndUIMultiColumn %}
|
|
||||||
{% EndUIPanel %}
|
|
||||||
|
|
||||||
{% if bDeletionNeeded %}
|
{% UIPanel Neutral { sTitle:'DataFeatureRemoval:Features:Title'|dict_s, sSubTitle: '' } %}
|
||||||
{% UIFieldSet Standard {sLegend:'DataFeatureRemoval:DeletionPlan:Title'|dict_s} %}
|
{% UIMultiColumn Standard {} %}
|
||||||
{% UIDataTable ForForm { sRef:'aDeletionPlanSummary', aColumns:aDeletionPlanSummary.Columns, aData:aDeletionPlanSummary.Data} %}{% EndUIDataTable %}
|
{% for iColumnIndex in 0..iColumnCount-1 %}
|
||||||
{% EndUIFieldSet %}
|
{% UIColumn Standard {} %}
|
||||||
{% if bDeletionPossible %}
|
{% for aExtension in aAvailableExtensions[iColumnIndex] %}
|
||||||
{% UIForm Standard {} %}
|
{% if aExtension['installed'] %}
|
||||||
{% UIInput ForHidden { sName:'transaction_id', sValue:sTransactionId} %}
|
{% UIExtensionDetails Installed { sCode : aExtension['code'], sLabel : aExtension['label'], sDescription : aExtension['description'], aMetaData : [aExtension['version'], aExtension['source']], aExtraFlags : aExtension['extra_flags']} %}{% EndUIExtensionDetails %}
|
||||||
{% UIInput ForHidden { sName:'operation', sValue:'DoDeletion'} %}
|
{% else %}
|
||||||
{% for sKey, sClass in aClasses %}
|
{% UIExtensionDetails NotInstalled { sCode : aExtension['code'], sLabel : aExtension['label'], sDescription : aExtension['description'], aMetaData : [aExtension['version'], aExtension['source']], aExtraFlags : aExtension['extra_flags']} %}{% EndUIExtensionDetails %}
|
||||||
{% UIInput ForHidden { sName:"classes[" ~ sKey ~ "]", sValue:sClass } %}
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
{% EndUIColumn %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{% for sCode, sLabel in aAddedExtensions %}
|
{% EndUIMultiColumn %}
|
||||||
{% UIInput ForHidden { sName:"aAddedExtensions[" ~ sCode ~ "]", sValue:sLabel } %}
|
{% EndUIPanel %}
|
||||||
{% endfor %}
|
|
||||||
{% for sCode, sLabel in aRemovedExtensions %}
|
|
||||||
{% UIInput ForHidden { sName:"aRemovedExtensions[" ~ sCode ~ "]", sValue:sLabel } %}
|
|
||||||
{% endfor %}
|
|
||||||
{% for sInputName, sValue in aHiddenInputs %}
|
|
||||||
{% UIInput ForHidden { sName:sInputName, sValue:sValue } %}
|
|
||||||
{% endfor %}
|
|
||||||
{% UIToolbar ForButton {} %}
|
|
||||||
{% UIButton ForPrimaryAction {sLabel:'UI:Button:DoDeletion'|dict_s, sName:'btn_deletion', sId:'btn_deletion', bIsSubmit:true} %}
|
|
||||||
{% EndUIToolbar %}
|
|
||||||
{% EndUIForm %}
|
|
||||||
{% else %}
|
|
||||||
{% UIAlert ForFailure { sContent: 'DataFeatureRemoval:DeletionPlan:Error:Issues'|dict_s } %}{% EndUIAlert %}
|
|
||||||
{% endif %}
|
|
||||||
{% else %}
|
{% else %}
|
||||||
{% UIAlert ForSuccess { sTitle:'DataFeatureRemoval:CleanupComplete:Title'|dict_s, sContent:'DataFeatureRemoval:CompilComplete'|dict_s, sId:value } %}{% EndUIAlert %}
|
{% UIPanel Neutral { sTitle:'DataFeatureRemoval:Features:Title'|dict_s, sSubTitle: '' } %}
|
||||||
|
{% UIMultiColumn Standard {} %}
|
||||||
|
{% for iColumnIndex in 0..iColumnCount-1 %}
|
||||||
|
{% UIColumn Standard {} %}
|
||||||
|
{% for aExtension in aAvailableExtensions[iColumnIndex] %}
|
||||||
|
{% if aExtension['installed'] %}
|
||||||
|
{% UIExtensionDetails Installed { sCode : aExtension['code'], sLabel : aExtension['label'], sDescription : aExtension['description'], aMetaData : [aExtension['version'], aExtension['source']], aExtraFlags : aExtension['extra_flags']} %}{% EndUIExtensionDetails %}
|
||||||
|
{% else %}
|
||||||
|
{% UIExtensionDetails NotInstalled { sCode : aExtension['code'], sLabel : aExtension['label'], sDescription : aExtension['description'], aMetaData : [aExtension['version'], aExtension['source']], aExtraFlags : aExtension['extra_flags']} %}{% EndUIExtensionDetails %}
|
||||||
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
{% EndUIColumn %}
|
||||||
|
{% endfor %}
|
||||||
|
{% EndUIMultiColumn %}
|
||||||
|
{% EndUIPanel %}
|
||||||
|
|
||||||
{% UIForm Standard {'sId':'launch-setup-form', Action:sLaunchSetupUrl, 'EncType': 'application/x-www-form-urlencoded'} %}
|
{% if bDeletionNeeded %}
|
||||||
{% for sKey, sValue in aSetupParams %}
|
{% UIFieldSet Standard {sLegend:'DataFeatureRemoval:DeletionPlan:Title'|dict_s} %}
|
||||||
{% UIInput ForHidden { sName:sKey, sValue:sValue } %}
|
{% UIDataTable ForForm { sRef:'aDeletionPlanSummary', aColumns:aDeletionPlanSummary.Columns, aData:aDeletionPlanSummary.Data} %}{% EndUIDataTable %}
|
||||||
{% endfor %}
|
{% EndUIFieldSet %}
|
||||||
{% UIButton ForPrimaryAction {sLabel:'UI:Button:Setup'|dict_s, sName:'btn_setup', sId:'btn_setup', bIsSubmit:true} %}
|
{% if bDeletionPossible %}
|
||||||
{% EndUIForm %}
|
{% UIForm Standard {} %}
|
||||||
{% endif %}
|
{% UIInput ForHidden { sName:'transaction_id', sValue:sTransactionId} %}
|
||||||
|
{% UIInput ForHidden { sName:'operation', sValue:'DoDeletion'} %}
|
||||||
|
{% for sKey, sClass in aClasses %}
|
||||||
|
{% UIInput ForHidden { sName:"classes[" ~ sKey ~ "]", sValue:sClass } %}
|
||||||
|
{% endfor %}
|
||||||
|
{% for sCode, sLabel in aAddedExtensions %}
|
||||||
|
{% UIInput ForHidden { sName:"aAddedExtensions[" ~ sCode ~ "]", sValue:sLabel } %}
|
||||||
|
{% endfor %}
|
||||||
|
{% for sCode, sLabel in aRemovedExtensions %}
|
||||||
|
{% UIInput ForHidden { sName:"aRemovedExtensions[" ~ sCode ~ "]", sValue:sLabel } %}
|
||||||
|
{% endfor %}
|
||||||
|
{% for sInputName, sValue in aHiddenInputs %}
|
||||||
|
{% UIInput ForHidden { sName:sInputName, sValue:sValue } %}
|
||||||
|
{% endfor %}
|
||||||
|
{% UIToolbar ForButton {} %}
|
||||||
|
{% UIButton ForPrimaryAction {sLabel:'UI:Button:DoDeletion'|dict_s, sName:'btn_deletion', sId:'btn_deletion', bIsSubmit:true} %}
|
||||||
|
{% EndUIToolbar %}
|
||||||
|
{% EndUIForm %}
|
||||||
|
{% else %}
|
||||||
|
{% UIAlert ForFailure { sContent: 'DataFeatureRemoval:DeletionPlan:Error:Issues'|dict_s } %}{% EndUIAlert %}
|
||||||
|
{% endif %}
|
||||||
|
{% else %}
|
||||||
|
{% UIAlert ForSuccess { sTitle:'DataFeatureRemoval:CleanupComplete:Title'|dict_s, sContent:'DataFeatureRemoval:CompilComplete'|dict_s, sId:value } %}{% EndUIAlert %}
|
||||||
|
|
||||||
{% if bHasDeletionExecution %}
|
{% UIForm Standard {'sId':'launch-setup-form', Action:sLaunchSetupUrl, 'EncType': 'application/x-www-form-urlencoded'} %}
|
||||||
{% UIFieldSet Standard {sLegend:'DataFeatureRemoval:Execution:Title'|dict_s} %}
|
{% for sKey, sValue in aSetupParams %}
|
||||||
{% UIDataTable ForForm { sRef:'aDeletionExecutionSummary', aColumns:aDeletionExecutionSummary.Columns, aData:aDeletionExecutionSummary.Data} %}{% EndUIDataTable %}
|
{% UIInput ForHidden { sName:sKey, sValue:sValue } %}
|
||||||
{% EndUIFieldSet %}
|
{% endfor %}
|
||||||
|
{% UIButton ForPrimaryAction {sLabel:'UI:Button:Setup'|dict_s, sName:'btn_setup', sId:'btn_setup', bIsSubmit:true} %}
|
||||||
|
{% EndUIForm %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if bHasDeletionExecution %}
|
||||||
|
{% UIFieldSet Standard {sLegend:'DataFeatureRemoval:Execution:Title'|dict_s} %}
|
||||||
|
{% UIDataTable ForForm { sRef:'aDeletionExecutionSummary', aColumns:aDeletionExecutionSummary.Columns, aData:aDeletionExecutionSummary.Data} %}{% EndUIDataTable %}
|
||||||
|
{% EndUIFieldSet %}
|
||||||
|
{% endif %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% UIForm Standard {} %}
|
{% UIForm Standard {} %}
|
||||||
|
|||||||
@@ -1,29 +0,0 @@
|
|||||||
{# @copyright Copyright (C) 2010-2026 Combodo SARL #}
|
|
||||||
{# @license http://opensource.org/licenses/AGPL-3.0 #}
|
|
||||||
|
|
||||||
{% UIPanel ForInformation { sTitle:'DataFeatureRemoval:DeletionPlan:Title'|dict_s, sSubTitle: 'DataFeatureRemoval:DeletionPlan:SubTitle'|dict_format(iQueryCount) } %}
|
|
||||||
{% UIDataTable ForForm { sRef:'aDeletionPlanSummary', aColumns:aDeletionPlanSummary.Columns, aData:aDeletionPlanSummary.Data} %}{% EndUIDataTable %}
|
|
||||||
{% EndUIPanel %}
|
|
||||||
|
|
||||||
{% if bDeletionPossible %}
|
|
||||||
{% UIForm Standard {} %}
|
|
||||||
{% UIInput ForHidden { sName:'transaction_id', sValue:sTransactionId} %}
|
|
||||||
{% UIInput ForHidden { sName:'operation', sValue:'DoDeletion'} %}
|
|
||||||
{% for sKey, sClass in aClasses %}
|
|
||||||
{% UIInput ForHidden { sName:"classes[" ~ sKey ~ "]", sValue:sClass } %}
|
|
||||||
{% endfor %}
|
|
||||||
{% UIToolbar ForButton {} %}
|
|
||||||
{% UIButton ForPrimaryAction {sLabel:'UI:Button:DoDeletion'|dict_s, sName:'btn_deletion', sId:'btn_deletion', bIsSubmit:true} %}
|
|
||||||
{% EndUIToolbar %}
|
|
||||||
{% EndUIForm %}
|
|
||||||
{% else %}
|
|
||||||
{{ 'DataFeatureRemoval:DeletionPlan:Error:Issues'|dict_s }}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% UIForm Standard {} %}
|
|
||||||
{% UIInput ForHidden { sName:'transaction_id', sValue:sTransactionId} %}
|
|
||||||
{% UIInput ForHidden { sName:'operation', sValue:'Main'} %}
|
|
||||||
{% UIToolbar ForButton {} %}
|
|
||||||
{% UIButton ForPrimaryAction {sLabel:'UI:Button:BackToMain'|dict_s, sName:'btn_back', sId:'btn_back', bIsSubmit:true} %}
|
|
||||||
{% EndUIToolbar %}
|
|
||||||
{% EndUIForm %}
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
{# @copyright Copyright (C) 2010-2024 Combodo SAS #}
|
|
||||||
{# @license http://opensource.org/licenses/AGPL-3.0 #}
|
|
||||||
|
|
||||||
{% if bHasData %}
|
|
||||||
{% UIPanel Neutral { sTitle:'DataFeatureRemoval:Analysis:Title'|dict_s, sSubTitle: 'DataFeatureRemoval:Analysis:SubTitle'|dict_format(iCount) } %}
|
|
||||||
{% UIDataTable ForForm { sRef:'aAnalysisDataTable', aColumns:aAnalysisDataTable.Columns, aData:aAnalysisDataTable.Data} %}{% EndUIDataTable %}
|
|
||||||
{% EndUIPanel %}
|
|
||||||
|
|
||||||
{% UIForm Standard {} %}
|
|
||||||
{% UIInput ForHidden { sName:'transaction_id', sValue:sTransactionId} %}
|
|
||||||
{% UIInput ForHidden { sName:'operation', sValue:'DeletionPlan'} %}
|
|
||||||
{% for sKey, sClass in aClasses %}
|
|
||||||
{% UIInput ForHidden { sName:"classes[" ~ sKey ~ "]", sValue:sClass } %}
|
|
||||||
{% endfor %}
|
|
||||||
{% UIToolbar ForButton {} %}
|
|
||||||
{% UIButton ForPrimaryAction {sLabel:'UI:Button:PlanDeletion'|dict_s, sName:'btn_plandeletion', sId:'btn_plandeletion', bIsSubmit:true} %}
|
|
||||||
{% EndUIToolbar %}
|
|
||||||
{% EndUIForm %}
|
|
||||||
{% endif %}
|
|
||||||
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
|
|
||||||
{% UIForm Standard {} %}
|
{% UIForm Standard {} %}
|
||||||
{% UIInput ForHidden {sName:'operation', sValue:'Analyze'} %}
|
{% UIInput ForHidden {sName:'operation', sValue:'AnalysisResult'} %}
|
||||||
{% UIInput ForHidden {sName:'transaction_id', sValue:sTransactionId} %}
|
{% UIInput ForHidden {sName:'transaction_id', sValue:sTransactionId} %}
|
||||||
|
|
||||||
{% UIPanel Neutral { sTitle:'DataFeatureRemoval:Features:Title'|dict_s, sSubTitle: '' } %}
|
{% UIPanel Neutral { sTitle:'DataFeatureRemoval:Features:Title'|dict_s, sSubTitle: '' } %}
|
||||||
|
|||||||
@@ -15,21 +15,5 @@
|
|||||||
{{ 'DataFeatureRemoval:Helper:Desc2'|dict_s }}
|
{{ 'DataFeatureRemoval:Helper:Desc2'|dict_s }}
|
||||||
{% EndUIAlert %}
|
{% EndUIAlert %}
|
||||||
|
|
||||||
{% if null != DataFeatureRemovalErrorMessage %}
|
|
||||||
<div id="feature_removal_error_msg_div" style="display:block">
|
|
||||||
{% UIAlert ForFailure { sTitle:'DataFeatureRemoval:Failure:Title'|dict_s, sId: 'feature_removal_error_msg', sContent:DataFeatureRemovalErrorMessage } %}
|
|
||||||
{% EndUIAlert %}
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% include 'Features.html.twig' %}
|
{% include 'Features.html.twig' %}
|
||||||
{% include 'ExtensionRemovalData.html.twig' %}
|
|
||||||
|
|
||||||
{% if not bHasData %}
|
|
||||||
{% UIToolbar ForButton {} %}
|
|
||||||
<a href="{{ sSetupUrl }}">
|
|
||||||
{% UIButton ForPrimaryAction {sLabel:'UI:Button:Setup'|dict_s, sName:'btn_setup', sId:'btn_setup', bIsSubmit:false} %}
|
|
||||||
</a>
|
|
||||||
{% EndUIToolbar %}
|
|
||||||
{% endif %}
|
|
||||||
{% EndUIPanel %}
|
{% EndUIPanel %}
|
||||||
|
|||||||
@@ -477,7 +477,7 @@ class RunTimeEnvironment
|
|||||||
}
|
}
|
||||||
|
|
||||||
$aModulesToLoad = $this->GetModulesToLoad($this->sFinalEnv, $aDirsToCompile);
|
$aModulesToLoad = $this->GetModulesToLoad($this->sFinalEnv, $aDirsToCompile);
|
||||||
$aAvailableModules = $this->AnalyzeInstallation($oSourceConfig, $aDirsToCompile, false, $aModulesToLoad);
|
$aAvailableModules = $this->AnalyzeInstallation($oSourceConfig, $aDirsToCompile, true, $aModulesToLoad);
|
||||||
|
|
||||||
// Do load the required modules
|
// Do load the required modules
|
||||||
//
|
//
|
||||||
|
|||||||
@@ -175,11 +175,12 @@ class WizStepModulesChoice extends AbstractWizStepInstall
|
|||||||
["class" => "WizStepUpgradeMiscParams","state" => ""],
|
["class" => "WizStepUpgradeMiscParams","state" => ""],
|
||||||
];
|
];
|
||||||
$i = 0;
|
$i = 0;
|
||||||
|
$this->aSteps = null;
|
||||||
while (null != $this->GetStepInfo($i)) {
|
while (null != $this->GetStepInfo($i)) {
|
||||||
|
$this->aSteps = null;
|
||||||
$aSteps [] = ["class" => "WizStepModulesChoice","state" => "$i"];
|
$aSteps [] = ["class" => "WizStepModulesChoice","state" => "$i"];
|
||||||
$i++;
|
$i++;
|
||||||
}
|
}
|
||||||
$aSteps [] = ["class" => "WizStepModulesChoice","state" => "$i"];
|
|
||||||
|
|
||||||
return $aSteps;
|
return $aSteps;
|
||||||
}
|
}
|
||||||
@@ -709,7 +710,6 @@ EOF
|
|||||||
// Found an "installation.xml" file, let's use this definition for the wizard
|
// Found an "installation.xml" file, let's use this definition for the wizard
|
||||||
$aParams = new XMLParameters($this->GetSourceFilePath());
|
$aParams = new XMLParameters($this->GetSourceFilePath());
|
||||||
$this->aSteps = $aParams->Get('steps', []);
|
$this->aSteps = $aParams->Get('steps', []);
|
||||||
|
|
||||||
if ($index + 1 >= count($this->aSteps)) {
|
if ($index + 1 >= count($this->aSteps)) {
|
||||||
//make sure we also cache next step as well
|
//make sure we also cache next step as well
|
||||||
$aOptions = $this->oExtensionsMap->GetAllExtensionsOptionInfo($bRemoteExtensionsShouldBeMandatory);
|
$aOptions = $this->oExtensionsMap->GetAllExtensionsOptionInfo($bRemoteExtensionsShouldBeMandatory);
|
||||||
|
|||||||
@@ -15,78 +15,42 @@ $aParams = [
|
|||||||
new ContextTag(ContextTag::TAG_SETUP);
|
new ContextTag(ContextTag::TAG_SETUP);
|
||||||
$sToken = SetupUtils::CreateSetupToken();
|
$sToken = SetupUtils::CreateSetupToken();
|
||||||
|
|
||||||
$aSelectedModules = [
|
function GetLastestInstallFile(): ?string
|
||||||
'authent-cas',
|
{
|
||||||
'authent-external',
|
$aFiles = glob(APPROOT.'/log/install-*.xml');
|
||||||
'authent-ldap',
|
rsort($aFiles);
|
||||||
'authent-local',
|
$iLatestCtime = 0;
|
||||||
'combodo-backoffice-darkmoon-theme',
|
$sLastFilePath = null;
|
||||||
'combodo-backoffice-fullmoon-high-contrast-theme',
|
foreach ($aFiles as $sFilePath) {
|
||||||
'combodo-backoffice-fullmoon-protanopia-deuteranopia-theme',
|
if (is_file($sFilePath)) {
|
||||||
'combodo-backoffice-fullmoon-tritanopia-theme',
|
$iCurrentCtime = filemtime($sFilePath);
|
||||||
'combodo-data-feature-removal',
|
if ($iCurrentCtime > $iLatestCtime) {
|
||||||
'itop-backup',
|
$iLatestCtime = $iCurrentCtime;
|
||||||
'itop-config',
|
$sLastFilePath = $sFilePath;
|
||||||
'itop-files-information',
|
}
|
||||||
'itop-portal-base',
|
}
|
||||||
'itop-profiles-itil',
|
}
|
||||||
'itop-sla-computation',
|
|
||||||
'itop-structure',
|
|
||||||
'itop-welcome-itil',
|
|
||||||
'itop-config-mgmt',
|
|
||||||
'itop-attachments',
|
|
||||||
'itop-tickets',
|
|
||||||
'combodo-db-tools',
|
|
||||||
'itop-core-update',
|
|
||||||
'itop-hub-connector',
|
|
||||||
'itop-oauth-client',
|
|
||||||
'itop-themes-compat',
|
|
||||||
'combodo-my-account',
|
|
||||||
'combodo-my-account-user-info',
|
|
||||||
'combodo-oauth2-client',
|
|
||||||
'itop-attribute-class-set',
|
|
||||||
'itop-attribute-encrypted-password',
|
|
||||||
'itop-ui-copypaste',
|
|
||||||
'itop-datacenter-mgmt',
|
|
||||||
'itop-endusers-devices',
|
|
||||||
'itop-storage-mgmt',
|
|
||||||
'itop-virtualization-mgmt',
|
|
||||||
'itop-bridge-cmdb-ticket',
|
|
||||||
'itop-bridge-virtualization-storage',
|
|
||||||
'itop-service-mgmt',
|
|
||||||
'itop-bridge-cmdb-services',
|
|
||||||
'itop-bridge-datacenter-mgmt-services',
|
|
||||||
'itop-bridge-endusers-devices-services',
|
|
||||||
'itop-bridge-storage-mgmt-services',
|
|
||||||
'itop-bridge-virtualization-mgmt-services',
|
|
||||||
'itop-request-mgmt',
|
|
||||||
'itop-portal',
|
|
||||||
'itop-change-mgmt',
|
|
||||||
'itop-faq-light',
|
|
||||||
'itop-knownerror-mgmt',
|
|
||||||
'itop-problem-mgmt',
|
|
||||||
'itop-system-information',
|
|
||||||
'itop-log-mgmt',
|
|
||||||
];
|
|
||||||
|
|
||||||
$aSelectedExtensions = [
|
return $sLastFilePath;
|
||||||
'itop-config-mgmt-core',
|
}
|
||||||
'itop-config-mgmt-datacenter',
|
|
||||||
'itop-config-mgmt-end-user',
|
|
||||||
'itop-config-mgmt-storage',
|
|
||||||
'itop-config-mgmt-virtualization',
|
|
||||||
'itop-service-mgmt-enterprise',
|
|
||||||
'itop-ticket-mgmt-simple-ticket',
|
|
||||||
'itop-ticket-mgmt-simple-ticket-enhanced-portal',
|
|
||||||
'itop-change-mgmt-simple',
|
|
||||||
'itop-kown-error-mgmt',
|
|
||||||
'itop-problem-mgmt',
|
|
||||||
'itop-system-information',
|
|
||||||
'itop-log-mgmt',
|
|
||||||
];
|
|
||||||
|
|
||||||
$aRemovedExtensions = ['itop-container-mgmt' => 'Containerization'];
|
$aRemovedExtensions = ['itop-container-mgmt' => 'Containerization'];
|
||||||
|
|
||||||
|
$sPath = GetLastestInstallFile();
|
||||||
|
if (is_null($sPath)) {
|
||||||
|
throw new Exception("$sPath no installation XM. Launch a setup....");
|
||||||
|
}
|
||||||
|
$aParams = new XMLParameters($sPath);
|
||||||
|
$aSelectedModules = array_filter($aParams->Get('selected_modules', []), static function ($element) {
|
||||||
|
global $aRemovedExtensions;
|
||||||
|
return ! array_key_exists($element, $aRemovedExtensions);
|
||||||
|
});
|
||||||
|
|
||||||
|
$aSelectedExtensions = array_filter($aParams->Get('selected_extensions', []), static function ($element) {
|
||||||
|
global $aRemovedExtensions;
|
||||||
|
return ! array_key_exists($element, $aRemovedExtensions);
|
||||||
|
});
|
||||||
|
|
||||||
$aPostParams = [
|
$aPostParams = [
|
||||||
"auth_user" => 'admin',
|
"auth_user" => 'admin',
|
||||||
"auth_pwd" => 'admin',
|
"auth_pwd" => 'admin',
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ use Combodo\iTop\Test\UnitTest\ItopTestCase;
|
|||||||
use iTopExtensionsMap;
|
use iTopExtensionsMap;
|
||||||
use iTopExtensionsMapFake;
|
use iTopExtensionsMapFake;
|
||||||
use ModuleDiscovery;
|
use ModuleDiscovery;
|
||||||
use WepPageFake;
|
|
||||||
use WizardController;
|
use WizardController;
|
||||||
use WizStepModulesChoiceFake;
|
use WizStepModulesChoiceFake;
|
||||||
use XMLParameters;
|
use XMLParameters;
|
||||||
@@ -1431,4 +1430,25 @@ HTML,
|
|||||||
}
|
}
|
||||||
$this->assertEquals($expected, $this->oWizStepModulesChoiceFake->GetWizardSteps());
|
$this->assertEquals($expected, $this->oWizStepModulesChoiceFake->GetWizardSteps());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testGetWizardStepsWithoutAnyExtension()
|
||||||
|
{
|
||||||
|
$this->oWizard->SetParameter('source_dir', __DIR__.'/ressources');
|
||||||
|
$oExtensionMap = $this->createMock(iTopExtensionsMap::class);
|
||||||
|
$oExtensionMap->expects(self::any())->method('GetAllExtensionsOptionInfo')->willReturn([]);
|
||||||
|
|
||||||
|
$this->oWizStepModulesChoiceFake->setExtensionMap($oExtensionMap);
|
||||||
|
|
||||||
|
$expected = [
|
||||||
|
["class" => "WizStepWelcome","state" => ""],
|
||||||
|
["class" => "WizStepInstallOrUpgrade","state" => ""],
|
||||||
|
["class" => "WizStepDetectedInfo","state" => ""],
|
||||||
|
["class" => "WizStepUpgradeMiscParams","state" => ""],
|
||||||
|
];
|
||||||
|
|
||||||
|
for ($i = 0;$i <= 4; $i++) {
|
||||||
|
$expected [] = ["class" => "WizStepModulesChoice","state" => "".$i];
|
||||||
|
}
|
||||||
|
$this->assertEquals($expected, $this->oWizStepModulesChoiceFake->GetWizardSteps());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user