0byt3m1n1
Path:
/
data
/
applications
/
aps
/
typo3
/
4.5.5-0
/
standard
/
htdocs
/
typo3
/
sysext
/
install
/
mod
/
[
Home
]
File: class.tx_install.php
<?php /*************************************************************** * Copyright notice * * (c) 1999-2011 Kasper Skårhøj (kasperYYYY@typo3.com) * All rights reserved * * This script is part of the TYPO3 project. The TYPO3 project 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. * * The GNU General Public License can be found at * http://www.gnu.org/copyleft/gpl.html. * A copy is found in the textfile GPL.txt and important notices to the license * from the author is found in LICENSE.txt distributed with these scripts. * * * This script 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. * * This copyright notice MUST APPEAR in all copies of the script! ***************************************************************/ /** * Contains the class for the Install Tool * * $Id$ * * @author Kasper Skårhøj <kasperYYYY@typo3.com> * @author Ingmar Schlecht <ingmar@typo3.org> */ /** * [CLASS/FUNCTION INDEX of SCRIPT] * * * * 162: class tx_install extends t3lib_install * 234: function tx_install() * 318: function checkPassword() * 362: function loginForm() * 396: function init() * 574: function stepOutput() * 836: function checkTheConfig() * 867: function typo3conf_edit() * 976: function phpinformation() * * SECTION: typo3temp/ manager * 1079: function typo3TempManager() * 1199: function getSelectorOptions($deleteType,$tt='') * * SECTION: cleanup manager * 1231: function cleanupManager() * * SECTION: CONFIGURATION FORM * 1299: function generateConfigForm($type='') * 1367: function getDefaultConfigArrayComments($string,$mainArray=array(),$commentArray=array()) * * SECTION: CHECK CONFIGURATION FUNCTIONS * 1419: function checkConfiguration() * 1572: function check_mail($cmd='') * 1611: function checkExtensions() * 1673: function checkDirs() * 1762: function checkImageMagick($paths) * 1837: function _checkImageMagickGifCapability($path) * 1880: function _checkImageMagick_getVersion($file, $path) * 1915: function checkDatabase() * 1977: function setupGeneral($cmd='') * 2166: function writeToLocalconf_control($lines="", $showOutput=1) * 2190: function outputExitBasedOnStep($content) * 2206: function setLabelValueArray($arr,$type) * 2246: function getFormElement($labels,$values,$fieldName,$default,$msg='') * 2266: function getDatabaseList() * 2290: function setupGeneralCalculate() * 2387: function isTTF() * * SECTION: ABOUT the isXXX functions. * 2436: function isGD() * 2447: function isGIF() * 2459: function isJPG() * 2470: function isPNG() * 2482: function ImageTypes() * 2493: function getGDSoftwareInfo() * 2505: function generallyAboutConfiguration() * * SECTION: IMAGE processing * 2565: function checkTheImageProcessing() * 3046: function isExtensionEnabled($ext, $headCode, $short) * 3062: function displayTwinImage ($imageFile, $IMcommands=array(), $note='') * 3130: function getTwinImageMessage($message, $label_1="", $label_2='') * 3146: function formatImCmds($arr) * 3167: function imagemenu() * * SECTION: DATABASE analysing * 3209: function checkTheDatabase() * 3849: function updateWizard() * 3873: function updateWizard_parts($action) * 3987: function getUpgradeObjInstance($className, $identifier) * 4000: function isBackendAdminUser() * 4023: function isBasicComplete($tLabel) * 4063: function generateUpdateDatabaseForm($type, $arr_update, $arr_remove, $action_type) * 4094: function getUpdateDbFormWrap($action_type, $content, $label='Write to database') * 4107: function displayFields($arr, $pre=0, $label='') * 4132: function displayFieldComp($arr, $arr_db) * 4174: function displaySuggestions($arr, $excludeList='') * 4204: function compareDatabaseAndTCA($FDsrc, $TCA, $onlyFields=0) * 4262: function compareTCAandDatabase($TCA, $FDcomp) * 4296: function suggestFieldDefinition($fieldInfo) * 4373: function getItemArrayType($arr) * 4401: function getItemBlobSize($len) * 4412: function suggestTCAFieldDefinition($fieldName,$fieldInfo) * 4555: function includeTCA() * * SECTION: GENERAL FUNCTIONS * 4597: function linkIt($url,$link='') * 4611: function message($head, $short_string='', $long_string='', $type=0, $force=0) * 4632: function printSection($head, $short_string, $long_string, $type) * 4673: function fw($str,$size=1) * 4696: function fwheader($str) * 4707: function wrapInCells($label,$content) * 4716: function printAll() * 4735: function outputWrapper($content) * 4801: function menu() * 4823: function stepHeader() * 4865: function note123() * 4879: function endNotes() * 4912: function securityRisk() * 4930: function alterPasswordForm() * 4946: function messageBasicFinished() * 4968: function setScriptName($type) * 4981: function formWidth($size=48,$textarea=0,$styleOverride='') * 5002: function formWidthText($size=48,$styleOverride='',$wrap='') * 5018: function getBackupFilename($filename) * * TOTAL FUNCTIONS: 82 * (This index is automatically created/updated by the extension "extdeveval") * */ // include requirements definition: require_once(t3lib_extMgm::extPath('install') . 'requirements.php'); // include update classes require_once(t3lib_extMgm::extPath('install') . 'updates/class.tx_coreupdates_charsetdefaults.php'); require_once(t3lib_extMgm::extPath('install').'updates/class.tx_coreupdates_compatversion.php'); require_once(t3lib_extMgm::extPath('install').'updates/class.tx_coreupdates_cscsplit.php'); require_once(t3lib_extMgm::extPath('install').'updates/class.tx_coreupdates_notinmenu.php'); require_once(t3lib_extMgm::extPath('install').'updates/class.tx_coreupdates_mergeadvanced.php'); require_once(t3lib_extMgm::extPath('install').'updates/class.tx_coreupdates_installsysexts.php'); require_once(t3lib_extMgm::extPath('install').'updates/class.tx_coreupdates_imagescols.php'); require_once(t3lib_extMgm::extPath('install').'updates/class.tx_coreupdates_installnewsysexts.php'); require_once(t3lib_extMgm::extPath('install') . 'mod/class.tx_install_session.php'); require_once(t3lib_extMgm::extPath('install') . 'updates/class.tx_coreupdates_statictemplates.php'); require_once(t3lib_extMgm::extPath('install') . 'updates/class.tx_coreupdates_t3skin.php'); require_once(t3lib_extMgm::extPath('install') . 'updates/class.tx_coreupdates_compressionlevel.php'); require_once(t3lib_extMgm::extPath('install') . 'updates/class.tx_coreupdates_migrateworkspaces.php'); require_once(t3lib_extMgm::extPath('install') . 'updates/class.tx_coreupdates_flagsfromsprite.php'); require_once(t3lib_extMgm::extPath('install') . 'updates/class.tx_coreupdates_addflexformstoacl.php'); require_once(t3lib_extMgm::extPath('install') . 'updates/class.tx_coreupdates_imagelink.php'); /** * Install Tool module * * $Id$ * * @author Kasper Skårhøj <kasperYYYY@typo3.com> * @author Ingmar Schlecht <ingmar@typo3.org> * @package TYPO3 * @subpackage tx_install */ class tx_install extends t3lib_install { var $templateFilePath = 'typo3/sysext/install/Resources/Private/Templates/'; var $template; var $javascript; var $stylesheets; var $markers = array(); var $messages = array(); var $errorMessages = array(); var $mailMessage = ''; var $action = ''; // The url that calls this script var $scriptSelf = 'index.php'; // The url that calls this script var $updateIdentity = 'TYPO3 Install Tool'; var $headerStyle =''; var $setAllCheckBoxesByDefault=0; var $allowFileEditOutsite_typo3conf_dir=0; var $INSTALL =array(); // In constructor: is set to global GET/POST var TYPO3_INSTALL var $checkIMlzw = 0; // If set, lzw capabilities of the available ImageMagick installs are check by actually writing a gif-file and comparing size var $checkIM = 0; // If set, ImageMagick is checked. var $dumpImCommands=1; // If set, the image Magick commands are always outputted in the image processing checker var $mode = ''; // If set to "123" then only most vital information is displayed. var $step = 0; // If set to 1,2,3 or GO it signifies various functions. var $totalSteps = 4; // Can be changed by hook to define the total steps in 123 mode // internal var $passwordOK=0; // This is set, if the password check was ok. The function init() will exit if this is not set var $silent=1; // If set, the check routines don't add to the message-array var $sections=array(); // Used to gather the message information. var $fatalError=0; // This is set if some error occured that will definitely prevent TYpo3 from running. var $sendNoCacheHeaders=1; var $config_array = array( // Flags are set in this array if the options are available and checked ok. 'gd'=>0, 'gd_gif'=>0, 'gd_png'=>0, 'gd_jpg'=>0, 'freetype' => 0, 'safemode' => 0, 'dir_typo3temp' => 0, 'dir_temp' => 0, 'im_versions' => array(), 'im' => 0, 'sql.safe_mode_user' => '', 'mysqlConnect' => 0, 'no_database' => 0 ); var $typo3temp_path=''; /** * the session handling object * * @var tx_install_session */ protected $session = NULL; /** * the form protection instance used for creating and verifying form tokens * * @var t3lib_formprotection_InstallToolFormProtection */ protected $formProtection = NULL; var $menuitems = array( 'config' => 'Basic Configuration', 'database' => 'Database Analyser', 'update' => 'Upgrade Wizard', 'images' => 'Image Processing', 'extConfig' => 'All Configuration', 'cleanup' => 'Clean up', 'phpinfo' => 'phpinfo()', 'typo3conf_edit' => 'Edit files in typo3conf/', 'about' => 'About', 'logout' => 'Logout from Install Tool', ); // PHP modules which are required. Can be changed by hook in getMissingPhpModules() protected $requiredPhpModules = array( 'filter', 'gd', 'json', 'mysql', 'pcre', 'session', 'SPL', 'standard', 'xml', 'zlib' ); /** * Constructor * * @return void */ function tx_install() { parent::t3lib_install(); if (!$GLOBALS['TYPO3_CONF_VARS']['BE']['installToolPassword']) { $this->outputErrorAndExit('Install Tool deactivated.<br />You must enable it by setting a password in typo3conf/localconf.php. If you insert the line below, the password will be \'joh316\':<br /><br />$TYPO3_CONF_VARS[\'BE\'][\'installToolPassword\'] = \'bacb98acf97e0b6112b1d1b650b84971\';', 'Fatal error'); } if ($this->sendNoCacheHeaders) { header('Expires: Mon, 26 Jul 1997 05:00:00 GMT'); header('Last-Modified: ' . gmdate('D, d M Y H:i:s') . ' GMT'); header('Expires: 0'); header('Cache-Control: no-cache, must-revalidate'); header('Pragma: no-cache'); } // **************************** // Initializing incoming vars. // **************************** $this->INSTALL = t3lib_div::_GP('TYPO3_INSTALL'); $this->mode = t3lib_div::_GP('mode'); if ($this->mode !== '123') { $this->mode = ''; } if (t3lib_div::_GP('step') === 'go') { $this->step = 'go'; } else { $this->step = intval(t3lib_div::_GP('step')); } // Let DBAL decide whether to load itself $dbalLoaderFile = $this->backPath . 'sysext/dbal/class.tx_dbal_autoloader.php'; if (@is_file($dbalLoaderFile)) { include($dbalLoaderFile); } if ($this->mode === '123') { // Check for mandatory PHP modules $missingPhpModules = $this->getMissingPhpModules(); if (count($missingPhpModules) > 0) { throw new RuntimeException('TYPO3 Installation Error: The following PHP module(s) is/are missing: <em>' . implode(', ', $missingPhpModules) . '</em><br /><br />You need to install and enable these modules first to be able to install TYPO3.' ); } } $this->redirect_url = t3lib_div::sanitizeLocalUrl(t3lib_div::_GP('redirect_url')); $this->INSTALL['type'] = ''; if ($_GET['TYPO3_INSTALL']['type']) { $allowedTypes = array( 'config', 'database', 'update', 'images', 'extConfig', 'cleanup', 'phpinfo', 'typo3conf_edit', 'about', 'logout' ); if (in_array($_GET['TYPO3_INSTALL']['type'], $allowedTypes)) { $this->INSTALL['type'] = $_GET['TYPO3_INSTALL']['type']; } } if ($this->step == 4) { $this->INSTALL['type'] = 'database'; } // Hook to raise the counter for the total steps in the 1-2-3 installer if (is_array ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install/mod/class.tx_install.php']['additionalSteps'])) { foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install/mod/class.tx_install.php']['additionalSteps'] as $classData) { $hookObject = t3lib_div::getUserObj($classData); $this->totalSteps += (integer) $hookObject->executeAdditionalSteps($this); } } if ($this->mode=='123') { $tempItems = $this->menuitems; $this->menuitems = array( 'config' => $tempItems['config'], 'database' => $tempItems['database'] ); if (!$this->INSTALL['type'] || !isset($this->menuitems[$this->INSTALL['type']])) { $this->INSTALL['type'] = 'config'; } } else { if (!$this->INSTALL['type'] || !isset($this->menuitems[$this->INSTALL['type']])) { $this->INSTALL['type'] = 'about'; } } $this->action = $this->scriptSelf . '?TYPO3_INSTALL[type]=' . $this->INSTALL['type'] . ($this->mode? '&mode=' . $this->mode : '') . ($this->step? '&step=' . $this->step : ''); $this->typo3temp_path = PATH_site.'typo3temp/'; if (!is_dir($this->typo3temp_path) || !is_writeable($this->typo3temp_path)) { $this->outputErrorAndExit('Install Tool needs to write to typo3temp/. Make sure this directory is writeable by your webserver: ' . htmlspecialchars($this->typo3temp_path), 'Fatal error'); } try { $this->session = t3lib_div::makeInstance('tx_install_session'); } catch (Exception $exception) { $this->outputErrorAndExit($exception->getMessage()); } // ******************* // Check authorization // ******************* if (!$this->session->hasSession()) { $this->session->startSession(); } if ($this->session->isAuthorized() || $this->checkPassword()) { $this->passwordOK=1; $this->session->refreshSession(); $enableInstallToolFile = PATH_typo3conf . 'ENABLE_INSTALL_TOOL'; if (is_file ($enableInstallToolFile)) { // Extend the age of the ENABLE_INSTALL_TOOL file by one hour @touch($enableInstallToolFile); } if($this->redirect_url) { t3lib_utility_Http::redirect($this->redirect_url); } $this->formProtection = t3lib_formProtection_Factory::get( 't3lib_formprotection_InstallToolFormProtection' ); $this->formProtection->injectInstallTool($this); } else { $this->loginForm(); } } /** * Returns true if submitted password is ok. * * If password is ok, set session as "authorized". * * @return boolean true if the submitted password was ok and session was * authorized, false otherwise */ function checkPassword() { $p = t3lib_div::_GP('password'); if ($p && md5($p)==$GLOBALS['TYPO3_CONF_VARS']['BE']['installToolPassword']) { $this->session->setAuthorized(); // Sending warning email $wEmail = $GLOBALS['TYPO3_CONF_VARS']['BE']['warning_email_addr']; if ($wEmail) { $subject="Install Tool Login at '".$GLOBALS['TYPO3_CONF_VARS']['SYS']['sitename']."'"; $email_body="There has been a Install Tool login at TYPO3 site '".$GLOBALS['TYPO3_CONF_VARS']['SYS']['sitename']."' (".t3lib_div::getIndpEnv('HTTP_HOST').") from remote address '".t3lib_div::getIndpEnv('REMOTE_ADDR')."' (".t3lib_div::getIndpEnv('REMOTE_HOST').')'; mail($wEmail, $subject, $email_body, 'From: TYPO3 Install Tool WARNING <>' ); } return true; } else { // Bad password, send warning: if ($p) { $wEmail = $GLOBALS['TYPO3_CONF_VARS']['BE']['warning_email_addr']; if ($wEmail) { $subject="Install Tool Login ATTEMPT at '".$GLOBALS['TYPO3_CONF_VARS']['SYS']['sitename']."'"; $email_body="There has been an Install Tool login attempt at TYPO3 site '".$GLOBALS['TYPO3_CONF_VARS']['SYS']['sitename']."' (".t3lib_div::getIndpEnv('HTTP_HOST')."). The MD5 hash of the last 5 characters of the password tried was '".substr(md5($p), -5)."' REMOTE_ADDR was '".t3lib_div::getIndpEnv('REMOTE_ADDR')."' (".t3lib_div::getIndpEnv('REMOTE_HOST').')'; mail($wEmail, $subject, $email_body, 'From: TYPO3 Install Tool WARNING <>' ); } } return false; } } /** * Create the HTML for the login form * * Reads and fills the template. * Substitutes subparts when wrong password has been given * or the session has expired * * @return void */ function loginForm() { $password = t3lib_div::_GP('password'); $redirect_url = $this->redirect_url ? $this->redirect_url : $this->action; // Get the template file $templateFile = @file_get_contents( PATH_site . $this->templateFilePath . 'LoginForm.html' ); // Get the template part from the file $template = t3lib_parsehtml::getSubpart( $templateFile, '###TEMPLATE###' ); // Password has been given, but this form is rendered again. // This means the given password was wrong if (!empty($password)) { // Get the subpart for the wrong password $wrongPasswordSubPart = t3lib_parsehtml::getSubpart( $template, '###WRONGPASSWORD###' ); // Define the markers content $wrongPasswordMarkers = array( 'passwordMessage' => 'The password you just tried has this md5-value:', 'password' => md5($password) ); // Fill the markers in the subpart $wrongPasswordSubPart = t3lib_parsehtml::substituteMarkerArray( $wrongPasswordSubPart, $wrongPasswordMarkers, '###|###', TRUE, TRUE ); } // Session has expired if (!$this->session->isAuthorized() && $this->session->isExpired()) { // Get the subpart for the expired session message $sessionExpiredSubPart = t3lib_parsehtml::getSubpart( $template, '###SESSIONEXPIRED###' ); // Define the markers content $sessionExpiredMarkers = array( 'message' => 'Your Install Tool session has expired' ); // Fill the markers in the subpart $sessionExpiredSubPart = t3lib_parsehtml::substituteMarkerArray( $sessionExpiredSubPart, $sessionExpiredMarkers, '###|###', TRUE, TRUE ); } // Substitute the subpart for the expired session in the template $template = t3lib_parsehtml::substituteSubpart( $template, '###SESSIONEXPIRED###', $sessionExpiredSubPart ); // Substitute the subpart for the wrong password in the template $template = t3lib_parsehtml::substituteSubpart( $template, '###WRONGPASSWORD###', $wrongPasswordSubPart ); // Define the markers content $markers = array( 'siteName' => 'Site: ' . htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['SYS']['sitename']), 'headTitle' => 'Login to TYPO3 ' . TYPO3_version . ' Install Tool', 'redirectUrl' => htmlspecialchars($redirect_url), 'enterPassword' => 'Password', 'login' => 'Login', 'message' => ' <p class="typo3-message message-information"> The Install Tool Password is <em>not</em> the admin password of TYPO3. <br /> The default password is <em>joh316</em>. Be sure to change it! <br /><br /> If you don\'t know the current password, you can set a new one by setting the value of $TYPO3_CONF_VARS[\'BE\'][\'installToolPassword\'] in typo3conf/localconf.php to the md5() hash value of the password you desire. </p> ' ); // Fill the markers in the template $content = t3lib_parsehtml::substituteMarkerArray( $template, $markers, '###|###', TRUE, TRUE ); // Send content to the page wrapper function $this->output($this->outputWrapper($content)); } /** * Calling function that checks system, IM, GD, dirs, database * and lets you alter localconf.php * * This method is called from init.php to start the Install Tool. * * @return void */ function init() { // Must be called after inclusion of init.php (or from init.php) if (!defined('PATH_typo3')) exit; if (!$this->passwordOK) exit; // Setting stuff... $this->check_mail(); $this->setupGeneral(); $this->generateConfigForm(); if (count($this->messages)) { t3lib_div::debug($this->messages); } if ($this->step) { $this->output($this->outputWrapper($this->stepOutput())); } else { // Menu... switch($this->INSTALL['type']) { case 'images': $this->checkIM=1; $this->checkTheConfig(); $this->silent=0; $this->checkTheImageProcessing(); break; case 'database': $this->checkTheConfig(); $this->silent=0; $this->checkTheDatabase(); break; case 'update': $this->checkDatabase(); $this->silent=0; $this->updateWizard(); break; case 'config': $this->silent=0; $this->checkIM=1; $this->message( 'About configuration', 'How to configure TYPO3', $this->generallyAboutConfiguration() ); $isPhpCgi = (PHP_SAPI == 'fpm-fcgi' || PHP_SAPI == 'cgi' || PHP_SAPI == 'isapi' || PHP_SAPI == 'cgi-fcgi'); $this->message( 'System Information', 'Your system has the following configuration', ' <dl id="systemInformation"> <dt>OS detected:</dt> <dd>' . (TYPO3_OS == 'WIN' ? 'WIN' : 'UNIX') .'</dd> <dt>CGI detected:</dt> <dd>' . ($isPhpCgi ? 'YES' : 'NO') . '</dd> <dt>PATH_thisScript:</dt> <dd>' . PATH_thisScript . '</dd> </dl> ' ); $this->checkTheConfig(); $ext = 'Write config to localconf.php'; if ($this->fatalError) { if ( $this->config_array['no_database'] || !$this->config_array['mysqlConnect'] ) { $this->message($ext, 'Database not configured yet!', ' <p> You need to specify database username, password and host as one of the first things. <br /> Next you\'ll have to select a database to use with TYPO3. </p> <p> Use the form below. </p> ', 2); } else { $this->message($ext, 'Fatal error encountered!', ' <p> Somewhere above a fatal configuration problem is encountered. Please make sure that you\'ve fixed this error before you submit the configuration. TYPO3 will not run if this problem is not fixed! <br /> You should also check all warnings that may appear. </p> ', 2); } } elseif ($this->mode=='123') { if (!$this->fatalError) { $this->message($ext, 'Basic configuration completed', ' <p> You have no fatal errors in your basic configuration. You may have warnings though. Please pay attention to them! However you may continue and install the database. </p> <p> <strong> <span style="color:#f00;">Step 2: </span> </strong> <a href="' . $this->scriptSelf . '?TYPO3_INSTALL[type]=database' . ($this->mode ? '&mode=' . rawurlencode($this->mode) : '') . '">Click here to install the database.</a> </p> ', -1, 1); } } $this->message($ext, 'Very Important: Changing Image Processing settings', ' <p> When you change the settings for Image Processing you <em>must</em> take into account that <em>old images</em> may still be in typo3temp/ folder and prevent new files from being generated! <br /> This is especially important to know, if you\'re trying to set up image processing for the very first time. <br /> The problem is solved by <a href="' . htmlspecialchars($this->setScriptName('cleanup')) . '">clearing the typo3temp/ folder</a>. Also make sure to clear the cache_pages table. </p> ', 1, 1); $this->message($ext, 'Very Important: Changing Encryption Key setting', ' <p> When you change the setting for the Encryption Key you <em>must</em> take into account that a change to this value might invalidate temporary information, URLs etc. <br /> The problem is solved by <a href="' . htmlspecialchars($this->setScriptName('cleanup')) . '">clearing the typo3temp/ folder</a>. Also make sure to clear the cache_pages table. </p> ', 1, 1); $this->message($ext, 'Update localconf.php', ' <p> This form updates the localconf.php file with the suggested values you see below. The values are based on the analysis above. <br /> You can change the values in case you have alternatives to the suggested defaults. <br /> By this final step you will configure TYPO3 for immediate use provided that you have no fatal errors left above. </p>' . $this->setupGeneral('get_form') . ' ', 0, 1); $this->output($this->outputWrapper($this->printAll())); break; case 'extConfig': $this->silent=0; $this->generateConfigForm('get_form'); // Get the template file $templateFile = @file_get_contents( PATH_site . $this->templateFilePath . 'InitExtConfig.html' ); // Get the template part from the file $template = t3lib_parsehtml::getSubpart( $templateFile, '###TEMPLATE###' ); // Define the markers content $markers = array( 'action' => $this->action, 'content' => $this->printAll(), 'write' => 'Write to localconf.php', 'notice' => 'NOTICE:', 'explanation' => ' By clicking this button, localconf.php is updated with new values for the parameters listed above! ' ); // Fill the markers in the template $content = t3lib_parsehtml::substituteMarkerArray( $template, $markers, '###|###', TRUE, FALSE ); // Send content to the page wrapper function $this->output($this->outputWrapper($content)); break; case 'cleanup': $this->checkTheConfig(); $this->silent=0; $this->cleanupManager(); break; case 'phpinfo': $this->silent=0; $this->phpinformation(); break; case 'typo3conf_edit': $this->silent=0; $this->typo3conf_edit(); break; case 'logout': $enableInstallToolFile = PATH_site . 'typo3conf/ENABLE_INSTALL_TOOL'; if (is_file($enableInstallToolFile) && trim(file_get_contents($enableInstallToolFile)) !== 'KEEP_FILE') { unlink(PATH_typo3conf . 'ENABLE_INSTALL_TOOL'); } $this->formProtection->clean(); $this->session->destroySession(); t3lib_utility_Http::redirect($this->scriptSelf); break; case 'about': default: $this->silent=0; $this->message('About', 'Warning - very important!', $this->securityRisk().$this->alterPasswordForm(),2); $this->message('About', 'Using this script', ' <p> Installing TYPO3 has always been a hot topic on the mailing list and forums. Therefore we\'ve developed this tool which will help you through configuration and testing. <br /> There are three primary steps for you to take: </p> <p> <strong>1: Basic Configuration</strong> <br /> In this step your PHP-configuration is checked. If there are any settings that will prevent TYPO3 from running correctly you\'ll get warnings and errors with a description of the problem. <br /> You\'ll have to enter a database username, password and hostname. Then you can choose to create a new database or select an existing one. <br /> Finally the image processing settings are entered and verified and you can choose to let the script update the configuration file, typo3conf/localconf.php with the suggested settings. </p> <p> <strong>2: Database Analyser</strong> <br /> In this step you can either install a new database or update the database from any previous TYPO3 version. <br /> You can also get an overview of extra/missing fields/tables in the database compared to a raw sql-file. <br /> The database is also verified against your \'tables.php\' configuration ($TCA) and you can even see suggestions to entries in $TCA or new fields in the database. </p> <p> <strong>3: Upgrade Wizard</strong> <br /> Here you will find update methods taking care of changes to the TYPO3 core which are not backwards compatible. <br /> It is recommended to run this wizard after every update to make sure everything will still work flawlessly. </p> <p> <strong>4: Image Processing</strong> <br /> This step is a visual guide to verify your configuration of the image processing software. <br /> You\'ll be presented to a list of images that should all match in pairs. If some irregularity appears, you\'ll get a warning. Thus you\'re able to track an error before you\'ll discover it on your website. </p> <p> <strong>5: All Configuration</strong> <br /> This gives you access to any of the configuration options in the TYPO3_CONF_VARS array. Every option is also presented with a comment explaining what it does. </p> <p> <strong>6: Cleanup</strong> <br /> Here you can clean up the temporary files in typo3temp/ folder and the tables used for caching of data in your database. </p> '); $this->message('About', 'Why is this script stand-alone?', ' <p> You would think that this script should rather be a module in the backend and access-controlled to only admin-users from the database. But that\'s not how it works. <br /> The reason is, that this script must not be depending on the success of the configuration of TYPO3 and whether or not there is a working database behind. Therefore the script is invoked from the backend init.php file, which allows access if the constant \'TYPO3_enterInstallScript\' has been defined and is not false. That is and should be the case <em>only</em> when calling the script \'typo3/install/index.php\' - this script! </p> '); $headCode='Header legend'; $this->message($headCode, 'Notice!', ' <p> Indicates that something is important to be aware of. <br /> This does <em>not</em> indicate an error. </p> ', 1); $this->message($headCode, 'Just information', ' <p> This is a simple message with some information about something. </p> '); $this->message($headCode, 'Check was successful', ' <p> Indicates that something was checked and returned an expected result. </p> ', -1); $this->message($headCode, 'Warning!', ' <p> Indicates that something may very well cause trouble and you should definitely look into it before proceeding. <br /> This indicates a <em>potential</em> error. </p> ', 2); $this->message($headCode, 'Error!', ' <p> Indicates that something is definitely wrong and that TYPO3 will most likely not perform as expected if this problem is not solved. <br /> This indicates an actual error. </p> ', 3); $this->output($this->outputWrapper($this->printAll())); break; } } } /** * Controls the step 1-2-3-go process * * @return string The content to output to the screen */ function stepOutput() { // Get the template file $templateFile = @file_get_contents( PATH_site . $this->templateFilePath . 'StepOutput.html' ); // Get the template part from the file $template = t3lib_parsehtml::getSubpart( $templateFile, '###TEMPLATE###' ); // Define the markers content $markers = array( 'stepHeader' => $this->stepHeader(), 'notice' => 'Skip this wizard (for power users only)', 'skip123' => $this->scriptSelf ); $this->checkTheConfig(); $error_missingConnect = ' <p class="typo3-message message-error"> <strong> There is no connection to the database! </strong> <br /> (Username: <em>' . htmlspecialchars(TYPO3_db_username) . '</em>, Host: <em>' . htmlspecialchars(TYPO3_db_host) . '</em>, Using Password: YES) <br /> Go to Step 1 and enter a valid username and password! </p> '; $error_missingDB = ' <p class="typo3-message message-error"> <strong> There is no access to the database (<em>' . htmlspecialchars(TYPO3_db) . '</em>)! </strong> <br /> Go to Step 2 and select a valid database! </p> '; // only get the number of tables if it is not the first two steps in the 123-installer // (= no DB connection yet) $whichTables = ($this->step != 1 && $this->step != 2 ? $this->getListOfTables() : array()); $error_emptyDB = ' <p class="typo3-message message-error"> <strong> The database is still empty. There are no tables! </strong> <br /> Go to Step 3 and import a database! </p> '; // Hook to override and add steps to the 1-2-3 installer if (is_array ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install/mod/class.tx_install.php']['stepOutput'])) { foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install/mod/class.tx_install.php']['stepOutput'] as $classData) { $hookObject = t3lib_div::getUserObj($classData); $hookObject->executeStepOutput($markers, $this->step, $this); } } // Use the default steps when there is no override if (!$markers['header'] && !$markers['step']) { switch(strtolower($this->step)) { case 1: // Get the subpart for the first step $step1SubPart = t3lib_parsehtml::getSubpart( $templateFile, '###STEP1###' ); // Add header marker for main template $markers['header'] = 'Welcome to the TYPO3 Install Tool'; // Define the markers content for the subpart $step1SubPartMarkers = array( 'llIntroduction' => ' <p> TYPO3 is an enterprise content management system that is powerful, yet easy to install. </p> <p> In three simple steps you\'ll be ready to add content to your website. </p> ', 'step' => $this->step + 1, 'action' => htmlspecialchars($this->action), 'continue' => 'Continue' ); // Add step marker for main template $markers['step'] = t3lib_parsehtml::substituteMarkerArray( $step1SubPart, $step1SubPartMarkers, '###|###', TRUE, FALSE ); break; case 2: // Get the subpart for the second step $step2SubPart = t3lib_parsehtml::getSubpart( $templateFile, '###STEP2###' ); // Add header marker for main template $markers['header'] = 'Connect to your database host'; // Define the markers content for the subpart $step2SubPartMarkers = array( 'step' => $this->step + 1, 'action' => htmlspecialchars($this->action), 'encryptionKey' => $this->createEncryptionKey(), 'branch' => TYPO3_branch, 'labelUsername' => 'Username', 'username' => htmlspecialchars(TYPO3_db_username), 'labelPassword' => 'Password', 'password' => htmlspecialchars(TYPO3_db_password), 'labelHost' => 'Host', 'host' => TYPO3_db_host ? htmlspecialchars(TYPO3_db_host) : 'localhost', 'continue' => 'Continue', 'llDescription' => 'If you have not already created a username and password to access the database, please do so now. This can be done using tools provided by your host.' ); // Add step marker for main template $markers['step'] = t3lib_parsehtml::substituteMarkerArray( $step2SubPart, $step2SubPartMarkers, '###|###', TRUE, FALSE ); break; case 3: // Add header marker for main template $markers['header'] = 'Select database'; // There should be a database host connection at this point if ($result = $GLOBALS['TYPO3_DB']->sql_pconnect( TYPO3_db_host, TYPO3_db_username, TYPO3_db_password )) { // Get the subpart for the third step $step3SubPart = t3lib_parsehtml::getSubpart( $templateFile, '###STEP3###' ); // Get the subpart for the database options $step3DatabaseOptionsSubPart = t3lib_parsehtml::getSubpart( $step3SubPart, '###DATABASEOPTIONS###' ); $dbArr = $this->getDatabaseList(); $dbIncluded = 0; $step3DatabaseOptions = array(); foreach ($dbArr as $dbname) { // Define the markers content for database options $step3DatabaseOptionMarkers = array( 'databaseValue' => htmlspecialchars($dbname), 'databaseSelected' => $dbname == TYPO3_db ? 'selected="selected"' : '', 'databaseName' => htmlspecialchars($dbname) ); // Add the option HTML to an array $step3DatabaseOptions[] = t3lib_parsehtml::substituteMarkerArray( $step3DatabaseOptionsSubPart, $step3DatabaseOptionMarkers, '###|###', TRUE, TRUE ); if ($dbname==TYPO3_db) $dbIncluded=1; } if (!$dbIncluded && TYPO3_db) { // // Define the markers content when no access $step3DatabaseOptionMarkers = array( 'databaseValue' => htmlspecialchars(TYPO3_db), 'databaseSelected' => 'selected="selected"', 'databaseName' => htmlspecialchars(TYPO3_db) . ' (NO ACCESS!)' ); // Add the option HTML to an array $step3DatabaseOptions[] = t3lib_parsehtml::substituteMarkerArray( $step3DatabaseOptionsSubPart, $step3DatabaseOptionMarkers, '###|###', TRUE, TRUE ); } // Substitute the subpart for the database options $content = t3lib_parsehtml::substituteSubpart( $step3SubPart, '###DATABASEOPTIONS###', implode(LF, $step3DatabaseOptions) ); // Define the markers content $step3SubPartMarkers = array( 'step' => $this->step + 1, 'llOptions' => 'You have two options:', 'action' => htmlspecialchars($this->action), 'llOption1' => 'Create a new database (recommended):', 'llRemark1' => 'Enter a name for your TYPO3 database.', 'llOption2' => 'Select an EMPTY existing database:', 'llRemark2' => 'Any tables used by TYPO3 will be overwritten.', 'continue' => 'Continue' ); // Add step marker for main template $markers['step'] = t3lib_parsehtml::substituteMarkerArray( $content, $step3SubPartMarkers, '###|###', TRUE, TRUE ); } else { // Add step marker for main template when no connection $markers['step'] = $error_missingConnect; } break; case 4: // Add header marker for main template $markers['header'] = 'Import the Database Tables'; // There should be a database host connection at this point if ($result = $GLOBALS['TYPO3_DB']->sql_pconnect( TYPO3_db_host, TYPO3_db_username, TYPO3_db_password )) { // The selected database should be accessible if ($GLOBALS['TYPO3_DB']->sql_select_db(TYPO3_db)) { // Get the subpart for the fourth step $step4SubPart = t3lib_parsehtml::getSubpart( $templateFile, '###STEP4###' ); // Get the subpart for the database type options $step4DatabaseTypeOptionsSubPart = t3lib_parsehtml::getSubpart( $step4SubPart, '###DATABASETYPEOPTIONS###' ); $sFiles = t3lib_div::getFilesInDir(PATH_typo3conf,'sql',1,1); // Check if default database scheme "database.sql" already exists, otherwise create it if (!strstr(implode(',',$sFiles).',', '/database.sql,')) { array_unshift($sFiles,'Default TYPO3 Tables'); } $step4DatabaseTypeOptions = array(); foreach ($sFiles as $f) { if ($f=='Default TYPO3 Tables') $key='CURRENT_TABLES+STATIC'; else $key=htmlspecialchars($f); // Define the markers content for database type subpart $step4DatabaseTypeOptionMarkers = array( 'databaseTypeValue' => 'import|' . $key, 'databaseName' => htmlspecialchars(basename($f)) ); // Add the option HTML to an array $step4DatabaseTypeOptions[] = t3lib_parsehtml::substituteMarkerArray( $step4DatabaseTypeOptionsSubPart, $step4DatabaseTypeOptionMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the database type options $content = t3lib_parsehtml::substituteSubpart( $step4SubPart, '###DATABASETYPEOPTIONS###', implode(LF, $step4DatabaseTypeOptions) ); // Define the markers content $step4SubPartMarkers = array( 'llSummary' => 'Database summary:', 'llUsername' => 'Username:', 'username' => htmlspecialchars(TYPO3_db_username), 'llHost' => 'Host:', 'host' => htmlspecialchars(TYPO3_db_host), 'llDatabase' => 'Database:', 'database' => htmlspecialchars(TYPO3_db), 'llNumberTables' => 'Number of tables:', 'numberTables' => count($whichTables), 'action' => htmlspecialchars($this->action), 'llDatabaseType' => 'Select database contents:', 'label' => 'Import database' ); // Add step marker for main template $markers['step'] = t3lib_parsehtml::substituteMarkerArray( $content, $step4SubPartMarkers, '###|###', TRUE, TRUE ); } else { // Add step marker for main template when no database $markers['step'] = $error_missingDB; } } else { // Add step marker for main template when no connection $markers['step'] = $error_missingConnect; } break; case 'go': // Add header marker for main template $markers['header'] = 'Congratulations!'; // There should be a database host connection at this point if ($result = $GLOBALS['TYPO3_DB']->sql_pconnect( TYPO3_db_host, TYPO3_db_username, TYPO3_db_password )) { // The selected database should be accessible if ($GLOBALS['TYPO3_DB']->sql_select_db(TYPO3_db)) { // The database should contain tables if (count($whichTables)) { // Get the subpart for the go step $stepGoSubPart = t3lib_parsehtml::getSubpart( $templateFile, '###STEPGO###' ); // Define the markers content $stepGoSubPartMarkers = array( 'messageBasicFinished' => $this->messageBasicFinished(), 'llImportant' => 'Important Security Warning', 'securityRisk' => $this->securityRisk(), 'llSwitchMode' => ' <a href="' . $this->scriptSelf . '"> Change the Install Tool password here </a> ' ); // Add step marker for main template $markers['step'] = t3lib_parsehtml::substituteMarkerArray( $stepGoSubPart, $stepGoSubPartMarkers, '###|###', TRUE, TRUE ); } else { // Add step marker for main template when empty database $markers['step'] = $error_emptyDB; } } else { // Add step marker for main template when no database $markers['step'] = $error_missingDB; } } else { // Add step marker for main template when no connection $markers['step'] = $error_missingConnect; } break; } } // Fill the markers in the template $content = t3lib_parsehtml::substituteMarkerArray( $template, $markers, '###|###', TRUE, FALSE ); return $content; } /** * Calling the functions that checks the system * * @return void */ function checkTheConfig() { // Order important: $this->checkDirs(); $this->checkConfiguration(); $this->checkExtensions(); if (TYPO3_OS=='WIN') { $paths=array($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_path_lzw'], $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_path'], 'c:\\php\\imagemagick\\', 'c:\\php\\GraphicsMagick\\', 'c:\\apache\\ImageMagick\\', 'c:\\apache\\GraphicsMagick\\'); if (!isset($_SERVER['PATH'])) { $serverPath = array_change_key_case($_SERVER, CASE_UPPER); $paths = array_merge($paths, explode(';', $serverPath['PATH'])); } else { $paths = array_merge($paths, explode(';', $_SERVER['PATH'])); } } else { $paths=array($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_path_lzw'], $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_path'], '/usr/local/bin/','/usr/bin/','/usr/X11R6/bin/', '/opt/local/bin/'); $paths = array_merge($paths, explode(':', $_SERVER['PATH'])); } $paths = array_unique($paths); asort($paths); if (t3lib_utility_PhpOptions::isSafeModeEnabled()) { $paths=array(ini_get('safe_mode_exec_dir'),'/usr/local/php/bin/'); } if ($this->INSTALL['checkIM']['lzw']) { $this->checkIMlzw=1; } if ($this->INSTALL['checkIM']['path']) { $paths[]=trim($this->INSTALL['checkIM']['path']); } if ($this->checkIM) $this->checkImageMagick($paths); $this->checkDatabase(); } /** * Editing files in typo3conf directory (or elsewhere if enabled) * * @return void */ function typo3conf_edit() { // default: $EDIT_path = PATH_typo3conf; if ($this->allowFileEditOutsite_typo3conf_dir && $this->INSTALL['FILE']['EDIT_path']) { if (t3lib_div::validPathStr($this->INSTALL['FILE']['EDIT_path']) && substr($this->INSTALL['FILE']['EDIT_path'],-1)=='/') { $tmp_path = PATH_site.$this->INSTALL['FILE']['EDIT_path']; if (is_dir($tmp_path)) { $EDIT_path=$tmp_path; } else { $this->errorMessages[] = ' \'' . $tmp_path . '\' was not directory '; } } else { $this->errorMessages[] = ' Bad directory name (must be like t3lib/ or media/script/) '; } } $headCode = 'Edit files in '.basename($EDIT_path).'/'; $messages = ''; if ($this->INSTALL['SAVE_FILE']) { $save_to_file = $this->INSTALL['FILE']['name']; if (@is_file($save_to_file)) { $save_to_file_md5 = md5($save_to_file); if (isset($this->INSTALL['FILE'][$save_to_file_md5]) && t3lib_div::isFirstPartOfStr($save_to_file,$EDIT_path.'') && substr($save_to_file,-1)!='~' && !strstr($save_to_file,'_bak')) { $this->INSTALL['typo3conf_files'] = $save_to_file; $save_fileContent = $this->INSTALL['FILE'][$save_to_file_md5]; if ($this->INSTALL['FILE']['win_to_unix_br']) { $save_fileContent = str_replace(CRLF, LF, $save_fileContent); } $backupFile = $this->getBackupFilename($save_to_file); if ($this->INSTALL['FILE']['backup']) { if (@is_file($backupFile)) { unlink($backupFile); } rename($save_to_file,$backupFile); $messages .= ' Backup written to <strong>' . $backupFile . '</strong> <br /> '; } t3lib_div::writeFile($save_to_file,$save_fileContent); $messages .= ' File saved: <strong>' . $save_to_file . '</strong> <br /> MD5-sum: ' . $this->INSTALL['FILE']['prevMD5'] . ' (prev) <br /> MD5-sum: ' . md5($save_fileContent) . ' (new) <br /> '; } } } // Filelist: // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'Typo3ConfEdit.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Get the subpart for the files $filesSubpart = t3lib_parsehtml::getSubpart($template, '###FILES###'); $files = array(); $typo3conf_files = t3lib_div::getFilesInDir($EDIT_path,'',1,1); $fileFound = 0; foreach ($typo3conf_files as $k => $file) { // Delete temp_CACHED files if option is set if ( $this->INSTALL['delTempCached'] && preg_match('|/temp_CACHED_[a-z0-9_]+\.php|', $file)) { unlink($file); continue; } if ($this->INSTALL['typo3conf_files'] && !strcmp($this->INSTALL['typo3conf_files'],$file)) { $fileFound = 1; } // Define the markers content for the files subpart $filesMarkers = array( 'editUrl' => $this->action . '&TYPO3_INSTALL[typo3conf_files]=' . rawurlencode($file) . ($this->allowFileEditOutsite_typo3conf_dir ? '&TYPO3_INSTALL[FILE][EDIT_path]=' . rawurlencode($this->INSTALL['FILE']['EDIT_path']) : '') . '#confEditFileList', 'fileName' => basename($file), 'fileSize' => t3lib_div::formatSize(filesize($file)), 'class' => $this->INSTALL['typo3conf_files'] && !strcmp($this->INSTALL['typo3conf_files'], $file) ? 'class="act"' : '' ); // Fill the markers in the subpart $files[] = t3lib_parsehtml::substituteMarkerArray( $filesSubpart, $filesMarkers, '###|###', TRUE, FALSE ); } if ($fileFound && @is_file($this->INSTALL['typo3conf_files'])) { $backupFile = $this->getBackupFilename($this->INSTALL['typo3conf_files']); $fileContent = t3lib_div::getUrl($this->INSTALL['typo3conf_files']); // Get the subpart to edit the files $fileEditTemplate = t3lib_parsehtml::getSubpart($template, '###FILEEDIT###'); $allowFileEditOutsideTypo3ConfDirSubPart = ''; if (substr($this->INSTALL['typo3conf_files'], -1) != '~' && !strstr($this->INSTALL['typo3conf_files'], '_bak')) { // Get the subpart to show the save button $showSaveButtonSubPart = t3lib_parsehtml::getSubpart($fileEditTemplate, '###SHOWSAVEBUTTON###'); } if ($this->allowFileEditOutsite_typo3conf_dir) { // Get the subpart to show if files are allowed outside the directory typo3conf $allowFileEditOutsideTypo3ConfDirSubPart = t3lib_parsehtml::getSubpart($fileEditTemplate, '###ALLOWFILEEDITOUTSIDETYPO3CONFDIR###'); } // Substitute the subpart for the save button $fileEditContent = t3lib_parsehtml::substituteSubpart( $fileEditTemplate, '###SHOWSAVEBUTTON###', $showSaveButtonSubPart ); // Substitute the subpart to show if files are allowed outside the directory typo3conf $fileEditContent = t3lib_parsehtml::substituteSubpart( $fileEditContent, '###ALLOWFILEEDITOUTSIDETYPO3CONFDIR###', $allowFileEditOutsideTypo3ConfDirSubPart ); // Define the markers content for subpart to edit the files $fileEditMarkers = array( 'messages' => !empty($messages) ? '<p class="typo3-message message-warning">' . $messages . '</p>' : '', 'action' => $this->action . '#fileEditHeader', 'saveFile' => 'Save file', 'close' => 'Close', 'llEditing' => 'Editing file:', 'file' => $this->INSTALL['typo3conf_files'], 'md5Sum' => 'MD5-sum: ' . md5($fileContent), 'fileName' => $this->INSTALL['typo3conf_files'], 'fileEditPath' => $this->INSTALL['FILE']['EDIT_path'], 'filePreviousMd5' => md5($fileContent), 'fileMd5' => md5($this->INSTALL['typo3conf_files']), 'fileContent' => t3lib_div::formatForTextarea($fileContent), 'winToUnixBrChecked' => TYPO3_OS == 'WIN' ? '' : 'checked="checked"', 'winToUnixBr' => 'Convert Windows linebreaks (13-10) to Unix (10)', 'backupChecked' => @is_file($backupFile) ? 'checked="checked"' : '', 'backup' => 'Make backup copy (rename to ' . basename($backupFile) . ')' ); // Fill the markers in the subpart to edit the files $fileEditContent = t3lib_parsehtml::substituteMarkerArray( $fileEditContent, $fileEditMarkers, '###|###', TRUE, FALSE ); } if ($this->allowFileEditOutsite_typo3conf_dir) { // Get the subpart to show if files are allowed outside the directory typo3conf $allowFileEditOutsideTypo3ConfDirSubPart = t3lib_parsehtml::getSubpart($template, '###ALLOWFILEEDITOUTSIDETYPO3CONFDIR###'); // Define the markers content $allowFileEditOutsideTypo3ConfDirMarkers = array( 'action' => $this->action, 'pathSite' => PATH_site, 'editPath' => $this->INSTALL['FILE']['EDIT_path'], 'set' => 'Set' ); // Fill the markers in the subpart $allowFileEditOutsideTypo3ConfDirSubPart = t3lib_parsehtml::substituteMarkerArray( $allowFileEditOutsideTypo3ConfDirSubPart, $allowFileEditOutsideTypo3ConfDirMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart to edit the file $fileListContent = t3lib_parsehtml::substituteSubpart( $template, '###FILEEDIT###', $fileEditContent ); // Substitute the subpart when files can be edited outside typo3conf directory $fileListContent = t3lib_parsehtml::substituteSubpart( $fileListContent, '###ALLOWFILEEDITOUTSIDETYPO3CONFDIR###', $allowFileEditOutsideTypo3ConfDirSubPart ); // Substitute the subpart for the files $fileListContent = t3lib_parsehtml::substituteSubpart( $fileListContent, '###FILES###', implode(LF, $files) ); // Define the markers content $fileListMarkers = array( 'editPath' => '(' . $EDIT_path . ')', 'deleteTempCachedUrl' => $this->action . '&TYPO3_INSTALL[delTempCached]=1', 'deleteTempCached' => 'Delete temp_CACHED* files' ); // Fill the markers $fileListContent = t3lib_parsehtml::substituteMarkerArray( $fileListContent, $fileListMarkers, '###|###', TRUE, FALSE ); // Add the content to the message array $this->message($headCode, 'Files in folder', $fileListContent); // Output the page $this->output($this->outputWrapper($this->printAll())); } /** * Outputs system information * * @return void */ function phpinformation() { $headCode = 'PHP information'; $sVar = t3lib_div::getIndpEnv('_ARRAY'); $sVar['CONST: PHP_OS']=PHP_OS; $sVar['CONST: TYPO3_OS']=TYPO3_OS; $sVar['CONST: PATH_thisScript']=PATH_thisScript; $sVar['CONST: php_sapi_name()']=PHP_SAPI; $sVar['OTHER: TYPO3_VERSION']=TYPO3_version; $sVar['OTHER: PHP_VERSION']=phpversion(); $sVar['imagecreatefromgif()']=function_exists('imagecreatefromgif'); $sVar['imagecreatefrompng()']=function_exists('imagecreatefrompng'); $sVar['imagecreatefromjpeg()']=function_exists('imagecreatefromjpeg'); $sVar['imagegif()']=function_exists('imagegif'); $sVar['imagepng()']=function_exists('imagepng'); $sVar['imagejpeg()']=function_exists('imagejpeg'); $sVar['imagettftext()']=function_exists('imagettftext'); $sVar['OTHER: IMAGE_TYPES']=function_exists('imagetypes') ? imagetypes() : 0; $sVar['OTHER: memory_limit']=ini_get('memory_limit'); $gE_keys = explode(',','SERVER_PORT,SERVER_SOFTWARE,GATEWAY_INTERFACE,SCRIPT_NAME,PATH_TRANSLATED'); foreach ($gE_keys as $k) { $sVar['SERVER: '.$k]=$_SERVER[$k]; } $gE_keys = explode(',','image_processing,gdlib,gdlib_png,im,im_path,im_path_lzw,im_version_5,im_negate_mask,im_imvMaskState,im_combine_filename'); foreach ($gE_keys as $k) { $sVar['T3CV_GFX: '.$k]=$GLOBALS['TYPO3_CONF_VARS']['GFX'][$k]; } $debugInfo = array( '### DEBUG SYSTEM INFORMATION - START ###' ); foreach ($sVar as $kkk => $vvv) { $debugInfo[]=str_pad(substr($kkk,0,20),20).': '.$vvv; } $debugInfo[]='### DEBUG SYSTEM INFORMATION - END ###'; // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'PhpInformation.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Define the markers content $markers = array( 'explanation' => 'Please copy/paste the information from this text field into an email or bug-report as "Debug System Information" whenever you wish to get support or report problems. This information helps others to check if your system has some obvious misconfiguration and you\'ll get your help faster!', 'debugInfo' => t3lib_div::formatForTextarea(implode(LF, $debugInfo)) ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $template, $markers, '###|###', TRUE, FALSE ); // Add the content to the message array $this->message($headCode, 'DEBUG information', $content); // Start with various server information $getEnvArray = array(); $gE_keys = explode(',','QUERY_STRING,HTTP_ACCEPT,HTTP_ACCEPT_ENCODING,HTTP_ACCEPT_LANGUAGE,HTTP_CONNECTION,HTTP_COOKIE,HTTP_HOST,HTTP_USER_AGENT,REMOTE_ADDR,REMOTE_HOST,REMOTE_PORT,SERVER_ADDR,SERVER_ADMIN,SERVER_NAME,SERVER_PORT,SERVER_SIGNATURE,SERVER_SOFTWARE,GATEWAY_INTERFACE,SERVER_PROTOCOL,REQUEST_METHOD,SCRIPT_NAME,PATH_TRANSLATED,HTTP_REFERER,PATH_INFO'); foreach ($gE_keys as $k) { $getEnvArray[$k] = getenv($k); } $this->message($headCode, 't3lib_div::getIndpEnv()', $this->viewArray(t3lib_div::getIndpEnv('_ARRAY'))); $this->message($headCode, 'getenv()', $this->viewArray($getEnvArray)); $this->message($headCode, '_ENV', $this->viewArray($_ENV)); $this->message($headCode, '_SERVER', $this->viewArray($_SERVER)); $this->message($headCode, '_COOKIE', $this->viewArray($_COOKIE)); $this->message($headCode, '_GET', $this->viewArray($_GET)); // Start with the phpinfo() part ob_start(); phpinfo(); $contents = explode('<body>',ob_get_contents()); ob_end_clean(); $contents = explode('</body>',$contents[1]); // Do code cleaning: phpinfo() is not XHTML1.1 compliant $phpinfo = str_replace('<font', '<span', $contents[0]); $phpinfo = str_replace('</font', '</span', $phpinfo); $phpinfo = str_replace('<img border="0"', '<img', $phpinfo); $phpinfo = str_replace('<a name=', '<a id=', $phpinfo); // Add phpinfo() to the message array $this->message($headCode, 'phpinfo()', ' <div class="phpinfo"> ' . $phpinfo . ' </div> '); // Output the page $this->output($this->outputWrapper($this->printAll())); } /******************************* * * cleanup manager * *******************************/ /** * Provides a tool cleaning up various tables in the database * * @return void * @author Robert Lemke <rl@robertlemke.de> * @todo Add more functionality ... */ function cleanupManager() { $headCode = 'Clean up your TYPO3 installation'; $this->message($headCode, 'Database cache tables', ' <p> <strong>Clear cached image sizes</strong> <br /> Clears the cache used for memorizing sizes of all images used in your website. This information is cached in order to gain performance and will be stored each time a new image is being displayed in the frontend. </p> <p> You should <em>Clear All Cache</em> in the backend after clearing this cache. </p> '); $tables = $this->getListOfTables(); $action = $this->INSTALL['cleanup_type']; if (($action == 'cache_imagesizes' || $action == 'all') && isset ($tables['cache_imagesizes'])) { $GLOBALS['TYPO3_DB']->exec_TRUNCATEquery('cache_imagesizes'); } $cleanupType = array ( 'all' => 'Clean up everything', ); // Get cache_imagesizes info if (isset ($tables['cache_imagesizes'])) { $cleanupType['cache_imagesizes'] = 'Clear cached image sizes only'; $cachedImageSizesCounter = intval($GLOBALS['TYPO3_DB']->exec_SELECTcountRows('*', 'cache_imagesizes')); } else { $this->message($headCode, 'Table cache_imagesizes does not exist!', ' <p> The table cache_imagesizes was not found. Please check your database settings in Basic Configuration and compare your table definition with the Database Analyzer. </p> ', 2); $cachedImageSizesCounter = 'unknown'; } // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'CleanUpManager.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Get the subpart for the 'Clean up' dropdown $cleanUpOptionsSubpart = t3lib_parsehtml::getSubpart($template, '###CLEANUPOPTIONS###'); $cleanUpOptions = array(); foreach ($cleanupType as $cleanUpKey => $cleanUpValue) { // Define the markers content $cleanUpMarkers = array( 'value' => htmlspecialchars($cleanUpKey), 'data' => htmlspecialchars($cleanUpValue) ); // Fill the markers in the subpart $cleanUpOptions[] = t3lib_parsehtml::substituteMarkerArray( $cleanUpOptionsSubpart, $cleanUpMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the 'Clean up' dropdown $content = t3lib_parsehtml::substituteSubpart( $template, '###CLEANUPOPTIONS###', implode(LF, $cleanUpOptions) ); // Define the markers content $markers = array( 'numberCached' => 'Number cached image sizes:', 'number' => $cachedImageSizesCounter, 'action' => $this->action, 'cleanUp' => 'Clean up', 'execute' => 'Execute' ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $content, $markers, '###|###', TRUE, FALSE ); // Add the content to the message array $this->message($headCode, 'Statistics', $content, 1); $this->message($headCode, 'typo3temp/ folder', ' <p> TYPO3 uses this directory for temporary files, mainly processed and cached images. <br /> The filenames are very cryptic; They are unique representations of the file properties made by md5-hashing a serialized array with information. <br /> Anyway this directory may contain many thousand files and a lot of them may be of no use anymore. </p> <p> With this test you can delete the files in this folder. When you do that, you should also clear the cache database tables afterwards. </p> '); if (!$this->config_array['dir_typo3temp']) { $this->message('typo3temp/ directory', 'typo3temp/ not writable!', ' <p> You must make typo3temp/ write enabled before you can proceed with this test. </p> ', 2); $this->output($this->outputWrapper($this->printAll())); return; } // Run through files $fileCounter = 0; $deleteCounter = 0; $criteriaMatch = 0; $tmap=array('day'=>1, 'week'=>7, 'month'=>30); $tt = $this->INSTALL['typo3temp_delete']; $subdir = $this->INSTALL['typo3temp_subdir']; if (strlen($subdir) && !preg_match('/^[[:alnum:]_]+\/$/',$subdir)) die('subdir "'.$subdir.'" was not allowed!'); $action = $this->INSTALL['typo3temp_action']; $d = @dir($this->typo3temp_path.$subdir); if (is_object($d)) { while($entry=$d->read()) { $theFile = $this->typo3temp_path.$subdir.$entry; if (@is_file($theFile)) { $ok = 0; $fileCounter++; if ($tt) { if (t3lib_div::testInt($tt)) { if (filesize($theFile) > $tt*1024) $ok=1; } else { if (fileatime($theFile) < $GLOBALS['EXEC_TIME'] - (intval($tmap[$tt]) * 60 * 60 * 24)) { $ok = 1; } } } else { $ok = 1; } if ($ok) { $hashPart=substr(basename($theFile),-14,10); // This is a kind of check that the file being deleted has a 10 char hash in it if (!preg_match('/[^a-f0-9]/',$hashPart) || substr($theFile,-6)==='.cache' || substr($theFile,-4)==='.tbl' || substr(basename($theFile),0,8)==='install_') { if ($action && $deleteCounter<$action) { $deleteCounter++; unlink($theFile); } else { $criteriaMatch++; } } } } } $d->close(); } // Find sub-dirs: $subdirRegistry = array(''=>''); $d = @dir($this->typo3temp_path); if (is_object($d)) { while($entry=$d->read()) { $theFile = $entry; if (@is_dir($this->typo3temp_path.$theFile) && $theFile!='..' && $theFile!='.') { $subdirRegistry[$theFile.'/'] = $theFile.'/ (Files: '.count(t3lib_div::getFilesInDir($this->typo3temp_path.$theFile)).')'; } } } $deleteType=array( '0' => 'All', 'day' => 'Last access more than a day ago', 'week' => 'Last access more than a week ago', 'month' => 'Last access more than a month ago', '10' => 'Filesize greater than 10KB', '50' => 'Filesize greater than 50KB', '100' => 'Filesize greater than 100KB' ); $actionType=array( '0' => "Don't delete, just display statistics", '100' => 'Delete 100', '500' => 'Delete 500', '1000' => 'Delete 1000' ); // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'Typo3TempManager.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Get the subpart for 'Delete files by condition' dropdown $deleteOptionsSubpart = t3lib_parsehtml::getSubpart($template, '###DELETEOPTIONS###'); $deleteOptions = array(); foreach ($deleteType as $deleteKey => $deleteValue) { // Define the markers content $deleteMarkers = array( 'value' => htmlspecialchars($deleteKey), 'selected' => !strcmp($deleteKey, $tt) ? 'selected="selected"' : '', 'data' => htmlspecialchars($deleteValue) ); // Fill the markers in the subpart $deleteOptions[] = t3lib_parsehtml::substituteMarkerArray( $deleteOptionsSubpart, $deleteMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for 'Delete files by condition' dropdown $content = t3lib_parsehtml::substituteSubpart( $template, '###DELETEOPTIONS###', implode(LF, $deleteOptions) ); // Get the subpart for 'Number of files at a time' dropdown $actionOptionsSubpart = t3lib_parsehtml::getSubpart($template, '###ACTIONOPTIONS###'); $actionOptions = array(); foreach ($actionType as $actionKey => $actionValue) { // Define the markers content $actionMarkers = array( 'value' => htmlspecialchars($actionKey), 'data' => htmlspecialchars($actionValue) ); // Fill the markers in the subpart $actionOptions[] = t3lib_parsehtml::substituteMarkerArray( $actionOptionsSubpart, $actionMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for 'Number of files at a time' dropdown $content = t3lib_parsehtml::substituteSubpart( $content, '###ACTIONOPTIONS###', implode(LF, $actionOptions) ); // Get the subpart for 'From sub-directory' dropdown $subDirectoryOptionsSubpart = t3lib_parsehtml::getSubpart($template, '###SUBDIRECTORYOPTIONS###'); $subDirectoryOptions = array(); foreach ($subdirRegistry as $subDirectoryKey => $subDirectoryValue) { // Define the markers content $subDirectoryMarkers = array( 'value' => htmlspecialchars($subDirectoryKey), 'selected' => !strcmp($subDirectoryKey, $this->INSTALL['typo3temp_subdir']) ? 'selected="selected"' : '', 'data' => htmlspecialchars($subDirectoryValue) ); // Fill the markers in the subpart $subDirectoryOptions[] = t3lib_parsehtml::substituteMarkerArray( $subDirectoryOptionsSubpart, $subDirectoryMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for 'From sub-directory' dropdown $content = t3lib_parsehtml::substituteSubpart( $content, '###SUBDIRECTORYOPTIONS###', implode(LF, $subDirectoryOptions) ); // Define the markers content $markers = array( 'numberTemporary' => 'Number of temporary files:', 'numberMatching' => 'Number matching:', 'numberDeleted' => 'Number deleted:', 'temporary' => ($fileCounter - $deleteCounter), 'matching' => $criteriaMatch, 'deleteType' => '<span>' . htmlspecialchars($deleteType[$tt]) . '</span>', 'deleted' => $deleteCounter, 'deleteCondition' => 'Delete files by condition', 'numberFiles' => 'Number of files at a time:', 'fromSubdirectory' => 'From sub-directory:', 'execute' => 'Execute', 'explanation' => ' <p> This tool will delete files only if the last 10 characters before the extension (3 chars+\'.\') are hexadecimal valid ciphers, which are lowercase a-f and 0-9. </p> ' ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $content, $markers, '###|###', TRUE, FALSE ); // Add the content to the message array $this->message($headCode, 'Statistics', $content, 1); // Output the page $this->output($this->outputWrapper($this->printAll())); } /******************************* * * CONFIGURATION FORM * ********************************/ /** * Creating the form for editing the TYPO3_CONF_VARS options. * * @param string $type If get_form, display form, otherwise checks and store in localconf.php * @return void */ function generateConfigForm($type='') { $default_config_content = t3lib_div::getUrl(PATH_t3lib.'config_default.php'); $commentArr = $this->getDefaultConfigArrayComments($default_config_content); switch($type) { case 'get_form': // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'GenerateConfigForm.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); foreach ($GLOBALS['TYPO3_CONF_VARS'] as $k => $va) { $ext='['.$k.']'; $this->message($ext, '$TYPO3_CONF_VARS[\''.$k.'\']',$commentArr[0][$k],1); foreach ($va as $vk => $value) { $textAreaSubpart = ''; $booleanSubpart = ''; $textLineSubpart = ''; $description = trim($commentArr[1][$k][$vk]); $isTextarea = preg_match('/^(<.*?>)?string \(textarea\)/i', $description) ? TRUE : FALSE; $doNotRender = preg_match('/^(<.*?>)?string \(exclude\)/i', $description) ? TRUE : FALSE; if (!is_array($value) && !$doNotRender && ($this->checkForBadString($value) || $isTextarea)) { $k2 = '['.$vk.']'; if ($isTextarea) { // Get the subpart for a textarea $textAreaSubpart = t3lib_parsehtml::getSubpart($template, '###TEXTAREA###'); // Define the markers content $textAreaMarkers = array( 'id' => $k . '-' . $vk, 'name' => 'TYPO3_INSTALL[extConfig]['.$k.']['.$vk.']', 'value' => str_replace(array("'.chr(10).'", "' . LF . '"), array(LF, LF), $value) ); $value = str_replace(array("'.chr(10).'", "' . LF . '"), array(' | ', ' | '), $value); // Fill the markers in the subpart $textAreaSubpart = t3lib_parsehtml::substituteMarkerArray( $textAreaSubpart, $textAreaMarkers, '###|###', TRUE, FALSE ); } elseif (preg_match('/^(<.*?>)?boolean/i', $description)) { // Get the subpart for a checkbox $booleanSubpart = t3lib_parsehtml::getSubpart($template, '###BOOLEAN###'); // Define the markers content $booleanMarkers = array( 'id' => $k . '-' . $vk, 'name' => 'TYPO3_INSTALL[extConfig]['.$k.']['.$vk.']', 'value' => $value && strcmp($value, '0') ? $value : 1, 'checked' => $value ? 'checked="checked"' : '' ); // Fill the markers in the subpart $booleanSubpart = t3lib_parsehtml::substituteMarkerArray( $booleanSubpart, $booleanMarkers, '###|###', TRUE, FALSE ); } else { // Get the subpart for an input text field $textLineSubpart = t3lib_parsehtml::getSubpart($template, '###TEXTLINE###'); // Define the markers content $textLineMarkers = array( 'id' => $k . '-' . $vk, 'name' => 'TYPO3_INSTALL[extConfig]['.$k.']['.$vk.']', 'value' => $value ); // Fill the markers in the subpart $textLineSubpart = t3lib_parsehtml::substituteMarkerArray( $textLineSubpart, $textLineMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for a textarea $content = t3lib_parsehtml::substituteSubpart( $template, '###TEXTAREA###', $textAreaSubpart ); // Substitute the subpart for a checkbox $content = t3lib_parsehtml::substituteSubpart( $content, '###BOOLEAN###', $booleanSubpart ); // Substitute the subpart for an input text field $content = t3lib_parsehtml::substituteSubpart( $content, '###TEXTLINE###', $textLineSubpart ); // Define the markers content $markers = array( 'description' => $description, 'key' => '[' . $k . '][' . $vk . ']', 'label' => htmlspecialchars(t3lib_div::fixed_lgd_cs($value, 40)) ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $content, $markers, '###|###', TRUE, FALSE ); // Add the content to the message array $this->message($ext, $k2, $content); } } } break; default: if (is_array($this->INSTALL['extConfig'])) { $lines = $this->writeToLocalconf_control(); foreach ($this->INSTALL['extConfig'] as $k => $va) { if (is_array($GLOBALS['TYPO3_CONF_VARS'][$k])) { foreach ($va as $vk => $value) { if (isset($GLOBALS['TYPO3_CONF_VARS'][$k][$vk])) { $doit=1; if ($k=='BE' && $vk=='installToolPassword') { if ($value) { if (isset($_POST['installToolPassword_check'])) { if (!$this->formProtection->validateToken( (string) $_POST['formToken'], 'installToolPassword', 'change' )) { $doit = FALSE; break; } if (!t3lib_div::_GP('installToolPassword_check') || strcmp(t3lib_div::_GP('installToolPassword_check'), $value) ) { $doit = FALSE; $this->errorMessages[] = 'The two passwords did not ' . 'match! The password was not changed.'; } } if (t3lib_div::_GP('installToolPassword_md5')) $value =md5($value); } else $doit=0; } $description = trim($commentArr[1][$k][$vk]); if (preg_match('/^string \(textarea\)/i', $description)) { // Force Unix linebreaks in textareas $value = str_replace(CR, '', $value); // Preserve linebreaks $value = str_replace(LF, "' . LF . '", $value); } if (preg_match('/^boolean/i', $description)) { // When submitting settings in the Install Tool, values that default to "false" or "true" // in config_default.php will be sent as "0" resp. "1". Therefore, reset the values // to their boolean equivalent. if ($GLOBALS['TYPO3_CONF_VARS'][$k][$vk] === false && $value === '0') { $value = false; } elseif ($GLOBALS['TYPO3_CONF_VARS'][$k][$vk] === true && $value === '1') { $value = true; } } if ($doit && strcmp($GLOBALS['TYPO3_CONF_VARS'][$k][$vk],$value)) $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\''.$k.'\'][\''.$vk.'\']', $value); } } } } $this->writeToLocalconf_control($lines); } break; } } /** * Make an array of the comments in the t3lib/config_default.php file * * @param string $string The contents of the config_default.php file * @param array $mainArray * @param array $commentArray * @return array */ function getDefaultConfigArrayComments($string,$mainArray=array(),$commentArray=array()) { $lines = explode(LF, $string); $in=0; $mainKey=''; foreach ($lines as $lc) { $lc = trim($lc); if ($in) { if (!strcmp($lc,');')) { $in=0; } else { if (preg_match('/["\']([[:alnum:]_-]*)["\'][[:space:]]*=>(.*)/i',$lc,$reg)) { preg_match('/,[\t\s]*\/\/(.*)/i',$reg[2],$creg); $theComment = trim($creg[1]); if (substr(strtolower(trim($reg[2])),0,5)=='array' && !strcmp($reg[1],strtoupper($reg[1]))) { $mainKey=trim($reg[1]); $mainArray[$mainKey]=$theComment; } elseif ($mainKey) { $commentArray[$mainKey][$reg[1]]=$theComment; } } } } if (!strcmp($lc, '$TYPO3_CONF_VARS = array(')) { $in=1; } } return array($mainArray,$commentArray); } /******************************* * * CHECK CONFIGURATION FUNCTIONS * *******************************/ /** * Checking php.ini configuration and set appropriate messages and flags. * * @return void */ function checkConfiguration() { $ext='php.ini configuration checked'; $this->message($ext); // ***************** // Incoming values: // ***************** // Includepath $incPaths = t3lib_div::trimExplode(TYPO3_OS=='WIN'?';':':', ini_get('include_path')); if (!in_array('.',$incPaths)) { $this->message($ext, 'Current directory (./) is not in include path!', ' <p> <em>include_path=' . ini_get('include_path') . '</em> <br /> Normally the current path, \'.\', is included in the include_path of PHP. Although TYPO3 does not rely on this, it is an unusual setting that may introduce problems for some extensions. </p> ', 1); } else $this->message($ext, 'Current directory in include path',"",-1); // ***************** // File uploads // ***************** if (!ini_get('file_uploads')) { $this->message($ext, 'File uploads not allowed', ' <p> <em>file_uploads=' . ini_get('file_uploads') . '</em> <br /> TYPO3 uses the ability to upload files from the browser in various cases. <br /> As long as this flag is disabled, you\'ll not be able to upload files. <br /> But it doesn\'t end here, because not only are files not accepted by the server - ALL content in the forms are discarded and therefore nothing at all will be editable if you don\'t set this flag! <br /> However if you cannot enable fileupload for some reason alternatively you change the default form encoding value with \$TYPO3_CONF_VARS[SYS][form_enctype]. </p> ', 3); } else $this->message($ext, 'File uploads allowed',"",-1); $upload_max_filesize = t3lib_div::getBytesFromSizeMeasurement(ini_get('upload_max_filesize')); $post_max_size = t3lib_div::getBytesFromSizeMeasurement(ini_get('post_max_size')); if ($upload_max_filesize<1024*1024*10) { $this->message($ext, 'Maximum upload filesize too small?', ' <p> <em>upload_max_filesize=' . ini_get('upload_max_filesize') . '</em> <br /> By default TYPO3 supports uploading, copying and moving files of sizes up to 10MB (You can alter the TYPO3 defaults by the config option TYPO3_CONF_VARS[BE][maxFileSize]). <br /> Your current value is below this, so at this point, PHP sets the limits for uploaded filesizes and not TYPO3. <br /> <strong>Notice:</strong> The limits for filesizes attached to database records are set in the tables.php configuration files (\$TCA) for each group/file field. You may override these values in localconf.php or by page TSconfig settings. </p> ', 1); } if ($upload_max_filesize > $post_max_size) { $this->message($ext, 'Maximum size for POST requests is smaller than max. upload filesize', ' <p> <em>upload_max_filesize=' . ini_get('upload_max_filesize') . ' , post_max_size=' . ini_get('post_max_size') . '</em> <br /> You have defined a maximum size for file uploads which exceeds the allowed size for POST requests. Therefore the file uploads can not be larger than ' . ini_get('post_max_size') . ' </p> ', 1); } // ***************** // Memory and functions // ***************** $memory_limit_value = t3lib_div::getBytesFromSizeMeasurement(ini_get('memory_limit')); if ($memory_limit_value && $memory_limit_value < t3lib_div::getBytesFromSizeMeasurement(TYPO3_REQUIREMENTS_MINIMUM_PHP_MEMORY_LIMIT)) { $this->message($ext, 'Memory limit below ' . TYPO3_REQUIREMENTS_MINIMUM_PHP_MEMORY_LIMIT, ' <p> <em>memory_limit=' . ini_get('memory_limit') . '</em> <br /> Your system is configured to enforce a memory limit of PHP scripts lower than ' . TYPO3_REQUIREMENTS_MINIMUM_PHP_MEMORY_LIMIT . '. The Extension Manager needs to include more PHP-classes than will fit into this memory space. There is nothing else to do than raise the limit. To be safe, ask the system administrator of the webserver to raise the limit to over ' . TYPO3_REQUIREMENTS_MINIMUM_PHP_MEMORY_LIMIT . '. </p> ', 3); } elseif(!$memory_limit_value) { $this->message($ext, 'Memory limit', ' <p> <em>No memory limit in effect.</em> </p> ', -1); } else { $this->message($ext, 'Memory limit: ' . ini_get('memory_limit'), '', -1); } if (ini_get('max_execution_time')<30) { $this->message($ext, 'Maximum execution time below 30 seconds', ' <p> <em>max_execution_time=' . ini_get('max_execution_time') . '</em> <br /> May impose problems if too low. </p> ', 1); } else { $this->message($ext, 'Maximum execution time: ' . ini_get('max_execution_time') . ' seconds', '', -1); } if (ini_get('disable_functions')) { $this->message($ext, 'Functions disabled!', ' <p> <em>disable_functions=' . ini_get('disable_functions') . '</em> <br /> The above list of functions are disabled. If TYPO3 use any of these there might be trouble. <br /> TYPO3 is designed to use the default set of PHP4.3.0+ functions plus the functions of GDLib. <br /> Possibly these functions are disabled due to security risks and most likely the list would include a function like <em>exec()</em> which is use by TYPO3 to access ImageMagick. </p> ', 2); } else { $this->message($ext, 'Functions disabled: none', '', -1); } // Mail tests if (TYPO3_OS == 'WIN') { $smtp = ini_get('SMTP'); $bad_smtp = false; if (!t3lib_div::validIP($smtp)) { $smtp_addr = @gethostbyname($smtp); $bad_smtp = ($smtp_addr == $smtp); } else { $smtp_addr = $smtp; } if (!$smtp || $bad_smtp || !t3lib_div::testInt(ini_get('smtp_port'))) { $this->message($ext, 'Mail configuration is not set correctly', ' <p> Mail configuration is not set <br /> PHP mail() function requires SMTP and smtp_port to have correct values on Windows. </p> ', 2); } else { if (($smtp_addr == '127.0.0.1' || $smtp_addr == '::1') && ($_SERVER['SERVER_ADDR'] == '127.0.0.1' || $_SERVER['SERVER_ADDR'] == '::1')) { $this->message($ext, 'Mail is configured (potential problem exists!)', ' <p> <em>SMTP=' . $smtp . '</em> - <strong>Note:</strong> this server! Are you sure it runs SMTP server? <br /> <em>smtp_port=' . ini_get('smtp_port') . '</em> </p>' . $this->check_mail('get_form') . ' ', 1); } else { $this->message($ext, 'Mail is configured', ' <p> <em>SMTP=' . $smtp . '</em> <br /> <em>smtp_port=' . ini_get('smtp_port') . '</em> </p>' . $this->check_mail('get_form') .' ', -1); } } } elseif (!ini_get('sendmail_path')) { $this->message($ext, 'Sendmail path not defined!', ' <p> This may be critical to TYPO3\'s use of the mail() function. Please be sure that the mail() function in your php-installation works! </p>' . $this->check_mail('get_form') . ' ', 1); } else { list($prg) = explode(' ', ini_get('sendmail_path')); if (!@is_executable($prg)) { $this->message($ext, 'Sendmail program not found or not executable?', ' <p> <em>sendmail_path=' . ini_get('sendmail_path') . '</em> <br /> This may be critical to TYPO3\'s use of the mail() function. Please be sure that the mail() function in your php-installation works! </p>' . $this->check_mail('get_form') .' ', 1); } else { $this->message($ext, 'Sendmail OK', ' <p> <em>sendmail_path=' . ini_get('sendmail_path') . '</em> </p>' . $this->check_mail('get_form') . ' ', -1); } } // ***************** // Safe mode related // ***************** if (t3lib_utility_PhpOptions::isSafeModeEnabled()) { $this->message($ext, 'Safe mode turned on', ' <p> <em>safe_mode=' . ini_get('safe_mode') . '</em> <br /> In safe_mode PHP is restricted in several ways. This is a good thing because it adds protection to your (and others) scripts. But it may also introduce problems. In TYPO3 this <em>may be</em> a problem in two areas: File administration and execution of external programs, in particular ImageMagick. <br /> If you just ignore this warning, you\'ll most likely find, that TYPO3 seems to work except from the image-generation. The problem in that case is that the external ImageMagick programs are not allowed to be executed from the regular paths like "/usr/bin/" or "/usr/X11R6/bin/". <br /> If you use safe_mode with TYPO3, you should disable use of external programs ([BE][disable_exec_function]=1). <br /> In safe mode you must ensure that all the php-scripts and upload folders are owned by the same user. </p> <p> <em>safe_mode_exec_dir=' . ini_get('safe_mode_exec_dir') . '</em> <br /> If the ImageMagick utilities are located in this directory, everything is fine. Below on this page, you can see if ImageMagick is found here. If not, ask you ISP to put the three ImageMagick programs, \'convert\', \'combine\'/\'composite\' and \'identify\' there (eg. with symlinks if Unix server) </p> <p> <strong>Example of safe_mode settings:</strong> <br /> Set this in the php.ini file: </p> <p> ; Safe Mode <br /> safe_mode = On <br /> safe_mode_exec_dir = /usr/bin/ </p> <p> ...and the ImageMagick \'/usr/bin/convert\' will be executable. <br /> The last slash is important (..../) and you can only specify one directory. </p> <p> <strong>Notice: </strong> <br /> ImageMagick 6 or GraphicsMagick is recommended and the binaries are normally installed in /usr/bin. <br /> Paths to ImageMagick are defined in localconf.php and may be something else than /usr/bin/, but this is default for ImageMagick 6+ </p> ', 2); if (ini_get('doc_root')) { $this->message($ext, 'doc_root set', ' <p> <em>doc_root=' . ini_get('doc_root') . '</em> <br /> PHP cannot execute scripts outside this directory. If that is a problem is please correct it. </p> ', 1); } $this->config_array['safemode']=1; } else { $this->message($ext, 'safe_mode: off',"",-1); } if (t3lib_utility_PhpOptions::isSqlSafeModeEnabled()) { $this->message($ext, 'sql.safe_mode is enabled', ' <p> <em>sql.safe_mode=' . ini_get('sql.safe_mode') . '</em> <br /> This means that you can only connect to the database with a username corresponding to the user of the webserver process or fileowner. Consult your ISP for information about this. Also see <a href="http://www.wrox.com/Consumer/Store/Books/2963/29632002.htm"> http://www.wrox.com/Consumer/Store/Books/2963/29632002.htm</a> <br /> The owner of the current file is: <strong>' . get_current_user () . '</strong> </p> ', 1); $this->config_array['sql.safe_mode_user'] = get_current_user(); } else { $this->message($ext, 'sql.safe_mode: off',"",-1); } if (ini_get('open_basedir')) { $this->message($ext, 'open_basedir set', ' <p> <em>open_basedir=' . ini_get('open_basedir') . '</em> <br /> This restricts TYPO3 to open and include files only in this path. Please make sure that this does not prevent TYPO3 from running. <br /> <strong>Notice (UNIX):</strong> Before checking a path according to open_basedir, PHP resolves all symbolic links. </p> ', 1); // ???? If this option was set falsely you probably didn't see this page in the // first place, but this option <strong>may spoil this configuration test</strong> // when checking for such as ImageMagick executables. } else { $this->message($ext, 'open_basedir: off',"",-1); } // Check availability of PHP session support if (extension_loaded('session')) { $this->message($ext, 'PHP sessions available', ' <p> <em>PHP Sessions available</em> <br /> PHP is compiled with session support and session support is available. </p> ', -1); } else { $this->message($ext, 'PHP Sessions not available', ' <p> PHP is not compiled with session support, or session support is disabled in php.ini. <br /> TYPO3 needs session support. </p> ', 3); } // Suhosin/Hardened PHP: $suhosinDescription = ' <p> Suhosin limits the number of elements that can be submitted in forms to the server. This will affect for example the "All configuration" section in the Install Tool or Inline Relational Record Editing (IRRE) with many child records. </p>'; if (extension_loaded('suhosin')) { $suhosinSuggestion = ' <p> At least a value of 400 is suggested. </p> '; $suhosinRequestMaxVars = ini_get('suhosin.request.max_vars'); $suhosinPostMaxVars = ini_get('suhosin.post.max_vars'); $suhosinRequestMaxVarsType = ($suhosinRequestMaxVars < 400 ? 2 : -1); $suhosinPostMaxVarsType = ($suhosinPostMaxVars < 400 ? 2 : -1); $suhosinType = ($suhosinRequestMaxVars < 400 || $suhosinPostMaxVars < 400 ? 2 : -1); $this->message($ext, 'Suhosin/Hardened PHP is loaded', $suhosinDescription, $suhosinType); $this->message($ext, 'suhosin.request.max_vars: ' . $suhosinRequestMaxVars, $suhosinSuggestion, $suhosinRequestMaxVarsType); $this->message($ext, 'suhosin.post.max_vars: ' . $suhosinPostMaxVars, $suhosinSuggestion, $suhosinPostMaxVarsType); } else { $this->message($ext, 'Suhosin/Hardened PHP is not loaded', $suhosinDescription, 0); } // Check for stripped PHPdoc comments that are required to evaluate annotations: $method = new ReflectionMethod('tx_install', 'check_mail'); if (strlen($method->getDocComment()) === 0) { $description = ' <p> The system extension Extbase evaluates annotations in PHPdoc comments and thus requires eAccelerator not to strip away these parts. However, this is currently the only part in the TYPO3 Core (beside deprecation log and unit tests). If Extbase is not used, recompiling eAccelerator is not required at all. <br/> <br/> If you do not want comments to be stripped by eAccelerator, please recompile with the following configuration setting (<a href="http://eaccelerator.net/ticket/229" target="_blank"> more details</a>): <br /> <em>--with-eaccelerator-doc-comment-inclusion</em> </p> '; $this->message($ext, 'PHPdoc comments are stripped', $description, 2); } } /** * Check if PHP function mail() works * * @param string $cmd If "get_form" then a formfield for the mail-address is shown. If not, it's checked if "check_mail" was in the INSTALL array and if so a test mail is sent to the recipient given. * @return string The mail form if it is requested with get_form */ function check_mail($cmd='') { switch($cmd) { case 'get_form': $out = ' <p id="checkMailForm"> You can check the t3lib_mail functionality by entering your email address here and press the button. You should then receive a testmail from "typo3installtool@example.org". </p> '; // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'CheckMail.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); if (!empty($this->mailMessage)) { // Get the subpart for the mail is sent message $mailSentSubpart = t3lib_parsehtml::getSubpart($template, '###MAILSENT###'); } // Substitute the subpart for the mail is sent message $template = t3lib_parsehtml::substituteSubpart( $template, '###MAILSENT###', $mailSentSubpart ); // Define the markers content $markers = array( 'message' => $this->mailMessage, 'enterEmail' => 'Enter the email address', 'actionUrl' => $this->action . '#checkMailForm', 'submit' => 'Send test mail' ); // Fill the markers $out .= t3lib_parsehtml::substituteMarkerArray( $template, $markers, '###|###', TRUE, TRUE ); break; default: if (trim($this->INSTALL['check_mail'])) { $subject = 'TEST SUBJECT'; $email = trim($this->INSTALL['check_mail']); /** @var $mailMessage t3lib_mail_Message */ $mailMessage = t3lib_div::makeInstance('t3lib_mail_Message'); $mailMessage->addTo($email) ->addFrom('typo3installtool@example.org', 'TYPO3 Install Tool') ->setSubject($subject) ->setBody('<html><body>HTML TEST CONTENT</body></html>'); $mailMessage->addPart('TEST CONTENT'); $mailMessage->send(); $this->mailMessage= 'Mail was sent to: ' . $email; } break; } return $out; } /** * Checking php extensions, specifically GDLib and Freetype * * @return void */ function checkExtensions() { $ext = 'GDLib'; $this->message($ext); $software_info=1; if (extension_loaded('gd') && $this->isGD()) { $this->config_array['gd']=1; $this->message($ext, 'GDLib found',"",-1); if ($this->isPNG()) { $this->config_array['gd_png']=1; $this->message($ext, 'PNG supported',"",-1); } if ($this->isGIF()) { $this->config_array['gd_gif']=1; $this->message($ext, 'GIF supported',"",-1); } if ($this->isJPG()) { $this->config_array['gd_jpg']=1; $this->message($ext, 'JPG supported (not used by TYPO3)',''); } if (!$this->config_array['gd_gif'] && !$this->config_array['gd_png']) { $this->message($ext, 'PNG or GIF not supported', ' <p> Your GDLib supports either GIF nor PNG. It must support either one of them. </p> ', 2); } else { $msg=array(); if ($this->config_array['gd_gif'] && $this->config_array['gd_png']) { $msg[] = ' <p> You can choose between generating GIF or PNG files, as your GDLib supports both. </p> '; } if ($this->config_array['gd_gif']) { $msg[] = ' <p> You should watch out for the generated size of the GIF-files because some versions of the GD library do not compress them with LZW, but RLE and ImageMagick is subsequently used to compress with LZW. But in the case of ImageMagick failing this task (eg. not being compiled with LZW which is the case with some versions) you\'ll end up with GIF-filesizes all too big! <br /> This Install Tool tests what kinds of GIF compression are available in the ImageMagick installations by a physical test. You can also check it manually by opening a TYPO3 generated gif-file with Photoshop and save it in a new file. If the file sizes of the original and the new file are almost the same, you\'re having LZW compression and everything is fine. </p> '; } if ($this->config_array['gd_png']) { $msg[] = ' <p> TYPO3 prefers the use of GIF-files and most likely your visitors on your website does too as not all browsers support PNG yet. </p> '; } $this->message($ext, 'GIF / PNG issues', implode(LF, $msg), 1); } if (!$this->isTTF()) { $this->message($ext, 'FreeType is apparently not installed', ' <p> It looks like the FreeType library is not compiled into GDLib. This is required when TYPO3 uses GDLib and you\'ll most likely get errors like \'ImageTTFBBox is not a function\' or \'ImageTTFText is not a function\'. </p> ', 2); } else { $this->message($ext, 'FreeType quick-test (' . ($this->isGIF() ? 'as GIF' : 'as PNG') . ')', ' <p> <img src="' . htmlspecialchars(t3lib_div::getIndpEnv('REQUEST_URI') . '&testingTrueTypeSupport=1') . '" alt="" /> <br /> (If the text is exceeding the image borders you are using Freetype 2 and need to set TYPO3_CONF_VARS[GFX][TTFdpi]=96. <br /> If there is no image at all Freetype is most likely NOT available and you can just as well disable GDlib for TYPO3...) </p> ', -1); $this->config_array['freetype']=1; } } else { $this->message($ext, 'GDLib2 not found', ' <p> GDLib2 is required if you want to use the GIFBUILDER object in TypoScript. GIFBUILDER is in charge of all advanced image generation in TypoScript, including graphical menuitems. <br /> GDLib2 is also used in the TYPO3 Backend (TBE) to generate record icons and new module tabs. <br /> It\'s highly recommended to install this library. Remember to compile GD with FreeType which is also required. <br /> If you choose not to install GDLib, you can disable it in the configuration with [GFX][gdlib]=0;. </p> ', 2); } $this->message($ext, 'GDLib software information', $this->getGDSoftwareInfo()); } /** * Checking and testing that the required writable directories are writable. * * @return void */ function checkDirs() { // Check typo3/temp/ $ext='Directories'; $this->message($ext); $uniqueName = md5(uniqid(microtime())); // The requirement level (the integer value, ie. the second value of the value array) has the following meanings: // -1 = not required, but if it exists may be writable or not // 0 = not required, if it exists the dir should be writable // 1 = required, don't has to be writable // 2 = required, has to be writable $checkWrite=array( 'typo3temp/' => array('This folder is used by both the frontend (FE) and backend (BE) interface for all kind of temporary and cached files.',2,'dir_typo3temp'), 'typo3temp/pics/' => array('This folder is part of the typo3temp/ section. It needs to be writable, too.',2,'dir_typo3temp'), 'typo3temp/temp/' => array('This folder is part of the typo3temp/ section. It needs to be writable, too.',2,'dir_typo3temp'), 'typo3temp/llxml/' => array('This folder is part of the typo3temp/ section. It needs to be writable, too.',2,'dir_typo3temp'), 'typo3temp/cs/' => array('This folder is part of the typo3temp/ section. It needs to be writable, too.',2,'dir_typo3temp'), 'typo3temp/GB/' => array('This folder is part of the typo3temp/ section. It needs to be writable, too.',2,'dir_typo3temp'), 'typo3temp/locks/' => array('This folder is part of the typo3temp/ section. It needs to be writable, too.',2,'dir_typo3temp'), 'typo3conf/' => array('This directory contains the local configuration files of your website. TYPO3 must be able to write to these configuration files during setup and when the Extension Manager (EM) installs extensions.',2), 'typo3conf/ext/' => array('Location for local extensions. Must be writable if the Extension Manager is supposed to install extensions for this website.',0), 'typo3conf/l10n/' => array('Location for translations. Must be writable if the Extension Manager is supposed to install translations for extensions.',0), TYPO3_mainDir.'ext/' => array('Location for global extensions. Must be writable if the Extension Manager is supposed to install extensions globally in the source.',-1), 'uploads/' => array('Location for uploaded files from RTE, in the subdirectories for uploaded files of content elements.',2), 'uploads/pics/' => array('Typical location for uploaded files (images especially).',0), 'uploads/media/' => array('Typical location for uploaded files (non-images especially).',0), 'uploads/tf/' => array('Typical location for uploaded files (TS template resources).',0), $GLOBALS['TYPO3_CONF_VARS']['BE']['fileadminDir'] => array('Location for local files such as templates, independent uploads etc.',-1), $GLOBALS['TYPO3_CONF_VARS']['BE']['fileadminDir'] . '_temp_/' => array('Typical temporary location for default upload of files by administrators.',0), ); foreach ($checkWrite as $relpath => $descr) { // Check typo3temp/ $general_message = $descr[0]; // If the directory is missing, try to create it if (!@is_dir(PATH_site.$relpath)) { t3lib_div::mkdir(PATH_site.$relpath); } if (!@is_dir(PATH_site.$relpath)) { if ($descr[1]) { // required... $this->message($ext, $relpath.' directory does not exist and could not be created', ' <p> <em>Full path: ' . PATH_site . $relpath . '</em> <br /> ' . $general_message . ' </p> <p> This error should not occur as ' . $relpath . ' must always be accessible in the root of a TYPO3 website. </p> ', 3); } else { if ($descr[1] == 0) { $msg = 'This directory does not necessarily have to exist but if it does it must be writable.'; } else { $msg = 'This directory does not necessarily have to exist and if it does it can be writable or not.'; } $this->message($ext, $relpath.' directory does not exist', ' <p> <em>Full path: ' . PATH_site . $relpath . '</em> <br /> ' . $general_message . ' </p> <p> ' . $msg . ' </p> ', 2); } } else { $file = PATH_site.$relpath.$uniqueName; @touch($file); if (@is_file($file)) { unlink($file); if ($descr[2]) { $this->config_array[$descr[2]]=1; } $this->message($ext, $relpath.' writable','',-1); } else { $severity = ($descr[1]==2 || $descr[1]==0) ? 3 : 2; if ($descr[1] == 0 || $descr[1] == 2) { $msg = 'The directory '.$relpath.' must be writable!'; } elseif ($descr[1] == -1 || $descr[1] == 1) { $msg = 'The directory '.$relpath.' does not neccesarily have to be writable.'; } $this->message($ext, $relpath .' directory not writable', ' <p> <em>Full path: ' . $file . '</em> <br /> ' . $general_message . ' </p> <p> Tried to write this file (with touch()) but didn\'t succeed. <br /> ' . $msg . ' </p> ', $severity); } } } } /** * Checking for existing ImageMagick installs. * * This tries to find available ImageMagick installations and tries to find the version numbers by executing "convert" without parameters. If the ->checkIMlzw is set, LZW capabilities of the IM installs are check also. * * @param array $paths Possible ImageMagick paths * @return void */ function checkImageMagick($paths) { $ext='Check Image Magick'; $this->message($ext); // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'CheckImageMagick.html'); $paths = array_unique($paths); $programs = explode(',','gm,convert,combine,composite,identify'); $isExt = TYPO3_OS=="WIN" ? ".exe" : ""; $this->config_array['im_combine_filename']='combine'; foreach ($paths as $k => $v) { if (!preg_match('/[\\/]$/',$v)) $v.='/'; foreach ($programs as $filename) { if (ini_get('open_basedir') || (file_exists($v)&&@is_file($v.$filename.$isExt))) { $version = $this->_checkImageMagick_getVersion($filename,$v); if($version > 0) { // Assume GraphicsMagick if($filename=='gm') { $index[$v]['gm']=$version; // No need to check for "identify" etc. continue; } else { // Assume ImageMagick $index[$v][$filename]=$version; } } } } if (count($index[$v])>=3 || $index[$v]['gm']) { $this->config_array['im'] = 1; } if ($index[$v]['gm'] || (!$index[$v]['composite'] && $index[$v]['combine'])) { $this->config_array['im_combine_filename']='combine'; } elseif ($index[$v]['composite'] && !$index[$v]['combine']) { $this->config_array['im_combine_filename']='composite'; } if (isset($index[$v]['convert']) && $this->checkIMlzw) { $index[$v]['gif_capability'] = ''.$this->_checkImageMagickGifCapability($v); } } $this->config_array['im_versions']=$index; if (!$this->config_array['im']) { $this->message($ext, 'No ImageMagick installation available', ' <p> It seems that there is no adequate ImageMagick installation available at the checked locations (' . implode(', ', $paths) . ') <br /> An \'adequate\' installation for requires \'convert\', \'combine\'/\'composite\' and \'identify\' to be available </p> ', 2); } else { // Get the subpart for the ImageMagick versions $theCode = t3lib_parsehtml::getSubpart($templateFile, '###VERSIONS###'); // Get the subpart for each ImageMagick version $rowsSubPart = t3lib_parsehtml::getSubpart($theCode, '###ROWS###'); $rows = array(); foreach ($this->config_array['im_versions'] as $p => $v) { $ka = array(); reset($v); while(list($ka[])=each($v)) {} // Define the markers content $rowsMarkers = array( 'file' => $p, 'type' => implode('<br />', $ka), 'version' => implode('<br />', $v) ); // Fill the markers in the subpart $rows[] = t3lib_parsehtml::substituteMarkerArray( $rowsSubPart, $rowsMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the ImageMagick versions $theCode = t3lib_parsehtml::substituteSubpart( $theCode, '###ROWS###', implode(LF, $rows) ); // Add the content to the message array $this->message($ext, 'Available ImageMagick/GraphicsMagick installations:', $theCode, -1); } // Get the template file $formSubPart = t3lib_parsehtml::getSubpart($templateFile, '###FORM###'); // Define the markers content $formMarkers = array( 'actionUrl' => $this->action, 'lzwChecked' => ($this->INSTALL['checkIM']['lzw'] ? 'checked="checked"' : ''), 'lzwLabel' => 'Check LZW capabilities.', 'checkPath' => 'Check this path for ImageMagick installation:', 'imageMagickPath' => htmlspecialchars($this->INSTALL['checkIM']['path']), 'comment' => '(Eg. "D:\wwwroot\im537\ImageMagick\" for Windows or "/usr/bin/" for Unix)', 'send' => 'Send' ); // Fill the markers $formSubPart = t3lib_parsehtml::substituteMarkerArray( $formSubPart, $formMarkers, '###|###', TRUE, FALSE ); // Add the content to the message array $this->message($ext, 'Search for ImageMagick:', $formSubPart, 0); } /** * Checking GIF-compression capabilities of ImageMagick install * * @param string $path Path of ImageMagick installation * @return string Type of compression */ function _checkImageMagickGifCapability($path) { if ($this->config_array['dir_typo3temp']) { // && !$this->config_array['safemode'] $tempPath = $this->typo3temp_path; $uniqueName = md5(uniqid(microtime())); $dest = $tempPath.$uniqueName.'.gif'; $src = $this->backPath.'gfx/typo3logo.gif'; if (@is_file($src) && !strstr($src,' ') && !strstr($dest,' ')) { $cmd = t3lib_div::imageMagickCommand('convert', $src.' '.$dest, $path); t3lib_utility_Command::exec($cmd); } else die('No typo3/gfx/typo3logo.gif file!'); $out=''; if (@is_file($dest)) { $new_info = @getimagesize($dest); clearstatcache(); $new_size = filesize($dest); $src_info = @getimagesize($src); clearstatcache(); $src_size = @filesize($src); if ($new_info[0]!=$src_info[0] || $new_info[1]!=$src_info[1] || !$new_size || !$src_size) { $out='error'; } else { // NONE-LZW ratio was 5.5 in test if ($new_size/$src_size > 4) { $out='NONE'; // NONE-RLE ratio was not tested } elseif ($new_size/$src_size > 1.5) { $out='RLE'; } else { $out='LZW'; } } unlink($dest); } return $out; } } /** * Extracts the version number for ImageMagick * * @param string $file The program name to execute in order to find out the version number * @param string $path Path for the above program * @return string Version number of the found ImageMagick instance */ function _checkImageMagick_getVersion($file, $path) { // Temporarily override some settings $im_version = $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5']; $combine_filename = $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_combine_filename']; if ($file=='gm') { $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5'] = 'gm'; // Work-around, preventing execution of "gm gm" $file = 'identify'; // Work-around - GM doesn't like to be executed without any arguments $parameters = '-version'; } else { $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5'] = 'im5'; // Override the combine_filename setting if ($file=='combine' || $file=='composite') { $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_combine_filename'] = $file; } } $cmd = t3lib_div::imageMagickCommand($file, $parameters, $path); $retVal = false; t3lib_utility_Command::exec($cmd, $retVal); $string = $retVal[0]; list(,$ver) = explode('Magick', $string); list($ver) = explode(' ',trim($ver)); // Restore the values $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5'] = $im_version; $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_combine_filename'] = $combine_filename; return trim($ver); } /** * Checks database username/password/host/database * * @return void */ function checkDatabase() { $ext='Check database'; $this->message($ext); if (!extension_loaded('mysql') && !t3lib_extMgm::isLoaded('dbal')) { $this->message($ext, 'MySQL not available', ' <p> PHP does not feature MySQL support (which is pretty unusual). </p> ', 2); } else { if (!TYPO3_db_host || !TYPO3_db_username) { $this->message($ext, 'Username, password or host not set', ' <p> You may need to enter data for these values: <br /> Username: <strong>' . htmlspecialchars(TYPO3_db_username) . '</strong> <br /> Host: <strong>' . htmlspecialchars(TYPO3_db_host) . '</strong> <br /> <br /> Use the form below. </p> ', 2); } if ($result = $GLOBALS['TYPO3_DB']->sql_pconnect(TYPO3_db_host, TYPO3_db_username, TYPO3_db_password)) { $this->message($ext, 'Connected to SQL database successfully', ' <dl id="t3-install-databaseconnected"> <dt> Username: </dt> <dd> ' . htmlspecialchars(TYPO3_db_username) . ' </dd> <dt> Host: </dt> <dd> ' . htmlspecialchars(TYPO3_db_host) . ' </dd> </dl> ', -1, 1); $this->config_array['mysqlConnect']=1; if (!TYPO3_db) { $this->message($ext, 'No database selected', ' <p> Currently you have no database selected. <br /> Please select one or create a new database. </p> ', 3); $this->config_array['no_database']=1; } elseif (!$GLOBALS['TYPO3_DB']->sql_select_db(TYPO3_db)) { $this->message($ext, 'Database', ' <p> \'' . htmlspecialchars(TYPO3_db) . '\' could not be selected as database! <br /> Please select another one or create a new database. </p> ', 3, 1); $this->config_array['no_database']=1; } else { $this->message($ext, 'Database', ' <p> <strong>' . htmlspecialchars(TYPO3_db) . '</strong> is selected as database. </p> ', 1, 1); } } else { $sqlSafeModeUser = ''; if ($this->config_array['sql.safe_mode_user']) { $sqlSafeModeUser = ' <strong>Notice:</strong> <em>sql.safe_mode</em> is turned on, so apparently your username to the database is the same as the scriptowner, which is ' . $this->config_array['sql.safe_mode_user']; } $this->message($ext, 'Could not connect to SQL database!', ' <p> Connecting to SQL database failed with these settings: <br /> Username: <strong>' . htmlspecialchars(TYPO3_db_username) . '</strong> <br /> Host: <strong>' . htmlspecialchars(TYPO3_db_host) . '</strong> </p> <p> Make sure you\'re using the correct set of data. <br /> ' . $sqlSafeModeUser . ' </p> ', 3); } } } /** * Prints form for updating localconf.php or updates localconf.php depending on $cmd * * @param string $cmd If "get_form" it outputs the form. Default is to write "localconf.php" based on input in ->INSTALL[localconf.php] array and flag ->setLocalconf * @return string Form HTML */ function setupGeneral($cmd='') { switch($cmd) { case 'get_form': // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'SetupGeneral.html'); // Get the template part from the file $form = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Get the subpart for all modes $allModesSubpart = t3lib_parsehtml::getSubpart($form, '###ALLMODES###'); // Define the markers content $formMarkers['actionUrl'] = $this->action; // Username if (TYPO3_db_username) { $username = TYPO3_db_username; } elseif ($this->config_array['sql.safe_mode_user']) { $username = $this->config_array['sql.safe_mode_user']; // Get the subpart for the sql safe mode user $sqlSafeModeUserSubpart = t3lib_parsehtml::getSubpart($allModesSubpart, '###SQLSAFEMODEUSERSUBPART###'); // Define the markers content $sqlSafeModeUserMarkers = array( 'labelSqlSafeModeUser' => 'sql.safe_mode_user:', 'sqlSafeModeUser' => $this->config_array['sql.safe_mode_user'] ); // Fill the markers in the subpart $sqlSafeModeUserSubpart = t3lib_parsehtml::substituteMarkerArray( $sqlSafeModeUserSubpart, $sqlSafeModeUserMarkers, '###|###', TRUE, FALSE ); } // Get the subpart for all modes $allModesSubpart = t3lib_parsehtml::substituteSubpart( $allModesSubpart, '###SQLSAFEMODEUSERSUBPART###', $sqlSafeModeUserSubpart ); // Define the markers content $allModesMarkers = array( 'labelUsername' => 'Username:', 'username' => htmlspecialchars($username), 'labelPassword' => 'Password:', 'password' => htmlspecialchars(TYPO3_db_password), 'labelHost' => 'Host:', 'host' => htmlspecialchars(TYPO3_db_host), 'labelDatabase' => 'Database:', 'labelCreateDatabase' => 'Create database?' ); // Get the subpart for the database list $databasesSubpart = t3lib_parsehtml::getSubpart($allModesSubpart, '###DATABASELIST###'); if ($this->config_array['mysqlConnect']) { // Get the subpart when database is available $databaseAvailableSubpart = t3lib_parsehtml::getSubpart($databasesSubpart, '###DATABASEAVAILABLE###'); // Get the subpart for each database table $databaseItemSubpart = t3lib_parsehtml::getSubpart($databaseAvailableSubpart, '###DATABASEITEM###'); $dbArr = $this->getDatabaseList(); $dbIncluded=0; $databaseItems = array(); foreach ($dbArr as $dbname) { // Define the markers content $databaseItemMarkers = array( 'databaseSelected' => '', 'databaseName' => htmlspecialchars($dbname), 'databaseValue' => htmlspecialchars($dbname) ); if ($dbname == TYPO3_db) { $databaseItemMarkers['databaseSelected'] = 'selected="selected"'; } // Fill the markers in the subpart $databaseItems[] = t3lib_parsehtml::substituteMarkerArray( $databaseItemSubpart, $databaseItemMarkers, '###|###', TRUE, FALSE ); if ($dbname==TYPO3_db) $dbIncluded=1; } if (!$dbIncluded && TYPO3_db) { $databaseItemMarkers['databaseName'] = htmlspecialchars(TYPO3_db); $databaseItemMarkers['databaseSelected'] = 'selected="selected"'; $databaseItemMarkers['databaseValue'] = htmlspecialchars(TYPO3_db) . ' (NO ACCESS!)'; // Fill the markers in the subpart $databaseItems[] = t3lib_parsehtml::substituteMarkerArray( $databaseItemSubpart, $databaseItemMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the database tables $databaseAvailableSubpart = t3lib_parsehtml::substituteSubpart( $databaseAvailableSubpart, '###DATABASEITEM###', implode(LF, $databaseItems) ); } else { // Get the subpart when the database is not available $databaseNotAvailableSubpart = t3lib_parsehtml::getSubpart($databasesSubpart, '###DATABASENOTAVAILABLE###'); $databaseNotAvailableMarkers = array( 'typo3Db' => htmlspecialchars(TYPO3_db), 'labelNoDatabase' => ' (Can only select database if username/password/host is correctly set first) ' ); // Fill the markers in the subpart $databaseNotAvailableSubpart = t3lib_parsehtml::substituteMarkerArray( $databaseNotAvailableSubpart, $databaseNotAvailableMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart when database is available $databasesSubpart = t3lib_parsehtml::substituteSubpart( $databasesSubpart, '###DATABASEAVAILABLE###', $databaseAvailableSubpart ); // Substitute the subpart when database is not available $databasesSubpart = t3lib_parsehtml::substituteSubpart( $databasesSubpart, '###DATABASENOTAVAILABLE###', $databaseNotAvailableSubpart ); // Substitute the subpart for the databases $allModesSubpart = t3lib_parsehtml::substituteSubpart( $allModesSubpart, '###DATABASELIST###', $databasesSubpart ); // Fill the markers in the subpart for all modes $allModesSubpart = t3lib_parsehtml::substituteMarkerArray( $allModesSubpart, $allModesMarkers, '###|###', TRUE, FALSE ); // Substitute the subpart for all modes $form = t3lib_parsehtml::substituteSubpart( $form, '###ALLMODES###', $allModesSubpart ); if ($this->mode!='123') { // Get the subpart for the regular mode $regularModeSubpart = t3lib_parsehtml::getSubpart($form, '###REGULARMODE###'); // Define the markers content $regularModeMarkers = array( 'labelSiteName' => 'Site name:', 'siteName' => htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['SYS']['sitename']), 'labelEncryptionKey' => 'Encryption key:', 'encryptionKey' => htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['SYS']['encryptionKey']), 'labelGenerateRandomKey' => 'Generate random key' ); // Other $fA = $this->setupGeneralCalculate(); $regularModeMarkers['labelCurrentValueIs'] = 'current value is'; // Disable exec function if (is_array($fA['disable_exec_function'])) { // Get the subpart for the disable exec function $disableExecFunctionSubpart = t3lib_parsehtml::getSubpart($regularModeSubpart, '###DISABLEEXECFUNCTIONSUBPART###'); $regularModeMarkers['labelDisableExecFunction'] = '[BE][disable_exec_function]='; $regularModeMarkers['strongDisableExecFunction'] = (integer) current($fA['disable_exec_function']); $regularModeMarkers['defaultDisableExecFunction'] = (integer) $GLOBALS['TYPO3_CONF_VARS']['BE']['disable_exec_function']; $regularModeMarkers['disableExecFunction'] = (integer) current($fA['disable_exec_function']); // Fill the markers in the subpart $disableExecFunctionSubpart = t3lib_parsehtml::substituteMarkerArray( $disableExecFunctionSubpart, $regularModeMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the disable exec function $regularModeSubpart = t3lib_parsehtml::substituteSubpart( $regularModeSubpart, '###DISABLEEXECFUNCTIONSUBPART###', $disableExecFunctionSubpart ); // GDlib if (is_array($fA['gdlib'])) { // Get the subpart for the disable gd lib $gdLibSubpart = t3lib_parsehtml::getSubpart($regularModeSubpart, '###DISABLEGDLIB###'); $regularModeMarkers['labelGdLib'] = '[GFX][gdlib]='; $regularModeMarkers['strongGdLib'] = (integer) current($fA['gdlib']); $regularModeMarkers['defaultGdLib'] = (integer) $GLOBALS['TYPO3_CONF_VARS']['GFX']['gdlib']; $regularModeMarkers['gdLib'] = (integer) current($fA['gdlib']); // Fill the markers in the subpart $gdLibSubpart = t3lib_parsehtml::substituteMarkerArray( $gdLibSubpart, $regularModeMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the disable gdlib $regularModeSubpart = t3lib_parsehtml::substituteSubpart( $regularModeSubpart, '###DISABLEGDLIB###', $gdLibSubpart ); // GDlib PNG if (is_array($fA['gdlib_png']) && $GLOBALS['TYPO3_CONF_VARS']['GFX']['gdlib']) { // Get the subpart for the gdlib png $gdLibPngSubpart = t3lib_parsehtml::getSubpart($regularModeSubpart, '###GDLIBPNGSUBPART###'); // Get the subpart for the dropdown options $gdLibPngOptionSubpart = t3lib_parsehtml::getSubpart($gdLibPngSubpart, '###GDLIBPNGOPTION###'); $gdLibPngLabels = $this->setLabelValueArray($fA['gdlib_png'], 2); reset($gdLibPngLabels); $regularModeMarkers['labelGdLibPng'] = '[GFX][gdlib_png]='; $regularModeMarkers['strongGdLibPng'] = (string) current($gdLibPngLabels); $regularModeMarkers['defaultGdLibPng'] = (integer) $GLOBALS['TYPO3_CONF_VARS']['GFX']['gdlib_png']; $gdLibPngOptions = array(); foreach ($gdLibPngLabels as $k => $v) { list($cleanV) = explode('|', $fA['gdlib_png'][$k]); $gdLibPngMarker['value'] = htmlspecialchars($fA['gdlib_png'][$k]); $gdLibPngMarker['data'] = htmlspecialchars($v); if (!strcmp($GLOBALS['TYPO3_CONF_VARS']['GFX']['gdlib_png'], $cleanV)) { $gdLibPngMarker['selected'] = 'selected="selected"'; } // Fill the markers in the subpart $gdLibPngOptions[] = t3lib_parsehtml::substituteMarkerArray( $gdLibPngOptionSubpart, $gdLibPngMarker, '###|###', TRUE, FALSE ); } // Substitute the subpart for the dropdown options $gdLibPngSubpart = t3lib_parsehtml::substituteSubpart( $gdLibPngSubpart, '###GDLIBPNGOPTION###', implode(LF, $gdLibPngOptions) ); // Fill the markers in the subpart $gdLibPngSubpart = t3lib_parsehtml::substituteMarkerArray( $gdLibPngSubpart, $regularModeMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the gdlib png $regularModeSubpart = t3lib_parsehtml::substituteSubpart( $regularModeSubpart, '###GDLIBPNGSUBPART###', $gdLibPngSubpart ); // ImageMagick if (is_array($fA['im'])) { // Get the subpart for ImageMagick $imageMagickSubpart = t3lib_parsehtml::getSubpart($regularModeSubpart, '###IMAGEMAGICKSUBPART###'); // Define the markers content $regularModeMarkers['labelImageMagick'] = '[GFX][im]='; $regularModeMarkers['strongImageMagick'] = (string) current($fA['im']); $regularModeMarkers['defaultImageMagick'] = (integer) $GLOBALS['TYPO3_CONF_VARS']['GFX']['im']; $regularModeMarkers['imageMagick'] = (integer) current($fA['im']); // Fill the markers in the subpart $imageMagickSubpart = t3lib_parsehtml::substituteMarkerArray( $imageMagickSubpart, $regularModeMarkers, '###|###', TRUE, FALSE ); // IM Combine Filename // Get the subpart for ImageMagick Combine filename $imCombineFileNameSubpart = t3lib_parsehtml::getSubpart($regularModeSubpart, '###IMCOMBINEFILENAMESUBPART###'); // Define the markers content $regularModeMarkers['labelImCombineFilename'] = '[GFX][im_combine_filename]'; $regularModeMarkers['strongImCombineFilename'] = (string) current($fA['im_combine_filename']); $regularModeMarkers['defaultImCombineFilename'] = (string) $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_combine_filename']; $regularModeMarkers['imCombineFilename'] = (string) ($fA['im_combine_filename'] ? current($fA['im_combine_filename']) : 'combine'); // Fill the markers in the subpart $imCombineFileNameSubpart = t3lib_parsehtml::substituteMarkerArray( $imCombineFileNameSubpart, $regularModeMarkers, '###|###', TRUE, FALSE ); // IM Version 5 // Get the subpart for ImageMagick Version 5 $imVersion5Subpart = t3lib_parsehtml::getSubpart($regularModeSubpart, '###IMVERSION5SUBPART###'); // Define the markers content $regularModeMarkers['labelImVersion5'] = '[GFX][im_version_5]='; $regularModeMarkers['strongImVersion5'] = (string) current($fA['im_version_5']); $regularModeMarkers['defaultImVersion5'] = (string) $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5']; $regularModeMarkers['imVersion5'] = (string) ($fA['im_version_5'] ? current($fA['im_version_5']) : ''); // Fill the markers in the subpart $imVersion5Subpart = t3lib_parsehtml::substituteMarkerArray( $imVersion5Subpart, $regularModeMarkers, '###|###', TRUE, FALSE ); if ($GLOBALS['TYPO3_CONF_VARS']['GFX']['im']) { // IM Path if (is_array($fA['im_path'])) { // Get the subpart for ImageMagick path $imPathSubpart = t3lib_parsehtml::getSubpart($regularModeSubpart, '###IMPATHSUBPART###'); $labelImPath = $this->setLabelValueArray($fA['im_path'], 1); reset($labelImPath); $imPath = $this->setLabelValueArray($fA['im_path'], 0); reset($imPath); // Define the markers content $regularModeMarkers['labelImPath'] = '[GFX][im_path]='; $regularModeMarkers['strongImPath'] = (string) current($labelImPath); $regularModeMarkers['defaultImPath'] = (string) $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_path']; $regularModeMarkers['ImPath'] = (string) current($imPath); // Fill the markers in the subpart $imPathSubpart = t3lib_parsehtml::substituteMarkerArray( $imPathSubpart, $regularModeMarkers, '###|###', TRUE, FALSE ); } // IM Path LZW if (is_array($fA['im_path_lzw'])) { // Get the subpart for ImageMagick lzw path $imPathLzwSubpart = t3lib_parsehtml::getSubpart($regularModeSubpart, '###IMPATHLZWSUBPART###'); // Get the subpart for ImageMagick lzw path dropdown options $imPathOptionSubpart = t3lib_parsehtml::getSubpart($regularModeSubpart, '###IMPATHLZWOPTION###'); $labelImPathLzw = $this->setLabelValueArray($fA['im_path_lzw'], 1); reset($labelImPathLzw); $imPathLzw = $this->setLabelValueArray($fA['im_path_lzw'], 0); reset($imPathLzw); // Define the markers content $regularModeMarkers['labelImPathLzw'] = '[GFX][im_path_lzw]='; $regularModeMarkers['strongImPathLzw'] = (string) current($labelImPathLzw); $regularModeMarkers['defaultImPathLzw'] = (string) $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_path_lzw']; $regularModeMarkers['ImPathLzw'] = (string) current($imPathLzw); $imPathLzwOptions = array(); foreach ($labelImPathLzw as $k => $v) { list($cleanV) = explode('|', $fA['im_path_lzw'][$k]); // Define the markers content $imPathLzwMarker = array( 'value' => htmlspecialchars($fA['im_path_lzw'][$k]), 'data' => htmlspecialchars($v) ); if (!strcmp($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_path_lzw'], $cleanV)) { $imPathLzwMarker['selected'] = 'selected="selected"'; } // Fill the markers in the subpart $imPathLzwOptions[] = t3lib_parsehtml::substituteMarkerArray( $imPathOptionSubpart, $imPathLzwMarker, '###|###', TRUE, FALSE ); } // Substitute the subpart for ImageMagick lzw path dropdown options $imPathLzwSubpart = t3lib_parsehtml::substituteSubpart( $imPathLzwSubpart, '###IMPATHLZWOPTION###', implode(LF, $imPathLzwOptions) ); // Fill the markers in the subpart $imPathLzwSubpart = t3lib_parsehtml::substituteMarkerArray( $imPathLzwSubpart, $regularModeMarkers, '###|###', TRUE, FALSE ); } } } // Substitute the subpart for ImageMagick $regularModeSubpart = t3lib_parsehtml::substituteSubpart( $regularModeSubpart, '###IMAGEMAGICKSUBPART###', $imageMagickSubpart ); // Substitute the subpart for ImageMagick Combine filename $regularModeSubpart = t3lib_parsehtml::substituteSubpart( $regularModeSubpart, '###IMCOMBINEFILENAMESUBPART###', $imCombineFileNameSubpart ); // Substitute the subpart for ImageMagick Version 5 $regularModeSubpart = t3lib_parsehtml::substituteSubpart( $regularModeSubpart, '###IMVERSION5SUBPART###', $imVersion5Subpart ); // Substitute the subpart for ImageMagick path $regularModeSubpart = t3lib_parsehtml::substituteSubpart( $regularModeSubpart, '###IMPATHSUBPART###', $imPathSubpart ); // Substitute the subpart for ImageMagick lzw path $regularModeSubpart = t3lib_parsehtml::substituteSubpart( $regularModeSubpart, '###IMPATHLZWSUBPART###', $imPathLzwSubpart ); // TrueType Font dpi // Get the subpart for TrueType dpi $ttfDpiSubpart = t3lib_parsehtml::getSubpart($regularModeSubpart, '###TTFDPISUBPART###'); // Define the markers content $regularModeMarkers['labelTtfDpi'] = '[GFX][TTFdpi]='; $regularModeMarkers['ttfDpi'] = htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['GFX']['TTFdpi']); // Fill the markers in the subpart $ttfDpiSubpart = t3lib_parsehtml::substituteMarkerArray( $ttfDpiSubpart, $regularModeMarkers, '###|###', TRUE, FALSE ); // Substitute the subpart for TrueType dpi $regularModeSubpart = t3lib_parsehtml::substituteSubpart( $regularModeSubpart, '###TTFDPISUBPART###', $ttfDpiSubpart ); // Fill the markers in the regular mode subpart $regularModeSubpart = t3lib_parsehtml::substituteMarkerArray( $regularModeSubpart, $regularModeMarkers, '###|###', TRUE, FALSE ); } $formMarkers['labelUpdateLocalConf'] = 'Update localconf.php'; $formMarkers['labelNotice'] = 'NOTICE:'; $formMarkers['labelCommentUpdateLocalConf'] = 'By clicking this button, localconf.php is updated with new values for the parameters listed above!'; // Substitute the subpart for regular mode $form = t3lib_parsehtml::substituteSubpart( $form, '###REGULARMODE###', $regularModeSubpart ); // Fill the markers $out = t3lib_parsehtml::substituteMarkerArray( $form, $formMarkers, '###|###', TRUE, FALSE ); break; default: if (is_array($this->INSTALL['localconf.php'])) { $lines = $this->writeToLocalconf_control(); // New database? if (trim($this->INSTALL['localconf.php']['NEW_DATABASE_NAME'])) { $newdbname=trim($this->INSTALL['localconf.php']['NEW_DATABASE_NAME']); if (!preg_match('/[^[:alnum:]_-]/',$newdbname)) { if ($result = $GLOBALS['TYPO3_DB']->sql_pconnect(TYPO3_db_host, TYPO3_db_username, TYPO3_db_password)) { if ($GLOBALS['TYPO3_DB']->admin_query('CREATE DATABASE '.$newdbname)) { $this->INSTALL['localconf.php']['typo_db'] = $newdbname; $this->messages[]= "Database '".$newdbname."' created"; } else { $this->errorMessages[] = ' Could not create database \'' . $newdbname . '\' (...not created) '; } } else { $this->errorMessages[] = ' Could not connect to database when creating database \'' . $newdbname . '\' (...not created) '; } } else { $this->errorMessages[] = ' The NEW database name \'' . $newdbname . '\' was not alphanumeric, a-zA-Z0-9_- (...not created) '; } } // Parsing values foreach ($this->INSTALL['localconf.php'] as $key => $value) { switch((string)$key) { case 'typo_db_username': if (strlen($value)<50) { if (strcmp(TYPO3_db_username,$value)) $this->setValueInLocalconfFile($lines, '$typo_db_username', trim($value)); } else { $this->errorMessages[] = ' Username \'' . $value . '\' was longer than 50 chars (...not saved) '; } break; case 'typo_db_password': if (strlen($value)<50) { if (strcmp(TYPO3_db_password,$value)) $this->setValueInLocalconfFile($lines, '$typo_db_password', trim($value)); } else { $this->errorMessages[] = ' Password was longer than 50 chars (...not saved) '; } break; case 'typo_db_host': if (preg_match('/^[a-zA-Z0-9_\.-]+(:.+)?$/',$value) && strlen($value)<50) { if (strcmp(TYPO3_db_host,$value)) $this->setValueInLocalconfFile($lines, '$typo_db_host', $value); } else { $this->errorMessages[] = ' Host \'' . $value . '\' was not alphanumeric (a-z, A-Z, 0-9 or _-.), or longer than 50 chars (...not saved) '; } break; case 'typo_db': if (strlen($value)<50) { if (strcmp(TYPO3_db,$value)) $this->setValueInLocalconfFile($lines, '$typo_db', trim($value)); } else { $this->errorMessages[] = ' Database name \'' . $value . '\' was longer than 50 chars (...not saved) '; } break; case 'disable_exec_function': if (strcmp($GLOBALS['TYPO3_CONF_VARS']['BE']['disable_exec_function'],$value)) $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\'BE\'][\'disable_exec_function\']', $value?1:0); break; case 'sitename': if (strcmp($GLOBALS['TYPO3_CONF_VARS']['SYS']['sitename'],$value)) $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\'SYS\'][\'sitename\']', $value); break; case 'encryptionKey': if (strcmp($GLOBALS['TYPO3_CONF_VARS']['SYS']['encryptionKey'],$value)) $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\'SYS\'][\'encryptionKey\']', $value); break; case 'compat_version': if (strcmp($GLOBALS['TYPO3_CONF_VARS']['SYS']['compat_version'],$value)) $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\'SYS\'][\'compat_version\']', $value); break; case 'im_combine_filename': if (strcmp($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_combine_filename'],$value)) $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\'GFX\'][\'im_combine_filename\']', $value); break; case 'gdlib': case 'gdlib_png': case 'im': if (strcmp($GLOBALS['TYPO3_CONF_VARS']['GFX'][$key], $value)) { $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\'GFX\'][\'' . $key . '\']', ($value ? 1 : 0)); } break; case 'im_path': list($value,$version) = explode('|',$value); if (strcmp($GLOBALS['TYPO3_CONF_VARS']['GFX'][$key], $value)) { $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\'GFX\'][\'' . $key . '\']', $value); } if (doubleval($version) > 0 && doubleval($version) < 4) { // Assume GraphicsMagick $value_ext = 'gm'; } elseif (doubleval($version) < 5) { // Assume ImageMagick 4.x $value_ext = ''; } elseif (doubleval($version) >= 6) { // Assume ImageMagick 6.x $value_ext = 'im6'; } else { // Assume ImageMagick 5.x $value_ext = 'im5'; } if (strcmp(strtolower($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5']), $value_ext)) { $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\'GFX\'][\'im_version_5\']', $value_ext); } break; case 'im_path_lzw': list($value) = explode('|',$value); if (strcmp($GLOBALS['TYPO3_CONF_VARS']['GFX'][$key], $value)) { $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\'GFX\'][\'' . $key . '\']', $value); } break; case 'TTFdpi': if (strcmp($GLOBALS['TYPO3_CONF_VARS']['GFX']['TTFdpi'],$value)) $this->setValueInLocalconfFile($lines, '$TYPO3_CONF_VARS[\'GFX\'][\'TTFdpi\']', $value); break; } } // Hook to modify localconf.php lines in the 1-2-3 installer if (is_array ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install/mod/class.tx_install.php']['writeLocalconf'])) { foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install/mod/class.tx_install.php']['writeLocalconf'] as $classData) { $hookObject = t3lib_div::getUserObj($classData); $hookObject->executeWriteLocalconf($lines, $this->step, $this); } } $this->writeToLocalconf_control($lines); } break; } return $out; } /** * Writes or returns lines from localconf.php * * @param array $lines Array of lines to write back to localconf.php. Possibly * @param boolean $showOutput If TRUE then print what has been done. * @return mixed If $lines is not an array it will return an array with the lines from localconf.php. Otherwise it will return a status string, either "continue" (updated) or "nochange" (not updated) * @see parent::writeToLocalconf_control() */ function writeToLocalconf_control($lines='', $showOutput=TRUE) { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'WriteToLocalConfControl.html'); $returnVal = parent::writeToLocalconf_control($lines); if ($showOutput) { switch($returnVal) { case 'continue': // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###CONTINUE###'); // Get the subpart for messages $messagesSubPart = t3lib_parsehtml::getSubpart($template, '###MESSAGES###'); $messages = array(); foreach ($this->messages as $message) { // Define the markers content $messagesMarkers['message'] = $message; // Fill the markers in the subpart $messages[] = t3lib_parsehtml::substituteMarkerArray( $messagesSubPart, $messagesMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for messages $content = t3lib_parsehtml::substituteSubpart( $template, '###MESSAGES###', implode(LF, $messages) ); // Define the markers content $markers = array( 'header' => 'Writing to \'localconf.php\'', 'action' => $this->action, 'label' => 'Click to continue...' ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $content, $markers, '###|###', TRUE, FALSE ); $this->outputExitBasedOnStep($content); break; case 'nochange': // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###NOCHANGE###'); // Define the markers content $markers = array( 'header' => 'Writing to \'localconf.php\'', 'message' => 'No values were changed, so nothing is updated!', 'action' => $this->action, 'label' => 'Click to continue...' ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $template, $markers, '###|###', TRUE, FALSE ); $this->outputExitBasedOnStep($content); break; } } return $returnVal; } /** * If in 1-2-3 mode, send a redirect header response with the action and exit * otherwise send output to output() function * * @param string $content The HTML to output * @return void */ function outputExitBasedOnStep($content) { if ($this->step) { t3lib_utility_Http::redirect($this->action); } else { $this->output($this->outputWrapper($content)); } exit; } /** * This appends something to value in the input array based on $type. Private. * * @param array $arr * @param integer $type * @return array */ function setLabelValueArray($arr,$type) { foreach ($arr as $k => $v) { if($this->config_array['im_versions'][$v]['gm']) { $program = 'gm'; } else { $program = 'convert'; } switch($type) { // value, im case 0: $arr[$k].='|'.$this->config_array['im_versions'][$v][$program]; break; // labels, im case 1: if($this->config_array['im_versions'][$v][$program]) { $arr[$k].= ' ('.$this->config_array['im_versions'][$v][$program]; $arr[$k].= ($this->config_array['im_versions'][$v]['gif_capability'] ? ', '.$this->config_array['im_versions'][$v]['gif_capability'] : ''); $arr[$k].= ')'; } else { $arr[$k].= ''; } break; // labels, gd case 2: $arr[$k].=' ('.($v==1?'PNG':'GIF').')'; break; } } return $arr; } /** * Returns the list of available databases (with access-check based on username/password) * * @return array List of available databases */ public function getDatabaseList() { $dbArr = array(); if ($result = $GLOBALS['TYPO3_DB']->sql_pconnect(TYPO3_db_host, TYPO3_db_username, TYPO3_db_password)) { $dbArr = $GLOBALS['TYPO3_DB']->admin_get_dbs(); } return $dbArr; } /** * Calculates the suggested setup that should be written to localconf.php * * If safe_mode * - disable_exec_function = 1 * - im = 0 * * if PNG/GIF/GD * - disable gdlib if nothing * - select png/gif if only one of them is available, else PNG/GIF selector, defaulting to GIF * - (safe_mode is on) * - im_path (default to 4.2.9, preferable with LZW) im_ver5-flag is set based on im_path being 4.2.9 or 5+ * - im_path_lzw (default to LZW version, pref. 4.2.9) * * @return array Suggested setup */ function setupGeneralCalculate() { $formArray['disable_exec_function']=array(0); $formArray['im_path']=array(''); $formArray['im_path_lzw']=array(''); $formArray['im_combine_filename']=array(''); $formArray['im_version_5']=array(''); $formArray['im']=array(1); $formArray['gdlib']=array(1); if ($this->config_array['gd'] && ($this->config_array['gd_gif'] || $this->config_array['gd_png'])) { if ($this->config_array['gd_gif'] && !$this->config_array['gd_png']) { $formArray['gdlib_png']=array(0); } elseif (!$this->config_array['gd_gif'] && $this->config_array['gd_png']) { $formArray['gdlib_png']=array(1); } else { $formArray['gdlib_png']=array(0,1); } } else { $formArray['gdlib']=array(0); } if ($this->config_array['safemode']) { $formArray['disable_exec_function']=array(1); } if ($this->config_array['im']) { $formArray['im'] = array(1); $found = $LZW_found = 0; $totalArr = array(); foreach ($this->config_array['im_versions'] as $path => $dat) { if (count($dat)>=3) { if (doubleval($dat['convert'])<5) { $formArray['im_version_5']=array(0); if ($dat['gif_capability']=='LZW') { $formArray['im_path']=array($path); $found=2; } elseif ($found<2) { $formArray['im_path']=array($path); $found=1; } } elseif (doubleval($dat['convert']) >= 6) { $formArray['im_version_5'] = array('im6'); if ($dat['gif_capability'] == 'LZW') { $formArray['im_path'] = array($path); $found = 2; } elseif ($found < 2) { $formArray['im_path'] = array($path); $found = 1; } } elseif (!$found) { $formArray['im_version_5']=array('im5'); $formArray['im_path']=array($path); $found=1; } } elseif ($dat['gm']) { $formArray['im_version_5']=array('gm'); if ($dat['gif_capability']=='LZW') { $formArray['im_path']=array($path); $found=2; } elseif ($found<2) { $formArray['im_path']=array($path); $found=1; } } if ($dat['gif_capability']=='LZW') { if (doubleval($dat['convert'])<5 || !$LZW_found) { $formArray['im_path_lzw']=array($path); $LZW_found=1; } } elseif ($dat['gif_capability']=="RLE" && !$LZW_found) { $formArray['im_path_lzw']=array($path); } $totalArr[]=$path; } $formArray['im_path']=array_unique(array_merge($formArray['im_path'],$totalArr)); $formArray['im_path_lzw']=array_unique(array_merge($formArray['im_path_lzw'],$totalArr)); $formArray['im_combine_filename']=array($this->config_array['im_combine_filename']); } else { $formArray['im']=array(0); } return $formArray; } /** * Returns true if TTF lib is installed. * * @return boolean TRUE if TrueType support */ function isTTF() { // Return right away if imageTTFtext does not exist. if (!function_exists('imagettftext')) { return 0; } // try, print truetype font: $im = @imagecreate(300, 50); $background_color = imagecolorallocate($im, 255, 255, 55); $text_color = imagecolorallocate($im, 233, 14, 91); $test = @imagettftext($im, t3lib_div::freetypeDpiComp(20), 0, 10, 20, $text_color, PATH_t3lib."/fonts/vera.ttf", 'Testing Truetype support'); if (t3lib_div::_GP('testingTrueTypeSupport')) { if ($this->isGIF()) { header('Content-type: image/gif'); imagegif($im); } else { header('Content-type: image/png'); imagepng($im); } exit; } return (is_array($test) ? 1 : 0); } /** * Checks if the essential PHP modules are loaded * * @return array list of modules which are missing */ protected function getMissingPhpModules() { // Hook to adjust the required PHP modules in the 1-2-3 installer $modules = $this->requiredPhpModules; if (is_array ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install/mod/class.tx_install.php']['requiredPhpModules'])) { foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install/mod/class.tx_install.php']['requiredPhpModules'] as $classData) { $hookObject = t3lib_div::getUserObj($classData); $modules = $hookObject->setRequiredPhpModules($modules, $this); } } $this->requiredPhpModules = $modules; $result = array(); foreach ($this->requiredPhpModules as $module) { if (is_array($module)) { $detectedSubmodules = FALSE; foreach ($module as $submodule) { if (extension_loaded($submodule)) { $detectedSubmodules = TRUE; } } if ($detectedSubmodules === FALSE) { $result[] = 'one of: (' . implode(', ', $module) . ')'; } } else { if (!extension_loaded($module)) { $result[] = $module; } } } return $result; } /***************************************** * * ABOUT the isXXX functions. * * I had a very real experience that these checks DID NOT fail eg PNG support if it didn't exist! * So first (1) we check if the functions are there. If they ARE we are going to make further investigations (2) by creating an actual image. * And if THAT succeeds also, then we can be certain of the support! */ /** * Check if GD module is available by checking the function imagecreate * * @return boolean TRUE if GD is available */ function isGD() { if (function_exists('imagecreatetruecolor')) { if (@imagecreatetruecolor(50, 100)) { return 1; } } return 0; } /** * Check if GIF functions are available * * @return boolean TRUE if GIF functions are available */ function isGIF() { // If GIF-functions exists, also do a real test of them: if (function_exists('imagecreatefromgif') && function_exists('imagegif') && ($this->ImageTypes() & IMG_GIF)) { $im = @imagecreatefromgif(t3lib_extMgm::extPath('install').'imgs/jesus.gif'); return ($im ? 1 : 0); } } /** * Check if JPG functions are available * * @return boolean TRUE if JPEG functions are available */ function isJPG() { if (function_exists('imagecreatefromjpeg') && function_exists('imagejpeg') && ($this->ImageTypes() & IMG_JPG)) { return 1; } } /** * Check if PNG functions are available * * @return boolean TRUE if PNG functions are available */ function isPNG() { if (function_exists('imagecreatefrompng') && function_exists('imagepng') && ($this->ImageTypes() & IMG_PNG)) { $im = imagecreatefrompng(t3lib_extMgm::extPath('install').'imgs/jesus.png'); return ($im ? 1 : 0); } } /** * Return the image types supported by this PHP build * * @return integer A bit-field corresponding to the image formats supported by the version of GD linked into PHP. The following bits are returned, IMG_GIF | IMG_JPG | IMG_PNG | IMG_WBMP | IMG_XPM. */ function ImageTypes() { return imagetypes(); } /** * Returns general information about GDlib * * @return string HTML with GD lib information */ function getGDSoftwareInfo() { return ' <p> You can get GDLib in the PNG version from <a href="http://www.libgd.org/">http://www.libgd.org/</a> <br /> FreeType is for download at <a href="http://www.freetype.org/">http://www.freetype.org/</a> <br /> Generally, TYPO3 packages are listed at <a href="' . TYPO3_URL_DOWNLOAD . '">' . TYPO3_URL_DOWNLOAD . '</a> </p> '; } /** * Returns general information about configuration of TYPO3. * * @return string HTML with the general information */ function generallyAboutConfiguration() { return ' <p> Local configuration is done by overriding default values in the included file, typo3conf/localconf.php. In this file you enter the database information along with values in the global array TYPO3_CONF_VARS. <br /> The options in the TYPO3_CONF_VARS array and how to use it for your own purposes is discussed in the base configuration file, t3lib/config_default.php. This file sets up the default values and subsequently includes the localconf.php file in which you can then override values. <br /> See this page for <a href="' . TYPO3_URL_SYSTEMREQUIREMENTS . '">more information about system requirements.</a> </p>'; } /********************** * * IMAGE processing * **********************/ /** * jesus.TIF: IBM/LZW * jesus.GIF: Save for web, 32 colors * jesus.JPG: Save for web, 30 quality * jesus.PNG: Save for web, PNG-24 * jesus.tga 24 bit TGA file * jesus.pcx * jesus.bmp 24 bit BMP file * jesus_ps6.PDF: PDF w/layers and vector data * typo3logo.ai: Illustrator 8 file * pdf_from_imagemagick.PDF PDF-file made by Acrobat Distiller from InDesign PS-file * * * Imagemagick * - Read formats * - Write png, gif, jpg * * Problems may arise from the use of safe_mode (eg. png) * In safemode you will automatically execute the program convert in the safe_mode_exec_path no matter what other path you specify * check fileexist before anything... * * - compare gif size * - scaling (by stdgraphic) * - combining (by stdgraphic) * * GDlib: * - create from:.... * - ttf text * * From TypoScript: (GD only, GD+IM, IM) * * @return void */ function checkTheImageProcessing() { $this->message('Image Processing', 'What is it?', ' <p> TYPO3 is known for its ability to process images on the server. <br /> In the backend interface (TBE) thumbnails are automatically generated (by ImageMagick in thumbs.php) as well as icons, menu items and pane tabs (by GDLib). <br /> In the TypoScript enabled frontend all kinds of graphical elements are processed. Typically images are scaled down to fit the pages (by ImageMagick) and menu items, graphical headers and such are generated automatically (by GDLib + ImageMagick). <br /> In addition TYPO3 is able to handle many file formats (thanks to ImageMagick), for example TIF, BMP, PCX, TGA, AI and PDF in addition to the standard web formats; JPG, GIF, PNG. </p> <p> In order to do this, TYPO3 uses two sets of tools: </p> <p> <strong>ImageMagick / GraphicsMagick:</strong> <br /> For conversion of non-web formats to webformats, combining images with alpha-masks, performing image-effects like blurring and sharpening. <br /> ImageMagick is a collection of external programs on the server called by the exec() function in PHP. TYPO3 uses three of these, namely \'convert\' (converting fileformats, scaling, effects), \'combine\'/\'composite\' (combining images with masks) and \'identify\' (returns image information). GraphicsMagick is an alternative to ImageMagick and can be enabled by setting [GFX][im_version_5] to \'gm\'. This is recommended and enabled by default. <br /> Because ImageMagick and Graphicsmagick are external programs, two requirements must be met: 1) The programs must be installed on the server and working and 2) if safe_mode is enabled, the programs must be located in the folder defined by the php.ini setting, <em>safe_mode_exec_dir</em> (else they are not executed). <br /> ImageMagick is available for both Windows and Unix. The current version is 6+. <br /> ImageMagick homepage is at <a href="http://www.imagemagick.org/">http://www.imagemagick.org/</a> </p> <p> <strong>GDLib:</strong> <br /> For drawing boxes and rendering text on images with truetype fonts. Also used for icons, menuitems and generally the TypoScript GIFBUILDER object is based on GDlib, but extensively utilizing ImageMagick to process intermediate results. <br /> GDLib is accessed through internal functions in PHP, so in this case, you have no safe_mode problems, but you\'ll need a version of PHP with GDLib compiled in. Also in order to use TrueType fonts with GDLib you\'ll need FreeType compiled in as well. <br /> </p> ' . $this->getGDSoftwareInfo() . ' <p> You can disable all image processing options in TYPO3 ([GFX][image_processing]=0), but that would seriously disable TYPO3. </p> '); $this->message('Image Processing', 'Verifying the image processing capabilities of your server', ' <p> This page performs image processing and displays the result. It\'s a thorough check that everything you\'ve configured is working correctly. <br /> It\'s quite simple to verify your installation; Just look down the page, the images in pairs should look like each other. If some images are not alike, something is wrong. You may also notice warnings and errors if this tool found signs of any problems. </p> <p> The image to the right is the reference image (how it should be) and to the left the image made by your server. <br /> The reference images are made with the classic ImageMagick install based on the 4.2.9 RPM and 5.2.3 RPM. If the version 5 flag is set, the reference images are made by the 5.2.3 RPM. </p> <p> This test will work only if your ImageMagick/GDLib configuration allows it to. The typo3temp/ folder must be writable for all the temporary image files. They are all prefixed \'install_\' so they are easy to recognize and delete afterwards. </p> '); $im_path = $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_path']; if ($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5']=='gm') { $im_path_version = $this->config_array['im_versions'][$im_path]['gm']; } else { $im_path_version = $this->config_array['im_versions'][$im_path]['convert']; } $im_path_lzw = $GLOBALS['TYPO3_CONF_VARS']['GFX']['im_path_lzw']; $im_path_lzw_version = $this->config_array['im_versions'][$im_path_lzw]['convert']; $msg = ' <dl id="t3-install-imageprocessingim"> <dt> ImageMagick enabled: </dt> <dd> ' . htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['GFX']['im']) . ' </dd> <dt> ImageMagick path: </dt> <dd> ' . htmlspecialchars($im_path) . ' <span>(' . htmlspecialchars($im_path_version) . ')</span> </dd> <dt> ImageMagick path/LZW: </dt> <dd> ' . htmlspecialchars($im_path_lzw) . ' <span>(' . htmlspecialchars($im_path_lzw_version) . ')</span> </dd> <dt> Version 5/GraphicsMagick flag: </dt> <dd> ' . ($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5'] ? htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5']) : ' ') . ' </dd> </dl> <dl id="t3-install-imageprocessingother"> <dt> GDLib enabled: </dt> <dd> ' . ($GLOBALS['TYPO3_CONF_VARS']['GFX']['gdlib'] ? htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['GFX']['gdlib']) : ' ') . ' </dd> <dt> GDLib using PNG: </dt> <dd> ' . ($GLOBALS['TYPO3_CONF_VARS']['GFX']['gdlib_png'] ? htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['GFX']['gdlib_png']) : ' ') . ' </dd> <dt> IM5 effects enabled: </dt> <dd> ' . htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_v5effects']) . ' <span>(Blurring/Sharpening with IM 5+)</span> </dd> <dt> Freetype DPI: </dt> <dd> ' . htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['GFX']['TTFdpi']) . ' <span>(Should be 96 for Freetype 2)</span> </dd> <dt> Mask invert: </dt> <dd> ' . htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_imvMaskState']) . ' <span>(Should be set for some IM versions approx. 5.4+)</span> </dd> </dl> <dl id="t3-install-imageprocessingfileformats"> <dt> File Formats: </dt> <dd> ' . htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['GFX']['imagefile_ext']) . ' </dd> </dl> '; // Various checks to detect IM/GM version mismatches $mismatch=false; switch (strtolower($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5'])) { case 'gm': if (doubleval($im_path_version)>=2) $mismatch=true; break; case 'im4': if (doubleval($im_path_version)>=5) $mismatch=true; break; default: if (($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5']?true:false) != (doubleval($im_path_version)>=5)) $mismatch=true; break; } if ($mismatch) { $msg .= ' <p> Warning: Mismatch between the version of ImageMagick' . ' (' . htmlspecialchars($im_path_version) . ') and the configuration of ' . '[GFX][im_version_5] (' . htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5']) . ') </p> '; $etype=2; } else $etype=1; if ($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5']=='gm') { $msg = str_replace('ImageMagick','GraphicsMagick',$msg); } $this->message('Image Processing', 'Current configuration',$msg,$etype); if (!$GLOBALS['TYPO3_CONF_VARS']['GFX']['image_processing']) { $this->message('Image Processing', 'Image Processing disabled!', ' <p> Image Processing is disabled by the config flag [GFX][image_processing] set to false (zero) </p> ', 2); $this->output($this->outputWrapper($this->printAll())); return; } if (!$this->config_array['dir_typo3temp']) { $this->message('Image Processing', 'typo3temp/ not writable!', ' <p> You must make typo3temp/ write enabled before you can proceed with this test. </p> ', 2); $this->output($this->outputWrapper($this->printAll())); return; } $msg = ' <p> <a id="testmenu"></a> Click each of these links in turn to test a topic. <strong> Please be aware that each test may take several seconds! </strong>: </p> ' . $this->imagemenu(); $this->message('Image Processing','Testmenu',$msg,''); $parseStart = t3lib_div::milliseconds(); $imageProc = t3lib_div::makeInstance('t3lib_stdGraphic'); $imageProc->init(); $imageProc->tempPath = $this->typo3temp_path; $imageProc->dontCheckForExistingTempFile=1; // $imageProc->filenamePrefix='install_'.($GLOBALS['TYPO3_CONF_VARS']['GFX']['im_version_5']?"v5":""); $imageProc->filenamePrefix='install_'; $imageProc->dontCompress=1; $imageProc->alternativeOutputKey='TYPO3_INSTALL_SCRIPT'; $imageProc->noFramePrepended=$GLOBALS['TYPO3_CONF_VARS']['GFX']['im_noFramePrepended']; // Very temporary!!! $imageProc->dontUnlinkTempFiles=0; $imActive = ($this->config_array['im'] && $im_path); $gdActive = ($this->config_array['gd'] && $GLOBALS['TYPO3_CONF_VARS']['GFX']['gdlib']); switch($this->INSTALL['images_type']) { case 'read': $refParseTime='5600'; // 4.2.9 $refParseTime='3300'; // 5.2.3 $headCode = 'Reading and converting images'; $this->message($headCode, 'Supported file formats', ' <p> This verifies that your ImageMagick installation is able to read the nine default file formats; JPG, GIF, PNG, TIF, BMP, PCX, TGA, PDF, AI. The tool \'identify\' will be used to read the pixeldimensions of non-web formats. The tool \'convert\' is used to read the image and write a temporary JPG-file </p> '); if ($imActive) { // Reading formats - writing JPG $extArr = explode(',','jpg,gif,png,tif,bmp,pcx,tga'); foreach ($extArr as $ext) { if ($this->isExtensionEnabled($ext, $headCode, "Read ".strtoupper($ext))) { $imageProc->IM_commands=array(); $theFile = t3lib_extMgm::extPath('install').'imgs/jesus.'.$ext; if (!@is_file($theFile)) die('Error: '.$theFile.' was not a file'); $imageProc->imageMagickConvert_forceFileNameBody='read_'.$ext; $fileInfo = $imageProc->imageMagickConvert($theFile,'jpg',"",'',"",'',"",1); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,"Read ".strtoupper($ext),$result[0],$result[1]); } } if ($this->isExtensionEnabled('pdf', $headCode, 'Read PDF')) { $imageProc->IM_commands=array(); $theFile = t3lib_extMgm::extPath('install').'imgs/pdf_from_imagemagick.pdf'; if (!@is_file($theFile)) die('Error: '.$theFile.' was not a file'); $imageProc->imageMagickConvert_forceFileNameBody='read_pdf'; $fileInfo = $imageProc->imageMagickConvert($theFile,'jpg',"170",'',"",'',"",1); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,'Read PDF',$result[0],$result[1]); } if ($this->isExtensionEnabled('ai', $headCode, 'Read AI')) { $imageProc->IM_commands=array(); $theFile = t3lib_extMgm::extPath('install').'imgs/typo3logotype.ai'; if (!@is_file($theFile)) die('Error: '.$theFile.' was not a file'); $imageProc->imageMagickConvert_forceFileNameBody='read_ai'; $fileInfo = $imageProc->imageMagickConvert($theFile,'jpg',"170",'',"",'',"",1); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,'Read AI',$result[0],$result[1]); } } else { $this->message($headCode, 'Test skipped', ' <p> Use of ImageMagick has been disabled in the configuration. <br /> Refer to section \'Basic Configuration\' to change or review you configuration settings </p> ', 2); } break; case 'write': $refParseTime='300'; // Writingformats - writing JPG $headCode = 'Writing images'; $this->message($headCode, 'Writing GIF and PNG', ' <p> This verifies that ImageMagick is able to write GIF and PNG files. <br /> The GIF-file is attempted compressed with LZW by the t3lib_div::gif_compress() function. </p> '); if ($imActive) { // Writing GIF $imageProc->IM_commands=array(); $theFile = t3lib_extMgm::extPath('install').'imgs/jesus.gif'; if (!@is_file($theFile)) die('Error: '.$theFile.' was not a file'); $imageProc->imageMagickConvert_forceFileNameBody='write_gif'; $fileInfo = $imageProc->imageMagickConvert($theFile,'gif',"",'',"",'',"",1); if ($GLOBALS['TYPO3_CONF_VARS']['GFX']['gif_compress']) { clearstatcache(); $prevSize=t3lib_div::formatSize(@filesize($fileInfo[3])); $returnCode = t3lib_div::gif_compress($fileInfo[3],''); clearstatcache(); $curSize=t3lib_div::formatSize(@filesize($fileInfo[3])); $note = array('Note on gif_compress() function:',"The 'gif_compress' method used was '".$returnCode."'.<br />Previous filesize: ".$prevSize.'. Current filesize:'.$curSize); } else $note=array('Note on gif_compress() function:','<em>Not used! Disabled by [GFX][gif_compress]</em>'); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands, $note); $this->message($headCode,'Write GIF',$result[0],$result[1]); // Writing PNG $imageProc->IM_commands=array(); $theFile = t3lib_extMgm::extPath('install').'imgs/jesus.gif'; $imageProc->imageMagickConvert_forceFileNameBody='write_png'; $fileInfo = $imageProc->imageMagickConvert($theFile,'png',"",'',"",'',"",1); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,'Write PNG',$result[0],$result[1]); } else { $this->message($headCode, 'Test skipped', ' <p> Use of ImageMagick has been disabled in the configuration. <br /> Refer to section \'Basic Configuration\' to change or review you configuration settings </p> ', 2); } break; case 'scaling': $refParseTime='650'; // Scaling $headCode = 'Scaling images'; $this->message($headCode, 'Scaling transparent images', ' <p> This shows how ImageMagick reacts when scaling transparent GIF and PNG files. </p> '); if ($imActive) { // Scaling transparent image $imageProc->IM_commands=array(); $theFile = t3lib_extMgm::extPath('install').'imgs/jesus2_transp.gif'; if (!@is_file($theFile)) die('Error: '.$theFile.' was not a file'); $imageProc->imageMagickConvert_forceFileNameBody='scale_gif'; $fileInfo = $imageProc->imageMagickConvert($theFile,'gif',"150",'',"",'',"",1); if ($GLOBALS['TYPO3_CONF_VARS']['GFX']['gif_compress']) { clearstatcache(); $prevSize=t3lib_div::formatSize(@filesize($fileInfo[3])); $returnCode = t3lib_div::gif_compress($fileInfo[3],''); clearstatcache(); $curSize=t3lib_div::formatSize(@filesize($fileInfo[3])); $note = array('Note on gif_compress() function:',"The 'gif_compress' method used was '".$returnCode."'.<br />Previous filesize: ".$prevSize.'. Current filesize:'.$curSize); } else $note=array('Note on gif_compress() function:','<em>Not used! Disabled by [GFX][gif_compress]</em>'); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands,$note); $this->message($headCode,'GIF to GIF, 150 pixels wide',$result[0],$result[1]); $imageProc->IM_commands=array(); $theFile = t3lib_extMgm::extPath('install').'imgs/jesus2_transp.png'; if (!@is_file($theFile)) die('Error: '.$theFile.' was not a file'); $imageProc->imageMagickConvert_forceFileNameBody='scale_png'; $fileInfo = $imageProc->imageMagickConvert($theFile,'png',"150",'',"",'',"",1); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,'PNG to PNG, 150 pixels wide',$result[0],$result[1]); $imageProc->IM_commands=array(); $theFile = t3lib_extMgm::extPath('install').'imgs/jesus2_transp.gif'; if (!@is_file($theFile)) die('Error: '.$theFile.' was not a file'); $imageProc->imageMagickConvert_forceFileNameBody='scale_jpg'; $fileInfo = $imageProc->imageMagickConvert($theFile,'jpg',"150",'',"",'',"",1); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,'GIF to JPG, 150 pixels wide',$result[0],$result[1]); } else { $this->message($headCode, 'Test skipped', ' <p> Use of ImageMagick has been disabled in the configuration. <br /> Refer to section \'Basic Configuration\' to change or review you configuration settings </p> ', 2); } break; case 'combining': $refParseTime='150'; // 4.2.9 $refParseTime='250'; // 5.2.3 // Combine $headCode = 'Combining images'; $this->message($headCode, 'Combining images', ' <p> This verifies that the ImageMagick tool, \'combine\'/\'composite\', is able to combine two images through a grayscale mask. <br /> If the masking seems to work but inverted, that just means you\'ll have to make sure the invert flag is set (some combination of im_negate_mask/im_imvMaskState) </p> '); if ($imActive) { $imageProc->IM_commands=array(); $input = t3lib_extMgm::extPath('install').'imgs/greenback.gif'; $overlay = t3lib_extMgm::extPath('install').'imgs/jesus.jpg'; $mask = t3lib_extMgm::extPath('install').'imgs/blackwhite_mask.gif'; if (!@is_file($input)) die('Error: '.$input.' was not a file'); if (!@is_file($overlay)) die('Error: '.$overlay.' was not a file'); if (!@is_file($mask)) die('Error: '.$mask.' was not a file'); $output = $imageProc->tempPath.$imageProc->filenamePrefix.t3lib_div::shortMD5($imageProc->alternativeOutputKey.'combine1').'.jpg'; $imageProc->combineExec($input,$overlay,$mask,$output, true); $fileInfo = $imageProc->getImageDimensions($output); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,'Combine using a GIF mask with only black and white',$result[0],$result[1]); // Combine $imageProc->IM_commands=array(); $input = t3lib_extMgm::extPath('install').'imgs/combine_back.jpg'; $overlay = t3lib_extMgm::extPath('install').'imgs/jesus.jpg'; $mask = t3lib_extMgm::extPath('install').'imgs/combine_mask.jpg'; if (!@is_file($input)) die('Error: '.$input.' was not a file'); if (!@is_file($overlay)) die('Error: '.$overlay.' was not a file'); if (!@is_file($mask)) die('Error: '.$mask.' was not a file'); $output = $imageProc->tempPath.$imageProc->filenamePrefix.t3lib_div::shortMD5($imageProc->alternativeOutputKey.'combine2').'.jpg'; $imageProc->combineExec($input,$overlay,$mask,$output, true); $fileInfo = $imageProc->getImageDimensions($output); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,'Combine using a JPG mask with graylevels',$result[0],$result[1]); } else { $this->message($headCode, 'Test skipped', ' <p> Use of ImageMagick has been disabled in the configuration. <br /> Refer to section \'Basic Configuration\' to change or review you configuration settings </p> ', 2); } break; case 'gdlib': // GIF / 4.2.9 / LZW (5.2.3) $refParseTime='1800'; // PNG / 4.2.9 / LZW (5.2.3) $refParseTime='2700'; // GIF / 5.2.3 / LZW (5.2.3) $refParseTime='1600'; // GDLibrary $headCode = 'GDLib'; $this->message($headCode, 'Testing GDLib', ' <p> This verifies that the GDLib installation works properly. </p> '); if ($gdActive) { // GD with box $imageProc->IM_commands=array(); $im = imagecreatetruecolor(170, 136); $Bcolor = ImageColorAllocate ($im, 0, 0, 0); ImageFilledRectangle($im, 0, 0, 170, 136, $Bcolor); $workArea=array(0,0,170,136); $conf=array( 'dimensions' => '10,50,150,36', 'color' => 'olive' ); $imageProc->makeBox($im,$conf,$workArea); $output = $imageProc->tempPath.$imageProc->filenamePrefix.t3lib_div::shortMD5('GDbox').'.'.$imageProc->gifExtension; $imageProc->ImageWrite($im,$output); $fileInfo = $imageProc->getImageDimensions($output); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,'Create simple image',$result[0],$result[1]); // GD from image with box $imageProc->IM_commands=array(); $input = t3lib_extMgm::extPath('install').'imgs/jesus.'.$imageProc->gifExtension; if (!@is_file($input)) die('Error: '.$input.' was not a file'); $im = $imageProc->imageCreateFromFile($input); $workArea=array(0,0,170,136); $conf=array(); $conf['dimensions']='10,50,150,36'; $conf['color']='olive'; $imageProc->makeBox($im,$conf,$workArea); $output = $imageProc->tempPath.$imageProc->filenamePrefix.t3lib_div::shortMD5('GDfromImage+box').'.'.$imageProc->gifExtension; $imageProc->ImageWrite($im,$output); $fileInfo = $imageProc->getImageDimensions($output); $GDWithBox_filesize = @filesize($output); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,'Create image from file',$result[0],$result[1]); // GD with text $imageProc->IM_commands=array(); $im = imagecreatetruecolor(170, 136); $Bcolor = ImageColorAllocate ($im, 128,128,150); ImageFilledRectangle($im, 0, 0, 170, 136, $Bcolor); $workArea=array(0,0,170,136); $conf=array( 'iterations' => 1, 'angle' => 0, 'antiAlias' => 1, 'text' => 'HELLO WORLD', 'fontColor' => '#003366', 'fontSize' => 18, 'fontFile' => $this->backPath.'../t3lib/fonts/vera.ttf', 'offset' => '17,40' ); $conf['BBOX'] = $imageProc->calcBBox($conf); $imageProc->makeText($im,$conf,$workArea); $output = $imageProc->tempPath.$imageProc->filenamePrefix.t3lib_div::shortMD5('GDwithText').'.'.$imageProc->gifExtension; $imageProc->ImageWrite($im,$output); $fileInfo = $imageProc->getImageDimensions($output); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands); $this->message($headCode,'Render text with TrueType font',$result[0],$result[1]); if ($imActive) { // extension: GD with text, niceText $conf['offset'] = '17,65'; $conf['niceText'] = 1; $imageProc->makeText($im,$conf,$workArea); $output = $imageProc->tempPath.$imageProc->filenamePrefix.t3lib_div::shortMD5('GDwithText-niceText').'.'.$imageProc->gifExtension; $imageProc->ImageWrite($im,$output); $fileInfo = $imageProc->getImageDimensions($output); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands, array("Note on 'niceText':","'niceText' is a concept that tries to improve the antialiasing of the rendered type by actually rendering the textstring in double size on a black/white mask, downscaling the mask and masking the text onto the image through this mask. This involves ImageMagick 'combine'/'composite' and 'convert'.")); $this->message($headCode, 'Render text with TrueType font using \'niceText\' option', ' <p> (If the image has another background color than the image above (eg. dark background color with light text) then you will have to set TYPO3_CONF_VARS[GFX][im_imvMaskState]=1) </p> ' . $result[0], $result[1]); } else { $this->message($headCode, 'Render text with TrueType font using \'niceText\' option', ' <p> <strong>Test is skipped!</strong> </p> <p> Use of ImageMagick has been disabled in the configuration. ImageMagick is needed to generate text with the niceText option. <br /> Refer to section \'Basic Configuration\' to change or review you configuration settings </p> ', 2); } if ($imActive) { // extension: GD with text, niceText AND shadow $conf['offset'] = '17,90'; $conf['niceText'] = 1; $conf['shadow.'] = array( 'offset'=>'2,2', 'blur' => $imageProc->V5_EFFECTS?"20":"90", 'opacity' => '50', 'color' => 'black' ); $imageProc->makeShadow($im,$conf['shadow.'],$workArea,$conf); $imageProc->makeText($im,$conf,$workArea); $output = $imageProc->tempPath.$imageProc->filenamePrefix.t3lib_div::shortMD5('GDwithText-niceText-shadow').'.'.$imageProc->gifExtension; $imageProc->ImageWrite($im,$output); $fileInfo = $imageProc->getImageDimensions($output); $result = $this->displayTwinImage($fileInfo[3],$imageProc->IM_commands, array('Note on drop shadows:','Drop shadows are done by using ImageMagick to blur a mask through which the drop shadow is generated. The blurring of the mask only works in ImageMagick 4.2.9 and <em>not</em> ImageMagick 5 - which is why you may see a hard and not soft shadow.')); $this->message($headCode, 'Render \'niceText\' with a shadow under', ' <p> (This test makes sense only if the above test had a correct output. But if so, you may not see a soft dropshadow from the third text string as you should. In that case you are most likely using ImageMagick 5 and should set the flag TYPO3_CONF_VARS[GFX][im_v5effects]. However this may cost server performance! </p> ' . $result[0], $result[1]); } else { $this->message($headCode, 'Render \'niceText\' with a shadow under', ' <p> <strong>Test is skipped!</strong> </p> <p> Use of ImageMagick has been disabled in the configuration. ImageMagick is needed to generate shadows. <br /> Refer to section \'Basic Configuration\' to change or review you configuration settings </p> ', 2); } if ($imageProc->gifExtension=='gif') { $buffer=20; $assess = "This assessment is based on the filesize from 'Create image from file' test, which were ".$GDWithBox_filesize.' bytes'; $goodNews = "If the image was LZW compressed you would expect to have a size of less than 9000 bytes. If you open the image with Photoshop and saves it from Photoshop, you'll a filesize like that.<br />The good news is (hopefully) that your [GFX][im_path_lzw] path is correctly set so the gif_compress() function will take care of the compression for you!"; if ($GDWithBox_filesize<8784+$buffer) { $msg = ' <p> <strong> Your GDLib appears to have LZW compression! </strong> <br /> This assessment is based on the filesize from \'Create image from file\' test, which were ' . $GDWithBox_filesize . ' bytes. <br /> This is a real advantage for you because you don\'t need to use ImageMagick for LZW compressing. In order to make sure that GDLib is used, <strong> please set the config option [GFX][im_path_lzw] to an empty string! </strong> <br /> When you disable the use of ImageMagick for LZW compressing, you\'ll see that the gif_compress() function has a return code of \'GD\' (for GDLib) instead of \'IM\' (for ImageMagick) </p> '; } elseif ($GDWithBox_filesize>19000) { $msg = ' <p> <strong> Your GDLib appears to have no compression at all! </strong> <br /> ' . $assess . ' <br /> ' . $goodNews . ' </p> '; } else { $msg = ' <p> Your GDLib appears to have RLE compression <br /> ' . $assess . ' <br /> ' . $goodNews . ' </p> '; } $this->message($headCode,'GIF compressing in GDLib'," ".$msg." ",1); } } else { $this->message($headCode, 'Test skipped', ' <p> Use of GDLib has been disabled in the configuration. <br /> Refer to section \'Basic Configuration\' to change or review you configuration settings </p> ', 2); } break; } if ($this->INSTALL['images_type']) { // End info if ($this->fatalError) { $this->message('Info', 'Errors', ' <p> It seems that you had some fatal errors in this test. Please make sure that your ImageMagick and GDLib settings are correct. Refer to the \'Basic Configuration\' section for more information and debugging of your settings. </p> '); } $parseMS = t3lib_div::milliseconds() - $parseStart; $this->message('Info', 'Parsetime', ' <p> ' . $parseMS . ' ms </p> '); } $this->output($this->outputWrapper($this->printAll())); } /** * Check if image file extension is enabled * Adds error message to the message array * * @param string $ext The image file extension * @param string $headCode The header for the message * @param string $short The short description for the message * @return boolean TRUE if extension is enabled */ function isExtensionEnabled($ext, $headCode, $short) { if (!t3lib_div::inList($GLOBALS['TYPO3_CONF_VARS']['GFX']['imagefile_ext'],$ext)) { $this->message($headCode, $short, ' <p> Skipped - extension not in the list of allowed extensions ([GFX][imagefile_ext]). </p> ',1); } else { return 1; } } /** * Generate the HTML after reading and converting images * Displays the verification and the converted image if succeeded * Adds error messages if needed * * @param string $imageFile The file name of the converted image * @param array $IMcommands The ImageMagick commands used * @param string $note Additional note for image operation * @return array Contains content and highest error level */ function displayTwinImage ($imageFile, $IMcommands=array(), $note='') { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'DisplayTwinImage.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); $content = ''; $errorLevels=array(-1); if ($imageFile) { // Get the subpart for the images $imageSubpart = t3lib_parsehtml::getSubpart($template, '###IMAGE###'); $verifyFile = t3lib_extMgm::extPath('install').'verify_imgs/'.basename($imageFile); $destImg = @getImageSize($imageFile); $destImgCode ='<img src="'.$this->backPath.'../'.substr($imageFile,strlen(PATH_site)).'" '.$destImg[3].'>'; $verifyImg = @getImageSize($verifyFile); $verifyImgCode = '<img src="'.$this->backPath.t3lib_extMgm::extRelPath('install').'verify_imgs/'.basename($verifyFile).'" '.$verifyImg[3].'>'; clearstatcache(); $destImg['filesize'] = @filesize($imageFile); clearstatcache(); $verifyImg['filesize'] = @filesize($verifyFile); // Define the markers content $imageMarkers = array( 'destWidth' => $destImg[0], 'destHeight' => $destImg[1], 'destUrl' => $this->backPath . '../' . substr($imageFile, strlen(PATH_site)), 'verifyWidth' => $verifyImg[0], 'verifyHeight' => $verifyImg[1], 'verifyUrl' => $this->backPath . t3lib_extMgm::extRelPath('install') . 'verify_imgs/' . basename($verifyFile), 'yourServer' => 'Your server:', 'yourServerInformation' => t3lib_div::formatSize($destImg['filesize']) . ', ' . $destImg[0] . 'x' . $destImg[1] . ' pixels', 'reference' => 'Reference:', 'referenceInformation' => t3lib_div::formatSize($verifyImg['filesize']) . ', ' . $verifyImg[0] . 'x' . $verifyImg[1] . ' pixels' ); if ($destImg[0]!=$verifyImg[0] || $destImg[1]!=$verifyImg[1]) { // Get the subpart for the different pixel dimensions message $differentPixelDimensionsSubpart = t3lib_parsehtml::getSubpart($imageSubpart, '###DIFFERENTPIXELDIMENSIONS###'); // Define the markers content $differentPixelDimensionsMarkers = array( 'message' => 'Pixel dimension are not equal!' ); // Fill the markers in the subpart $differentPixelDimensionsSubpart = t3lib_parsehtml::substituteMarkerArray( $differentPixelDimensionsSubpart, $differentPixelDimensionsMarkers, '###|###', TRUE, FALSE ); $errorLevels[]=2; } // Substitute the subpart for different pixel dimensions message $imageSubpart = t3lib_parsehtml::substituteSubpart( $imageSubpart, '###DIFFERENTPIXELDIMENSIONS###', $differentPixelDimensionsSubpart ); if ($note) { // Get the subpart for the note $noteSubpart = t3lib_parsehtml::getSubpart($imageSubpart, '###NOTE###'); // Define the markers content $noteMarkers = array( 'message' => $note[0], 'label' => $note[1] ); // Fill the markers in the subpart $noteSubpart = t3lib_parsehtml::substituteMarkerArray( $noteSubpart, $noteMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the note $imageSubpart = t3lib_parsehtml::substituteSubpart( $imageSubpart, '###NOTE###', $noteSubpart ); if ($this->dumpImCommands && count($IMcommands)) { $commands = $this->formatImCmds($IMcommands); // Get the subpart for the ImageMagick commands $imCommandsSubpart = t3lib_parsehtml::getSubpart($imageSubpart, '###IMCOMMANDS###'); // Define the markers content $imCommandsMarkers = array( 'message' => 'ImageMagick commands executed:', 'rows' => t3lib_div::intInRange(count($commands), 2, 10), 'commands' => htmlspecialchars(implode(LF, $commands)) ); // Fill the markers in the subpart $imCommandsSubpart = t3lib_parsehtml::substituteMarkerArray( $imCommandsSubpart, $imCommandsMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the ImageMagick commands $imageSubpart = t3lib_parsehtml::substituteSubpart( $imageSubpart, '###IMCOMMANDS###', $imCommandsSubpart ); // Fill the markers $imageSubpart = t3lib_parsehtml::substituteMarkerArray( $imageSubpart, $imageMarkers, '###|###', TRUE, FALSE ); } else { // Get the subpart when no image has been generated $noImageSubpart = t3lib_parsehtml::getSubpart($template, '###NOIMAGE###'); $commands = $this->formatImCmds($IMcommands); if (count($commands)) { // Get the subpart for the ImageMagick commands $commandsSubpart = t3lib_parsehtml::getSubpart($noImageSubpart, '###COMMANDSAVAILABLE###'); // Define the markers content $commandsMarkers = array( 'rows' => t3lib_div::intInRange(count($commands), 2, 10), 'commands' => htmlspecialchars(implode(LF, $commands)) ); // Fill the markers in the subpart $commandsSubpart = t3lib_parsehtml::substituteMarkerArray( $commandsSubpart, $commandsMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the ImageMagick commands $noImageSubpart = t3lib_parsehtml::substituteSubpart( $noImageSubpart, '###COMMANDSAVAILABLE###', $commandsSubpart ); // Define the markers content $noImageMarkers = array( 'message' => 'There was no result from the ImageMagick operation', 'label' => 'Below there\'s a dump of the ImageMagick commands executed:' ); // Fill the markers $noImageSubpart = t3lib_parsehtml::substituteMarkerArray( $noImageSubpart, $noImageMarkers, '###|###', TRUE, FALSE ); $errorLevels[]=3; } // Substitute the subpart when image has been generated $content = t3lib_parsehtml::substituteSubpart( $template, '###IMAGE###', $imageSubpart ); // Substitute the subpart when no image has been generated $content = t3lib_parsehtml::substituteSubpart( $content, '###NOIMAGE###', $noImageSubpart ); return array($content, max($errorLevels)); } /** * Format ImageMagick commands for use in HTML * * @param array $arr The ImageMagick commands * @return string The formatted commands */ function formatImCmds($arr) { $out = array(); if (is_array($arr)) { foreach ($arr as $k => $v) { $out[] = $v[1]; if ($v[2]) { $out[]=' RETURNED: ' . $v[2]; } } } return $out; } /** * Generate the menu for the test menu in 'image processing' * * @return string The HTML for the test menu */ function imagemenu() { // Get the template file $template = @file_get_contents(PATH_site . $this->templateFilePath . 'ImageMenu.html'); // Get the subpart for the menu $menuSubPart = t3lib_parsehtml::getSubpart($template, '###MENU###'); // Get the subpart for the single item in the menu $menuItemSubPart = t3lib_parsehtml::getSubpart($menuSubPart, '###MENUITEM###'); $menuitems = array( 'read' => 'Reading image formats', 'write' => 'Writing GIF and PNG', 'scaling' => 'Scaling images', 'combining' => 'Combining images', 'gdlib' => 'GD library functions' ); $c = 0; $items = array(); foreach ($menuitems as $k => $v) { // Define the markers content $markers = array( 'backgroundColor' => $this->INSTALL['images_type'] == $k ? 'activeMenu' : 'generalTableBackground', 'url' => htmlspecialchars($this->action . '&TYPO3_INSTALL[images_type]=' . $k . '#imageMenu'), 'item' => $v ); // Fill the markers in the subpart $items[] = t3lib_parsehtml::substituteMarkerArray( $menuItemSubPart, $markers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the single item in the menu $menuSubPart = t3lib_parsehtml::substituteSubpart( $menuSubPart, '###MENUITEM###', implode(LF, $items) ); return $menuSubPart; } /********************** * * DATABASE analysing * **********************/ /** * The Database Analyzer * * @return void */ function checkTheDatabase() { if (!$this->config_array['mysqlConnect']) { $this->message('Database Analyser', 'Your database connection failed', ' <p> Please go to the \'Basic Configuration\' section and correct this problem first. </p> ', 2); $this->output($this->outputWrapper($this->printAll())); return; } if ($this->config_array['no_database']) { $this->message('Database Analyser', 'No database selected', ' <p> Please go to the \'Basic Configuration\' section and correct this problem first. </p> ', 2); $this->output($this->outputWrapper($this->printAll())); return; } // Getting current tables $whichTables=$this->getListOfTables(); // Getting number of static_template records if ($whichTables['static_template']) { $static_template_count = $GLOBALS['TYPO3_DB']->exec_SELECTcountRows('uid', 'static_template'); } $static_template_count=intval($static_template_count); $headCode ='Database Analyser'; $this->message($headCode, 'What is it?', ' <p> In this section you can get an overview of your currently selected database compared to sql-files. You can also import sql-data directly into the database or upgrade tables from earlier versions of TYPO3. </p> ', 0); $this->message($headCode, 'Connected to SQL database successfully', ' <dl id="t3-install-databaseconnected"> <dt> Username: </dt> <dd> ' . htmlspecialchars(TYPO3_db_username) . ' </dd> <dt> Host: </dt> <dd> ' . htmlspecialchars(TYPO3_db_host) . ' </dd> </dl> ', -1, 1); $this->message($headCode, 'Database', ' <p> <strong>' . htmlspecialchars(TYPO3_db) . '</strong> is selected as database. <br /> Has <strong>' . count($whichTables) . '</strong> tables. </p> ', -1, 1); // Menu $sql_files = array_merge( t3lib_div::getFilesInDir(PATH_typo3conf,'sql',1,1), array() ); $action_type = $this->INSTALL['database_type']; $actionParts = explode('|',$action_type); if (count($actionParts)<2) { $action_type=''; } // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'CheckTheDatabaseMenu.html'); // Get the template part from the file $menu = t3lib_parsehtml::getSubpart($templateFile, '###MENU###'); $menuMarkers = array( 'action' => $this->action, 'updateRequiredTables' => 'Update required tables', 'compare' => 'COMPARE', 'noticeCmpFileCurrent' => ($action_type == 'cmpFile|CURRENT_TABLES' ? ' class="notice"' : ''), 'dumpStaticData' => 'Dump static data', 'import' => 'IMPORT', 'noticeImportCurrent' => ($action_type == 'import|CURRENT_STATIC' ? ' class="notice"' : ''), 'noticeCmpTca' => ($action_type == "cmpTCA|" ? ' class="notice"' : ''), 'compareWithTca' => 'Compare with $TCA', 'noticeAdminUser' => ($action_type == "adminUser|" ? ' class="notice"' : ''), 'createAdminUser' => 'Create "admin" user', 'noticeUc' => ($action_type == "UC|" ? ' class="notice"' : ''), 'resetUserPreferences' => 'Reset user preferences', 'noticeCache' => ($action_type == "cache|" ? ' class="notice"' : ''), 'clearTables' => 'Clear tables' ); // Get the subpart for extra SQL $extraSql = t3lib_parsehtml::getSubpart($menu, '###EXTRASQL###'); $directJump=''; $extraSqlFiles = array(); foreach ($sql_files as $k => $file) { if ($this->mode=="123" && !count($whichTables) && strstr($file,'_testsite')) { $directJump = $this->action.'&TYPO3_INSTALL[database_type]=import|'.rawurlencode($file); } $lf=t3lib_div::testInt($k); $fShortName = substr($file,strlen(PATH_site)); $spec1 = $spec2 = ''; // Define the markers content $extraSqlMarkers = array( 'fileShortName' => $fShortName, 'fileSize' => t3lib_div::formatSize(filesize($file)), 'noticeCmpFile' => ($action_type == 'cmpFile|' . $file ? ' class="notice"' : ''), 'file' => rawurlencode($file), 'noticeImport' => ($action_type == 'import|' . $file ? ' class="notice"' : ''), 'specs' => $spec1 . $spec2, 'noticeView' => ($action_type == 'view|' . $file ? ' class="notice"' : ''), 'view' => 'VIEW' ); // Fill the markers in the subpart $extraSqlFiles[] = t3lib_parsehtml::substituteMarkerArray( $extraSql, $extraSqlMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for extra SQL $menu = t3lib_parsehtml::substituteSubpart( $menu, '###EXTRASQL###', implode(LF, $extraSqlFiles) ); // Fill the markers $menu = t3lib_parsehtml::substituteMarkerArray( $menu, $menuMarkers, '###|###', TRUE, FALSE ); if ($directJump) { if (!$action_type) { $this->message($headCode, 'Menu', ' <script language="javascript" type="text/javascript"> window.location.href = "' . $directJump . '"; </script>', 0, 1); } } else { $this->message($headCode, 'Menu', ' <p> From this menu you can select which of the available SQL files you want to either compare or import/merge with the existing database. </p> <dl id="t3-install-checkthedatabaseexplanation"> <dt> COMPARE: </dt> <dd> Compares the tables and fields of the current database and the selected file. It also offers to \'update\' the difference found. </dd> <dt> IMPORT: </dt> <dd> Imports the SQL-dump file into the current database. You can either dump the raw file or choose which tables to import. In any case, you\'ll see a new screen where you must confirm the operation. </dd> <dt> VIEW: </dt> <dd> Shows the content of the SQL-file, limiting characters on a single line to a reader-friendly amount. </dd> </dl> <p> The SQL-files are selected from typo3conf/ (here you can put your own) and t3lib/stddb/ (TYPO3 distribution). The SQL-files should be made by the <em>mysqldump</em> tool or at least be formatted like that tool would do. </p> ' . $menu, 0, 1); } if ($action_type) { switch($actionParts[0]) { case 'cmpFile': $tblFileContent=''; if (!strcmp($actionParts[1],'CURRENT_TABLES')) { $tblFileContent = t3lib_div::getUrl(PATH_t3lib.'stddb/tables.sql'); foreach ($GLOBALS['TYPO3_LOADED_EXT'] as $loadedExtConf) { if (is_array($loadedExtConf) && $loadedExtConf['ext_tables.sql']) { $tblFileContent .= LF . LF . LF . LF . t3lib_div::getUrl($loadedExtConf['ext_tables.sql']); } } } elseif (@is_file($actionParts[1])) { $tblFileContent = t3lib_div::getUrl($actionParts[1]); } if ($tblFileContent) { $fileContent = implode( LF, $this->getStatementArray($tblFileContent,1,'^CREATE TABLE ') ); $FDfile = $this->getFieldDefinitions_fileContent($fileContent); if (!count($FDfile)) { die ("Error: There were no 'CREATE TABLE' definitions in the provided file"); } // Updating database... if (is_array($this->INSTALL['database_update'])) { $FDdb = $this->getFieldDefinitions_database(); $diff = $this->getDatabaseExtra($FDfile, $FDdb); $update_statements = $this->getUpdateSuggestions($diff); $diff = $this->getDatabaseExtra($FDdb, $FDfile); $remove_statements = $this->getUpdateSuggestions($diff,'remove'); $results = array(); $results[] = $this->performUpdateQueries($update_statements['clear_table'], $this->INSTALL['database_update']); $results[] = $this->performUpdateQueries($update_statements['add'], $this->INSTALL['database_update']); $results[] = $this->performUpdateQueries($update_statements['change'], $this->INSTALL['database_update']); $results[] = $this->performUpdateQueries($remove_statements['change'], $this->INSTALL['database_update']); $results[] = $this->performUpdateQueries($remove_statements['drop'], $this->INSTALL['database_update']); $results[] = $this->performUpdateQueries($update_statements['create_table'], $this->INSTALL['database_update']); $results[] = $this->performUpdateQueries($remove_statements['change_table'], $this->INSTALL['database_update']); $results[] = $this->performUpdateQueries($remove_statements['drop_table'], $this->INSTALL['database_update']); $this->databaseUpdateErrorMessages = array(); foreach ($results as $resultSet) { if (is_array($resultSet)) { foreach ($resultSet as $key => $errorMessage) { $this->databaseUpdateErrorMessages[$key] = $errorMessage; } } } } // Init again / first time depending... $FDdb = $this->getFieldDefinitions_database(); $diff = $this->getDatabaseExtra($FDfile, $FDdb); $update_statements = $this->getUpdateSuggestions($diff); $diff = $this->getDatabaseExtra($FDdb, $FDfile); $remove_statements = $this->getUpdateSuggestions($diff,'remove'); $tLabel = 'Update database tables and fields'; if ($remove_statements || $update_statements) { $formContent = $this->generateUpdateDatabaseForm('get_form',$update_statements,$remove_statements,$action_type); $this->message($tLabel, 'Table and field definitions should be updated', ' <p> There seems to be a number of differencies between the database and the selected SQL-file. <br /> Please select which statements you want to execute in order to update your database: </p> ' . $formContent, 2); } else { $formContent = $this->generateUpdateDatabaseForm('get_form',$update_statements,$remove_statements,$action_type); $this->message($tLabel, 'Table and field definitions are OK.', ' <p> The tables and fields in the current database corresponds perfectly to the database in the selected SQL-file. </p> ', -1); } } break; case 'cmpTCA': $this->includeTCA(); $FDdb = $this->getFieldDefinitions_database(); // Displaying configured fields which are not in the database $tLabel='Tables and fields in $TCA, but not in database'; $cmpTCA_DB = $this->compareTCAandDatabase($GLOBALS['TCA'],$FDdb); if (!count($cmpTCA_DB['extra'])) { $this->message($tLabel, 'Table and field definitions OK', ' <p> All fields and tables configured in $TCA appeared to exist in the database as well </p> ', -1); } else { $this->message($tLabel, 'Invalid table and field definitions in $TCA!', ' <p> There are some tables and/or fields configured in the $TCA array which do not exist in the database! <br /> This will most likely cause you trouble with the TYPO3 backend interface! </p> ', 3); foreach ($cmpTCA_DB['extra'] as $tableName => $conf) { $this->message($tLabel, $tableName,$this->displayFields($conf['fields'],0,'Suggested database field:'),2); } } // Displaying tables that are not setup in $cmpDB_TCA = $this->compareDatabaseAndTCA($FDdb,$GLOBALS['TCA']); $excludeTables='be_sessions,fe_session_data,fe_sessions'; if (TYPO3_OS=='WIN') {$excludeTables = strtolower($excludeTables);} $excludeFields = array( 'be_users' => 'uc,lastlogin,usergroup_cached_list', 'fe_users' => 'uc,lastlogin,fe_cruser_id', 'pages' => 'SYS_LASTCHANGED', 'sys_dmail' => 'mailContent', 'tt_board' => 'doublePostCheck', 'tt_guest' => 'doublePostCheck', 'tt_products' => 'ordered' ); $tCount=0; $fCount=0; $tLabel="Tables from database, but not in \$TCA"; $fLabel="Fields from database, but not in \$TCA"; $this->message($tLabel); if (is_array($cmpDB_TCA['extra'])) { foreach ($cmpDB_TCA['extra'] as $tableName => $conf) { if (!t3lib_div::inList($excludeTables,$tableName) && substr($tableName,0,4)!="sys_" && substr($tableName,-3)!="_mm" && substr($tableName,0,6)!="index_" && substr($tableName,0,6)!='cache_') { if ($conf['whole_table']) { $this->message($tLabel, $tableName,$this->displayFields($conf['fields']),1); $tCount++; } else { list($theContent, $fC) = $this->displaySuggestions($conf['fields'],$excludeFields[$tableName]); $fCount+=$fC; if ($fC) { $this->message($fLabel, $tableName,$theContent,1); } } } } } if (!$tCount) { $this->message($tLabel, 'Correct number of tables in the database', ' <p> There are no extra tables in the database compared to the configured tables in the $TCA array. </p> ', -1); } else { $this->message($tLabel, 'Extra tables in the database', ' <p> There are some tables in the database which are not configured in the $TCA array. <br /> You should probably not worry about this, but please make sure that you know what these tables are about and why they are not configured in $TCA. </p> ', 2); } if (!$fCount) { $this->message($fLabel, 'Correct number of fields in the database', ' <p> There are no additional fields in the database tables compared to the configured fields in the $TCA array. </p> ', -1); } else { $this->message($fLabel, 'Extra fields in the database', ' <p> There are some additional fields the database tables which are not configured in the $TCA array. <br /> You should probably not worry about this, but please make sure that you know what these fields are about and why they are not configured in $TCA. </p> ', 2); } // Displaying actual and suggested field database defitions if (is_array($cmpTCA_DB['matching'])) { $tLabel="Comparison between database and \$TCA"; $this->message($tLabel, 'Actual and suggested field definitions', ' <p> This table shows you the suggested field definitions which are calculated based on the configuration in $TCA. <br /> If the suggested value differs from the actual current database value, you should not panic, but simply check if the datatype of that field is sufficient compared to the data, you want TYPO3 to put there. </p> ', 0); foreach ($cmpTCA_DB['matching'] as $tableName => $conf) { $this->message($tLabel, $tableName,$this->displayFieldComp($conf['fields'], $FDdb[$tableName]['fields']),1); } } break; case 'import': $mode123Imported=0; $tblFileContent=''; if (preg_match('/^CURRENT_/', $actionParts[1])) { if (!strcmp($actionParts[1],'CURRENT_TABLES') || !strcmp($actionParts[1],'CURRENT_TABLES+STATIC')) { $tblFileContent = t3lib_div::getUrl(PATH_t3lib.'stddb/tables.sql'); foreach ($GLOBALS['TYPO3_LOADED_EXT'] as $loadedExtConf) { if (is_array($loadedExtConf) && $loadedExtConf['ext_tables.sql']) { $tblFileContent.= LF . LF . LF . LF . t3lib_div::getUrl($loadedExtConf['ext_tables.sql']); } } } if (!strcmp($actionParts[1],'CURRENT_STATIC') || !strcmp($actionParts[1],'CURRENT_TABLES+STATIC')) { foreach ($GLOBALS['TYPO3_LOADED_EXT'] as $loadedExtConf) { if (is_array($loadedExtConf) && $loadedExtConf['ext_tables_static+adt.sql']) { $tblFileContent.= LF . LF . LF . LF . t3lib_div::getUrl($loadedExtConf['ext_tables_static+adt.sql']); } } } } elseif (@is_file($actionParts[1])) { $tblFileContent = t3lib_div::getUrl($actionParts[1]); } if ($tblFileContent) { $tLabel='Import SQL dump'; // Getting statement array from $statements = $this->getStatementArray($tblFileContent,1); list($statements_table, $insertCount) = $this->getCreateTables($statements,1); // Updating database... if ($this->INSTALL['database_import_all']) { $r=0; foreach ($statements as $k=>$v) { $res = $GLOBALS['TYPO3_DB']->admin_query($v); $r++; } // Make a database comparison because some tables that are defined twice have // not been created at this point. This applies to the "pages.*" // fields defined in sysext/cms/ext_tables.sql for example. $fileContent = implode( LF, $this->getStatementArray($tblFileContent,1,'^CREATE TABLE ') ); $FDfile = $this->getFieldDefinitions_fileContent($fileContent); $FDdb = $this->getFieldDefinitions_database(); $diff = $this->getDatabaseExtra($FDfile, $FDdb); $update_statements = $this->getUpdateSuggestions($diff); if (is_array($update_statements['add'])) { foreach ($update_statements['add'] as $statement) { $res = $GLOBALS['TYPO3_DB']->admin_query($statement); } } if ($this->mode=='123') { // Create default be_user admin/password $username = 'admin'; $pass = 'password'; $count = $GLOBALS['TYPO3_DB']->exec_SELECTcountRows( 'uid', 'be_users', 'username=' . $GLOBALS['TYPO3_DB']->fullQuoteStr($username, 'be_users') ); if (!$count) { $insertFields = array( 'username' => $username, 'password' => md5($pass), 'admin' => 1, 'uc' => '', 'fileoper_perms' => 0, 'tstamp' => $GLOBALS['EXEC_TIME'], 'crdate' => $GLOBALS['EXEC_TIME'] ); $GLOBALS['TYPO3_DB']->exec_INSERTquery('be_users', $insertFields); } } $this->message($tLabel, 'Imported ALL', ' <p> Queries: ' . $r . ' </p> ', 1, 1); if (t3lib_div::_GP('goto_step')) { $this->action.='&step='.t3lib_div::_GP('goto_step'); t3lib_utility_Http::redirect($this->action); } } elseif (is_array($this->INSTALL['database_import'])) { // Traverse the tables foreach ($this->INSTALL['database_import'] as $table => $md5str) { if ($md5str==md5($statements_table[$table])) { $res = $GLOBALS['TYPO3_DB']->admin_query('DROP TABLE IF EXISTS '.$table); $res = $GLOBALS['TYPO3_DB']->admin_query($statements_table[$table]); if ($insertCount[$table]) { $statements_insert = $this->getTableInsertStatements($statements, $table); foreach ($statements_insert as $k => $v) { $res = $GLOBALS['TYPO3_DB']->admin_query($v); } } $this->message($tLabel, 'Imported \'' . $table . '\'', ' <p> Rows: ' . $insertCount[$table] . ' </p> ', 1, 1); } } } $mode123Imported=$this->isBasicComplete($tLabel); if (!$mode123Imported) { // Re-Getting current tables - may have been changed during import $whichTables=$this->getListOfTables(); if (count($statements_table)) { reset($statements_table); $out=''; // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'CheckTheDatabaseImport.html'); // Get the template part from the file $content = t3lib_parsehtml::getSubpart($templateFile, '###IMPORT###'); if ($this->mode!='123') { $tables = array(); // Get the subpart for regular mode $regularModeSubpart = t3lib_parsehtml::getSubpart($content, '###REGULARMODE###'); foreach ($statements_table as $table => $definition) { // Get the subpart for rows $tableSubpart = t3lib_parsehtml::getSubpart($content, '###ROWS###'); // Fill the 'table exists' part when it exists $exist=isset($whichTables[$table]); if ($exist) { // Get the subpart for table exists $existSubpart = t3lib_parsehtml::getSubpart($tableSubpart, '###EXIST###'); // Define the markers content $existMarkers = array ( 'tableExists' => 'Table exists!', 'backPath' => $this->backPath ); // Fill the markers in the subpart $existSubpart = t3lib_parsehtml::substituteMarkerArray( $existSubpart, $existMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for table exists $tableHtml = t3lib_parsehtml::substituteSubpart( $tableSubpart, '###EXIST###', $existSubpart ); // Define the markers content $tableMarkers = array ( 'table' => $table, 'definition' => md5($definition), 'count' => $insertCount[$table] ? $insertCount[$table] : '', 'rowLabel' => $insertCount[$table] ? 'Rows: ' : '', 'tableExists' => 'Table exists!', 'backPath' => $this->backPath ); // Fill the markers $tables[] = t3lib_parsehtml::substituteMarkerArray( $tableHtml, $tableMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the rows $regularModeSubpart = t3lib_parsehtml::substituteSubpart( $regularModeSubpart, '###ROWS###', implode(LF, $tables) ); } // Substitute the subpart for the regular mode $content = t3lib_parsehtml::substituteSubpart( $content, '###REGULARMODE###', $regularModeSubpart ); // Define the markers content $contentMarkers = array( 'checked' => ($this->mode == '123' || t3lib_div::_GP('presetWholeTable') ? 'checked="checked"' : ''), 'label' => 'Import the whole file \'' . basename($actionParts[1]) . '\' directly (ignores selections above)' ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $content, $contentMarkers, '###|###', TRUE, FALSE ); $form = $this->getUpdateDbFormWrap($action_type, $content); $this->message($tLabel, 'Select tables to import', ' <p> This is an overview of the CREATE TABLE definitions in the SQL file. <br /> Select which tables you want to dump to the database. <br /> Any table you choose dump to the database is dropped from the database first, so you\'ll lose all data in existing tables. </p> ' . $form, 1, 1); } else { $this->message($tLabel, 'No tables', ' <p> There seems to be no CREATE TABLE definitions in the SQL file. <br /> This tool is intelligently creating one table at a time and not just dumping the whole content of the file uncritically. That\'s why there must be defined tables in the SQL file. </p> ', 3, 1); } } } break; case 'view': if (@is_file($actionParts[1])) { $tLabel = 'Import SQL dump'; // Getting statement array from $fileContent = t3lib_div::getUrl($actionParts[1]); $statements = $this->getStatementArray($fileContent, 1); $maxL = 1000; $strLen = strlen($fileContent); $maxlen = 200+($maxL-t3lib_div::intInRange(($strLen-20000)/100,0,$maxL)); if (count($statements)) { $out = ''; foreach ($statements as $statement) { $out.= '<p>' . nl2br(htmlspecialchars(t3lib_div::fixed_lgd_cs($statement,$maxlen))) . '</p>'; } } $this->message($tLabel,'Content of '.basename($actionParts[1]),$out,1); } break; // Create admin user case 'adminUser': if ($whichTables['be_users']) { if (is_array($this->INSTALL['database_adminUser'])) { $username = preg_replace('/[^\da-z._-]/i', '', trim($this->INSTALL['database_adminUser']['username'])); $pass = trim($this->INSTALL['database_adminUser']['password']); $pass2 = trim($this->INSTALL['database_adminUser']['password2']); if ($username && $pass && $pass2) { if ($pass != $pass2) { $this->message($headCode, 'Passwords are not equal!', ' <p> The passwords entered twice are not equal. </p> ', 2, 1); } else { $res = $GLOBALS['TYPO3_DB']->exec_SELECTquery('uid', 'be_users', 'username='.$GLOBALS['TYPO3_DB']->fullQuoteStr($username, 'be_users')); if (!$GLOBALS['TYPO3_DB']->sql_num_rows($res)) { $insertFields = array( 'username' => $username, 'password' => md5($pass), 'admin' => 1, 'uc' => '', 'fileoper_perms' => 0, 'tstamp' => $GLOBALS['EXEC_TIME'], 'crdate' => $GLOBALS['EXEC_TIME'] ); $result = $GLOBALS['TYPO3_DB']->exec_INSERTquery('be_users', $insertFields); $this->isBasicComplete($headCode); if ($result) { $this->message($headCode, 'User created', ' <p> Username: <strong>' . htmlspecialchars($username) . ' </strong> </p> ', 1, 1); } else { $this->message($headCode, 'User not created', ' <p> Error: <strong>' . htmlspecialchars($GLOBALS['TYPO3_DB']->sql_error()) . ' </strong> </p> ', 3, 1); } } else { $this->message($headCode, 'Username not unique!', ' <p> The username, <strong>' . htmlspecialchars($username) . ' </strong> , was not unique. </p> ', 2, 1); } } } else { $this->message($headCode, 'Missing data!', ' <p> Not all required form fields have been filled. </p> ', 2, 1); } } // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'CheckTheDatabaseAdminUser.html'); // Get the template part from the file $content = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Define the markers content $contentMarkers = array( 'userName' => 'username - unique, no space, lowercase', 'password' => 'password', 'repeatPassword' => 'password (repeated)' ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $content, $contentMarkers, '###|###', TRUE, FALSE ); $form = $this->getUpdateDbFormWrap($action_type, $content); $this->message($headCode, 'Create admin user', ' <p> Enter username and password for a new admin user. <br /> You should use this function only if there are no admin users in the database, for instance if this is a blank database. <br /> After you\'ve created the user, log in and add the rest of the user information, like email and real name. </p> ' . $form, 0, 1); } else { $this->message($headCode, 'Required table not in database', ' <p> \'be_users\' must be a table in the database! </p> ', 3, 1); } break; // clear uc case 'UC': if ($whichTables['be_users']) { if (!strcmp($this->INSTALL['database_UC'],1)) { $GLOBALS['TYPO3_DB']->exec_UPDATEquery('be_users', '', array('uc' => '')); $this->message($headCode, 'Clearing be_users.uc', ' <p> Done. </p> ', 1); } // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'CheckTheDatabaseUc.html'); // Get the template part from the file $content = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Define the markers content $contentMarkers = array( 'clearBeUsers' => 'Clear be_users preferences ("uc" field)' ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $content, $contentMarkers, '###|###', TRUE, FALSE ); $form = $this->getUpdateDbFormWrap($action_type, $content); $this->message($headCode, 'Clear user preferences', ' <p> If you press this button all backend users from the tables be_users will have their user preferences cleared (field \'uc\' set to an empty string). <br /> This may come in handy in rare cases where that configuration may be corrupt. <br /> Clearing this will clear all user settings from the \'Setup\' module. </p> ' . $form); } else { $this->message($headCode, 'Required table not in database', ' <p> \'be_users\' must be a table in the database! </p> ', 3); } break; case 'cache': $tableListArr = explode(',','cache_pages,cache_pagesection,cache_hash,cache_imagesizes,--div--,sys_log,sys_history,--div--,be_sessions,fe_sessions,fe_session_data'. (t3lib_extMgm::isLoaded('indexed_search') ? ',--div--,index_words,index_rel,index_phash,index_grlist,index_section,index_fulltext' : ''). (t3lib_extMgm::isLoaded('tt_products') ? ',--div--,sys_products_orders,sys_products_orders_mm_tt_products' : ''). (t3lib_extMgm::isLoaded('direct_mail') ? ',--div--,sys_dmail_maillog' : ''). (t3lib_extMgm::isLoaded('sys_stat') ? ',--div--,sys_stat' : '') ); if (is_array($this->INSTALL['database_clearcache'])) { $qList = array(); // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'CheckTheDatabaseCache.html'); // Get the subpart for emptied tables $emptiedTablesSubpart = t3lib_parsehtml::getSubpart($templateFile, '###EMPTIEDTABLES###'); // Get the subpart for table $tableSubpart = t3lib_parsehtml::getSubpart($emptiedTablesSubpart, '###TABLE###'); foreach ($tableListArr as $table) { if ($table!='--div--') { $table_c = TYPO3_OS=='WIN' ? strtolower($table) : $table; if ($this->INSTALL['database_clearcache'][$table] && $whichTables[$table_c]) { $GLOBALS['TYPO3_DB']->exec_TRUNCATEquery($table); // Define the markers content $emptiedTablesMarkers = array( 'tableName' => $table ); // Fill the markers in the subpart $qList[] = t3lib_parsehtml::substituteMarkerArray( $tableSubpart, $emptiedTablesMarkers, '###|###', TRUE, FALSE ); } } } // Substitute the subpart for table $emptiedTablesSubpart = t3lib_parsehtml::substituteSubpart( $emptiedTablesSubpart, '###TABLE###', implode(LF, $qList) ); if (count($qList)) { $this->message($headCode, 'Clearing cache', ' <p> The following tables were emptied: </p> ' . $emptiedTablesSubpart, 1); } } // Count entries and make checkboxes $labelArr = array( 'cache_pages' => 'Pages', 'cache_pagesection' => 'TS template related information', 'cache_hash' => 'Multipurpose md5-hash cache', 'cache_imagesizes' => 'Cached image sizes', 'sys_log' => 'Backend action logging', 'sys_stat' => 'Page hit statistics', 'sys_history' => 'Addendum to the sys_log which tracks ALL changes to content through TCE. May become huge by time. Is used for rollback (undo) and the WorkFlow engine.', 'be_sessions' => 'Backend User sessions', 'fe_sessions' => 'Frontend User sessions', 'fe_session_data' => 'Frontend User sessions data', 'sys_dmail_maillog' => 'Direct Mail log', 'sys_products_orders' => 'tt_product orders', 'sys_products_orders_mm_tt_products' => 'relations between tt_products and sys_products_orders' ); $countEntries=array(); reset($tableListArr); // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'CheckTheDatabaseCache.html'); // Get the subpart for table list $tableListSubpart = t3lib_parsehtml::getSubpart($templateFile, '###TABLELIST###'); // Get the subpart for the group separator $groupSubpart = t3lib_parsehtml::getSubpart($tableListSubpart, '###GROUP###'); // Get the subpart for a single table $singleTableSubpart = t3lib_parsehtml::getSubpart($tableListSubpart, '###SINGLETABLE###'); $checkBoxes = array(); foreach ($tableListArr as $table) { if ($table!='--div--') { $table_c = TYPO3_OS=='WIN' ? strtolower($table) : $table; if ($whichTables[$table_c]) { $countEntries[$table] = $GLOBALS['TYPO3_DB']->exec_SELECTcountRows('*', $table); // Checkboxes: if ( $this->INSTALL['database_clearcache'][$table] || $_GET['PRESET']['database_clearcache'][$table] ) { $checked = 'checked="checked"'; } else { $checked = ''; } // Define the markers content $singleTableMarkers = array( 'table' => $table, 'checked' => $checked, 'count' => '(' . $countEntries[$table] . ' rows)', 'label' => $labelArr[$table] ); // Fill the markers in the subpart $checkBoxes[] = t3lib_parsehtml::substituteMarkerArray( $singleTableSubpart, $singleTableMarkers, '###|###', TRUE, FALSE ); } } else { $checkBoxes[] = $groupSubpart; } } // Substitute the subpart for the single tables $content = t3lib_parsehtml::substituteSubpart( $tableListSubpart, '###SINGLETABLE###', implode(LF, $checkBoxes) ); // Substitute the subpart for the group separator $content = t3lib_parsehtml::substituteSubpart( $content, '###GROUP###', '' ); $form = $this->getUpdateDbFormWrap($action_type, $content); $this->message($headCode, 'Clear out selected tables', ' <p> Pressing this button will delete all records from the selected tables. </p> ' . $form); break; } } $this->output($this->outputWrapper($this->printAll())); } /** * Generates update wizard * * @return void */ function updateWizard() { // clear cache files t3lib_extMgm::removeCacheFiles(t3lib_extMgm::getCacheFilePrefix()); // call wizard $action = ($this->INSTALL['database_type'] ? $this->INSTALL['database_type'] : 'checkForUpdate'); $this->updateWizard_parts($action); $this->output($this->outputWrapper($this->printAll())); } /** * Implements the steps for the update wizard * * @param string $action Which should be done. * @return void */ function updateWizard_parts($action) { $content = ''; $updateItems = array(); // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'UpdateWizardParts.html'); switch ($action) { // first step - check for updates available case 'checkForUpdate': // Get the subpart for check for update $checkForUpdateSubpart = t3lib_parsehtml::getSubpart($templateFile, '###CHECKFORUPDATE###'); $title = 'Step 1 - Introduction'; $updateWizardBoxes = ''; if (!$GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install']['update']) { $updatesAvailableSubpart = ' <p> <strong> No updates registered! </strong> </p> '; } else { // step through list of updates, and check if update is needed and if yes, output an explanation $updatesAvailableSubpart = t3lib_parsehtml::getSubpart($checkForUpdateSubpart, '###UPDATESAVAILABLE###'); $updateWizardBoxesSubpart = t3lib_parsehtml::getSubpart($updatesAvailableSubpart, '###UPDATEWIZARDBOXES###'); $singleUpdateWizardBoxSubpart = t3lib_parsehtml::getSubpart($updateWizardBoxesSubpart, '###SINGLEUPDATEWIZARDBOX###'); $singleUpdate = array(); foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install']['update'] as $identifier => $className) { $tmpObj = $this->getUpgradeObjInstance($className, $identifier); if ($tmpObj->shouldRenderWizard()) { $explanation = ''; $tmpObj->checkForUpdate($explanation); $updateMarkers = array( 'next' => '<button type="submit" name="TYPO3_INSTALL[update][###IDENTIFIER###]"> Next <span class="t3-install-form-button-icon-positive"> </span> </button>', 'identifier' => $identifier, 'title' => $tmpObj->getTitle(), 'explanation' => $explanation, ); // only display the message, no button if (!$tmpObj->shouldRenderNextButton()) { $updateMarkers['next'] = ''; } $singleUpdate[] = t3lib_parsehtml::substituteMarkerArray( $singleUpdateWizardBoxSubpart, $updateMarkers, '###|###', TRUE, FALSE ); } } if (!empty($singleUpdate)) { $updateWizardBoxesSubpart = t3lib_parsehtml::substituteSubpart( $updateWizardBoxesSubpart, '###SINGLEUPDATEWIZARDBOX###', implode(LF, $singleUpdate) ); $updateWizardBoxesMarkers = array( 'action' => $this->action ); $updateWizardBoxesSubpart = t3lib_parsehtml::substituteMarkerArray( $updateWizardBoxesSubpart, $updateWizardBoxesMarkers, '###|###', TRUE, FALSE ); } else { $updateWizardBoxesSubpart = ' <p> <strong> No updates to perform! </strong> </p> '; } $updatesAvailableSubpart = t3lib_parsehtml::substituteSubpart( $updatesAvailableSubpart, '###UPDATEWIZARDBOXES###', $updateWizardBoxesSubpart ); $updatesAvailableMarkers = array( 'finalStep' => 'Final Step', 'finalStepExplanation' => ' <p> When all updates are done you should check your database for required updates. <br /> Perform <strong> COMPARE DATABASE </strong> as often until no more changes are required. <br /> <br /> </p> ', 'compareDatabase' => 'COMPARE DATABASE' ); $updatesAvailableSubpart = t3lib_parsehtml::substituteMarkerArray( $updatesAvailableSubpart, $updatesAvailableMarkers, '###|###', TRUE, FALSE ); } $content = t3lib_parsehtml::substituteSubpart( $checkForUpdateSubpart, '###UPDATESAVAILABLE###', $updatesAvailableSubpart ); break; // second step - get user input and ask for final confirmation case 'getUserInput': $title = 'Step 2 - Configuration of updates'; $getUserInputSubpart = t3lib_parsehtml::getSubpart($templateFile, '###GETUSERINPUT###'); $markers = array( 'introduction' => 'The following updates will be performed:', 'showDatabaseQueries' => 'Show database queries performed', 'performUpdates' => 'Perform updates!', 'action' => $this->action ); if (!$this->INSTALL['update']) { $noUpdatesAvailableSubpart = t3lib_parsehtml::getSubpart($getUserInputSubpart, '###NOUPDATESAVAILABLE###'); $noUpdateMarkers['noUpdates'] = 'No updates selected!'; $noUpdatesAvailableSubpart = t3lib_parsehtml::substituteMarkerArray( $noUpdatesAvailableSubpart, $noUpdateMarkers, '###|###', TRUE, FALSE ); break; } else { // update methods might need to get custom data $updatesAvailableSubpart = t3lib_parsehtml::getSubpart($getUserInputSubpart, '###UPDATESAVAILABLE###'); $updateItems = array(); foreach ($this->INSTALL['update'] as $identifier => $tmp) { $updateMarkers = array(); $className = $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install']['update'][$identifier]; $tmpObj = $this->getUpgradeObjInstance($className, $identifier); $updateMarkers['identifier'] = $identifier; $updateMarkers['title'] = $tmpObj->getTitle(); if (method_exists($tmpObj,'getUserInput')) { $updateMarkers['identifierMethod'] = $tmpObj->getUserInput('TYPO3_INSTALL[update][' . $identifier . ']'); } $updateItems[] = t3lib_parsehtml::substituteMarkerArray( $updatesAvailableSubpart, $updateMarkers, '###|###', TRUE, TRUE ); } $updatesAvailableSubpart = implode(LF, $updateItems); } $content = t3lib_parsehtml::substituteSubpart( $getUserInputSubpart, '###NOUPDATESAVAILABLE###', $noUpdatesAvailableSubpart ); $content = t3lib_parsehtml::substituteSubpart( $content, '###UPDATESAVAILABLE###', $updatesAvailableSubpart ); $content = t3lib_parsehtml::substituteMarkerArray( $content, $markers, '###|###', TRUE, FALSE ); break; case 'performUpdate': // third step - perform update $title = 'Step 3 - Perform updates'; $performUpdateSubpart = t3lib_parsehtml::getSubpart($templateFile, '###PERFORMUPDATE###'); $updateItemsSubpart = t3lib_parsehtml::getSubpart($performUpdateSubpart, '###UPDATEITEMS###'); $checkUserInputSubpart = t3lib_parsehtml::getSubpart($updateItemsSubpart, '###CHECKUSERINPUT###'); $updatePerformedSubpart = t3lib_parsehtml::getSubpart($updateItemsSubpart, '###UPDATEPERFORMED###'); $noPerformUpdateSubpart = t3lib_parsehtml::getSubpart($updateItemsSubpart, '###NOPERFORMUPDATE###'); $databaseQueriesSubpart = t3lib_parsehtml::getSubpart($updatePerformedSubpart, '###DATABASEQUERIES###'); $customOutputSubpart = t3lib_parsehtml::getSubpart($updatePerformedSubpart, '###CUSTOMOUTPUT###'); if (!$this->INSTALL['update']['extList']) { break; } $GLOBALS['TYPO3_DB']->store_lastBuiltQuery = TRUE; foreach ($this->INSTALL['update']['extList'] as $identifier) { $className = $GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install']['update'][$identifier]; $tmpObj = $this->getUpgradeObjInstance($className, $identifier); $updateItemsMarkers['identifier'] = $identifier; $updateItemsMarkers['title'] = $tmpObj->getTitle(); // check user input if testing method is available if (method_exists($tmpObj,'checkUserInput') && !$tmpObj->checkUserInput($customOutput)) { $customOutput = ''; $userInputMarkers = array( 'customOutput' => ($customOutput ? $customOutput : 'Something went wrong'), 'goBack' => 'Go back to update configuration' ); $checkUserInput = t3lib_parsehtml::substituteMarkerArray( $checkUserInputSubpart, $userInputMarkers, '###|###', TRUE, FALSE ); } else { if (method_exists($tmpObj,'performUpdate')) { $customOutput = ''; $dbQueries = array(); $databaseQueries = array(); if ($tmpObj->performUpdate($dbQueries, $customOutput)) { $performUpdateMarkers['updateStatus'] = 'Update successful!'; } else { $performUpdateMarkers['updateStatus'] = 'Update FAILED!'; } if ($this->INSTALL['update']['showDatabaseQueries']) { $content .= '<br />' . implode('<br />',$dbQueries); foreach($dbQueries as $query) { $databaseQueryMarkers['query'] = $query; $databaseQueries[] = t3lib_parsehtml::substituteMarkerArray( $databaseQueriesSubpart, $databaseQueryMarkers, '###|###', TRUE, FALSE ); } } if (strlen($customOutput)) { $content.= '<br />' . $customOutput; $customOutputMarkers['custom'] = $customOutput; $customOutputItem = t3lib_parsehtml::substituteMarkerArray( $customOutputSubpart, $customOutputMarkers, '###|###', TRUE, FALSE ); } $updatePerformed = t3lib_parsehtml::substituteSubpart( $updatePerformedSubpart, '###DATABASEQUERIES###', implode(LF, $databaseQueries) ); $updatePerformed = t3lib_parsehtml::substituteSubpart( $updatePerformed, '###CUSTOMOUTPUT###', $customOutputItem ); $updatePerformed = t3lib_parsehtml::substituteMarkerArray( $updatePerformed, $performUpdateMarkers, '###|###', TRUE, FALSE ); } else { $noPerformUpdateMarkers['noUpdateMethod'] = 'No update method available!'; $noPerformUpdate = t3lib_parsehtml::substituteMarkerArray( $noPerformUpdateSubpart, $noPerformUpdateMarkers, '###|###', TRUE, FALSE ); } } $updateItem = t3lib_parsehtml::substituteSubpart( $updateItemsSubpart, '###CHECKUSERINPUT###', $checkUserInput ); $updateItem = t3lib_parsehtml::substituteSubpart( $updateItem, '###UPDATEPERFORMED###', $updatePerformed ); $updateItem = t3lib_parsehtml::substituteSubpart( $updateItem, '###NOPERFORMUPDATE###', $noPerformUpdate ); $updateItem = t3lib_parsehtml::substituteSubpart( $updateItem, '###UPDATEITEMS###', implode(LF, $updateItems) ); $updateItems[] = t3lib_parsehtml::substituteMarkerArray( $updateItem, $updateItemsMarkers, '###|###', TRUE, FALSE ); } $content = t3lib_parsehtml::substituteSubpart( $performUpdateSubpart, '###UPDATEITEMS###', implode(LF, $updateItems) ); $GLOBALS['TYPO3_DB']->store_lastBuiltQuery = FALSE; // also render the link to the next update wizard, if available $nextUpdateWizard = $this->getNextUpdadeWizardInstance($tmpObj); if ($nextUpdateWizard) { $content = t3lib_parsehtml::substituteMarkerArray( $content, array('NEXTIDENTIFIER' => $nextUpdateWizard->getIdentifier()), '###|###', TRUE, FALSE ); } else { // no next wizard, also hide the button to the next update wizard $content = t3lib_parsehtml::substituteSubpart( $content, '###NEXTUPDATEWIZARD###', '' ); } break; } $this->message('Upgrade Wizard', $title, $content); } /** * Creates instance of an upgrade object, setting the pObj, versionNumber and pObj * * @param string $className The class name * @param string $identifier The identifier of upgrade object - needed to fetch user input * @return object Newly instanciated upgrade object */ function getUpgradeObjInstance($className, $identifier) { $tmpObj = t3lib_div::getUserObj($className); $tmpObj->setIdentifier($identifier); $tmpObj->versionNumber = t3lib_div::int_from_ver(TYPO3_version); $tmpObj->pObj = $this; $tmpObj->userInput = $this->INSTALL['update'][$identifier]; return $tmpObj; } /** * Returns the next upgrade wizard object. * * Used to show the link/button to the next upgrade wizard * @param object $currentObj current Upgrade Wizard Object * @return mixed Upgrade Wizard instance or FALSE */ protected function getNextUpdadeWizardInstance($currentObj) { $isPreviousRecord = TRUE; foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['ext/install']['update'] as $identifier => $className) { // first, find the current update wizard, and then start validating the next ones if ($currentObj->getIdentifier() == $identifier) { $isPreviousRecord = FALSE; continue; } if (!$isPreviousRecord) { $nextUpdateWizard = $this->getUpgradeObjInstance($className, $identifier); if ($nextUpdateWizard->shouldRenderWizard()) { return $nextUpdateWizard; } } } return FALSE; } /** * Check if at lease one backend admin user has been created * * @return integer Amount of backend users in the database */ function isBackendAdminUser() { return $GLOBALS['TYPO3_DB']->exec_SELECTcountRows('uid', 'be_users', 'admin=1'); } /** * Check if the basic settings are complete * Only used by 1-2-3 mode * * @param string $tLabel The header for the message * @return boolean TRUE if complete */ function isBasicComplete($tLabel) { if ($this->mode=='123') { $tables = $this->getListOfTables(); if (count($tables)) { $beuser = $this->isBackendAdminUser(); } if (count($tables) && $beuser) { $mode123Imported=1; $this->message($tLabel, 'Basic Installation Completed', $this->messageBasicFinished(), -1, 1); $this->message($tLabel,'Security Risk!',$this->securityRisk().$this->alterPasswordForm(),2,1); } else { $this->message($tLabel,'Still missing something?',nl2br(' You may be missing one of these points before your TYPO3 installation is complete: '.(count($tables)?'':'- You haven\'t imported any tables yet. ') .($beuser?'':'- You haven\'t created an admin user yet. ') .' You\'re about to import a database with a complete site in it, these three points should be met. '),-1,1); } } return $mode123Imported; } /** * Generate the contents for the form for 'Database Analyzer' * when the 'COMPARE' still contains errors * * @param string $type get_form if the form needs to be generated * @param array $arr_update The tables/fields which needs an update * @param array $arr_remove The tables/fields which needs to be removed * @param string $action_type The action type * @return string HTML for the form */ function generateUpdateDatabaseForm($type, $arr_update, $arr_remove, $action_type) { $content = ''; switch($type) { case 'get_form': $content.= $this->generateUpdateDatabaseForm_checkboxes($arr_update['clear_table'],'Clear tables (use with care!)',false,true); $content.= $this->generateUpdateDatabaseForm_checkboxes($arr_update['add'],'Add fields'); $content.= $this->generateUpdateDatabaseForm_checkboxes($arr_update['change'],'Changing fields',(t3lib_extMgm::isLoaded('dbal')?0:1),0,$arr_update['change_currentValue']); $content.= $this->generateUpdateDatabaseForm_checkboxes($arr_remove['change'],'Remove unused fields (rename with prefix)',$this->setAllCheckBoxesByDefault,1); $content.= $this->generateUpdateDatabaseForm_checkboxes($arr_remove['drop'],'Drop fields (really!)',$this->setAllCheckBoxesByDefault); $content.= $this->generateUpdateDatabaseForm_checkboxes($arr_update['create_table'],'Add tables'); $content.= $this->generateUpdateDatabaseForm_checkboxes($arr_remove['change_table'],'Removing tables (rename with prefix)',$this->setAllCheckBoxesByDefault,1,$arr_remove['tables_count'],1); $content.= $this->generateUpdateDatabaseForm_checkboxes($arr_remove['drop_table'],'Drop tables (really!)',$this->setAllCheckBoxesByDefault,0,$arr_remove['tables_count'],1); $content = $this->getUpdateDbFormWrap($action_type, $content); break; default: break; } return $content; } /** * Form wrap for 'Database Analyzer' * when the 'COMPARE' still contains errors * * @param string $action_type The action type * @param string $content The form content * @param string $label The submit button label * @return string HTML of the form */ function getUpdateDbFormWrap($action_type, $content, $label='Write to database') { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'GetUpdateDbFormWrap.html'); // Get the template part from the file $form = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Define the markers content $formMarkers = array( 'action' => $this->action, 'actionType' => htmlspecialchars($action_type), 'content' => $content, 'label' => $label ); // Fill the markers $form = t3lib_parsehtml::substituteMarkerArray( $form, $formMarkers, '###|###', TRUE, FALSE ); return $form; } /** * Generates an HTML table for the setup of database tables * Used in 'Database analyzer > Compare with $TCA' * * @param array $arr Description of the table with fieldname and fieldcontent * @param boolean $pre TRUE if the field content needs to be wrapped with a <pre> tag * @param string $label The header label * @return string HTML of the table */ function displayFields($arr, $pre=0, $label='') { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'DisplayFields.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Define the markers content $templateMarkers = array( 'headerFieldName' => 'Field name:', 'headerLabel' => $label ? $label : 'Info:' ); if (is_array($arr)) { $rows = array(); // Get the subpart for rows $rowsSubpart = t3lib_parsehtml::getSubpart($template, '###ROWS###'); foreach ($arr as $fieldname => $fieldContent) { if ($pre) { $fieldContent = '<pre>'.trim($fieldContent).'</pre>'; } // Define the markers content $rowsMarkers = array( 'fieldName' => $fieldname, 'fieldContent' => $fieldContent ); // Fill the markers in the subpart $rows[] = t3lib_parsehtml::substituteMarkerArray( $rowsSubpart, $rowsMarkers, '###|###', TRUE, FALSE ); } } // Substitute the subpart for rows $template = t3lib_parsehtml::substituteSubpart( $template, '###ROWS###', implode(LF, $rows) ); // Fill the markers $template = t3lib_parsehtml::substituteMarkerArray( $template, $templateMarkers, '###|###', TRUE, FALSE ); return $template; } /** * Generates an HTML table with comparison between database and $TCA * Used in 'Database analyzer > Compare with $TCA' * * @param array $arr Description of the table with fieldname and fieldcontent * @param array $arr_db The actual content of a field in the database * @return string HTML of the table */ function displayFieldComp($arr, $arr_db) { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'DisplayFieldComp.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Define the markers content $templateMarkers = array( 'headerFieldName' => 'Field name:', 'headerSuggested' => 'Suggested value from $TCA:', 'headerActual' => 'Actual value from database:' ); $rows = array(); if (is_array($arr)) { // Get the subpart for rows $rowsSubpart = t3lib_parsehtml::getSubpart($template, '###ROWS###'); foreach ($arr as $fieldname => $fieldContent) { // This tries to equalize the types tinyint and int $str1 = $fieldContent; $str2 = trim($arr_db[$fieldname]); $str1 = str_replace('tinyint(3)','tinyint(4)',$str1); $str2 = str_replace('tinyint(3)','tinyint(4)',$str2); $str1 = str_replace('int(10)','int(11)',$str1); $str2 = str_replace('int(10)','int(11)',$str2); // Compare: if (strcmp($str1,$str2)) { $bgcolor=' class="warning"'; } else { $bgcolor=''; } // Define the markers content $rowsMarkers = array( 'fieldName' => $fieldname, 'fieldContent' => $fieldContent, 'fieldContentDb' => $arr_db[$fieldname], 'class' => $bgcolor ); // Fill the markers in the subpart $rows[] = t3lib_parsehtml::substituteMarkerArray( $rowsSubpart, $rowsMarkers, '###|###', TRUE, FALSE ); } } // Substitute the subpart for rows $template = t3lib_parsehtml::substituteSubpart( $template, '###ROWS###', implode(LF, $rows) ); // Fill the markers $out = t3lib_parsehtml::substituteMarkerArray( $template, $templateMarkers, '###|###', TRUE, FALSE ); return $out; } /** * Generates an HTML table with $TCA suggestions looking at the type of field * Used in 'Database analyzer > Compare with $TCA' * * @param array $arr Description of the table with fieldname and fieldcontent * @param string $excludeList Comma separated list of fields which should be excluded from this table * @return string HTML of the table */ function displaySuggestions($arr, $excludeList='') { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'DisplaySuggestions.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); $templateMarkers = array(); $fC=0; $rows = array(); if (is_array($arr)) { // Get the subpart for rows $rowsSubpart = t3lib_parsehtml::getSubpart($template, '###ROWS###'); foreach ($arr as $fieldname => $fieldContent) { if (!t3lib_div::inList($excludeList,$fieldname) && substr($fieldname,0,strlen($this->deletedPrefixKey))!=$this->deletedPrefixKey && substr($fieldname,-1)!='.') { if ($arr[$fieldname.'.']) { // Get the subpart for pre $preSubpart = t3lib_parsehtml::getSubpart($rowsSubpart, '###PRE###'); // Define the markers content $preMarkers = array( 'code' => '<pre>' . trim($arr[$fieldname.'.']) . '</pre>' ); // Fill the markers in the subpart $preSubpart = t3lib_parsehtml::substituteMarkerArray( $preSubpart, $preMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for pre $row = t3lib_parsehtml::substituteSubpart( $rowsSubpart, '###PRE###', $preSubpart ); // Define the markers content $rowsMarkers = array( 'headerFieldName' => 'Field name:', 'headerLabel' => 'Info:', 'headerSuggestion' => 'Suggestion for the field:', 'fieldName' => $fieldname, 'fieldContent' => $fieldContent, ); // Fill the markers in the subpart $rows[] = t3lib_parsehtml::substituteMarkerArray( $row, $rowsMarkers, '###|###', TRUE, FALSE ); $fC++; } } } // Substitute the subpart for rows $template = t3lib_parsehtml::substituteSubpart( $template, '###ROWS###', implode(LF, $rows) ); // Fill the markers $out = t3lib_parsehtml::substituteMarkerArray( $template, $templateMarkers, '###|###', TRUE, FALSE ); return array($out,$fC); } /** * Compares an array with field definitions with $TCA array * * @param array $FDsrc Field definition source * @param array $TCA The TCA array * @param boolean $onlyFields * @return array */ function compareDatabaseAndTCA($FDsrc, $TCA, $onlyFields=0) { $extraArr=array(); if (is_array($FDsrc)) { foreach ($FDsrc as $table => $info) { if (!isset($TCA[$table])) { if (!$onlyFields) { // If the table was not in the FDcomp-array, the result array is loaded with that table. $extraArr[$table]=$info; $extraArr[$table]['whole_table']=1; unset($extraArr[$table]['keys']); } } else { $theKey='fields'; $excludeListArr=array(); if (is_array($TCA[$table]['ctrl']['enablecolumns'])) { $excludeListArr[]=$TCA[$table]['ctrl']['enablecolumns']; } $excludeListArr[]=$TCA[$table]['ctrl']['tstamp']; $excludeListArr[]=$TCA[$table]['ctrl']['sortby']; $excludeListArr[]=$TCA[$table]['ctrl']['delete']; $excludeListArr[]=$TCA[$table]['ctrl']['cruser_id']; $excludeListArr[]=$TCA[$table]['ctrl']['crdate']; $excludeListArr[]='uid'; $excludeListArr[]='pid'; if ($table=='pages') { $excludeListArr[]='perms_userid'; $excludeListArr[]='perms_groupid'; $excludeListArr[]='perms_user'; $excludeListArr[]='perms_group'; $excludeListArr[]='perms_everybody'; } if ($table=='sys_dmail') { $excludeListArr[]='scheduled'; $excludeListArr[]='scheduled_begin'; $excludeListArr[]='scheduled_end'; $excludeListArr[]='query_info'; } if (is_array($info[$theKey])) { foreach ($info[$theKey] as $fieldN => $fieldC) { if (!isset($TCA[$table]['columns'][$fieldN]) && !in_array($fieldN,$excludeListArr)) { $extraArr[$table][$theKey][$fieldN] = $info['fields'][$fieldN]; $extraArr[$table][$theKey][$fieldN.'.']=$this->suggestTCAFieldDefinition($fieldN,$fieldC); } } } } } } return array('extra'=>$extraArr); } /** * Compares the $TCA array with a field definition array * * @param array $TCA The TCA * @param array $FDcomp Field definition comparison * @return array */ function compareTCAandDatabase($TCA, $FDcomp) { $matchingArr = $extraArr = array(); if (is_array($TCA)) { foreach ($TCA as $table => $info) { if (!isset($FDcomp[$table])) { // If the table was not in the FDcomp-array, the result array is loaded with that table. //$extraArr[$table]=$info; $extraArr[$table]['whole_table']=1; } else { foreach ($info['columns'] as $fieldN => $fieldC) { $fieldDef = $this->suggestFieldDefinition($fieldC); if (!is_array($fieldDef)) { if (!isset($FDcomp[$table]['fields'][$fieldN])) { $extraArr[$table]['fields'][$fieldN]=$fieldDef; } else { $matchingArr[$table]['fields'][$fieldN]=$fieldDef; } } } } } } return array('extra'=>$extraArr, 'matching'=>$matchingArr); } /** * Suggests a field definition for a TCA config array. * * @param array $fieldInfo Info of a field * @return string The suggestion */ function suggestFieldDefinition($fieldInfo) { $out=''; switch($fieldInfo['config']['type']) { case 'input': if (preg_match('/date|time|int|year/',$fieldInfo['config']['eval'])) { $out = "int(11) NOT NULL default '0'"; } else { $max = intval($fieldInfo['config']['max']); if ($max>0 && $max<200) { $out = 'varchar('.$max.") NOT NULL default ''"; } else { $out = 'tinytext'; } } break; case 'text': $out = 'text'; break; case 'check': if (is_array($fieldInfo['config']['items']) && count($fieldInfo['config']['items'])>8) { $out = "int(11) NOT NULL default '0'"; } else { $out = "tinyint(3) NOT NULL default '0'"; } break; case 'radio': if (is_array($fieldInfo['config']['items'])) { $out = $this->getItemArrayType($fieldInfo['config']['items']); } else { $out = 'ERROR: Radiobox did not have items!'; } break; case 'group': if ($fieldInfo['config']['internal_type']=='db') { $max = t3lib_div::intInRange($fieldInfo['config']['maxitems'],1,10000); if (count(explode(',',$fieldInfo['config']['allowed']))>1) { // Tablenames are 10, "_" 1, uid's 5, comma 1 $len = $max*(10+1+5+1); $out=$this->getItemBlobSize($len); } elseif ($max<=1) { $out = "int(11) NOT NULL default '0'"; } else { // uid's 5, comma 1 $len = $max*(5+1); $out=$this->getItemBlobSize($len); } } if ($fieldInfo['config']['internal_type']=='file') { $max = t3lib_div::intInRange($fieldInfo['config']['maxitems'],1,10000); // Filenames is 30+ chars.... $len = $max*(30+1); $out=$this->getItemBlobSize($len); } break; case 'select': $max = t3lib_div::intInRange($fieldInfo['config']['maxitems'],1,10000); if ($max<=1) { if ($fieldInfo['config']['foreign_table']) { $out = "int(11) NOT NULL default '0'"; } else { $out = $this->getItemArrayType($fieldInfo['config']['items']); } } else { // five chars (special=10) + comma: $len = $max*(($fieldInfo['config']['special']?10:5)+1); $out=$this->getItemBlobSize($len); } break; default: break; } return $out?$out:$fieldInfo; } /** * Check if field needs to be varchar or int * Private * * @param array $arr * @return string The definition */ function getItemArrayType($arr) { if (is_array($arr)) { $type[] = $intSize[] = 0; foreach ($arr as $item) { if (!t3lib_div::testInt($item[1]) && $item[1]!='--div--') { $type[]=strlen($item[1]); } else { $intSize[]=$item[1]; } } $us = min($intSize)>=0 ? ' unsigned' : ''; if (max($type)>0) { $out = 'varchar('.max($type).") NOT NULL default ''"; } else { $out = "int(11) NOT NULL default '0'"; } } return $out; } /** * Defines the blob size of an item by a given length * Private * * @param integer $len The length * @return string The blob definition */ function getItemBlobSize($len) { return ($len>255 ? 'tiny' : '').'blob'; } /** * Should suggest a TCA configuration for a specific field. * * @param string $fieldName The field name * @param string $fieldInfo The field information * @return string Suggested TCA configuration */ function suggestTCAFieldDefinition($fieldName,$fieldInfo) { list($type,$len) = preg_split('/ |\(|\)/', $fieldInfo, 3); switch($type) { case 'int': $out=" '".$fieldName."' => array ( 'label' => '".strtoupper($fieldName).":', 'exclude' => 0, 'config' => array ( 'type' => 'input', 'size' => '8', 'max' => '20', 'eval' => 'date', 'default' => '0', 'checkbox' => '0' ) ), ----- OR ----- '".$fieldName."' => array ( 'label' => '".strtoupper($fieldName).":', 'exclude' => 0, 'config' => array ( 'type' => 'select', 'items' => array ( array('[nothing]', 0), array('Extra choice! Only negative values here.', -1), array('__Divider:__', '--div--') ), 'foreign_table' => '[some_table_name]' ) ),"; break; case 'varchar': if ($len>10) { $out=" '".$fieldName."' => array ( 'label' => '".strtoupper($fieldName).":', 'exclude' => 0, 'config' => array ( 'type' => 'input', 'size' => '8', 'max' => '".$len."', 'eval' => 'trim', 'default' => '' ) ),"; } else { $out=" '".$fieldName."' => array ( 'label' => '".strtoupper($fieldName).":', 'exclude' => 0, 'config' => array ( 'type' => 'select', 'items' => array ( array('Item number 1', 'key1'), array('Item number 2', 'key2'), array('-----', '--div--'), array('Item number 3', 'key3') ), 'default' => '1' ) ),"; } break; case 'tinyint': $out=" '".$fieldName."' => array ( 'label' => '".strtoupper($fieldName).":', 'exclude' => 0, 'config' => array ( 'type' => 'select', 'items' => array ( array('Item number 1', '1'), array('Item number 2', '2'), array('-----', '--div--'), array('Item number 3', '3') ), 'default' => '1' ) ), ----- OR ----- '".$fieldName."' => array ( 'label' => '".strtoupper($fieldName).":', 'exclude' => 0, 'config' => array ( 'type' => 'check', 'default' => '1' ) ),"; break; case 'tinytext': $out=" '".$fieldName."' => array ( 'label' => '".strtoupper($fieldName).":', 'exclude' => 0, 'config' => array ( 'type' => 'input', 'size' => '40', 'max' => '255', 'eval' => '', 'default' => '' ) ),"; break; case 'text': case 'mediumtext': $out=" '".$fieldName."' => array ( 'label' => '".strtoupper($fieldName).":', 'config' => array ( 'type' => 'text', 'cols' => '48', 'rows' => '5' ) ),"; break; default: $out=" '".$fieldName."' => array ( 'label' => '".strtoupper($fieldName).":', 'exclude' => 0, 'config' => array ( 'type' => 'input', 'size' => '30', 'max' => '', 'eval' => '', 'default' => '' ) ),"; break; } return $out?$out:$fieldInfo; } /** * Includes TCA * * @return void */ function includeTCA() { global $TCA; include (TYPO3_tables_script ? PATH_typo3conf.TYPO3_tables_script : PATH_t3lib.'stddb/tables.php'); // Extension additions if ($GLOBALS['TYPO3_LOADED_EXT']['_CACHEFILE']) { include(PATH_typo3conf.$GLOBALS['TYPO3_LOADED_EXT']['_CACHEFILE'].'_ext_tables.php'); } else { include(PATH_t3lib.'stddb/load_ext_tables.php'); } if (TYPO3_extTableDef_script) { include (PATH_typo3conf.TYPO3_extTableDef_script); } foreach ($TCA as $table => $conf) { t3lib_div::loadTCA($table); } } /********************** * * GENERAL FUNCTIONS * **********************/ /** * Setting a message in the message-log and sets the fatalError flag if error type is 3. * * @param string $head Section header * @param string $short_string A short description * @param string $long_string A long (more detailed) description * @param integer $type -1=OK sign, 0=message, 1=notification, 2=warning, 3=error * @param boolean $force Print message also in "Advanced" mode (not only in 1-2-3 mode) * @return void */ function message($head, $short_string='', $long_string='', $type=0, $force=0) { // Return directly if mode-123 is enabled. if (!$force && $this->mode=='123' && $type<2) { return; } if ($type==3) { $this->fatalError=1; } $long_string = trim($long_string); if (!$this->silent) $this->printSection($head, $short_string, $long_string, $type); } /** * This "prints" a section with a message to the ->sections array * * @param string $head Section header * @param string $short_string A short description * @param string $long_string A long (more detailed) description * @param integer $type -1=OK sign, 0=message, 1=notification, 2=warning , 3=error * @return void */ function printSection($head, $short_string, $long_string, $type) { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'PrintSection.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); switch($type) { case 3: $messageType = 'message-error'; break; case 2: $messageType = 'message-warning'; break; case 1: $messageType = 'message-notice'; break; case 0: $messageType = 'message-information'; break; case -1: $messageType = 'message-ok'; break; } if (!trim($short_string)) { $content = ''; } else { if (trim($long_string)) { // Get the subpart for the long string $longStringSubpart = t3lib_parsehtml::getSubpart($template, '###LONGSTRINGAVAILABLE###'); } // Substitute the subpart for the long string $content = t3lib_parsehtml::substituteSubpart( $template, '###LONGSTRINGAVAILABLE###', $longStringSubpart ); // Define the markers content $markers = array( 'messageType' => $messageType, 'shortString' => $short_string, 'longString' => $long_string ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $content, $markers, '###|###', TRUE, FALSE ); } $this->sections[$head][] = $content; } /** * This prints all the messages in the ->section array * * @return string HTML of all the messages */ function printAll() { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'PrintAll.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); $sections = array(); foreach ($this->sections as $header => $valArray) { // Get the subpart for sections $sectionSubpart = t3lib_parsehtml::getSubpart($template, '###SECTIONS###'); // Define the markers content $sectionMarkers = array( 'header' => $header . ':', 'sectionContent' => implode(LF, $valArray) ); // Fill the markers in the subpart $sections[] = t3lib_parsehtml::substituteMarkerArray( $sectionSubpart, $sectionMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the sections $content = t3lib_parsehtml::substituteSubpart( $template, '###SECTIONS###', implode(LF, $sections) ); return $content; } /** * This wraps and returns the main content of the page into proper html-code. * * @param string $content The page content * @return string The full HTML page */ function outputWrapper($content) { // Get the template file if (!$this->passwordOK) { $this->template = @file_get_contents(PATH_site . $this->templateFilePath . 'Install_login.html'); } elseif ($this->mode == '123') { $this->template = @file_get_contents(PATH_site . $this->templateFilePath . 'Install_123.html'); } else { $this->template = @file_get_contents(PATH_site . $this->templateFilePath . 'Install.html'); } // Add prototype to javascript array for output $this->javascript[] = '<script type="text/javascript" src="' . t3lib_div::createVersionNumberedFilename( '../contrib/prototype/prototype.js' ) . '"></script>'; // Add JS functions for output $this->javascript[] = '<script type="text/javascript" src="' . t3lib_div::createVersionNumberedFilename( '../sysext/install/Resources/Public/Javascript/install.js' ) . '"></script>'; // Include the default stylesheets $this->stylesheets[] = '<link rel="stylesheet" type="text/css" href="' . t3lib_div::createVersionNumberedFilename($this->backPath . 'sysext/install/Resources/Public/Stylesheets/reset.css' ) . '" />'; $this->stylesheets[] = '<link rel="stylesheet" type="text/css" href="' . t3lib_div::createVersionNumberedFilename($this->backPath . 'sysext/install/Resources/Public/Stylesheets/general.css' ) . '" />'; // Get the browser info $browserInfo = t3lib_utility_Client::getBrowserInfo(t3lib_div::getIndpEnv('HTTP_USER_AGENT')); // Add the stylesheet for Internet Explorer if ($browserInfo['browser'] === 'msie') { // IE7 if (intval($browserInfo['version']) === 7) { $this->stylesheets[] = '<link rel="stylesheet" type="text/css" href="' . t3lib_div::createVersionNumberedFilename($this->backPath . 'sysext/install/Resources/Public/Stylesheets/ie7.css' ) . '" />'; // IE6 } elseif (intval($browserInfo['version']) < 7) { $this->stylesheets[] = '<link rel="stylesheet" type="text/css" href="' . t3lib_div::createVersionNumberedFilename($this->backPath . 'sysext/install/Resources/Public/Stylesheets/ie6.css' ) . '" />'; } } // Include the stylesheets based on screen if ($this->mode == '123') { $this->stylesheets[] = '<link rel="stylesheet" type="text/css" href="' . t3lib_div::createVersionNumberedFilename($this->backPath . 'sysext/install/Resources/Public/Stylesheets/install_123.css' ) . '" />'; } elseif ($this->passwordOK) { $this->stylesheets[] = '<link rel="stylesheet" type="text/css" href="' . t3lib_div::createVersionNumberedFilename($this->backPath . 'sysext/install/Resources/Public/Stylesheets/install.css' ) . '" />'; } else { $this->stylesheets[] = '<link rel="stylesheet" type="text/css" href="' . t3lib_div::createVersionNumberedFilename($this->backPath . 'sysext/install/Resources/Public/Stylesheets/install.css' ) . '" />'; $this->stylesheets[] = '<link rel="stylesheet" type="text/css" href="' . t3lib_div::createVersionNumberedFilename($this->backPath . 'sysext/install/Resources/Public/Stylesheets/install_login.css' ) . '" />'; } // Define the markers content if ($this->mode == '123') { $this->markers['headTitle'] = 'Installing TYPO3 ' . TYPO3_branch; } else { $this->markers['headTitle'] = ' TYPO3 ' . TYPO3_version . ' Install Tool on site: ' . htmlspecialchars($GLOBALS['TYPO3_CONF_VARS']['SYS']['sitename']) . ' '; } $this->markers['title'] = 'TYPO3 ' . TYPO3_version; $this->markers['javascript'] = implode(LF, $this->javascript); $this->markers['stylesheets'] = implode(LF, $this->stylesheets); $this->markers['llErrors'] = 'The following errors occured'; $this->markers['copyright'] = $this->copyright(); $this->markers['charset'] = $GLOBALS['TYPO3_CONF_VARS']['BE']['forceCharset'] ? $GLOBALS['TYPO3_CONF_VARS']['BE']['forceCharset'] : 'iso-8859-1'; $this->markers['backendUrl'] = '../index.php'; $this->markers['backend'] = 'Backend admin'; $this->markers['frontendUrl'] = '../../index.php'; $this->markers['frontend'] = 'Frontend website'; $this->markers['metaCharset'] = 'Content-Type" content="text/html; charset='; if (!empty($GLOBALS['TYPO3_CONF_VARS']['BE']['forceCharset'])) { $this->markers['metaCharset'] .= $GLOBALS['TYPO3_CONF_VARS']['BE']['forceCharset']; } else { $this->markers['metaCharset'] .= 'iso-8859-1'; } // Add the error messages if (!empty($this->errorMessages)) { // Get the subpart for all error messages $errorMessagesSubPart = t3lib_parsehtml::getSubpart($this->template, '###ERRORMESSAGES###'); // Get the subpart for a single error message $errorMessageSubPart = t3lib_parsehtml::getSubpart($errorMessagesSubPart, '###MESSAGES###'); $errors = array(); foreach ($this->errorMessages as $errorMessage) { // Define the markers content $errorMessageMarkers = array( 'message' => $errorMessage ); // Fill the markers in the subpart $errors[] = t3lib_parsehtml::substituteMarkerArray( $errorMessageSubPart, $errorMessageMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for a single message $errorMessagesSubPart = t3lib_parsehtml::substituteSubpart( $errorMessagesSubPart, '###MESSAGES###', implode(LF, $errors) ); } // Version subpart is only allowed when password is ok if ($this->passwordOK) { // Get the subpart for the version $versionSubPart = t3lib_parsehtml::getSubpart($this->template, '###VERSIONSUBPART###'); // Define the markers content $versionSubPartMarkers['version'] = 'Version: ' . TYPO3_version; // Fill the markers in the subpart $versionSubPart = t3lib_parsehtml::substituteMarkerArray( $versionSubPart, $versionSubPartMarkers, '###|###', TRUE, FALSE ); } // Substitute the version subpart $this->template = t3lib_parsehtml::substituteSubpart( $this->template, '###VERSIONSUBPART###', $versionSubPart ); // Substitute the menu subpart $this->template = t3lib_parsehtml::substituteSubpart( $this->template, '###MENU###', $this->menu() ); // Substitute the error messages subpart $this->template = t3lib_parsehtml::substituteSubpart( $this->template, '###ERRORMESSAGES###', $errorMessagesSubPart ); // Substitute the content subpart $this->template = t3lib_parsehtml::substituteSubpart( $this->template, '###CONTENT###', $content ); // Fill the markers $this->template = t3lib_parsehtml::substituteMarkerArray( $this->template, $this->markers, '###|###', TRUE, FALSE ); return $this->template; } /** * Outputs an error and dies. * Should be used by all errors that occur before even starting the install tool process. * * @param string The content of the error * @return void */ protected function outputErrorAndExit($content, $title = 'Install Tool error') { // Define the stylesheet $stylesheet = '<link rel="stylesheet" type="text/css" href="' . '../stylesheets/install/install.css" />'; $javascript = '<script type="text/javascript" src="' . '../contrib/prototype/prototype.js"></script>' . LF; $javascript .= '<script type="text/javascript" src="' . '../sysext/install/Resources/Public/Javascript/install.js"></script>'; // Get the template file $template = @file_get_contents(PATH_site . '/typo3/templates/install.html'); // Define the markers content $markers = array( 'styleSheet' => $stylesheet, 'javascript' => $javascript, 'title' => $title, 'content' => $content, ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $template, $markers, '###|###', 1, 1 ); // Output the warning message and exit header('Content-Type: text/html; charset=utf-8'); header('Cache-Control: no-cache, must-revalidate'); header('Pragma: no-cache'); echo $content; exit(); } /** * Sends the page to the client. * * @param string $content The HTML page * @return void */ function output($content) { header ('Content-Type: text/html; charset=' . ($GLOBALS['TYPO3_CONF_VARS']['BE']['forceCharset']?$GLOBALS['TYPO3_CONF_VARS']['BE']['forceCharset']:'iso-8859-1')); echo $content; } /** * Generates the main menu * * @return string HTML of the main menu */ function menu() { if ($this->mode !='123') { if (!$this->passwordOK) return; $c = 0; $items = array(); // Get the subpart for the main menu $menuSubPart = t3lib_parsehtml::getSubpart($this->template, '###MENU###'); // Get the subpart for each single menu item $menuItemSubPart = t3lib_parsehtml::getSubpart($this->template, '###MENUITEM###'); foreach ($this->menuitems as $k => $v) { // Define the markers content $markers = array( 'class' => ($this->INSTALL['type']==$k ? 'class="act"' : ''), 'id' => 't3-install-menu-' . $k, 'url' => htmlspecialchars( $this->scriptSelf . '?TYPO3_INSTALL[type]=' . $k . ($this->mode ? '&mode=' . rawurlencode($this->mode) : '') ), 'item' => $v ); // Fill the markers in the subpart $items[] = t3lib_parsehtml::substituteMarkerArray( $menuItemSubPart, $markers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the single menu items $menuSubPart = t3lib_parsehtml::substituteSubpart( $menuSubPart, '###MENUITEM###', implode(LF, $items) ); return $menuSubPart; } } /** * Generates the step header for 1-2-3 mode, the numbers at the top * * @return string HTML for the step header */ function stepHeader() { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'StepHeader.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Get the subpart for each item $stepItemSubPart = t3lib_parsehtml::getSubpart($template, '###STEPITEM###'); $steps = array(); for ($counter = 2; $counter <= $this->totalSteps; $counter++) { $state = ''; if ($this->step === $counter) { $state = 'act'; } elseif ($this->step === 'go' || $counter < $this->step) { $state = 'done'; } // Define the markers content $stepItemMarkers = array( 'class' => 'class="step' . ($counter - 1) . ($state ? ' ' . $state : '') . '"', 'url' => $this->scriptSelf . '?mode=' . $this->mode . '&step=' . $counter, 'step' => $counter ); // Fill the markers in the subpart $steps[] = t3lib_parsehtml::substituteMarkerArray( $stepItemSubPart, $stepItemMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for the items $content = t3lib_parsehtml::substituteSubpart( $template, '###STEPITEM###', implode(LF, $steps) ); return $content; } /** * Generate HTML for the security risk message * * @return string HTML for the security risk message */ function securityRisk() { return ' <p> <strong>An unsecured Install Tool presents a security risk.</strong> Minimize the risk with the following actions: </p> <ul> <li> Change the Install Tool password. </li> <li> Delete the ENABLE_INSTALL_TOOL file in the /typo3conf folder. This can be done manually or through User tools > User settings in the backend. </li> <li> For additional security, the /typo3/install/ folder can be renamed, deleted, or password protected with a .htaccess file. </li> </ul> '; } /** * Generates the form to alter the password of the Install Tool * * @return string HTML of the form */ function alterPasswordForm() { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'AlterPasswordForm.html'); // Get the template part from the file $template = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Define the markers content $markers = array( 'action' => $this->scriptSelf.'?TYPO3_INSTALL[type]=extConfig', 'enterPassword' => 'Enter new password:', 'enterAgain' => 'Enter again:', 'submit' => 'Set new password', 'formToken' => $this->formProtection->generateToken( 'installToolPassword', 'change' ), ); // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $template, $markers, '###|###', TRUE, FALSE ); return $content; } /** * Generate HTML for the copyright * * @return string HTML of the copyright */ function copyright() { $content = ' <p> <strong>TYPO3 CMS.</strong> Copyright © 1998-' . date('Y') . ' Kasper Skårhøj. Extensions are copyright of their respective owners. Go to <a href="' . TYPO3_URL_GENERAL . '">' . TYPO3_URL_GENERAL . '</a> for details. TYPO3 comes with ABSOLUTELY NO WARRANTY; <a href="' . TYPO3_URL_LICENSE . '">click</a> for details. This is free software, and you are welcome to redistribute it under certain conditions; <a href="' . TYPO3_URL_LICENSE . '">click</a> for details. Obstructing the appearance of this notice is prohibited by law. </p> <p> <a href="' . TYPO3_URL_DONATE . '"><strong>Donate</strong></a> | <a href="' . TYPO3_URL_ORG . '">TYPO3.org</a> </p> '; return $content; } /** * Generate HTML for the message that the basic setup has been finished * * @return string HTML of the message */ function messageBasicFinished() { return ' <p> You have completed the basic setup of the TYPO3 Content Management System. Choose between these options to continue: </p> <ul> <li> <a href="' . $this->scriptSelf . '">Configure TYPO3</a> (Recommended) <br /> This will let you analyze and verify that everything in your installation is in order. In addition, you can configure advanced TYPO3 options in this step. </li> <li> <a href="../../index.php"> Visit the frontend </a> </li> <li> <a href="../index.php"> Login to the backend </a> <br /> (Default username: <em>admin</em>, default password: <em>password</em>.) </li> </ul> '; } /** * Make the url of the script according to type, mode and step * * @param string $type The type * @return string The url */ function setScriptName($type) { $value = $this->scriptSelf.'?TYPO3_INSTALL[type]='.$type.($this->mode?'&mode='.rawurlencode($this->mode):'').($this->step?'&step='.rawurlencode($this->step):''); return $value; } /** * Return the filename that will be used for the backup. * It is important that backups of PHP files still stay as a PHP file, otherwise they could be viewed un-parsed in clear-text. * * @param string $filename Full path to a file * @return string The name of the backup file (again, including the full path) */ function getBackupFilename($filename) { if (preg_match('/\.php$/', $filename)) { $backupFile = str_replace('.php', '_bak.php', $filename); } else { $backupFile = $filename.'~'; } return $backupFile; } /** * Creates a table which checkboxes for updating database. * * @param array $arr Array of statements (key / value pairs where key is used for the checkboxes) * @param string $label Label for the table. * @param boolean $checked If set, then checkboxes are set by default. * @param boolean $iconDis If set, then icons are shown. * @param array $currentValue Array of "current values" for each key/value pair in $arr. Shown if given. * @param boolean $cVfullMsg If set, will show the prefix "Current value" if $currentValue is given. * @return string HTML table with checkboxes for update. Must be wrapped in a form. */ function generateUpdateDatabaseForm_checkboxes($arr,$label,$checked=1,$iconDis=0,$currentValue=array(),$cVfullMsg=0) { $out = array(); $tableId = uniqid('table'); $templateMarkers = array(); if (is_array($arr)) { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'GenerateUpdateDatabaseFormCheckboxes.html'); // Get the template part from the file $content = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Define the markers content $templateMarkers = array( 'label' => $label, 'tableId' => $tableId ); // Select/Deselect All if (count($arr) > 1) { // Get the subpart for multiple tables $multipleTablesSubpart = t3lib_parsehtml::getSubpart($content, '###MULTIPLETABLES###'); // Define the markers content $multipleTablesMarkers = array( 'label' => $label, 'tableId' => $tableId, 'checked' => ($checked ? ' checked="checked"' : ''), 'selectAllId' => 't3-install-' . $tableId . '-checkbox', 'selectDeselectAll' => 'select/deselect all' ); // Fill the markers in the subpart $multipleTablesSubpart = t3lib_parsehtml::substituteMarkerArray( $multipleTablesSubpart, $multipleTablesMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for multiple tables $content = t3lib_parsehtml::substituteSubpart( $content, '###MULTIPLETABLES###', $multipleTablesSubpart ); // Rows foreach ($arr as $key => $string) { // Get the subpart for rows $rowsSubpart = t3lib_parsehtml::getSubpart($content, '###ROWS###'); $currentSubpart = ''; $ico = ''; $warnings = array(); // Define the markers content $rowsMarkers = array( 'checkboxId' => 't3-install-db-' . $key, 'name' => $this->dbUpdateCheckboxPrefix . '[' . $key . ']', 'checked' => ($checked ? 'checked="checked"' : ''), 'string' => htmlspecialchars($string) ); if ($iconDis) { $iconMarkers['backPath'] = $this->backPath; if (preg_match('/^TRUNCATE/i', $string)) { $iconMarkers['iconText'] = ''; $warnings['clear_table_info'] = 'Clearing the table is sometimes neccessary when adding new keys. In case of cache_* tables this should not hurt at all. However, use it with care.'; } elseif (stristr($string,' user_')) { $iconMarkers['iconText'] = '(USER)'; } elseif (stristr($string,' app_')) { $iconMarkers['iconText'] = '(APP)'; } elseif (stristr($string,' ttx_') || stristr($string,' tx_')) { $iconMarkers['iconText'] = '(EXT)'; } if (!empty($iconMarkers)) { // Get the subpart for icons $iconSubpart = t3lib_parsehtml::getSubpart($content, '###ICONAVAILABLE###'); // Fill the markers in the subpart $iconSubpart = t3lib_parsehtml::substituteMarkerArray( $iconSubpart, $iconMarkers, '###|###', TRUE, TRUE ); } } // Substitute the subpart for icons $rowsSubpart = t3lib_parsehtml::substituteSubpart( $rowsSubpart, '###ICONAVAILABLE###', $iconSubpart ); if (isset($currentValue[$key])) { // Get the subpart for current $currentSubpart = t3lib_parsehtml::getSubpart($rowsSubpart, '###CURRENT###'); // Define the markers content $currentMarkers = array ( 'message' => (!$cVfullMsg ? 'Current value:': ''), 'value' => $currentValue[$key] ); // Fill the markers in the subpart $currentSubpart = t3lib_parsehtml::substituteMarkerArray( $currentSubpart, $currentMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for current $rowsSubpart = t3lib_parsehtml::substituteSubpart( $rowsSubpart, '###CURRENT###', $currentSubpart ); $errorSubpart = ''; if (isset($this->databaseUpdateErrorMessages[$key])) { // Get the subpart for current $errorSubpart = t3lib_parsehtml::getSubpart($rowsSubpart, '###ERROR###'); // Define the markers content $currentMarkers = array ( 'errorMessage' => $this->databaseUpdateErrorMessages[$key], ); // Fill the markers in the subpart $errorSubpart = t3lib_parsehtml::substituteMarkerArray( $errorSubpart, $currentMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for error messages $rowsSubpart = t3lib_parsehtml::substituteSubpart( $rowsSubpart, '###ERROR###', $errorSubpart ); // Fill the markers in the subpart $rowsSubpart = t3lib_parsehtml::substituteMarkerArray( $rowsSubpart, $rowsMarkers, '###|###', TRUE, FALSE ); $rows[] = $rowsSubpart; } // Substitute the subpart for rows $content = t3lib_parsehtml::substituteSubpart( $content, '###ROWS###', implode(LF, $rows) ); if (count($warnings)) { // Get the subpart for warnings $warningsSubpart = t3lib_parsehtml::getSubpart($content, '###WARNINGS###'); $warningItems = array(); foreach ($warnings as $warning) { // Get the subpart for single warning items $warningItemSubpart = t3lib_parsehtml::getSubpart($warningsSubpart, '###WARNINGITEM###'); // Define the markers content $warningItemMarker['warning'] = $warning; // Fill the markers in the subpart $warningItems[] = t3lib_parsehtml::substituteMarkerArray( $warningItemSubpart, $warningItemMarker, '###|###', TRUE, FALSE ); } // Substitute the subpart for single warning items $warningsSubpart = t3lib_parsehtml::substituteSubpart( $warningsSubpart, '###WARNINGITEM###', implode(LF, $warningItems) ); } // Substitute the subpart for warnings $content = t3lib_parsehtml::substituteSubpart( $content, '###WARNINGS###', $warningsSubpart ); } // Fill the markers $content = t3lib_parsehtml::substituteMarkerArray( $content, $templateMarkers, '###|###', TRUE, FALSE ); return $content; } /** * Returns HTML-code, which is a visual representation of a multidimensional array * Returns false if $array_in is not an array * * @param mixed $incomingValue Array to view * @return string HTML output */ function viewArray($incomingValue) { // Get the template file $templateFile = @file_get_contents(PATH_site . $this->templateFilePath . 'ViewArray.html'); if (is_array($incomingValue) && !empty($incomingValue)) { // Get the template part from the file $content = t3lib_parsehtml::getSubpart($templateFile, '###TEMPLATE###'); // Get the subpart for a single item $itemSubpart = t3lib_parsehtml::getSubpart($content, '###ITEM###'); foreach ($incomingValue as $key => $value) { if (is_array($value)) { $description = $this->viewArray($value); } elseif (is_object($value)) { $description = get_class($value); if (method_exists($value, '__toString')) { $description .= ': ' . (string)$value; } } else { if (gettype($value) == 'object') { $description = 'Unknown object'; } else { $description = htmlspecialchars((string) $value); } } // Define the markers content $itemMarkers = array( 'key' => htmlspecialchars((string) $key), 'description' => !empty($description) ? $description : ' ' ); // Fill the markers in the subpart $items[] = t3lib_parsehtml::substituteMarkerArray( $itemSubpart, $itemMarkers, '###|###', TRUE, FALSE ); } // Substitute the subpart for single item $content = t3lib_parsehtml::substituteSubpart( $content, '###ITEM###', implode(LF, $items) ); } return $content; } /** * Returns a newly created TYPO3 encryption key with a given length. * * @param integer $keyLength Desired key length * @return string The encryption key */ public function createEncryptionKey($keyLength = 96) { $bytes = t3lib_div::generateRandomBytes($keyLength); return substr(bin2hex($bytes), -96); } /** * Adds an error message that should be displayed. * * @param string $messageText * the text of the message to display, must not be empty */ public function addErrorMessage($messageText) { if ($messageText == '') { throw new InvalidArgumentException('$messageText must not be empty.'); } $this->errorMessages[] = $messageText; } } if (defined('TYPO3_MODE') && isset($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['XCLASS']['ext/install/mod/class.tx_install.php'])) { include_once($GLOBALS['TYPO3_CONF_VARS'][TYPO3_MODE]['XCLASS']['ext/install/mod/class.tx_install.php']); } ?>
© 2017 -
ZeroByte.ID
.