mirror of
https://github.com/salesagility/SuiteCRM.git
synced 2024-11-21 23:47:57 +00:00
409 lines
16 KiB
PHP
Executable File
409 lines
16 KiB
PHP
Executable File
<?php
|
|
/**
|
|
*
|
|
* SugarCRM Community Edition is a customer relationship management program developed by
|
|
* SugarCRM, Inc. Copyright (C) 2004-2013 SugarCRM Inc.
|
|
*
|
|
* SuiteCRM is an extension to SugarCRM Community Edition developed by SalesAgility Ltd.
|
|
* Copyright (C) 2011 - 2018 SalesAgility Ltd.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it under
|
|
* the terms of the GNU Affero General Public License version 3 as published by the
|
|
* Free Software Foundation with the addition of the following permission added
|
|
* to Section 15 as permitted in Section 7(a): FOR ANY PART OF THE COVERED WORK
|
|
* IN WHICH THE COPYRIGHT IS OWNED BY SUGARCRM, SUGARCRM DISCLAIMS THE WARRANTY
|
|
* OF NON INFRINGEMENT OF THIRD PARTY RIGHTS.
|
|
*
|
|
* This program 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 Affero General Public License for more
|
|
* details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License along with
|
|
* this program; if not, see http://www.gnu.org/licenses or write to the Free
|
|
* Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
|
* 02110-1301 USA.
|
|
*
|
|
* You can contact SugarCRM, Inc. headquarters at 10050 North Wolfe Road,
|
|
* SW2-130, Cupertino, CA 95014, USA. or at email address contact@sugarcrm.com.
|
|
*
|
|
* The interactive user interfaces in modified source and object code versions
|
|
* of this program must display Appropriate Legal Notices, as required under
|
|
* Section 5 of the GNU Affero General Public License version 3.
|
|
*
|
|
* In accordance with Section 7(b) of the GNU Affero General Public License version 3,
|
|
* these Appropriate Legal Notices must retain the display of the "Powered by
|
|
* SugarCRM" logo and "Supercharged by SuiteCRM" logo. If the display of the logos is not
|
|
* reasonably feasible for technical reasons, the Appropriate Legal Notices must
|
|
* display the words "Powered by SugarCRM" and "Supercharged by SuiteCRM".
|
|
*/
|
|
|
|
require_once 'include/utils/sugar_file_utils.php';
|
|
|
|
#[\AllowDynamicProperties]
|
|
class UpgradeMetaHelper
|
|
{
|
|
public $upgrade_dir;
|
|
public $debug_mode;
|
|
public $upgrade_modules;
|
|
public $customized_modules;
|
|
public $source_dir;
|
|
public $dest_dir ;
|
|
public $evparser;
|
|
public $dvparser;
|
|
public $path_to_master_copy;
|
|
/**
|
|
* UpgradeMetaHelper
|
|
* This is the constructor for the UpgradeMetaHelper class
|
|
* @param $dir The root upgrade directory (where to copy working files)
|
|
* @param $masterCopyDirectory The root master directory (where SugarCRM 5.x files reside)
|
|
* @param $debugMode Debug mode, default is false
|
|
*
|
|
*/
|
|
public function __construct($dir='upgrade', $masterCopyDirecotry='modules_50', $debugMode = false)
|
|
{
|
|
$this->upgrade_dir = $dir;
|
|
$this->debug_mode = $debugMode;
|
|
$this->upgrade_modules = $this->getModifiedModules();
|
|
|
|
if ((is_countable($this->upgrade_modules) ? count($this->upgrade_modules) : 0) > 0) {
|
|
$_SESSION['Upgraded_Modules'] = $this->upgrade_modules;
|
|
$this->create_upgrade_directory();
|
|
$this->path_to_master_copy = $masterCopyDirecotry;
|
|
$this->runParser();
|
|
}
|
|
|
|
$this->customized_modules = $this->getAllCustomizedModulesBeyondStudio();
|
|
if ((is_countable($this->customized_modules) ? count($this->customized_modules) : 0) > 0) {
|
|
$_SESSION['Customized_Modules'] = $this->customized_modules;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* getModifiedModules
|
|
* This method returns a two-dimensional Array of Studio enabled
|
|
* modules that have been modified. The second Array index is an
|
|
* Array of files that do not match the md5 checksum for the module
|
|
* @return $return_array Two-dimensional Array of [module][modified file(s) Array]
|
|
*/
|
|
public function getModifiedModules()
|
|
{
|
|
$md5_string = array();
|
|
if (file_exists(clean_path(getcwd().'/files.md5'))) {
|
|
require(clean_path(getcwd().'/files.md5'));
|
|
}
|
|
|
|
$return_array = array();
|
|
$modules = $this->loadStudioModules();
|
|
foreach ($modules as $mod) {
|
|
$editView = "modules/$mod/EditView.html";
|
|
$detailView = "modules/$mod/DetailView.html";
|
|
$searchForm = "modules/$mod/SearchForm.html";
|
|
if (file_exists($editView) && isset($md5_string['./' . $editView])) {
|
|
$fileContents = file_get_contents($editView);
|
|
if (md5($fileContents) != $md5_string['./' . $editView]) {
|
|
$return_array[$mod][] = $editView;
|
|
}
|
|
}
|
|
|
|
if (file_exists($detailView) && isset($md5_string['./' . $detailView])) {
|
|
$fileContents = file_get_contents($detailView);
|
|
if (md5($fileContents) != $md5_string['./' . $detailView]) {
|
|
$return_array[$mod][] = $detailView;
|
|
}
|
|
}
|
|
|
|
if (file_exists($searchForm) && isset($md5_string['./' . $searchForm])) {
|
|
$fileContents = file_get_contents($searchForm);
|
|
if (md5($fileContents) != $md5_string['./' . $searchForm]) {
|
|
$return_array[$mod][] = $searchForm;
|
|
}
|
|
}
|
|
} //foreach
|
|
|
|
return $return_array;
|
|
}
|
|
|
|
public function saveMatchingFilesQueries($currStep, $value)
|
|
{
|
|
$file_queries = [];
|
|
$upgrade_progress_dir = sugar_cached('upgrades/temp');
|
|
if (!is_dir($upgrade_progress_dir) && !mkdir($upgrade_progress_dir) && !is_dir($upgrade_progress_dir)) {
|
|
throw new \RuntimeException(sprintf('Directory "%s" was not created', $upgrade_progress_dir));
|
|
}
|
|
$file_queries_file = $upgrade_progress_dir.'/files_queries.php';
|
|
if (file_exists($file_queries_file)) {
|
|
include($file_queries_file);
|
|
} else {
|
|
if (function_exists('sugar_fopen')) {
|
|
sugar_fopen($file_queries_file, 'w+');
|
|
} else {
|
|
fopen($file_queries_file, 'wb+');
|
|
}
|
|
}
|
|
if (!isset($files_queries) || $files_queries == null) {
|
|
$files_queries = array();
|
|
}
|
|
$files_queries[$currStep]=$value;
|
|
if (is_writable($file_queries_file) && write_array_to_file(
|
|
"file_queries",
|
|
$file_queries,
|
|
$file_queries_file
|
|
)) {
|
|
//writing to the file
|
|
}
|
|
}
|
|
|
|
public function getAllCustomizedModulesBeyondStudio()
|
|
{
|
|
require_once('modules/UpgradeWizard/uw_utils.php');
|
|
$md5_string = array();
|
|
if (file_exists(clean_path(getcwd().'/files.md5'))) {
|
|
require(clean_path(getcwd().'/files.md5'));
|
|
}
|
|
$return_array = array();
|
|
$modules = $this->loadStudioModules();
|
|
$modulesAll = getAllModules(); //keep all modules as well
|
|
$allOtherModules = array_diff($modulesAll, $modules);
|
|
foreach ($modules as $mod) {
|
|
if (!is_dir('modules/'.$mod)) {
|
|
continue;
|
|
}
|
|
$editView = "modules/$mod/EditView.html";
|
|
$detailView ="modules/$mod/DetailView.html";
|
|
$exclude_files[]=$editView;
|
|
$exclude_files[]=$detailView;
|
|
$allModFiles = array();
|
|
$allModFiles = findAllFiles('modules/'.$mod, $allModFiles);
|
|
foreach ($allModFiles as $file) {
|
|
//$file_md5_ref = str_replace(clean_path(getcwd()),'',$file);
|
|
if (file_exists($file) && !in_array($file, $exclude_files)) {
|
|
if (isset($md5_string['./'.$file])) {
|
|
$fileContents = file_get_contents($file);
|
|
if (md5($fileContents) != $md5_string['./'.$file]) {
|
|
$return_array[$mod][] = $file;
|
|
} else {
|
|
//keep in the array to be deleted later
|
|
$_SESSION['removeMd5MatchingFiles'][] = $file;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} //foreach
|
|
//also check out other non-studio modules by taking the difference between
|
|
//allMOdules and
|
|
foreach ($allOtherModules as $mod) {
|
|
if (!is_dir('modules/'.$mod)) {
|
|
continue;
|
|
}
|
|
$allModFiles = array();
|
|
$allModFiles = findAllFiles('modules/'.$mod, $allModFiles);
|
|
foreach ($allModFiles as $file) {
|
|
//$file_md5_ref = str_replace(clean_path(getcwd()),'',$file);
|
|
if (file_exists($file)) {
|
|
if (isset($md5_string['./'.$file])) {
|
|
$fileContents = file_get_contents($file);
|
|
if (md5($fileContents) == $md5_string['./'.$file]) {
|
|
$_SESSION['removeMd5MatchingFiles'][] = $file;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
//Also store in a file
|
|
//saveMatchingFilesQueries('removeMd5MatchingFiles',$_SESSION['removeMd5MatchingFiles']);
|
|
return $return_array;
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
* Get all the customized modules. Compare the file md5s with the base md5s
|
|
* If a file has been modified then put the module in the list of customized
|
|
* modules. Show the list in the preflight check UI.
|
|
*/
|
|
public function getAllCustomizedModules()
|
|
{
|
|
require_once('files.md5');
|
|
|
|
$return_array = array();
|
|
$modules = getAllModules();
|
|
foreach ($modules as $mod) {
|
|
//find all files in each module if the files have been modified
|
|
//as compared to the base version then add the module to the
|
|
//customized modules array
|
|
$modFiles = findAllFiles(clean_path(getcwd())."/modules/$mod", array());
|
|
foreach ($modFiles as $file) {
|
|
$fileContents = file_get_contents($file);
|
|
$file = str_replace(clean_path(getcwd()), '', (string) $file);
|
|
if ($md5_string['./' . $file]) {
|
|
if (md5($fileContents) != $md5_string['./' . $file]) {
|
|
//A file has been customized in the module. Put the module into the
|
|
// customized modules array.
|
|
echo 'Changed File'.$file;
|
|
$return_array[$mod];
|
|
break;
|
|
}
|
|
} else {
|
|
// This is a new file in user's version and indicates that module has been
|
|
//customized. Put the module in the customized array.
|
|
echo 'New File'.$file;
|
|
$return_array[$mod];
|
|
break;
|
|
}
|
|
}
|
|
} //foreach
|
|
|
|
return $return_array;
|
|
}
|
|
|
|
/**
|
|
* loadStudioModules
|
|
* This method returns an Array of all modules where a studio.php file is
|
|
* present in the metadata directory of the module
|
|
* @return $modules Array of modules that are studio enabled
|
|
*/
|
|
public function loadStudioModules()
|
|
{
|
|
$modules = array();
|
|
$d = dir('modules');
|
|
while ($e = $d->read()) {
|
|
if (substr($e, 0, 1) == '.' || !is_dir('modules/' . $e)) {
|
|
continue;
|
|
}
|
|
if (file_exists('modules/' . $e . '/metadata/studio.php')) {
|
|
$modules[] = $e;
|
|
}
|
|
}
|
|
return $modules;
|
|
}
|
|
|
|
|
|
/**
|
|
* runParser
|
|
* This method creates the EditView and DetailView parser intances
|
|
* and runs the parsing for the modules to upgrade
|
|
*
|
|
*/
|
|
public function runParser()
|
|
{
|
|
require_once('include/SugarFields/Parsers/EditViewMetaParser.php');
|
|
require_once('include/SugarFields/Parsers/DetailViewMetaParser.php');
|
|
require_once('include/SugarFields/Parsers/SearchFormMetaParser.php');
|
|
|
|
$this->evparser = new EditViewMetaParser();
|
|
$this->dvparser = new DetailViewMetaParser();
|
|
$this->svparser = new SearchFormMetaParser();
|
|
|
|
foreach ($this->upgrade_modules as $module_name=>$files) {
|
|
$this->parseFile($module_name, $files);
|
|
} //foreach
|
|
}
|
|
|
|
|
|
/**
|
|
* parseFile
|
|
* Hanldes parsing the files for given module where Studio or manual
|
|
* changes have been detected.
|
|
* @param $module_name The module to parse
|
|
* @param $files Array of files found to parse
|
|
*
|
|
*/
|
|
public function parseFile($module_name, $files)
|
|
{
|
|
global $beanList, $dictionary;
|
|
foreach ($files as $file) {
|
|
if (preg_match('/(EditView|DetailView|SearchForm|QuickCreate)(\.html|\.tpl)$/s', (string) $file, $matches)) {
|
|
$view = $matches[1];
|
|
|
|
switch ($view) {
|
|
case 'EditView': $parser = $this->evparser; break;
|
|
case 'DetailView': $parser = $this->dvparser; break;
|
|
case 'SearchForm': $parser = $this->svparser; break;
|
|
}
|
|
|
|
$lowerCaseView = $view == 'SearchForm' ? 'search' : strtolower($view);
|
|
|
|
include('modules/'.$module_name.'/vardefs.php');
|
|
$bean_name = $beanList[$module_name];
|
|
$newFile = $this->upgrade_dir.'/modules/'.$module_name.'/metadata/'.$lowerCaseView.'defs.php';
|
|
|
|
$bean_name = $bean_name == 'aCase' ? 'Case' : $bean_name;
|
|
sugar_file_put_contents($newFile, $parser->parse(
|
|
$file,
|
|
$dictionary[$bean_name]['fields'],
|
|
$module_name,
|
|
true,
|
|
$this->path_to_master_copy.'/modules/'.$module_name.'/metadata/'.$lowerCaseView.'defs.php'
|
|
));
|
|
} //if
|
|
} //foreach
|
|
}
|
|
|
|
|
|
/**
|
|
* create_upgrade_directory
|
|
* Creates a directory called upgrade to house all the modules that are studio enabled.
|
|
* Also creates subdirectories for all studio enabled modules.
|
|
*
|
|
*/
|
|
public function create_upgrade_directory()
|
|
{
|
|
$dir = $this->upgrade_dir . '/modules';
|
|
if (!file_exists($this->upgrade_dir) && !mkdir($concurrentDirectory = $this->upgrade_dir,
|
|
0755) && !is_dir($concurrentDirectory)) {
|
|
throw new \RuntimeException(sprintf('Directory "%s" was not created', $concurrentDirectory));
|
|
}
|
|
if (!file_exists($dir) && !mkdir($dir, 0755) && !is_dir($dir)) {
|
|
throw new \RuntimeException(sprintf('Directory "%s" was not created', $dir));
|
|
}
|
|
|
|
foreach ($this->upgrade_modules as $module => $files) {
|
|
if (!file_exists($dir . '/' . $module) && !mkdir($concurrentDirectory = $dir . '/' . $module,
|
|
0755) && !is_dir($concurrentDirectory)) {
|
|
throw new \RuntimeException(sprintf('Directory "%s" was not created', $concurrentDirectory));
|
|
}
|
|
if (!file_exists($dir . '/' . $module . '/metadata') && !mkdir($concurrentDirectory = $dir . '/' . $module . '/metadata',
|
|
0755) && !is_dir($concurrentDirectory)) {
|
|
throw new \RuntimeException(sprintf('Directory "%s" was not created', $concurrentDirectory));
|
|
}
|
|
|
|
foreach ($files as $file) {
|
|
if (file_exists($file) && is_file($file)) {
|
|
copy($file, $this->upgrade_dir . '/' . $file);
|
|
} //if
|
|
} //foreach
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* verifyMetaData
|
|
* This function does some quick checks to make sure the metadataFile at
|
|
* least has an Array panel
|
|
*
|
|
* @param $metadataFile The path to the metadata file to verify
|
|
* @param $module The module to verify
|
|
* @param $view The view (EditView or DetailView)
|
|
* @return boolean true if verification check is okay; false otherwise
|
|
*/
|
|
public function verifyMetaData($metadataFile, $module, $view)
|
|
{
|
|
if (!file_exists($metadataFile) || !is_file($metadataFile)) {
|
|
return false;
|
|
}
|
|
|
|
include($metadataFile);
|
|
|
|
if (isset($viewdefs) && isset($viewdefs[$module][$view]['panels']) && is_array($viewdefs[$module][$view]['panels'])) {
|
|
return true;
|
|
}
|
|
|
|
if (isset($searchdefs) && isset($searchdefs[$module]) && is_array($searchdefs[$module])) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|