printMetadata(false); } else { foreach ($workspaces as $workspace) { echo "\n"; passthru(PHP_BINARY . " processmaker info " . $workspace->name); } } } /** * We will upgrade the CONTENT table * If we apply the command for all workspaces, we will need to execute one by one by redefining the constants * @param string $args, workspaceName that we need to apply the upgrade-content * @param string $opts * * @return void */ function run_upgrade_content($args, $opts) { //Check if the command is executed by a specific workspace if (count($args) === 1) { upgradeContent($args, $opts); } else { $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { passthru(PHP_BINARY . ' processmaker upgrade-content ' . $workspace->name); } } } /** * This function will upgrade the CONTENT table for a workspace * This function is executed only for one workspace * @param array $args, workspaceName that we will to apply the command * @param array $opts, we can send additional parameters * * @return void */ function upgradeContent($args, $opts) { try { //Load the attributes for the workspace $arrayWorkspace = get_workspaces_from_args($args); //Loop, read all the attributes related to the one workspace $wsName = $arrayWorkspace[key($arrayWorkspace)]->name; Bootstrap::setConstantsRelatedWs($wsName); $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { try { G::outRes("Upgrading content for " . pakeColor::colorize($workspace->name, "INFO") . "\n"); $workspace->upgradeContent($workspace->name, true); } catch (Exception $e) { G::outRes("Errors upgrading content of workspace " . CLI::info($workspace->name) . ": " . CLI::error($e->getMessage()) . "\n"); } } } catch (Exception $e) { G::outRes(CLI::error($e->getMessage()) . "\n"); } } /** * We will repair the translation in the languages defined in the workspace * Verify if we need to execute an external program for each workspace * If we apply the command for all workspaces, we will need to execute one by one by redefining the constants * @param string $args, workspaceName that we need to apply the database-upgrade * @param string $opts * * @return void */ function run_translation_upgrade($args, $opts) { $noXml = array_key_exists('noxml', $opts) ? '--no-xml' : ''; $noMafe = array_key_exists('nomafe', $opts) ? '--no-mafe' : ''; if (!empty($noXml)) { $noMafe = ' ' . $noMafe; } //Check if the command is executed by a specific workspace if (count($args) === 1) { translation_upgrade($args, $opts); } else { $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { passthru(PHP_BINARY . ' processmaker translation-repair ' . $noXml . $noMafe . ' ' . $workspace->name); } } } /** * This function will regenerate the translation for a workspace * This function is executed only for one workspace * @param array $args, workspaceName that we will to apply the command * @param array $opts, noxml and nomafe flags * * @return void */ function translation_upgrade($args, $opts) { try { //Load the attributes for the workspace $arrayWorkspace = get_workspaces_from_args($args); //Loop, read all the attributes related to the one workspace $wsName = $arrayWorkspace[key($arrayWorkspace)]->name; Bootstrap::setConstantsRelatedWs($wsName); $workspaces = get_workspaces_from_args($args); $flagUpdateXml = (!array_key_exists('noxml', $opts)); $flagUpdateMafe = (!array_key_exists('nomafe', $opts)); foreach ($workspaces as $workspace) { try { G::outRes("Upgrading translation for " . pakeColor::colorize($workspace->name, "INFO") . "\n"); $workspace->upgradeTranslation($flagUpdateXml, $flagUpdateMafe); } catch (Exception $e) { G::outRes("Errors upgrading translation of workspace " . CLI::info($workspace->name) . ": " . CLI::error($e->getMessage()) . "\n"); } } } catch (Exception $e) { G::outRes(CLI::error($e->getMessage()) . "\n"); } } function run_cacheview_upgrade($args, $opts) { $filter = new InputFilter(); $opts = $filter->xssFilterHard($opts); $args = $filter->xssFilterHard($args); $workspaces = get_workspaces_from_args($args); $lang = array_key_exists("lang", $opts) ? $opts['lang'] : 'en'; foreach ($workspaces as $workspace) { try { G::outRes("Upgrading cache view for " . pakeColor::colorize($workspace->name, "INFO") . "\n"); $workspace->upgradeCacheView(true, false, $lang); } catch (Exception $e) { G::outRes("Errors upgrading cache view of workspace " . CLI::info($workspace->name) . ": " . CLI::error($e->getMessage()) . "\n"); } } } function run_plugins_database_upgrade($args, $opts) { $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { try { CLI::logging("Upgrading plugins database for " . CLI::info($workspace->name) . "\n"); $workspace->upgradePluginsDatabase(); } catch (Exception $e) { CLI::logging("Errors upgrading plugins database: " . CLI::error($e->getMessage())); } } } function run_database_export($args, $opts) { if (count($args) < 2) { throw new Exception("Please provide a workspace name and a directory for export"); } $workspace = new WorkspaceTools($args[0]); $workspace->exportDatabase($args[1]); } function run_database_import($args, $opts) { throw new Exception("Not implemented"); } /** * Check if we need to execute an external program for each workspace * If we apply the command for all workspaces we will need to execute one by one by redefining the constants * @param string $args, workspaceName that we need to apply the database-upgrade * * @return void */ function run_database_upgrade($args) { //Check if the command is executed by a specific workspace if (count($args) === 1) { database_upgrade($args); } else { $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { passthru(PHP_BINARY . ' processmaker database-upgrade ' . $workspace->name); } } } function run_migrate_new_cases_lists($args, $opts) { migrate_new_cases_lists("migrate", $args, $opts); } function run_migrate_counters($args, $opts) { migrate_counters("migrate", $args); } function run_migrate_list_unassigned($args, $opts) { migrate_list_unassigned("migrate", $args, $opts); } /** * This function is executed only by one workspace * @param array $args, workspaceName for to apply the database-upgrade * * @return void */ function database_upgrade($args) { // Sanitize parameters sent $filter = new InputFilter(); $args = $filter->xssFilterHard($args); // Load the attributes for the workspace $workspaces = get_workspaces_from_args($args); // Get the name of the first workspace $wsName = $workspaces[key($workspaces)]->name; // Initialize workspace values Bootstrap::setConstantsRelatedWs($wsName); // Print a informative message print_r("Upgrading database in " . pakeColor::colorize($wsName, "INFO") . "\n"); // Loop to update the databases of all workspaces foreach ($workspaces as $workspace) { try { $workspace->upgradeDatabase(); $workspace->close(); } catch (Exception $e) { G::outRes("> Error: " . CLI::error($e->getMessage()) . "\n"); } } } function delete_app_from_table($con, $tableName, $appUid, $col = "APP_UID") { $stmt = $con->createStatement(); $sql = "DELETE FROM " . $tableName . " WHERE " . $col . "='" . $appUid . "'"; $rs = $stmt->executeQuery($sql, ResultSet::FETCHMODE_NUM); } function run_drafts_clean($args, $opts) { echo "Cleaning drafts\n"; if (count($args) < 1) { throw new Exception("Please specify a workspace name"); } $workspace = $args[0]; if (!file_exists(PATH_DB . $workspace . '/db.php')) { throw new Exception('Could not find workspace ' . $workspace); } $allDrafts = false; if (count($args) < 2) { echo "Cases older them this much days will be deleted (ENTER for all): "; $days = rtrim(fgets(STDIN), "\n"); if ($days == "") { $allDrafts = true; } } else { $days = $args[1]; if (strcmp($days, "all") == 0) { $allDrafts = true; } } if (!$allDrafts && (!is_numeric($days) || intval($days) <= 0)) { throw new Exception("Days value is not valid: " . $days); } if ($allDrafts) { echo "Removing all drafts\n"; } else { echo "Removing drafts older than " . $days . " days\n"; } /* Load the configuration from the workspace */ require_once(PATH_DB . $workspace . '/db.php'); require_once(PATH_THIRDPARTY . 'propel/Propel.php'); PROPEL::Init(PATH_METHODS . 'dbConnections/rootDbConnections.php'); $con = Propel::getConnection("root"); $stmt = $con->createStatement(); if (!$allDrafts) { $dateSql = "AND DATE_SUB(CURDATE(),INTERVAL " . $days . " DAY) >= APP_CREATE_DATE"; } else { $dateSql = ""; } /* Search for all the draft cases */ $sql = "SELECT APP_UID FROM APPLICATION WHERE APP_STATUS='DRAFT'" . $dateSql; $appRows = $stmt->executeQuery($sql, ResultSet::FETCHMODE_ASSOC); /* Tables to remove the cases from */ $tables = array( "APPLICATION", "APP_DELEGATION", "APP_CACHE_VIEW", "APP_THREAD", "APP_DOCUMENT", "APP_EVENT", "APP_HISTORY", "APP_MESSAGE" ); echo "Found " . $appRows->getRecordCount() . " cases to remove"; foreach ($appRows as $row) { echo "."; $appUid = $row['APP_UID']; foreach ($tables as $table) { delete_app_from_table($con, $table, $appUid); } delete_app_from_table($con, "CONTENT", $appUid, "CON_ID"); if (file_exists(PATH_DB . $workspace . '/files/' . $appUid)) { echo "\nRemoving files from " . $appUid . "\n"; G::rm_dir(PATH_DB . $workspace . '/files/' . $appUid); } } echo "\n"; } function run_workspace_backup($args, $opts) { $workspaces = array(); if (sizeof($args) > 2) { $filename = array_pop($args); foreach ($args as $arg) { $workspaces[] = new WorkspaceTools($arg); } } elseif (sizeof($args) > 0) { $workspace = new WorkspaceTools($args[0]); $workspaces[] = $workspace; if (sizeof($args) == 2) { $filename = $args[1]; } else { $filename = "{$workspace->name}.tar"; } } else { throw new Exception("No workspace specified for backup"); } foreach ($workspaces as $workspace) { if (!$workspace->workspaceExists()) { throw new Exception("Workspace '{$workspace->name}' not found"); } } //If this is a relative path, put the file in the backups directory if (strpos($filename, "/") === false && strpos($filename, '\\') === false) { $filename = PATH_DATA . "backups/$filename"; } CLI::logging("Backing up to $filename\n"); $filesize = array_key_exists("filesize", $opts) ? $opts['filesize'] : -1; if ($filesize >= 0) { if (!Bootstrap::isLinuxOs()) { CLI::error("This is not a Linux enviroment, cannot use this filesize [-s] feature.\n"); return; } $multipleBackup = new MultipleFilesBackup($filename, $filesize); //if filesize is 0 the default size will be took //using new method foreach ($workspaces as $workspace) { $multipleBackup->addToBackup($workspace); } $multipleBackup->letsBackup(); } else { //ansient method to backup into one large file $backup = WorkspaceTools::createBackup($filename); foreach ($workspaces as $workspace) { $workspace->backup($backup); } } CLI::logging("\n"); WorkspaceTools::printSysInfo(); foreach ($workspaces as $workspace) { CLI::logging("\n"); $workspace->printMetadata(false); } } function run_workspace_restore($args, $opts) { if (sizeof($args) > 0) { $filename = $args[0]; G::verifyPath(PATH_DATA . 'upgrade', true); if (isset($args[1]) && strlen($args[1]) >= 30) { eprintln("Invalid workspace name, insert a maximum of 30 characters.", 'red'); return; } if (strpos($filename, "/") === false && strpos($filename, '\\') === false) { $filename = PATH_DATA . "backups/$filename"; if (!file_exists($filename) && substr_compare($filename, ".tar", -4, 4, true) != 0) { $filename .= ".tar"; } } $info = array_key_exists("info", $opts); $lang = array_key_exists("lang", $opts) ? $opts['lang'] : 'en'; $port = array_key_exists("port", $opts) ? $opts['port'] : ''; $optionMigrateHistoryData = [ /*----------------------------------********---------------------------------*/ 'keepDynContent' => array_key_exists('keep_dyn_content', $args) /*----------------------------------********---------------------------------*/ ]; if ($info) { WorkspaceTools::getBackupInfo($filename); } else { CLI::logging("Restoring from $filename\n"); $workspace = array_key_exists("workspace", $opts) ? $opts['workspace'] : null; $overwrite = array_key_exists("overwrite", $opts); $multiple = array_key_exists("multiple", $opts); $dstWorkspace = isset($args[1]) ? $args[1] : null; if (!empty($multiple)) { if (!Bootstrap::isLinuxOs()) { CLI::error("This is not a Linux enviroment, cannot use this multiple [-m] feature.\n"); return; } MultipleFilesBackup::letsRestore($filename, $workspace, $dstWorkspace, $overwrite); } else { $anotherExtention = ".*"; //if there are files with and extra extention: e.g. .tar.number $multiplefiles = glob($filename . $anotherExtention); // example: //shared/workflow_data/backups/myWorkspace.tar.* if (count($multiplefiles) > 0) { CLI::error("Processmaker found these files: .\n"); foreach ($multiplefiles as $index => $value) { CLI::logging($value . "\n"); } CLI::error("Please, you should use -m parameter to restore them.\n"); return; } WorkspaceTools::restore($filename, $workspace, $dstWorkspace, $overwrite, $lang, $port, $optionMigrateHistoryData); } } } else { throw new Exception("No workspace specified for restore"); } } /** * Migrating cases folders of the workspaces * * @param array $command * @param array $args */ function runStructureDirectories($command, $args) { $workspaces = get_workspaces_from_args($command); if (count($command) === 1) { try { $workspace = $workspaces[0]; CLI::logging(": " . CLI::info($workspace->name) . "\n"); $workspace->updateStructureDirectories($workspace->name); $workspace->close(); } catch (Exception $e) { CLI::logging("Errors upgrading workspace " . CLI::info($workspace->name) . ": " . CLI::error($e->getMessage()) . "\n"); } } else { $count = count($workspaces); $countWorkspace = 0; foreach ($workspaces as $index => $workspace) { $countWorkspace++; CLI::logging("Updating workspaces ($countWorkspace/$count)"); passthru(PHP_BINARY . " processmaker migrate-cases-folders " . $workspace->name); } } } function run_database_verify_consistency($args, $opts) { verifyAppCacheConsistency($args); } function run_database_verify_migration_consistency($args, $opts) { verifyMigratedDataConsistency($args); } function verifyAppCacheConsistency($args) { $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { verifyWorkspaceConsistency($workspace); } } function verifyWorkspaceConsistency($workspace) { $isConsistent = true; print_r("Verifying data in workspace " . pakeColor::colorize($workspace->name, "INFO") . "\n"); $inconsistentUsers = $workspace->hasMissingUsers(); $inconsistentTasks = $workspace->hasMissingTasks(); $inconsistentProcesses = $workspace->hasMissingProcesses(); $inconsistentDelegations = $workspace->hasMissingAppDelegations(); if ($inconsistentUsers || $inconsistentTasks || $inconsistentProcesses || $inconsistentDelegations) { $isConsistent = false; } return $isConsistent; } function verifyMigratedDataConsistency($args) { $workspaces = get_workspaces_from_args($args); $inconsistentRecords = 0; foreach ($workspaces as $workspace) { print_r("Verifying data in workspace " . pakeColor::colorize($workspace->name, "INFO") . "\n"); $lists = array( 'LIST_CANCELLED', 'LIST_COMPLETED', 'LIST_INBOX', 'LIST_PARTICIPATED_HISTORY', 'LIST_PARTICIPATED_LAST', 'LIST_MY_INBOX', 'LIST_UNASSIGNED', ); foreach ($lists as $list) { $inconsistentRecords += $workspace->verifyListData($list); } } return $inconsistentRecords; } function run_migrate_itee_to_dummytask($args, $opts) { $filter = new InputFilter(); $opts = $filter->xssFilterHard($opts); $args = $filter->xssFilterHard($args); $arrayWorkspace = get_workspaces_from_args($args); foreach ($arrayWorkspace as $workspace) { try { $ws = new WorkspaceTools($workspace->name); $res = $ws->migrateIteeToDummytask($workspace->name); } catch (Exception $e) { G::outRes("> Error: " . CLI::error($e->getMessage()) . "\n"); } } } /*----------------------------------********---------------------------------*/ /** * Check if we need to execute an external program for each workspace * If we apply the command for all workspaces we will need to execute one by one by redefining the constants * @param string $args, workspaceName that we need to apply the database-upgrade * @param string $opts * * @return void */ function run_check_workspace_disabled_code($args, $opts) { //Check if the command is executed by a specific workspace if (count($args) === 1) { check_workspace_disabled_code($args, $opts); } else { $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { passthru(PHP_BINARY . ' processmaker check-workspace-disabled-code ' . $workspace->name); } } } /** * This function is executed only by one workspace * Code Security Scanner related to the custom blacklist * * @param array $args, the specific actions must be: upgrade|check * @param array $opts, workspaceName for to apply the database-upgrade * * @return void */ function check_workspace_disabled_code($args, $opts) { try { //Load the attributes for the workspace $arrayWorkspace = get_workspaces_from_args($args); //Loop, read all the attributes related to the one workspace $wsName = $arrayWorkspace[key($arrayWorkspace)]->name; Bootstrap::setConstantsRelatedWs($wsName); foreach ($arrayWorkspace as $value) { $workspace = $value; if (!$workspace->pmLicensedFeaturesVerifyFeature("B0oWlBLY3hHdWY0YUNpZEtFQm5CeTJhQlIwN3IxMEkwaG4=")) { throw new Exception("Error: This command cannot be used because your license does not include it."); } echo "> Workspace: " . $workspace->name . "\n"; try { $arrayFoundDisabledCode = $workspace->getDisabledCode(); if (!empty($arrayFoundDisabledCode)) { $strFoundDisabledCode = ""; foreach ($arrayFoundDisabledCode as $value2) { $arrayProcessData = $value2; $strFoundDisabledCode .= ($strFoundDisabledCode != "") ? "\n" : ""; $strFoundDisabledCode .= " Process: " . $arrayProcessData["processTitle"] . "\n"; $strFoundDisabledCode .= " Triggers:\n"; foreach ($arrayProcessData["triggers"] as $value3) { $arrayTriggerData = $value3; $strCodeAndLine = ""; foreach ($arrayTriggerData["disabledCode"] as $key4 => $value4) { $strCodeAndLine .= (($strCodeAndLine != "") ? ", " : "") . $key4 . " (Lines " . implode(", ", $value4) . ")"; } $strFoundDisabledCode .= " - " . $arrayTriggerData["triggerTitle"] . ": " . $strCodeAndLine . "\n"; } } echo $strFoundDisabledCode . "\n"; } else { echo "The workspace it's OK\n\n"; } } catch (Exception $e) { G::outRes("Errors to check disabled code: " . CLI::error($e->getMessage()) . "\n\n"); } $workspace->close(); } echo "Done!\n\n"; } catch (Exception $e) { G::outRes(CLI::error($e->getMessage()) . "\n"); } } function migrate_new_cases_lists($command, $args, $opts) { $filter = new InputFilter(); $opts = $filter->xssFilterHard($opts); $args = $filter->xssFilterHard($args); $lang = array_key_exists("lang", $opts) ? $opts['lang'] : 'en'; $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { print_r("Upgrading database in " . pakeColor::colorize($workspace->name, "INFO") . "\n"); try { $workspace->migrateList(true, $lang); echo "> List tables are done\n"; } catch (Exception $e) { G::outRes("> Error: " . CLI::error($e->getMessage()) . "\n"); } } } function migrate_counters($command, $args) { $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { print_r("Regenerating counters in: " . pakeColor::colorize($workspace->name, "INFO") . "\n"); try { $workspace->migrateCounters($workspace->name, true); echo "> Counters are done\n"; } catch (Exception $e) { G::outRes("> Error: " . CLI::error($e->getMessage()) . "\n"); } } } function migrate_list_unassigned($command, $args) { $filter = new InputFilter(); $args = $filter->xssFilterHard($args); $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { print_r("Upgrading Unassigned List in" . pakeColor::colorize($workspace->name, "INFO") . "\n"); try { $workspace->runRegenerateListUnassigned(); echo "> Unassigned List is done\n"; } catch (Exception $e) { G::outRes("> Error: " . CLI::error($e->getMessage()) . "\n"); } } } /*----------------------------------********---------------------------------*/ /** * Check if we need to execute an external program for each workspace * If we apply the command for all workspaces we will need to execute one by one by redefining the constants * @param string $args, workspaceName that we need to apply the database-upgrade * @param string $opts, specify the language * * @return void */ function run_migrate_content($args, $opts) { //Check the additional parameters $lang = array_key_exists("lang", $opts) ? '--lang=' . $opts['lang'] : '--lang=' . SYS_LANG; //Check if the command is executed by a specific workspace if (count($args) === 1) { migrate_content($args, $opts); } else { $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { passthru(PHP_BINARY . ' processmaker migrate-content ' . $lang . ' ' . $workspace->name); } } } /** * This function is executed only by one workspace * @param array $args, workspaceName for to apply the migrate-content * @param array $opts, specify the language * * @return void */ function migrate_content($args, $opts) { $filter = new InputFilter(); $args = $filter->xssFilterHard($args); $workspaces = get_workspaces_from_args($args); $lang = array_key_exists("lang", $opts) ? $opts['lang'] : SYS_LANG; $start = microtime(true); //We defined the constants related the workspace $wsName = $workspaces[key($workspaces)]->name; Bootstrap::setConstantsRelatedWs($wsName); //Loop, read all the attributes related to the one workspace CLI::logging("> Optimizing content data...\n"); foreach ($workspaces as $workspace) { print_r('Regenerating content in: ' . pakeColor::colorize($workspace->name, 'INFO') . "\n"); CLI::logging("-> Regenerating content \n"); $workspace->migrateContentRun($lang); } $stop = microtime(true); CLI::logging("<*> Optimizing content data Process took " . ($stop - $start) . " seconds.\n"); } function run_migrate_self_service_value($args, $opts) { $filter = new InputFilter(); $args = $filter->xssFilterHard($args); $workspaces = get_workspaces_from_args($args); $start = microtime(true); CLI::logging("> Optimizing Self-Service data...\n"); foreach ($workspaces as $workspace) { print_r('Migrating records in: ' . pakeColor::colorize($workspace->name, 'INFO') . "\n"); CLI::logging("-> Migrating Self-Service records \n"); $workspace->migrateSelfServiceRecordsRun($workspace->name); } $stop = microtime(true); CLI::logging("<*> Migrating Self-Service records Process took " . ($stop - $start) . " seconds.\n"); } function run_migrate_indexing_acv($args, $opts) { $filter = new InputFilter(); $args = $filter->xssFilterHard($args); $workspaces = get_workspaces_from_args($args); $start = microtime(true); CLI::logging("> Migrating and populating indexing for avoiding the use of table APP_CACHE_VIEW...\n"); foreach ($workspaces as $workspace) { print_r('Indexing for APP_CACHE_VIEW: ' . pakeColor::colorize($workspace->name, 'INFO') . "\n"); $workspace->migratePopulateIndexingACV($workspace->name); } $stop = microtime(true); CLI::logging("<*> Migrating and populating indexing for avoiding the use of table APP_CACHE_VIEW process took " . ($stop - $start) . " seconds.\n"); } function run_migrate_plugin($args, $opts) { $workspaces = get_workspaces_from_args($args); //Check if the command is executed by a specific workspace /** @var WorkspaceTools $workspace */ if (count($workspaces) === 1) { $workspace = array_shift($workspaces); CLI::logging('Regenerating Singleton in: ' . pakeColor::colorize($workspace->name, 'INFO') . "\n"); $workspace->migrateSingleton($workspace->name); CLI::logging("-> Regenerating Singleton \n"); } else { CLI::logging("> Migrating and populating data...\n"); $start = microtime(true); foreach ($workspaces as $workspace) { passthru(PHP_BINARY . ' processmaker migrate-plugins-singleton-information ' . $workspace->name); } $stop = microtime(true); CLI::logging("<*> Migrating and populating data Singleton took " . ($stop - $start) . " seconds.\n"); } } /** * This method recursively finds all PHP files that reference the path PATH_DATA * incorrectly, which is caused by importing processes where the data directory * of ProcessMaker has different routes. Modified files are backed up with the * extension '.backup' in the same directory. * * @param array $args * @param array $opts * @throws Exception * @return void */ function regenerate_pmtable_classes($args, $opts) { if (sizeof($args) > 0) { $start = microtime(true); CLI::logging("> Updating generated class files for PM Tables...\n"); Bootstrap::setConstantsRelatedWs($args[0]); $workspaceTools = new WorkspaceTools($args[0]); $workspaceTools->fixReferencePathFiles(PATH_DATA_SITE . "classes", PATH_DATA); $stop = microtime(true); CLI::logging("<*> Updating generated class files for PM Tables took " . ($stop - $start) . " seconds.\n"); } else { throw new Exception("No workspace specified for updating generated class files."); } } /*----------------------------------********---------------------------------*/ /** * This method migrates data from APP_HISTORY table to APP_DATA_CHANGE_LOG table. * * @param array $args * @param string $opts */ function migrate_history_data($args, $opts) { if (count($args) === 1) { $start = microtime(true); CLI::logging("> Migrating history data...\n"); Bootstrap::setConstantsRelatedWs($args[0]); $workspaceTools = new WorkspaceTools($args[0]); $workspaceTools->migrateAppHistoryToAppDataChangeLog(true); $stop = microtime(true); CLI::logging("<*> Migrating history data took " . ($stop - $start) . " seconds.\n"); } else { $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { passthru(PHP_BINARY . ' processmaker migrate-history-data ' . $workspace->name); } } } /*----------------------------------********---------------------------------*/ /** * Will be clean the History of use from the table * Will be remove the DYN_CONTENT_HISTORY from APP_HISTORY * * @param array $args * @param array $opts * * @return void */ function run_clear_dyn_content_history_data($args, $opts) { $workspaces = get_workspaces_from_args($args); $start = microtime(true); CLI::logging("> Cleaning history data from APP_HISTORY...\n"); foreach ($workspaces as $workspace) { CLI::logging('Remove history of use: ' . pakeColor::colorize($workspace->name, 'INFO') . "\n"); $workspace->clearDynContentHistoryData(true); } $stop = microtime(true); CLI::logging("<*> Cleaning history data from APP_HISTORY process took " . ($stop - $start) . " seconds.\n"); } /** * Sync JSON definition of the Forms with Input Documents information * * @param array $args * @param array $opts * * @return void * @see workflow/engine/bin/tasks/cliWorkspaces.php CLI::taskRun() */ function run_sync_forms_with_info_from_input_documents($args, $opts) { if (count($args) === 1) { //This variable is not defined and does not involve its value in this //task, it is removed at the end of the method. $_SERVER['REQUEST_URI'] = ''; if (!defined('SYS_SKIN')) { $config = System::getSystemConfiguration(); define('SYS_SKIN', $config['default_skin']); } CLI::logging('Sync JSON definition of the Forms with Input Documents information from workspace: ' . pakeColor::colorize($args[0], 'INFO') . "\n"); $workspaceTools = new WorkspaceTools($args[0]); $workspaceTools->syncFormsWithInputDocumentInfo(); unset($_SERVER['REQUEST_URI']); } else { $workspaces = get_workspaces_from_args($args); foreach ($workspaces as $workspace) { passthru(PHP_BINARY . ' processmaker sync-forms-with-info-from-input-documents ' . $workspace->name); } } } /** * Remove the deprecated files * * @return void * @see workflow/engine/bin/tasks/cliWorkspaces.php CLI::taskRun() * @link https://wiki.processmaker.com/3.3/processmaker_command */ function remove_deprecated_files() { //The constructor requires an argument, so we send an empty value in order to use the class. $workspaceTools = new WorkspaceTools(''); $workspaceTools->removeDeprecatedFiles(); CLI::logging("<*> The deprecated files has been removed. \n"); } /** * This function review the queries for each workspace or for an specific workspace * * @param array $args * * @return void */ function run_check_queries_incompatibilities($args) { try { $workspaces = get_workspaces_from_args($args); if (count($args) === 1) { CLI::logging("> Workspace: " . $workspaces[0]->name . PHP_EOL); check_queries_incompatibilities($workspaces[0]->name); } else { foreach ($workspaces as $workspace) { passthru(PHP_BINARY . " processmaker check-queries-incompatibilities " . $workspace->name); } } echo "Done!\n\n"; } catch (Exception $e) { G::outRes(CLI::error($e->getMessage()) . "\n"); } } /** * Check for the incompatibilities in the queries for the specific workspace * * @param string $wsName */ function check_queries_incompatibilities($wsName) { Bootstrap::setConstantsRelatedWs($wsName); require_once(PATH_DB . $wsName . '/db.php'); System::initLaravel(); $query = Process::query()->select('PRO_UID', 'PRO_TITLE'); $processesToCheck = $query->get()->values()->toArray(); $obj = new MySQL57(); $resTriggers = $obj->checkIncompatibilityTriggers($processesToCheck); if (!empty($resTriggers)) { foreach ($resTriggers as $trigger) { echo ">> The \"" . $trigger['PRO_TITLE'] . "\" process has a trigger called: \"" . $trigger['TRI_TITLE'] . "\" that contains UNION queries. Review the code to discard incompatibilities with MySQL5.7." . PHP_EOL; } } else { echo ">> No MySQL 5.7 incompatibilities in triggers found for this workspace." . PHP_EOL; } $resDynaforms = $obj->checkIncompatibilityDynaforms($processesToCheck); if (!empty($resDynaforms)) { foreach ($resDynaforms as $dynaform) { echo ">> The \"" . $dynaform['PRO_TITLE'] . "\" process has a dynaform called: \"" . $dynaform['DYN_TITLE'] . "\" that contains UNION queries. Review the code to discard incompatibilities with MySQL5.7." . PHP_EOL; } } else { echo ">> No MySQL 5.7 incompatibilities in dynaforms found for this workspace." . PHP_EOL; } $resVariables = $obj->checkIncompatibilityVariables($processesToCheck); if (!empty($resVariables)) { foreach ($resVariables as $variable) { echo ">> The \"" . $variable['PRO_TITLE'] . "\" process has a variable called: \"" . $variable['VAR_NAME'] . "\" that contains UNION queries. Review the code to discard incompatibilities with MySQL5.7." . PHP_EOL; } } else { echo ">> No MySQL 5.7 incompatibilities in variables found for this workspace." . PHP_EOL; } } /** * This function obtains the connection parameters and passes them to the artisan. * All artisan options can be applied. For more information on artisan options use * php artisan --help * @param array $args */ function run_artisan($args) { $jobsManager = JobsManager::getSingleton()->init(); $workspace = $jobsManager->getOptionValueFromArguments($args, "--workspace"); if ($workspace !== false) { config(['system.workspace' => $workspace]); $sw = in_array($args[0], ['queue:work', 'queue:listen']); $tries = $jobsManager->getOptionValueFromArguments($args, "--tries"); if ($sw === true && $tries === false) { $tries = $jobsManager->getTries(); array_push($args, "--tries={$tries}"); } array_push($args, "--processmakerPath=" . PROCESSMAKER_PATH); $command = "artisan " . implode(" ", $args); CLI::logging("> {$command}\n"); passthru(PHP_BINARY . " {$command}"); } else { CLI::logging("> The --workspace option is undefined.\n"); } } /** * Add new font to be used in Documents generation (TinyMCE editor and TCPDF library for now) * * @param array $args * @param array $options */ function documents_add_font($args, $options) { try { // Validate the main required argument if (empty($args)) { throw new Exception('Please send the font filename.'); } // Load and initialize optional arguments and options $fontFileName = $args[0]; $fontFriendlyName = $args[1] ?? ''; $fontProperties = $args[2] ?? ''; $fontType = $options['font_type'] ?? 'TrueType'; $name = ''; // Check fonts path OutputDocument::checkTcPdfFontsPath(); // Check if the font file exist if (!file_exists(PATH_DATA . 'fonts' . PATH_SEP . $fontFileName)) { throw new Exception("Font '{$fontFileName}' not exists."); } // Check if the font file was already added if (OutputDocument::existTcpdfFont($fontFileName)) { throw new Exception("Font '{$fontFileName}' already added."); } // Check if the friendly font name is valid if (preg_match('/[^0-9A-Za-z ]/', $fontFriendlyName)) { throw new Exception('The friendly font name is using an incorrect format please use only letters, numbers and spaces.'); } // Check if the font type is valid if (!in_array($fontType, ['TrueType', 'TrueTypeUnicode'])) { throw new Exception("Font type '{$fontType}' is invalid."); } // Convert TTF file to the format required by TCPDF library $tcPdfFont = TCPDF_FONTS::addTTFfont(PATH_DATA . 'fonts' . PATH_SEP . $fontFileName, $fontType); // Check if the conversion was successful if ($tcPdfFont === false) { throw new Exception("The font file '{$fontFileName}' cannot be converted."); } // Include font definition, in order to use the variable $name require_once K_PATH_FONTS . $tcPdfFont . '.php'; // Build the font family name to be used in the styles $fontFamilyName = strtolower($name); $fontFamilyName = str_replace('-', ' ', $fontFamilyName); $fontFamilyName = str_replace(['bold', 'oblique', 'italic', 'regular'], '', $fontFamilyName); $fontFamilyName = trim($fontFamilyName); // Add new font $font = [ 'fileName' => $fontFileName, 'familyName' => $fontFamilyName, 'friendlyName' => !empty($fontFriendlyName) ? $fontFriendlyName : $fontFamilyName, 'properties' => $fontProperties ]; OutputDocument::addTcPdfFont($font); // Print finalization message CLI::logging("Font '{$fontFileName}' added successfully." . PHP_EOL . PHP_EOL); } catch (Exception $e) { // Display the error message CLI::logging($e->getMessage() . PHP_EOL . PHP_EOL); } } /** * Remove a font used in Documents generation (TinyMCE editor and TCPDF library for now) * * @param array $args */ function documents_remove_font($args) { try { // Validate the main required argument if (empty($args)) { throw new Exception('Please send the font filename.'); } // Load arguments $fontFileName = $args[0]; // Check fonts path OutputDocument::checkTcPdfFontsPath(); // Check if the font file exist if (!file_exists(PATH_DATA . 'fonts' . PATH_SEP . $fontFileName)) { throw new Exception("Font '{$fontFileName}' not exists."); } // Check if the font file was registered if (!OutputDocument::existTcpdfFont($fontFileName)) { throw new Exception("Font '{$fontFileName}' was not registered."); } // This line get the filename of the previous converted font $tcPdfFont = TCPDF_FONTS::addTTFfont(PATH_DATA . 'fonts' . PATH_SEP . $fontFileName); // Remove TCPDF font files $extensions = ['ctg.z', 'php', 'z']; foreach ($extensions as $extension) { if (file_exists(PATH_DATA . 'fonts' . PATH_SEP . 'tcpdf' . PATH_SEP . $tcPdfFont . '.' . $extension)) { unlink(PATH_DATA . 'fonts' . PATH_SEP . 'tcpdf' . PATH_SEP . $tcPdfFont . '.' . $extension); } } // Remove font OutputDocument::removeTcPdfFont($fontFileName); // Print finalization message CLI::logging("Font '{$fontFileName}' removed successfully." . PHP_EOL . PHP_EOL); } catch (Exception $e) { // Display the error message CLI::logging($e->getMessage() . PHP_EOL . PHP_EOL); } }