Files
luos/workflow/engine/methods/setup/pluginsImportFile.php

332 lines
14 KiB
PHP
Raw Normal View History

<?php
/**
*
* processes_ImportFile.php
*
2019-09-06 15:49:07 -04:00
* If the feature is enable and the code_scanner_scope was enable with the argument import_plugin, will check the code
* Review when a plugin was enable
*
2019-09-06 15:49:07 -04:00
* @link https://wiki.processmaker.com/3.0/Plugins#Import_a_Plugin
*/
2017-08-14 16:13:46 -04:00
use ProcessMaker\Core\System;
2017-08-01 12:16:06 -04:00
use ProcessMaker\Plugins\PluginRegistry;
2018-08-28 09:34:11 -04:00
use ProcessMaker\Validation\ValidationUploadedFiles;
global $RBAC;
2017-12-04 13:25:35 +00:00
$RBAC->requirePermissions('PM_SETUP_ADVANCE');
try {
2018-12-13 14:42:05 -04:00
ValidationUploadedFiles::getValidationUploadedFiles()->dispatch(function($validator) {
2018-08-28 09:34:11 -04:00
throw new Exception($validator->getMessage());
});
//load the variables
2017-12-04 13:25:35 +00:00
if (!isset($_FILES['form']['error']['PLUGIN_FILENAME']) || $_FILES['form']['error']['PLUGIN_FILENAME'] == 1) {
throw (new Exception(G::loadTranslation('ID_ERROR_UPLOADING_PLUGIN_FILENAME')));
}
2014-12-01 12:33:00 -04:00
//save the file
if ($_FILES['form']['error']['PLUGIN_FILENAME'] == 0) {
$filename = $_FILES['form']['name']['PLUGIN_FILENAME'];
$path = PATH_DOCUMENT . 'input' . PATH_SEP;
$tempName = $_FILES['form']['tmp_name']['PLUGIN_FILENAME'];
2017-12-04 13:25:35 +00:00
G::uploadFile($tempName, $path, $filename);
2014-12-01 12:33:00 -04:00
}
//save the files Enterprise
if ($_FILES['form']['error']['PLUGIN_FILENAME'] == 0) {
$filename = $_FILES['form']['name']['PLUGIN_FILENAME'];
$path = PATH_DOCUMENT . 'input' . PATH_SEP;
if (strpos($filename, 'enterprise') !== false) {
2017-12-04 13:25:35 +00:00
$tar = new Archive_Tar($path . $filename);
$sFileName = substr($filename, 0, strrpos($filename, '.'));
$sClassName = substr($filename, 0, strpos($filename, '-'));
$sClassName = !empty($sClassName) ? $sClassName : $sFileName;
$files = $tar->listContent();
$licenseName = '';
$listFiles = array();
foreach ($files as $key => $val) {
if (strpos(trim($val['filename']), 'enterprise/data/') !== false) {
$listFiles[] = trim($val['filename']);
}
if (strpos(trim($val['filename']), 'license_') !== false) {
$licenseName = trim($val['filename']);
}
}
2017-12-04 13:25:35 +00:00
$tar->extractList($listFiles, PATH_PLUGINS . 'data');
$tar->extractList($licenseName, PATH_PLUGINS);
2017-08-01 12:16:06 -04:00
$pluginRegistry = PluginRegistry::loadSingleton();
$autoPlugins = glob(PATH_PLUGINS . "data/enterprise/data/*.tar");
$autoPluginsA = array();
foreach ($autoPlugins as $filePath) {
$plName = basename($filePath);
//if (!(in_array($plName, $def))) {
if (strpos($plName, 'enterprise') === false) {
$autoPluginsA[]["sFilename"] = $plName;
}
}
$aPlugins = $autoPluginsA;
2017-12-04 13:25:35 +00:00
foreach ($aPlugins as $key => $aPlugin) {
$sClassName = substr($aPlugin["sFilename"], 0, strpos($aPlugin["sFilename"], "-"));
$oTar = new Archive_Tar(PATH_PLUGINS . "data/enterprise/data/" . $aPlugin["sFilename"]);
$oTar->extract(PATH_PLUGINS);
if (!(class_exists($sClassName))) {
2017-12-04 13:25:35 +00:00
require_once(PATH_PLUGINS . $sClassName . ".php");
}
$pluginDetail = $pluginRegistry->getPluginDetails($sClassName . ".php");
2017-08-01 12:16:06 -04:00
$pluginRegistry->installPlugin($pluginDetail->getNamespace()); //error
$pluginRegistry->savePlugin($pluginDetail->getNamespace());
}
$licfile = glob(PATH_PLUGINS . "*.dat");
2017-12-04 13:25:35 +00:00
if ((isset($licfile[0])) && (is_file($licfile[0]))) {
$licfilename = basename($licfile[0]);
@copy($licfile[0], PATH_DATA_SITE . $licfilename);
@unlink($licfile[0]);
}
2017-12-04 13:25:35 +00:00
require_once('classes/model/AddonsStore.php');
AddonsStore::checkLicenseStore();
2017-12-04 13:25:35 +00:00
$licenseManager = PmLicenseManager::getSingleton();
AddonsStore::updateAll(false);
2017-12-04 13:25:35 +00:00
$message = G::loadTranslation('ID_ENTERPRISE_INSTALLED') . ' ' . G::loadTranslation('ID_LOG_AGAIN');
G::SendMessageText($message, "INFO");
2017-12-04 13:25:35 +00:00
$licenseManager = PmLicenseManager::getSingleton();
die('<script type="text/javascript">parent.parent.location = "../login/login";</script>');
}
}
//save the packages plugins
if ($_FILES['form']['error']['PLUGIN_FILENAME'] == 0) {
$filename = $_FILES['form']['name']['PLUGIN_FILENAME'];
$path = PATH_DOCUMENT . 'input' . PATH_SEP;
if (strpos($filename, 'plugins-') !== false) {
2017-12-04 13:25:35 +00:00
$tar = new Archive_Tar($path . $filename);
$sFileName = substr($filename, 0, strrpos($filename, '.'));
$sClassName = substr($filename, 0, strpos($filename, '-'));
$sClassName = !empty($sClassName) ? $sClassName : $sFileName;
$files = $tar->listContent();
$licenseName = '';
$listFiles = array();
foreach ($files as $key => $val) {
if (strpos(trim($val['filename']), 'plugins/') !== false) {
$listFiles[] = trim($val['filename']);
}
if (strpos(trim($val['filename']), 'license_') !== false) {
$licenseName = trim($val['filename']);
}
}
2017-12-04 13:25:35 +00:00
$tar->extractList($listFiles, PATH_PLUGINS . 'data');
$tar->extractList($licenseName, PATH_PLUGINS);
2017-08-01 12:16:06 -04:00
$pluginRegistry = PluginRegistry::loadSingleton();
$autoPlugins = glob(PATH_PLUGINS . "data/plugins/*.tar");
$autoPluginsA = array();
foreach ($autoPlugins as $filePath) {
$plName = basename($filePath);
if (strpos($plName, 'enterprise') === false) {
$autoPluginsA[]["sFilename"] = $plName;
}
}
$aPlugins = $autoPluginsA;
2017-12-04 13:25:35 +00:00
foreach ($aPlugins as $key => $aPlugin) {
$sClassName = substr($aPlugin["sFilename"], 0, strpos($aPlugin["sFilename"], "-"));
$oTar = new Archive_Tar(PATH_PLUGINS . "data/plugins/" . $aPlugin["sFilename"]);
$oTar->extract(PATH_PLUGINS);
if (!(class_exists($sClassName))) {
2017-12-04 13:25:35 +00:00
require_once(PATH_PLUGINS . $sClassName . ".php");
}
$pluginDetail = $pluginRegistry->getPluginDetails($sClassName . ".php");
2017-08-01 12:16:06 -04:00
$pluginRegistry->installPlugin($pluginDetail->getNamespace()); //error
$pluginRegistry->savePlugin($pluginDetail->getNamespace());
}
$licfile = glob(PATH_PLUGINS . "*.dat");
2017-12-04 13:25:35 +00:00
if ((isset($licfile[0])) && (is_file($licfile[0]))) {
$licfilename = basename($licfile[0]);
@copy($licfile[0], PATH_DATA_SITE . $licfilename);
@unlink($licfile[0]);
}
2017-12-04 13:25:35 +00:00
require_once('classes/model/AddonsStore.php');
AddonsStore::checkLicenseStore();
2017-12-04 13:25:35 +00:00
$licenseManager = PmLicenseManager::getSingleton();
AddonsStore::updateAll(false);
2017-12-04 13:25:35 +00:00
$message = G::loadTranslation('ID_ENTERPRISE_INSTALLED') . ' ' . G::loadTranslation('ID_LOG_AGAIN');
G::SendMessageText($message, "INFO");
2017-12-04 13:25:35 +00:00
$licenseManager = PmLicenseManager::getSingleton();
die('<script type="text/javascript">parent.parent.location = "../login/login";</script>');
}
}
2014-12-01 12:33:00 -04:00
2017-12-04 13:25:35 +00:00
if (!$_FILES['form']['type']['PLUGIN_FILENAME'] == 'application/octet-stream') {
$pluginFilename = $_FILES['form']['type']['PLUGIN_FILENAME'];
2017-12-04 13:25:35 +00:00
throw new Exception(G::loadTranslation('ID_FILES_INVALID_PLUGIN_FILENAME', SYS_LANG, array("pluginFilename" => $pluginFilename
)));
}
2017-12-04 13:25:35 +00:00
$tar = new Archive_Tar($path . $filename);
$sFileName = substr($filename, 0, strrpos($filename, '.'));
$sClassName = substr($filename, 0, strpos($filename, '-'));
$sClassName = !empty($sClassName) ? $sClassName : $sFileName;
$aFiles = $tar->listContent();
$bMainFile = false;
$bClassFile = false;
2017-12-04 13:25:35 +00:00
if (!is_array($aFiles)) {
throw new Exception(G::loadTranslation('ID_FAILED_IMPORT_PLUGINS', SYS_LANG, array("filename" => $filename
)));
}
foreach ($aFiles as $key => $val) {
2017-12-04 13:25:35 +00:00
if (trim($val['filename']) == $sClassName . '.php') {
$bMainFile = true;
2017-12-04 13:25:35 +00:00
}
if (trim($val['filename']) == $sClassName . PATH_SEP . 'class.' . $sClassName . '.php') {
$bClassFile = true;
2017-12-04 13:25:35 +00:00
}
}
$partnerFlag = (defined('PARTNER_FLAG')) ? PARTNER_FLAG : false;
if (($sClassName == 'enterprise') && ($partnerFlag)) {
$pathFileFlag = PATH_DATA . 'flagNewLicence';
file_put_contents($pathFileFlag, 'New Enterprise');
}
2017-08-01 12:16:06 -04:00
$oPluginRegistry = PluginRegistry::loadSingleton();
$pluginFile = $sClassName . '.php';
if ($bMainFile && $bClassFile) {
$sAux = $sClassName . 'Plugin';
$fVersionOld = 0.0;
2017-12-04 13:25:35 +00:00
if (file_exists(PATH_PLUGINS . $pluginFile)) {
if (!class_exists($sAux) && !class_exists($sClassName . 'plugin')) {
include PATH_PLUGINS . $pluginFile;
}
2017-12-04 13:25:35 +00:00
if (!class_exists($sAux)) {
$sAux = $sClassName . 'plugin';
}
2017-12-04 13:25:35 +00:00
$oClass = new $sAux($sClassName);
$fVersionOld = $oClass->iVersion;
2017-12-04 13:25:35 +00:00
unset($oClass);
}
2017-12-04 13:25:35 +00:00
$res = $tar->extract($path);
//Check if is enterprise plugin
if ($oPluginRegistry->isEnterprisePlugin($sClassName, $path)) {
throw new Exception(G::LoadTranslation('ID_PMPLUGIN_IMPORT_PLUGIN_IS_ENTERPRISE', [$filename]));
}
/*----------------------------------********---------------------------------*/
if (PMLicensedFeatures::getSingleton()->verifyfeature("B0oWlBLY3hHdWY0YUNpZEtFQm5CeTJhQlIwN3IxMEkwaG4=")) {
//Check disabled code
2019-09-06 15:49:07 -04:00
$arrayFoundDisabledCode = [];
2017-10-10 12:33:25 -04:00
$cs = new CodeScanner(config("system.workspace"));
2019-09-06 15:49:07 -04:00
if (in_array('import_plugin', $cs->getScope())) {
$arrayFoundDisabledCode = array_merge($cs->checkDisabledCode("FILE", $path . $pluginFile),
$cs->checkDisabledCode("PATH", $path . $sClassName));
}
PM-473 "Analisis de los resultados de escaneo de las..." SOLVED Issue: Analisis de los resultados de escaneo de las funciones en ProcessMaker. Plugin/trigger code scanner. Cause: Nueva solicitud de funciones Solution: Se ha implementado esta nueva funcionalidad, que consta de lo siguiente: - Escaneo de codigo al importar un plugin (no se aplica a plugins enterprise) - Escaneo de codigo al habilitar un plugin (si el plugin ya se encuentra fisicamente en el directorio de los plugins) - Escaneo de codigo al importar un proceso - Escaneo de codigo al crear/modificar codigo de un trigger - Escaneo de codigo al ejecutar un caso que tenga seteados triggers en sus steps (si el trigger tiene codigo no deseado, no se ejecuta el trigger) - Se ha agregado la opcion "check-plugin-disabled-code" al comando "./gulliver", el mismo muestra informacion sobre los plugins con codigo no deseado. Ej: $ ./gulliver check-plugin-disabled-code [enterprise-plugin|custom-plugin|all|<plugin-name>] - Se ha agregado la opcion "check-workspace-disabled-code" al comando "./processmaker", el mismo muestra informacion sobre los workspaces con codigo no deseado en sus triggers. Ej: $ ./processmaker check-workspace-disabled-code <myWorkspace> - Por defecto ProcessMaker no realiza el escaneo de codigo, si se desea escanear codigo no deseado, se debera definir el atributo "enable_blacklist = 1" en el archivo "env.ini", este atributo no se aplica a las nuevas opciones creadas para los comandos "./gulliver" y "./processmaker" - Para una configuracion personalizada de codigo no deseado (lista negra), se pueden definir las mismas en el archivo "path/to/processmaker/workflow/engine/config/blacklist.ini" (si no existe el archivo se puede crear), o tambien en el atributo "disable_functions" esto en el archivo "php.ini" Ejemplo de "blacklist.ini": ;Classes ;======= DashletInterface ;Functions ;========= eval exec ;date ;echo strlen
2014-11-19 16:47:22 -04:00
if (!empty($arrayFoundDisabledCode)) {
throw new Exception(G::LoadTranslation("ID_DISABLED_CODE_PLUGIN"));
}
PM-473 "Analisis de los resultados de escaneo de las..." SOLVED Issue: Analisis de los resultados de escaneo de las funciones en ProcessMaker. Plugin/trigger code scanner. Cause: Nueva solicitud de funciones Solution: Se ha implementado esta nueva funcionalidad, que consta de lo siguiente: - Escaneo de codigo al importar un plugin (no se aplica a plugins enterprise) - Escaneo de codigo al habilitar un plugin (si el plugin ya se encuentra fisicamente en el directorio de los plugins) - Escaneo de codigo al importar un proceso - Escaneo de codigo al crear/modificar codigo de un trigger - Escaneo de codigo al ejecutar un caso que tenga seteados triggers en sus steps (si el trigger tiene codigo no deseado, no se ejecuta el trigger) - Se ha agregado la opcion "check-plugin-disabled-code" al comando "./gulliver", el mismo muestra informacion sobre los plugins con codigo no deseado. Ej: $ ./gulliver check-plugin-disabled-code [enterprise-plugin|custom-plugin|all|<plugin-name>] - Se ha agregado la opcion "check-workspace-disabled-code" al comando "./processmaker", el mismo muestra informacion sobre los workspaces con codigo no deseado en sus triggers. Ej: $ ./processmaker check-workspace-disabled-code <myWorkspace> - Por defecto ProcessMaker no realiza el escaneo de codigo, si se desea escanear codigo no deseado, se debera definir el atributo "enable_blacklist = 1" en el archivo "env.ini", este atributo no se aplica a las nuevas opciones creadas para los comandos "./gulliver" y "./processmaker" - Para una configuracion personalizada de codigo no deseado (lista negra), se pueden definir las mismas en el archivo "path/to/processmaker/workflow/engine/config/blacklist.ini" (si no existe el archivo se puede crear), o tambien en el atributo "disable_functions" esto en el archivo "php.ini" Ejemplo de "blacklist.ini": ;Classes ;======= DashletInterface ;Functions ;========= eval exec ;date ;echo strlen
2014-11-19 16:47:22 -04:00
}
/*----------------------------------********---------------------------------*/
//Get contents of plugin file
2017-12-04 13:25:35 +00:00
$sContent = file_get_contents($path . $pluginFile);
$sContent = str_ireplace($sAux, $sAux . '_', $sContent);
$sContent = str_ireplace('PATH_PLUGINS', "'" . $path . "'", $sContent);
$sContent = preg_replace("/\\\$oPluginRegistry\s*=\s*&\s*PMPluginRegistry::getSingleton\s*\(\s*\)\s*;/i", null, $sContent);
$sContent = preg_replace("/\\\$oPluginRegistry->registerPlugin\s*\(\s*[\"\']" . $sClassName . "[\"\']\s*,\s*__FILE__\s*\)\s*;/i", null, $sContent);
//header('Content-Type: text/plain');var_dump($sClassName, $sContent);die;
2017-12-04 13:25:35 +00:00
file_put_contents($path . $pluginFile, $sContent);
$sAux = $sAux . '_';
2017-12-04 13:25:35 +00:00
include($path . $pluginFile);
2017-12-04 13:25:35 +00:00
$oClass = new $sAux($sClassName);
$fVersionNew = $oClass->iVersion;
2017-12-04 13:25:35 +00:00
if (!isset($oClass->iPMVersion)) {
$oClass->iPMVersion = 0;
}
if ($oClass->iPMVersion > 0) {
2017-08-14 16:13:46 -04:00
if (System::getVersion() > 0) {
if ($oClass->iPMVersion > System::getVersion()) {
//throw new Exception('This plugin needs version ' . $oClass->iPMVersion . ' or higher of ProcessMaker');
}
}
}
2017-12-04 13:25:35 +00:00
if (!isset($oClass->aDependences)) {
$oClass->aDependences = null;
}
2017-12-04 13:25:35 +00:00
if (!empty($oClass->aDependences)) {
foreach ($oClass->aDependences as $aDependence) {
2017-12-04 13:25:35 +00:00
if (file_exists(PATH_PLUGINS . $aDependence['sClassName'] . '.php')) {
require_once PATH_PLUGINS . $aDependence['sClassName'] . '.php';
2017-12-04 13:25:35 +00:00
if (!$oPluginRegistry->getPluginDetails($aDependence['sClassName'] . '.php')) {
$sDependence = $aDependence['sClassName'];
2017-12-04 13:25:35 +00:00
throw new Exception(G::loadTranslation('ID_PLUGIN_DEPENDENCE_PLUGIN', SYS_LANG, array("Dependence" => $sDependence
)));
}
} else {
$sDependence = $aDependence['sClassName'];
2017-12-04 13:25:35 +00:00
throw new Exception(G::loadTranslation('ID_PLUGIN_DEPENDENCE_PLUGIN', SYS_LANG, array("Dependence" => $sDependence
)));
}
}
}
2017-12-04 13:25:35 +00:00
unset($oClass);
if ($fVersionOld > $fVersionNew) {
2017-12-04 13:25:35 +00:00
throw new Exception(G::loadTranslation('ID_RECENT_VERSION_PLUGIN'));
}
2017-12-04 13:25:35 +00:00
$res = $tar->extract(PATH_PLUGINS);
} else {
2017-12-04 13:25:35 +00:00
throw new Exception(G::loadTranslation('ID_FILE_CONTAIN_CLASS_PLUGIN', SYS_LANG, array("filename" => $filename, "className" => $sClassName
)));
}
2017-12-04 13:25:35 +00:00
if (!file_exists(PATH_PLUGINS . $sClassName . '.php')) {
throw new Exception(G::loadTranslation('ID_FILE_PLUGIN_NOT_EXISTS', SYS_LANG, array("pluginFile" => $pluginFile
)));
}
2017-12-04 13:25:35 +00:00
require_once(PATH_PLUGINS . $pluginFile);
2017-12-04 13:25:35 +00:00
$oPluginRegistry->registerPlugin($sClassName, PATH_PLUGINS . $sClassName . ".php");
2017-12-04 13:25:35 +00:00
$details = $oPluginRegistry->getPluginDetails($pluginFile);
2017-08-01 12:16:06 -04:00
$oPluginRegistry->installPlugin($details->getNamespace());
$oPluginRegistry->setupPlugins(); //get and setup enabled plugins
2017-08-01 12:16:06 -04:00
$oPluginRegistry->savePlugin($details->getNamespace());
PM-473 "Analisis de los resultados de escaneo de las..." SOLVED Issue: Analisis de los resultados de escaneo de las funciones en ProcessMaker. Plugin/trigger code scanner. Cause: Nueva solicitud de funciones Solution: Se ha implementado esta nueva funcionalidad, que consta de lo siguiente: - Escaneo de codigo al importar un plugin (no se aplica a plugins enterprise) - Escaneo de codigo al habilitar un plugin (si el plugin ya se encuentra fisicamente en el directorio de los plugins) - Escaneo de codigo al importar un proceso - Escaneo de codigo al crear/modificar codigo de un trigger - Escaneo de codigo al ejecutar un caso que tenga seteados triggers en sus steps (si el trigger tiene codigo no deseado, no se ejecuta el trigger) - Se ha agregado la opcion "check-plugin-disabled-code" al comando "./gulliver", el mismo muestra informacion sobre los plugins con codigo no deseado. Ej: $ ./gulliver check-plugin-disabled-code [enterprise-plugin|custom-plugin|all|<plugin-name>] - Se ha agregado la opcion "check-workspace-disabled-code" al comando "./processmaker", el mismo muestra informacion sobre los workspaces con codigo no deseado en sus triggers. Ej: $ ./processmaker check-workspace-disabled-code <myWorkspace> - Por defecto ProcessMaker no realiza el escaneo de codigo, si se desea escanear codigo no deseado, se debera definir el atributo "enable_blacklist = 1" en el archivo "env.ini", este atributo no se aplica a las nuevas opciones creadas para los comandos "./gulliver" y "./processmaker" - Para una configuracion personalizada de codigo no deseado (lista negra), se pueden definir las mismas en el archivo "path/to/processmaker/workflow/engine/config/blacklist.ini" (si no existe el archivo se puede crear), o tambien en el atributo "disable_functions" esto en el archivo "php.ini" Ejemplo de "blacklist.ini": ;Classes ;======= DashletInterface ;Functions ;========= eval exec ;date ;echo strlen
2014-11-19 16:47:22 -04:00
2017-08-01 12:16:06 -04:00
$response = $oPluginRegistry->verifyTranslation($details->getNamespace());
G::auditLog("InstallPlugin", "Plugin Name: " . $details->getNamespace());
2017-12-04 13:25:35 +00:00
G::header("Location: pluginsMain");
die();
} catch (Exception $e) {
$_SESSION['__PLUGIN_ERROR__'] = $e->getMessage();
2017-12-04 13:25:35 +00:00
G::header('Location: pluginsMain');
die();
}