Files
MYSOPHAL/inc/software.class.php
2025-08-07 13:15:31 +01:00

1118 lines
37 KiB
PHP

<?php
/**
* ---------------------------------------------------------------------
* GLPI - Gestionnaire Libre de Parc Informatique
* Copyright (C) 2015-2020 Teclib' and contributors.
*
* http://glpi-project.org
*
* based on GLPI - Gestionnaire Libre de Parc Informatique
* Copyright (C) 2003-2014 by the INDEPNET Development Team.
*
* ---------------------------------------------------------------------
*
* LICENSE
*
* This file is part of GLPI.
*
* GLPI 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, or
* (at your option) any later version.
*
* GLPI 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with GLPI. If not, see <http://www.gnu.org/licenses/>.
* ---------------------------------------------------------------------
*/
if (!defined('GLPI_ROOT')) {
die("Sorry. You can't access this file directly");
}
/** Software Class
**/
class Software extends CommonDBTM {
use Glpi\Features\Clonable;
// From CommonDBTM
public $dohistory = true;
static protected $forward_entity_to = ['Infocom', 'ReservationItem', 'SoftwareVersion'];
static $rightname = 'software';
protected $usenotepad = true;
public function getCloneRelations() :array {
return [
Infocom::class,
Contract_Item::class,
Document_Item::class,
KnowbaseItem_Item::class
];
}
static function getTypeName($nb = 0) {
return _n('Software', 'Software', $nb);
}
/**
* @see CommonGLPI::getMenuShorcut()
*
* @since 0.85
**/
static function getMenuShorcut() {
return 's';
}
function getTabNameForItem(CommonGLPI $item, $withtemplate = 0) {
if (!$withtemplate) {
switch ($item->getType()) {
case __CLASS__ :
if ($item->isRecursive()
&& $item->can($item->fields['id'], UPDATE)) {
return __('Merging');
}
break;
}
}
return '';
}
static function displayTabContentForItem(CommonGLPI $item, $tabnum = 1, $withtemplate = 0) {
if ($item->getType() == __CLASS__) {
$item->showMergeCandidates();
}
return true;
}
function defineTabs($options = []) {
$ong = [];
$this->addDefaultFormTab($ong);
$this->addImpactTab($ong, $options);
$this->addStandardTab('SoftwareVersion', $ong, $options);
$this->addStandardTab('SoftwareLicense', $ong, $options);
$this->addStandardTab('Item_SoftwareVersion', $ong, $options);
$this->addStandardTab('Infocom', $ong, $options);
$this->addStandardTab('Contract_Item', $ong, $options);
$this->addStandardTab('Document_Item', $ong, $options);
$this->addStandardTab('KnowbaseItem_Item', $ong, $options);
$this->addStandardTab('Ticket', $ong, $options);
$this->addStandardTab('Item_Problem', $ong, $options);
$this->addStandardTab('Change_Item', $ong, $options);
$this->addStandardTab('Link', $ong, $options);
$this->addStandardTab('Notepad', $ong, $options);
$this->addStandardTab('Reservation', $ong, $options);
$this->addStandardTab('Domain_Item', $ong, $options);
$this->addStandardTab('Appliance_Item', $ong, $options);
$this->addStandardTab('Log', $ong, $options);
$this->addStandardTab(__CLASS__, $ong, $options);
return $ong;
}
function prepareInputForUpdate($input) {
if (isset($input['is_update']) && !$input['is_update']) {
$input['softwares_id'] = 0;
}
return $input;
}
function prepareInputForAdd($input) {
if (isset($input['is_update']) && !$input['is_update']) {
$input['softwares_id'] = 0;
}
if (isset($input["id"]) && ($input["id"] > 0)) {
$input["_oldID"] = $input["id"];
}
unset($input['id']);
unset($input['withtemplate']);
//If category was not set by user (when manually adding a user)
if (!isset($input["softwarecategories_id"]) || !$input["softwarecategories_id"]) {
$softcatrule = new RuleSoftwareCategoryCollection();
$result = $softcatrule->processAllRules(null, null, Toolbox::stripslashes_deep($input));
if (!empty($result)) {
if (isset($result['_ignore_import'])) {
$input["softwarecategories_id"] = 0;
} else if (isset($result["softwarecategories_id"])) {
$input["softwarecategories_id"] = $result["softwarecategories_id"];
} else if (isset($result["_import_category"])) {
$softCat = new SoftwareCategory();
$input["softwarecategories_id"]
= $softCat->importExternal($input["_system_category"]);
}
} else {
$input["softwarecategories_id"] = 0;
}
}
return $input;
}
function cleanDBonPurge() {
// SoftwareLicense does not extends CommonDBConnexity
$sl = new SoftwareLicense();
$sl->deleteByCriteria(['softwares_id' => $this->fields['id']]);
$this->deleteChildrenAndRelationsFromDb(
[
Item_Project::class,
SoftwareVersion::class,
]
);
}
/**
* Update validity indicator of a specific software
*
* @param $ID ID of the licence
*
* @since 0.85
*
* @return void
**/
static function updateValidityIndicator($ID) {
$soft = new self();
if ($soft->getFromDB($ID)) {
$valid = 1;
if (countElementsInTable('glpi_softwarelicenses',
['softwares_id'=>$ID,
'NOT' => [ 'is_valid']]) > 0) {
$valid = 0;
}
if ($valid != $soft->fields['is_valid']) {
$soft->update(['id' => $ID,
'is_valid' => $valid]);
}
}
}
/**
* Print the Software form
*
* @param $ID integer ID of the item
* @param $options array of possible options:
* - target filename : where to go when done.
* - withtemplate boolean : template or basic item
*
*@return boolean item found
**/
function showForm($ID, $options = []) {
$this->initForm($ID, $options);
$this->showFormHeader($options);
$canedit = $this->canEdit($ID);
echo "<tr class='tab_bg_1'>";
echo "<td>" . __('Name') . "</td>";
echo "<td>";
Html::autocompletionTextField($this, "name");
echo "</td>";
echo "<td>" . __('Publisher')."</td><td>";
Manufacturer::dropdown(['value' => $this->fields["manufacturers_id"]]);
echo "</td></tr>\n";
echo "<tr class='tab_bg_1'>";
echo "<td>" . Location::getTypeName(1) . "</td><td>";
Location::dropdown(['value' => $this->fields["locations_id"],
'entity' => $this->fields["entities_id"]]);
echo "</td>";
echo "<td>" . __('Category') . "</td><td>";
SoftwareCategory::dropdown(['value' => $this->fields["softwarecategories_id"]]);
echo "</td></tr>\n";
echo "<tr class='tab_bg_1'>";
echo "<td>" . __('Technician in charge of the software') . "</td><td>";
User::dropdown(['name' => 'users_id_tech',
'value' => $this->fields["users_id_tech"],
'right' => 'own_ticket',
'entity' => $this->fields["entities_id"]]);
echo "</td>";
echo "<td>" . __('Associable to a ticket') . "</td><td>";
Dropdown::showYesNo('is_helpdesk_visible', $this->fields['is_helpdesk_visible']);
echo "</td></tr>\n";
echo "<tr class='tab_bg_1'>";
echo "<td>".__('Group in charge of the software')."</td>";
echo "<td>";
Group::dropdown([
'name' => 'groups_id_tech',
'value' => $this->fields['groups_id_tech'],
'entity' => $this->fields['entities_id'],
'condition' => ['is_assign' => 1]
]);
echo "</td>";
echo "<td rowspan='4' class='middle'>".__('Comments') . "</td>";
echo "<td class='center middle' rowspan='4'>";
echo "<textarea cols='45' rows='8' name='comment' >".$this->fields["comment"]."</textarea>";
echo "</td></tr>\n";
echo "<tr class='tab_bg_1'>";
echo "<td >" . User::getTypeName(1) . "</td>";
echo "<td >";
User::dropdown(['value' => $this->fields["users_id"],
'entity' => $this->fields["entities_id"],
'right' => 'all']);
echo "</td></tr>\n";
echo "<tr class='tab_bg_1'>";
echo "<td>" . Group::getTypeName(1) . "</td><td>";
Group::dropdown([
'value' => $this->fields["groups_id"],
'entity' => $this->fields["entities_id"],
'condition' => ['is_itemgroup' => 1]
]);
echo "</td></tr>\n";
// UPDATE
echo "<tr class='tab_bg_1'>";
//TRANS: a noun, (ex : this software is an upgrade of..)
echo "<td>" . __('Upgrade') . "</td><td>";
Dropdown::showYesNo("is_update", $this->fields['is_update']);
echo "&nbsp;" . __('from') . "&nbsp;";
Software::dropdown(['value' => $this->fields["softwares_id"]]);
echo "</td></tr>\n";
$this->showFormButtons($options);
return true;
}
function getEmpty() {
global $CFG_GLPI;
parent::getEmpty();
$this->fields["is_helpdesk_visible"] = $CFG_GLPI["default_software_helpdesk_visible"];
}
/**
* @see CommonDBTM::getSpecificMassiveActions()
**/
function getSpecificMassiveActions($checkitem = null) {
$isadmin = static::canUpdate();
$actions = parent::getSpecificMassiveActions($checkitem);
if ($isadmin
&& (countElementsInTable("glpi_rules", ['sub_type'=>'RuleSoftwareCategory']) > 0)) {
$actions[__CLASS__.MassiveAction::CLASS_ACTION_SEPARATOR.'compute_software_category']
= "<i class='ma-icon fas fa-calculator'></i>".
__('Recalculate the category');
}
if (Session::haveRightsOr("rule_dictionnary_software", [CREATE, UPDATE])
&& (countElementsInTable("glpi_rules", ['sub_type'=>'RuleDictionnarySoftware']) > 0)) {
$actions[__CLASS__.MassiveAction::CLASS_ACTION_SEPARATOR.'replay_dictionnary']
= "<i class='ma-icon fas fa-undo'></i>".
__('Replay the dictionary rules');
}
if ($isadmin) {
KnowbaseItem_Item::getMassiveActionsForItemtype($actions, __CLASS__, 0, $checkitem);
}
return $actions;
}
/**
* @since 0.85
*
* @see CommonDBTM::processMassiveActionsForOneItemtype()
**/
static function processMassiveActionsForOneItemtype(MassiveAction $ma, CommonDBTM $item,
array $ids) {
switch ($ma->getAction()) {
case 'merge' :
$input = $ma->getInput();
if (isset($input['item_items_id'])) {
$items = [];
foreach ($ids as $id) {
$items[$id] = 1;
}
if ($item->can($input['item_items_id'], UPDATE)) {
if ($item->merge($items)) {
$ma->itemDone($item->getType(), $ids, MassiveAction::ACTION_OK);
} else {
$ma->itemDone($item->getType(), $ids, MassiveAction::ACTION_KO);
$ma->addMessage($item->getErrorMessage(ERROR_ON_ACTION));
}
} else {
$ma->itemDone($item->getType(), $ids, MassiveAction::ACTION_NORIGHT);
$ma->addMessage($item->getErrorMessage(ERROR_RIGHT));
}
} else {
$ma->itemDone($item->getType(), $ids, MassiveAction::ACTION_KO);
}
return;
case 'compute_software_category' :
$softcatrule = new RuleSoftwareCategoryCollection();
foreach ($ids as $id) {
$params = [];
//Get software name and manufacturer
if ($item->can($id, UPDATE)) {
$params["name"] = $item->fields["name"];
$params["manufacturers_id"] = $item->fields["manufacturers_id"];
$params["comment"] = $item->fields["comment"];
$output = [];
$output = $softcatrule->processAllRules(null, $output, $params);
//Process rules
if (isset($output['softwarecategories_id'])
&& $item->update(['id' => $id,
'softwarecategories_id'
=> $output['softwarecategories_id']])) {
$ma->itemDone($item->getType(), $id, MassiveAction::ACTION_OK);
} else {
$ma->itemDone($item->getType(), $id, MassiveAction::ACTION_KO);
$ma->addMessage($item->getErrorMessage(ERROR_ON_ACTION));
}
} else {
$ma->itemDone($item->getType(), $ids, MassiveAction::ACTION_NORIGHT);
$ma->addMessage($item->getErrorMessage(ERROR_RIGHT));
}
}
return;
case 'replay_dictionnary' :
$softdictionnayrule = new RuleDictionnarySoftwareCollection();
$allowed_ids = [];
foreach ($ids as $id) {
if ($item->can($id, UPDATE)) {
$allowed_ids[] = $id;
} else {
$ma->itemDone($item->getType(), $ids, MassiveAction::ACTION_NORIGHT);
$ma->addMessage($item->getErrorMessage(ERROR_RIGHT));
}
}
if ($softdictionnayrule->replayRulesOnExistingDB(0, 0, $allowed_ids)>0) {
$ma->itemDone($item->getType(), $allowed_ids, MassiveAction::ACTION_OK);
} else {
$ma->itemDone($item->getType(), $allowed_ids, MassiveAction::ACTION_KO);
}
return;
}
parent::processMassiveActionsForOneItemtype($ma, $item, $ids);
}
function rawSearchOptions() {
// Only use for History (not by search Engine)
$tab = parent::rawSearchOptions();
$tab[] = [
'id' => '2',
'table' => $this->getTable(),
'field' => 'id',
'name' => __('ID'),
'massiveaction' => false,
'datatype' => 'number'
];
$tab = array_merge($tab, Location::rawSearchOptionsToAdd());
$tab[] = [
'id' => '16',
'table' => $this->getTable(),
'field' => 'comment',
'name' => __('Comments'),
'datatype' => 'text'
];
$tab[] = [
'id' => '62',
'table' => 'glpi_softwarecategories',
'field' => 'completename',
'name' => __('Category'),
'datatype' => 'dropdown'
];
$tab[] = [
'id' => '19',
'table' => $this->getTable(),
'field' => 'date_mod',
'name' => __('Last update'),
'datatype' => 'datetime',
'massiveaction' => false
];
$tab[] = [
'id' => '121',
'table' => $this->getTable(),
'field' => 'date_creation',
'name' => __('Creation date'),
'datatype' => 'datetime',
'massiveaction' => false
];
$tab[] = [
'id' => '23',
'table' => 'glpi_manufacturers',
'field' => 'name',
'name' => __('Publisher'),
'datatype' => 'dropdown'
];
$tab[] = [
'id' => '24',
'table' => 'glpi_users',
'field' => 'name',
'linkfield' => 'users_id_tech',
'name' => __('Technician in charge of the software'),
'datatype' => 'dropdown',
'right' => 'own_ticket'
];
$tab[] = [
'id' => '49',
'table' => 'glpi_groups',
'field' => 'completename',
'linkfield' => 'groups_id_tech',
'name' => __('Group in charge of the software'),
'condition' => ['is_assign' => 1],
'datatype' => 'dropdown'
];
$tab[] = [
'id' => '64',
'table' => $this->getTable(),
'field' => 'template_name',
'name' => __('Template name'),
'datatype' => 'text',
'massiveaction' => false,
'nosearch' => true,
'nodisplay' => true,
'autocomplete' => true,
];
$tab[] = [
'id' => '70',
'table' => 'glpi_users',
'field' => 'name',
'name' => User::getTypeName(1),
'datatype' => 'dropdown',
'right' => 'all'
];
$tab[] = [
'id' => '71',
'table' => 'glpi_groups',
'field' => 'completename',
'name' => Group::getTypeName(1),
'condition' => ['is_itemgroup' => 1],
'datatype' => 'dropdown'
];
$tab[] = [
'id' => '61',
'table' => $this->getTable(),
'field' => 'is_helpdesk_visible',
'name' => __('Associable to a ticket'),
'datatype' => 'bool'
];
$tab[] = [
'id' => '63',
'table' => $this->getTable(),
'field' => 'is_valid',
//TRANS: Indicator to know is all licenses of the software are valids
'name' => __('Valid licenses'),
'datatype' => 'bool'
];
$tab[] = [
'id' => '80',
'table' => 'glpi_entities',
'field' => 'completename',
'name' => Entity::getTypeName(1),
'massiveaction' => false,
'datatype' => 'dropdown'
];
$newtab = [
'id' => '72',
'table' => 'glpi_items_softwareversions',
'field' => 'id',
'name' => _x('quantity', 'Number of installations'),
'forcegroupby' => true,
'usehaving' => true,
'datatype' => 'count',
'nometa' => true,
'massiveaction' => false,
'joinparams' => [
'jointype' => 'child',
'beforejoin' => [
'table' => 'glpi_softwareversions',
'joinparams' => ['jointype' => 'child'],
],
'condition' => "AND NEWTABLE.`is_deleted_item` = 0
AND NEWTABLE.`is_deleted` = 0
AND NEWTABLE.`is_template_item` = 0",
]
];
if (Session::getLoginUserID()) {
$newtab['joinparams']['condition'] .= getEntitiesRestrictRequest(' AND', 'NEWTABLE');
}
$tab[] = $newtab;
$tab = array_merge($tab, SoftwareLicense::rawSearchOptionsToAdd());
$name = _n('Version', 'Versions', Session::getPluralNumber());
$tab[] = [
'id' => 'versions',
'name' => $name
];
$tab[] = [
'id' => '5',
'table' => 'glpi_softwareversions',
'field' => 'name',
'name' => __('Name'),
'forcegroupby' => true,
'massiveaction' => false,
'displaywith' => ['softwares_id'],
'joinparams' => [
'jointype' => 'child'
],
'datatype' => 'dropdown'
];
$tab[] = [
'id' => '31',
'table' => 'glpi_states',
'field' => 'completename',
'name' => __('Status'),
'datatype' => 'dropdown',
'forcegroupby' => true,
'massiveaction' => false,
'joinparams' => [
'beforejoin' => [
'table' => 'glpi_softwareversions',
'joinparams' => [
'jointype' => 'child'
]
]
]
];
$tab[] = [
'id' => '170',
'table' => 'glpi_softwareversions',
'field' => 'comment',
'name' => __('Comments'),
'forcegroupby' => true,
'datatype' => 'text',
'massiveaction' => false,
'joinparams' => [
'jointype' => 'child'
]
];
$tab[] = [
'id' => '4',
'table' => 'glpi_operatingsystems',
'field' => 'name',
'datatype' => 'dropdown',
'name' => OperatingSystem::getTypeName(1),
'forcegroupby' => true,
'joinparams' => [
'beforejoin' => [
'table' => 'glpi_softwareversions',
'joinparams' => [
'jointype' => 'child'
]
]
]
];
// add objectlock search options
$tab = array_merge($tab, ObjectLock::rawSearchOptionsToAdd(get_class($this)));
$tab = array_merge($tab, Notepad::rawSearchOptionsToAdd());
return $tab;
}
/**
* Make a select box for software to install
*
* @param $myname select name
* @param $entity_restrict restrict to a defined entity
*
* @return integer random part of elements id
**/
static function dropdownSoftwareToInstall($myname, $entity_restrict) {
global $CFG_GLPI;
// Make a select box
$rand = mt_rand();
$where = getEntitiesRestrictCriteria('glpi_softwares', 'entities_id',
$entity_restrict, true);
$rand = Dropdown::show('Software', ['condition' => $where]);
$paramsselsoft = ['softwares_id' => '__VALUE__',
'myname' => $myname];
Ajax::updateItemOnSelectEvent("dropdown_softwares_id$rand", "show_".$myname.$rand,
$CFG_GLPI["root_doc"]."/ajax/dropdownInstallVersion.php",
$paramsselsoft);
echo "<span id='show_".$myname.$rand."'>&nbsp;</span>\n";
return $rand;
}
/**
* Make a select box for license software to associate
*
* @param $myname select name
* @param $entity_restrict restrict to a defined entity
*
* @return integer random part of elements id
**/
static function dropdownLicenseToInstall($myname, $entity_restrict) {
global $CFG_GLPI, $DB;
$iterator = $DB->request([
'SELECT' => [
'glpi_softwares.id',
'glpi_softwares.name'
],
'DISTINCT' => true,
'FROM' => 'glpi_softwares',
'INNER JOIN' => [
'glpi_softwarelicenses' => [
'ON' => [
'glpi_softwarelicenses' => 'softwares_id',
'glpi_softwares' => 'id'
]
]
],
'WHERE' => [
'glpi_softwares.is_deleted' => 0,
'glpi_softwares.is_template' => 0
] + getEntitiesRestrictCriteria('glpi_softwarelicenses', 'entities_id', $entity_restrict, true),
'ORDERBY' => 'glpi_softwares.name'
]);
$values = [];
while ($data = $iterator->next()) {
$softwares_id = $data["id"];
$values[$softwares_id] = $data["name"];
}
$rand = Dropdown::showFromArray('softwares_id', $values, ['display_emptychoice' => true]);
$paramsselsoft = ['softwares_id' => '__VALUE__',
'entity_restrict' => $entity_restrict,
'myname' => $myname];
Ajax::updateItemOnSelectEvent("dropdown_softwares_id$rand", "show_".$myname.$rand,
$CFG_GLPI["root_doc"]."/ajax/dropdownSoftwareLicense.php",
$paramsselsoft);
echo "<span id='show_".$myname.$rand."'>&nbsp;</span>\n";
return $rand;
}
/**
* Create a new software
*
* @param name the software's name (need to be addslashes)
* @param manufacturer_id id of the software's manufacturer
* @param entity the entity in which the software must be added
* @param comment (default '')
* @param is_recursive boolean must the software be recursive (false by default)
* @param is_helpdesk_visible show in helpdesk, default : from config (false by default)
*
* @return the software's ID
**/
function addSoftware($name, $manufacturer_id, $entity, $comment = '',
$is_recursive = false, $is_helpdesk_visible = null) {
global $CFG_GLPI;
$input["name"] = $name;
$input["manufacturers_id"] = $manufacturer_id;
$input["entities_id"] = $entity;
$input["is_recursive"] = ($is_recursive ? 1 : 0);
// No comment
if (is_null($is_helpdesk_visible)) {
$input["is_helpdesk_visible"] = $CFG_GLPI["default_software_helpdesk_visible"];
} else {
$input["is_helpdesk_visible"] = $is_helpdesk_visible;
}
//Process software's category rules
$softcatrule = new RuleSoftwareCategoryCollection();
$result = $softcatrule->processAllRules(null, null, Toolbox::stripslashes_deep($input));
if (!empty($result)) {
if (isset($result['_ignore_import'])) {
$input["softwarecategories_id"] = 0;
} else if (isset($result["softwarecategories_id"])) {
$input["softwarecategories_id"] = $result["softwarecategories_id"];
} else if (isset($result["_import_category"])) {
$softCat = new SoftwareCategory();
$input["softwarecategories_id"]
= $softCat->importExternal($input["_system_category"]);
}
} else {
$input["softwarecategories_id"] = 0;
}
return $this->add($input);
}
/**
* Add a software. If already exist in trashbin restore it
*
* @param name the software's name
* @param manufacturer the software's manufacturer
* @param entity the entity in which the software must be added
* @param comment comment (default '')
* @param is_recursive boolean must the software be recursive (false by default)
* @param is_helpdesk_visible show in helpdesk, default = config value (false by default)
*/
function addOrRestoreFromTrash($name, $manufacturer, $entity, $comment = '',
$is_recursive = false, $is_helpdesk_visible = null) {
global $DB;
//Look for the software by his name in GLPI for a specific entity
$manufacturer_id = 0;
if ($manufacturer != '') {
$manufacturer_id = Dropdown::import('Manufacturer', ['name' => $manufacturer]);
}
$iterator = $DB->request([
'SELECT' => [
'glpi_softwares.id',
'glpi_softwares.is_deleted'
],
'FROM' => 'glpi_softwares',
'WHERE' => [
'name' => $name,
'manufacturers_id' => $manufacturer_id,
'is_template' => 0
] + getEntitiesRestrictCriteria('glpi_softwares', 'entities_id', $entity, true)
]);
if (count($iterator)) {
//Software already exists for this entity, get his ID
$data = $iterator->next();
$ID = $data["id"];
// restore software
if ($data['is_deleted']) {
$this->removeFromTrash($ID);
}
} else {
$ID = 0;
}
if (!$ID) {
$ID = $this->addSoftware($name, $manufacturer_id, $entity, $comment, $is_recursive,
$is_helpdesk_visible);
}
return $ID;
}
/**
* Put software in trashbin because it's been removed by GLPI software dictionnary
*
* @param $ID the ID of the software to put in trashbin
* @param $comment the comment to add to the already existing software's comment (default '')
*
* @return boolean (success)
**/
function putInTrash($ID, $comment = '') {
global $CFG_GLPI;
$this->getFromDB($ID);
$input["id"] = $ID;
$input["is_deleted"] = 1;
//change category of the software on deletion (if defined in glpi_configs)
if (isset($CFG_GLPI["softwarecategories_id_ondelete"])
&& ($CFG_GLPI["softwarecategories_id_ondelete"] != 0)) {
$input["softwarecategories_id"] = $CFG_GLPI["softwarecategories_id_ondelete"];
}
//Add dictionnary comment to the current comment
$input["comment"] = (($this->fields["comment"] != '') ? "\n" : '') . $comment;
return $this->update($input);
}
/**
* Restore a software from trashbin
*
* @param $ID the ID of the software to put in trashbin
*
* @return boolean (success)
**/
function removeFromTrash($ID) {
$res = $this->restore(["id" => $ID]);
$softcatrule = new RuleSoftwareCategoryCollection();
$result = $softcatrule->processAllRules(null, null, $this->fields);
if (!empty($result)
&& isset($result['softwarecategories_id'])
&& ($result['softwarecategories_id'] != $this->fields['softwarecategories_id'])) {
$this->update(['id' => $ID,
'softwarecategories_id' => $result['softwarecategories_id']]);
}
return $res;
}
/**
* Show softwares candidates to be merged with the current
*
* @return void
**/
function showMergeCandidates() {
global $DB;
$ID = $this->getField('id');
$this->check($ID, UPDATE);
$rand = mt_rand();
echo "<div class='center'>";
$iterator = $DB->request([
'SELECT' => [
'glpi_softwares.id',
'glpi_softwares.name',
'glpi_entities.completename AS entity'
],
'FROM' => 'glpi_softwares',
'LEFT JOIN' => [
'glpi_entities' => [
'ON' => [
'glpi_softwares' => 'entities_id',
'glpi_entities' => 'id'
]
]
],
'WHERE' => [
'glpi_softwares.id' => ['!=', $ID],
'glpi_softwares.name' => addslashes($this->fields['name']),
'glpi_softwares.is_deleted' => 0,
'glpi_softwares.is_template' => 0
] + getEntitiesRestrictCriteria(
'glpi_softwares',
'entities_id',
getSonsOf("glpi_entities", $this->fields["entities_id"]),
false
),
'ORDERBY' => 'entity'
]);
$nb = count($iterator);
if ($nb) {
$link = Toolbox::getItemTypeFormURL('Software');
Html::openMassiveActionsForm('mass'.__CLASS__.$rand);
$massiveactionparams
= ['num_displayed' => min($_SESSION['glpilist_limit'], $nb),
'container' => 'mass'.__CLASS__.$rand,
'specific_actions'
=> [__CLASS__.MassiveAction::CLASS_ACTION_SEPARATOR.'merge'
=> __('Merge')],
'item' => $this];
Html::showMassiveActions($massiveactionparams);
echo "<table class='tab_cadre_fixehov'>";
echo "<tr><th width='10'>";
echo Html::getCheckAllAsCheckbox('mass'.__CLASS__.$rand);
echo "</th>";
echo "<th>".__('Name')."</th>";
echo "<th>".Entity::getTypeName(1)."</th>";
echo "<th>"._n('Installation', 'Installations', Session::getPluralNumber())."</th>";
echo "<th>".SoftwareLicense::getTypeName(Session::getPluralNumber())."</th></tr>";
while ($data = $iterator->next()) {
echo "<tr class='tab_bg_2'>";
echo "<td>".Html::getMassiveActionCheckBox(__CLASS__, $data["id"])."</td>";
echo "<td><a href='".$link."?id=".$data["id"]."'>".$data["name"]."</a></td>";
echo "<td>".$data["entity"]."</td>";
echo "<td class='right'>".Item_SoftwareVersion::countForSoftware($data["id"])."</td>";
echo "<td class='right'>".SoftwareLicense::countForSoftware($data["id"])."</td></tr>\n";
}
echo "</table>\n";
$massiveactionparams['ontop'] =false;
Html::showMassiveActions($massiveactionparams);
Html::closeForm();
} else {
echo __('No item found');
}
echo "</div>";
}
/**
* Merge softwares with current
*
* @param $item array of software ID to be merged
* @param boolean display html progress bar
*
* @return boolean about success
**/
function merge($item, $html = true) {
global $DB;
$ID = $this->getField('id');
if ($html) {
echo "<div class='center'>";
echo "<table class='tab_cadrehov'><tr><th>".__('Merging')."</th></tr>";
echo "<tr class='tab_bg_2'><td>";
Html::createProgressBar(__('Work in progress...'));
echo "</td></tr></table></div>\n";
}
$item = array_keys($item);
// Search for software version
$req = $DB->request("glpi_softwareversions", ["softwares_id" => $item]);
$i = 0;
if ($nb = $req->numrows()) {
foreach ($req as $from) {
$found = false;
foreach ($DB->request("glpi_softwareversions",
["softwares_id" => $ID,
"name" => $from["name"]]) as $dest) {
// Update version ID on License
$DB->update(
'glpi_softwarelicenses', [
'softwareversions_id_buy' => $dest['id']
], [
'softwareversions_id_buy' => $from['id']
]
);
$DB->update(
'glpi_softwarelicenses', [
'softwareversions_id_use' => $dest['id']
], [
'softwareversions_id_use' => $from['id']
]
);
// Move installation to existing version in destination software
$found = $DB->update(
'glpi_items_softwareversions', [
'softwareversions_id' => $dest['id']
], [
'softwareversions_id' => $from['id']
]
);
}
if ($found) {
// Installation has be moved, delete the source version
$DB->delete(
'glpi_softwareversions', [
'id' => $from['id']
]
);
} else {
// Move version to destination software
$result = $DB->update(
'glpi_softwareversions', [
'softwares_id' => $ID,
'entities_id' => $this->getField('entities_id')
], [
'id' => $from['id']
]
);
}
if ($result) {
$i++;
}
if ($html) {
Html::changeProgressBarPosition($i, $nb+1);
}
}
}
// Move software license
$result = $DB->update(
'glpi_softwarelicenses', [
'softwares_id' => $ID
], [
'softwares_id' => $item
]
);
if ($result) {
$i++;
}
if ($i == ($nb+1)) {
//error_log ("All merge operations ok.");
$soft = new self();
foreach ($item as $old) {
$soft->putInTrash($old, __('Software deleted after merging'));
}
}
if ($html) {
Html::changeProgressBarPosition($i, $nb+1, __('Task completed.'));
}
return $i == ($nb+1);
}
static function getDefaultSearchRequest() {
return [
'sort' => 0
];
}
static function getIcon() {
return "fas fa-cube";
}
}