Source for file configtest.php

Documentation is available at configtest.php

  1. <?php
  2.  
  3. /**
  4.  * SquirrelMail configtest script
  5.  *
  6.  * @copyright &copy; 2003-2006 The SquirrelMail Project Team
  7.  * @license http://opensource.org/licenses/gpl-license.php GNU Public License
  8.  * @version $Id: configtest.php,v 1.52 2006/10/07 11:21:17 tokul Exp $
  9.  * @package squirrelmail
  10.  * @subpackage config
  11.  */
  12.  
  13. /**********************************************************
  14.  * NOTE: you do not need to change this script!             *
  15.  * If it throws errors you need to adjust your config.      *
  16.  ************************************************************/
  17.  
  18.  
  19. // This script could really use some restructuring as it has grown quite rapidly
  20. // but is not very 'clean'. Feel free to get some structure into this thing.
  21.  
  22. /** force verbose error reporting and turn on display of errors */
  23. ini_set('display_errors',1);
  24.  
  25. /** Blockcopy from init.php. Cleans globals. */
  26. if ((bool) ini_get('register_globals'&&
  27.     strtolower(ini_get('register_globals'))!='off'{
  28.     /**
  29.      * Remove all globals that are not reserved by PHP
  30.      * 'value' and 'key' are used by foreach. Don't unset them inside foreach.
  31.      */
  32.     foreach ($GLOBALS as $key => $value{
  33.         switch($key{
  34.         case 'HTTP_POST_VARS':
  35.         case '_POST':
  36.         case 'HTTP_GET_VARS':
  37.         case '_GET':
  38.         case 'HTTP_COOKIE_VARS':
  39.         case '_COOKIE':
  40.         case 'HTTP_SERVER_VARS':
  41.         case '_SERVER':
  42.         case 'HTTP_ENV_VARS':
  43.         case '_ENV':
  44.         case 'HTTP_POST_FILES':
  45.         case '_FILES':
  46.         case '_REQUEST':
  47.         case 'HTTP_SESSION_VARS':
  48.         case '_SESSION':
  49.         case 'GLOBALS':
  50.         case 'key':
  51.         case 'value':
  52.             break;
  53.         default:
  54.             unset($GLOBALS[$key]);
  55.         }
  56.     }
  57.     // Unset variables used in foreach
  58.         unset($GLOBALS['key']);
  59.     unset($GLOBALS['value']);
  60. }
  61.  
  62.  
  63. /**
  64.  * Displays error messages and warnings
  65.  * @param string $str message
  66.  * @param boolean $fatal fatal error or only warning
  67.  */
  68. function do_err($str$fatal TRUE{
  69.     global $IND$warnings;
  70.     $level $fatal 'FATAL ERROR:' 'WARNING:';
  71.     echo '<p>'.$IND.'<font color="red"><b>' $level '</b></font> ' .$str"</p>\n";
  72.     if($fatal{
  73.         echo '</body></html>';
  74.         exit;
  75.     else {
  76.         $warnings++;
  77.     }
  78. }
  79.  
  80. /** @ignore */
  81. define('SM_PATH''../');
  82. /** load minimal function set */
  83. require(SM_PATH 'functions/global.php');
  84. require(SM_PATH 'functions/strings.php');
  85.  
  86. /** set default value in order to block remote access */
  87. $allow_remote_configtest=false;
  88.  
  89. /** Load all configuration files before output begins */
  90.  
  91. /* load default configuration */
  92. require(SM_PATH 'config/config_default.php');
  93. /* reset arrays in default configuration */
  94. $ldap_server array();
  95. $plugins array();
  96. $fontsets array();
  97. $theme array();
  98. $theme[0]['PATH'SM_PATH 'themes/default_theme.php';
  99. $theme[0]['NAME''Default';
  100. $aTemplateSet array();
  101. $aTemplateSet[0]['ID''default';
  102. $aTemplateSet[0]['NAME''Default';
  103. /* load site configuration */
  104. if (file_exists(SM_PATH 'config/config.php')) {
  105.     require(SM_PATH 'config/config.php');
  106. }
  107. /* load local configuration overrides */
  108. if (file_exists(SM_PATH 'config/config_local.php')) {
  109.     require(SM_PATH 'config/config_local.php');
  110. }
  111.  
  112. /** Warning counter */
  113. $warnings 0;
  114.  
  115. /** indent */
  116. $IND str_repeat('&nbsp;',4);
  117.  
  118. /**
  119.  * get_location starts session and must be run before output is started.
  120.  */
  121. $test_location get_location();
  122.  
  123. ?><!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
  124.   "http://www.w3.org/TR/1999/REC-html401-19991224/loose.dtd">
  125. <html>
  126. <head>
  127.   <meta name="robots" content="noindex,nofollow">
  128.   <title>SquirrelMail configtest</title>
  129. </head>
  130. <body>
  131. <h1>SquirrelMail configtest</h1>
  132.  
  133. <p>This script will try to check some aspects of your SquirrelMail configuration
  134. and point you to errors whereever it can find them. You need to go run <tt>conf.pl</tt>
  135. in the <tt>config/</tt> directory first before you run this script.</p>
  136.  
  137. <?php
  138.  
  139. $included array_map('basename'get_included_files() );
  140. if(!in_array('config.php'$included)) {
  141.     if(!file_exists(SM_PATH 'config/config.php')) {
  142.         do_err('Config file '.SM_PATH 'config/config.php does not exist!<br />'.
  143.                 'You need to run <tt>conf.pl</tt> first.');
  144.     }
  145.     do_err('Could not read '.SM_PATH.'config/config.php! Check file permissions.');
  146. }
  147. if(!in_array('strings.php'$included)) {
  148.     do_err('Could not include '.SM_PATH.'functions/strings.php!<br />'.
  149.             'Check permissions on that file.');
  150. }
  151.  
  152. /* Block remote use of script */
  153. if ($allow_remote_configtest{
  154.     sqGetGlobalVar('REMOTE_ADDR',$client_ip,SQ_SERVER);
  155.     sqGetGlobalVar('SERVER_ADDR',$server_ip,SQ_SERVER);
  156.  
  157.     if ((isset($client_ip|| $client_ip!='127.0.0.1'&&
  158.             (isset($client_ip|| isset($server_ip|| $client_ip!=$server_ip)) {
  159.         do_err('Enable "Allow remote configtest" option in squirrelmail configuration in order to use this script.');
  160.     }
  161. }
  162. /* checking PHP specs */
  163.  
  164. echo "<p><table>\n<tr><td>SquirrelMail version:</td><td><b>" $version "</b></td></tr>\n" .
  165.     '<tr><td>Config file version:</td><td><b>' $config_version "</b></td></tr>\n" .
  166.     '<tr><td>Config file last modified:</td><td><b>' .
  167.     date ('d F Y H:i:s'filemtime(SM_PATH 'config/config.php')) .
  168.     "</b></td></tr>\n</table>\n</p>\n\n";
  169.  
  170. /* check $config_version */
  171. if ($config_version!='1.5.0'{
  172.     do_err('Configuration file version does not match required version. Please update your configuration file.');
  173. }
  174.  
  175. echo "Checking PHP configuration...<br />\n";
  176.  
  177. if(!check_php_version(4,1,0)) {
  178.     do_err('Insufficient PHP version: 'PHP_VERSION '! Minimum required: 4.1.0');
  179. }
  180.  
  181. echo $IND 'PHP version ' PHP_VERSION ' OK. (You have: ' phpversion(". Minimum: 4.1.0)<br />\n";
  182. /* test for boolean false and any string that is not equal to 'off' */
  183. if ((bool) ini_get('register_globals'&& 
  184.     strtolower(ini_get('register_globals'))!='off'{
  185.     do_err('You have register_globals turned on.  This is not an error, but it CAN be a security hazard.  Consider turning register_globals off.'false);
  186. }
  187. $php_exts array('session','pcre');
  188. $diff array_diff($php_extsget_loaded_extensions());
  189. if(count($diff)) {
  190.     do_err('Required PHP extensions missing: '.implode(', ',$diff) );
  191. }
  192.  
  193. echo $IND "PHP extensions OK.<br />\n";
  194.  
  195. /* dangerous php settings */
  196. /**
  197.  * mbstring.func_overload allows to replace original string and regexp functions
  198.  * with their equivalents from php mbstring extension. It causes problems when
  199.  * scripts analyze 8bit strings byte after byte or use 8bit strings in regexp tests.
  200.  * Setting can be controlled in php.ini (php 4.2.0), webserver config (php 4.2.0)
  201.  * and .htaccess files (php 4.3.5).
  202.  */
  203. if (function_exists('mb_internal_encoding'&&
  204.     check_php_version(4,2,0&&
  205.     (int)ini_get('mbstring.func_overload')!=0{
  206.     $mb_error='You have enabled mbstring overloading.'
  207.         .' It can cause problems with SquirrelMail scripts that rely on single byte string functions.';
  208.     do_err($mb_error);
  209. }
  210.  
  211. /* checking paths */
  212.  
  213. echo "Checking paths...<br />\n";
  214.  
  215. if(!file_exists($data_dir)) {
  216.     // data_dir is not that important in db_setups.
  217.         if (isset($prefs_dsn&& empty($prefs_dsn)) {
  218.         $data_dir_error "Data dir ($data_dirdoes not exist!\n";
  219.         echo $IND .'<font color="red"><b>ERROR:</b></font> ' $data_dir_error;
  220.     else {
  221.         do_err("Data dir ($data_dirdoes not exist!");
  222.     }
  223. }
  224. // don't check if errors
  225. if(!isset($data_dir_error&& !is_dir($data_dir)) {
  226.     if (isset($prefs_dsn&& empty($prefs_dsn)) {
  227.         $data_dir_error "Data dir ($data_diris not a directory!\n";
  228.         echo $IND '<font color="red"><b>ERROR:</b></font> ' $data_dir_error;
  229.     else {
  230.         do_err("Data dir ($data_diris not a directory!");
  231.     }
  232. }
  233. // datadir should be executable - but no clean way to test on that
  234. if(!isset($data_dir_error&& !is_writable($data_dir)) {
  235.     if (isset($prefs_dsn&& empty($prefs_dsn)) {
  236.         $data_dir_error "Data dir ($data_diris not writable!\n";
  237.         echo $IND '<font color="red"><b>ERROR:</b></font> ' $data_dir_error;
  238.     else {
  239.         do_err("Data dir ($data_diris not writable!");
  240.     }
  241. }
  242.  
  243. if (isset($data_dir_error)) {
  244.     echo " Some plugins might need access to data directory.<br />\n";
  245. else {
  246.     // todo_ornot: actually write something and read it back.
  247.         echo $IND "Data dir OK.<br />\n";
  248. }
  249.  
  250. if($data_dir == $attachment_dir{
  251.     echo $IND "Attachment dir is the same as data dir.<br />\n";
  252.     if (isset($data_dir_error)) {
  253.         do_err($data_dir_error);
  254.     }
  255. else {
  256.     if(!file_exists($attachment_dir)) {
  257.         do_err("Attachment dir ($attachment_dirdoes not exist!");
  258.     }
  259.     if (!is_dir($attachment_dir)) {
  260.         do_err("Attachment dir ($attachment_diris not a directory!");
  261.     }
  262.     if (!is_writable($attachment_dir)) {
  263.         do_err("I cannot write to attachment dir ($attachment_dir)!");
  264.     }
  265.     echo $IND "Attachment dir OK.<br />\n";
  266. }
  267.  
  268.  
  269. /* check plugins and themes */
  270. $bad_plugins array(
  271.         'attachment_common',      // Integrated into SquirrelMail 1.2 core
  272.                 'auto_prune_sent',        // Obsolete: See Proon Automatic Folder Pruning plugin
  273.                 'compose_new_window',     // Integrated into SquirrelMail 1.4 core
  274.                 'delete_move_next',       // Integrated into SquirrelMail 1.5 core
  275.                 'disk_quota',             // Obsolete: See Check Quota plugin
  276.                 'email_priority',         // Integrated into SquirrelMail 1.2 core
  277.                 'emoticons',              // Obsolete: See HTML Mail plugin
  278.                 'focus_change',           // Integrated into SquirrelMail 1.2 core
  279.                 'folder_settings',        // Integrated into SquirrelMail 1.5.1 core
  280.                 'global_sql_addressbook'// Integrated into SquirrelMail 1.4 core
  281.                 'hancock',                // Not Working: See Random Signature Taglines plugin
  282.                 'msg_flags',              // Integrated into SquirrelMail 1.5.1 core
  283.                 'message_source',         // Added to SquirrelMail 1.4 Core Plugins (message_details)
  284.                 'motd',                   // Integrated into SquirrelMail 1.2 core
  285.                 'paginator',              // Integrated into SquirrelMail 1.2 core
  286.                 'printer_friendly',       // Integrated into SquirrelMail 1.2 core
  287.                 'procfilter',             // Obsolete: See Server Side Filter plugin
  288.                 'redhat_php_cgi_fix',     // Integrated into SquirrelMail 1.1.1 core
  289.                 'send_to_semicolon',      // Integrated into SquirrelMail 1.4.1 core
  290.                 'spamassassin',           // Not working beyond SquirrelMail 1.2.7: See Spamassassin SpamFilter (Frontend) v2 plugin
  291.                 'sqcalendar',             // Added to SquirrelMail 1.2 Core Plugins (calendar)
  292.                 'sqclock',                // Integrated into SquirrelMail 1.2 core
  293.                 'sql_squirrel_logger',    // Obsolete: See Squirrel Logger plugin
  294.                 'tmda',                   // Obsolete: See TMDA Tools plugin
  295.                 'vacation',               // Obsolete: See Vacation Local plugin
  296.                 'view_as_html',           // Integrated into SquirrelMail 1.5.1 core
  297.                 'xmailer'                 // Integrated into SquirrelMail 1.2 core
  298.                 );
  299.  
  300. if (isset($plugins[0])) {
  301.     foreach($plugins as $plugin{
  302.         if(!file_exists(SM_PATH .'plugins/'.$plugin)) {
  303.             do_err('You have enabled the <i>'.$plugin.'</i> plugin, but I cannot find it.'FALSE);
  304.         elseif (!is_readable(SM_PATH .'plugins/'.$plugin.'/setup.php')) {
  305.             do_err('You have enabled the <i>'.$plugin.'</i> plugin, but I cannot read its setup.php file.'FALSE);
  306.         elseif (in_array($plugin$bad_plugins)) {
  307.             do_err('You have enabled the <i>'.$plugin.'</i> plugin, which causes problems with this version of SquirrelMail. Please check the ReleaseNotes or other documentation for more information.'false);
  308.         }
  309.     }
  310.     // load plugin functions
  311.         include_once(SM_PATH 'functions/plugin.php');
  312.     // turn on output buffering in order to prevent output of new lines
  313.         ob_start();
  314.     foreach ($plugins as $name{
  315.         use_plugin($name);
  316.     }
  317.     // get output and remove whitespace
  318.         $output trim(ob_get_contents());
  319.     ob_end_clean();
  320.     // if plugins output more than newlines and spacing, stop script execution.
  321.         if (!empty($output)) {
  322.         $plugin_load_error 'Some output is produced when plugins are loaded.'
  323.             .' Usually it means error. Output said: '.htmlspecialchars($output);
  324.         do_err($plugin_load_error);
  325.     }
  326.     /**
  327.      * Hook is added in 1.5.2. Plugins should print error message and return true
  328.      * if there is an error in plugin.
  329.      */
  330.     $plugin_err boolean_hook_function('configtest');
  331.     if($plugin_err{
  332.         do_err('Some plugin tests failed.');
  333.     else {
  334.         echo $IND "Plugins OK.<br />\n";
  335.     }
  336. else {
  337.     echo $IND "Plugins are not enabled in config.<br />\n";
  338. }
  339. foreach($theme as $thm{
  340.     if(!file_exists($thm['PATH'])) {
  341.         do_err('You have enabled the <i>'.$thm['NAME'].'</i> theme but I cannot find it ('.$thm['PATH'].').'FALSE);
  342.     elseif(!is_readable($thm['PATH'])) {
  343.         do_err('You have enabled the <i>'.$thm['NAME'].'</i> theme but I cannot read it ('.$thm['PATH'].').'FALSE);
  344.     }
  345. }
  346.  
  347. echo $IND "Themes OK.<br />\n";
  348.  
  349. if $squirrelmail_default_language != 'en_US' {
  350.     $loc_path SM_PATH .'locale/'.$squirrelmail_default_language.'/LC_MESSAGES/squirrelmail.mo';
  351.     iffile_exists$loc_path ) ) {
  352.         do_err('You have set <i>' $squirrelmail_default_language .
  353.                 '</i> as your default language, but I cannot find this translation (should be '.
  354.                 'in <tt>' $loc_path '</tt>). Please note that you have to download translations '.
  355.                 'separately from the main SquirrelMail package.'FALSE);
  356.     elseif is_readable$loc_path ) ) {
  357.         do_err('You have set <i>' $squirrelmail_default_language .
  358.                 '</i> as your default language, but I cannot read this translation (file '.
  359.                 'in <tt>' $loc_path '</tt> unreadable).'FALSE);
  360.     else {
  361.         echo $IND "Default language OK.<br />\n";
  362.     }
  363. else {
  364.     echo $IND "Default language OK.<br />\n";
  365. }
  366.  
  367. echo $IND "Base URL detected as: <tt>" htmlspecialchars($test_location.
  368.     "</tt> (location base " (empty($config_location_base'autodetected' 'set to <tt>' .
  369.     htmlspecialchars($config_location_base)."</tt>"")<br />\n";
  370.  
  371. /* check minimal requirements for other security options */
  372.  
  373. /* imaps or ssmtp */
  374. if($use_smtp_tls == || $use_imap_tls == 1{
  375.     if(!check_php_version(4,3,0)) {
  376.         do_err('You need at least PHP 4.3.0 for SMTP/IMAP TLS!');
  377.     }
  378.     if(!extension_loaded('openssl')) {
  379.         do_err('You need the openssl PHP extension to use SMTP/IMAP TLS!');
  380.     }
  381. }
  382. /* starttls extensions */
  383. if($use_smtp_tls === || $use_imap_tls === 2{
  384.     if (function_exists('stream_socket_enable_crypto')) {
  385.         do_err('If you want to use STARTTLS extension, you need stream_socket_enable_crypto() function from PHP 5.1.0 and newer.');
  386.     }
  387. }
  388. /* digest-md5 */
  389. if ($smtp_auth_mech=='digest-md5' || $imap_auth_mech =='digest-md5'{
  390.     if (!extension_loaded('xml')) {
  391.         do_err('You need the PHP XML extension to use Digest-MD5 authentication!');
  392.     }
  393. }
  394.  
  395. /* check outgoing mail */
  396.  
  397. echo "Checking outgoing mail service....<br />\n";
  398.  
  399. if($useSendmail{
  400.     // is_executable also checks for existance, but we want to be as precise as possible with the errors
  401.         if(!file_exists($sendmail_path)) {
  402.         do_err("Location of sendmail program incorrect ($sendmail_path)!");
  403.     }
  404.     if(!is_executable($sendmail_path)) {
  405.         do_err("I cannot execute the sendmail program ($sendmail_path)!");
  406.     }
  407.  
  408.     echo $IND "sendmail OK<br />\n";
  409. else {
  410.     $stream fsockopen( ($use_smtp_tls==1?'tls://':'').$smtpServerAddress$smtpPort,
  411.             $errorNumber$errorString);
  412.     if(!$stream{
  413.         do_err("Error connecting to SMTP server \"$smtpServerAddress:$smtpPort\".".
  414.                 "Server error: ($errorNumber".htmlspecialchars($errorString));
  415.     }
  416.  
  417.     // check for SMTP code; should be 2xx to allow us access
  418.         $smtpline fgets($stream1024);
  419.     if(((int) $smtpline{0}3{
  420.         do_err("Error connecting to SMTP server. Server error: ".
  421.                 htmlspecialchars($smtpline));
  422.     }
  423.  
  424.     /* smtp starttls checks */
  425.     if ($use_smtp_tls===2{
  426.         // if something breaks, script should close smtp connection on exit.
  427.  
  428.         // say helo
  429.                 fwrite($stream,"EHLO $client_ip\r\n");
  430.  
  431.         $ehlo=array();
  432.         $ehlo_error false;
  433.         while ($line=fgets($stream1024)){
  434.             if (preg_match("/^250(-|\s)(\S*)\s+(\S.*)/",$line,$match)||
  435.                     preg_match("/^250(-|\s)(\S*)\s+/",$line,$match)) {
  436.                 if (!isset($match[3])) {
  437.                     // simple one word extension
  438.                                         $ehlo[strtoupper($match[2])]='';
  439.                 else {
  440.                     // ehlo-keyword + ehlo-param
  441.                                         $ehlo[strtoupper($match[2])]=trim($match[3]);
  442.                 }
  443.                 if ($match[1]==' '{
  444.                     $ret $line;
  445.                     break;
  446.                 }
  447.             else {
  448.                 // 
  449.                                 $ehlo_error true;
  450.                 $ehlo[]=$line;
  451.                 break;
  452.             }
  453.         }
  454.         if ($ehlo_error{
  455.             do_err('SMTP EHLO failed. You need ESMTP support for SMTP STARTTLS');
  456.         elseif (!array_key_exists('STARTTLS',$ehlo)) {
  457.             do_err('STARTTLS support is not declared by SMTP server.');
  458.         }
  459.  
  460.         fwrite($stream,"STARTTLS\r\n");
  461.         $starttls_response=fgets($stream1024);
  462.         if ($starttls_response[0]!=2{
  463.             $starttls_cmd_err 'SMTP STARTTLS failed. Server replied: '
  464.                 .htmlspecialchars($starttls_response);
  465.             do_err($starttls_cmd_err);
  466.         elseif(stream_socket_enable_crypto($stream,true,STREAM_CRYPTO_METHOD_TLS_CLIENT)) {
  467.             do_err('Failed to enable encryption on SMTP STARTTLS connection.');
  468.         else {
  469.             echo $IND "SMTP STARTTLS extension looks OK.<br />\n";
  470.         }
  471.         // According to RFC we should second ehlo call here.
  472.         }
  473.  
  474.     fputs($stream'QUIT');
  475.     fclose($stream);
  476.     echo $IND 'SMTP server OK (<tt><small>'.
  477.             trim(htmlspecialchars($smtpline))."</small></tt>)<br />\n";
  478.  
  479.     /* POP before SMTP */
  480.     if($pop_before_smtp{
  481.         $stream fsockopen($smtpServerAddress110$err_no$err_str);
  482.         if (!$stream{
  483.             do_err("Error connecting to POP Server ($smtpServerAddress:110) "
  484.                 . $err_no ' : ' htmlspecialchars($err_str));
  485.         }
  486.  
  487.         $tmp fgets($stream1024);
  488.         if (substr($tmp03!= '+OK'{
  489.             do_err("Error connecting to POP Server ($smtpServerAddress:110)"
  490.                 . ' '.htmlspecialchars($tmp));
  491.         }
  492.         fputs($stream'QUIT');
  493.         fclose($stream);
  494.         echo $IND "POP-before-SMTP OK.<br />\n";
  495.     }
  496. }
  497.  
  498. /**
  499.  * Check the IMAP server
  500.  */
  501. echo "Checking IMAP service....<br />\n";
  502.  
  503. /** Can we open a connection? */
  504. $stream fsockopen( ($use_imap_tls==1?'tls://':'').$imapServerAddress$imapPort,
  505.         $errorNumber$errorString);
  506. if(!$stream{
  507.     do_err("Error connecting to IMAP server \"$imapServerAddress:$imapPort\".".
  508.             "Server error: ($errorNumber".
  509.             htmlspecialchars($errorString));
  510. }
  511.  
  512. /** Is the first response 'OK'? */
  513. $imapline fgets($stream1024);
  514. if(substr($imapline0,4!= '* OK'{
  515.     do_err('Error connecting to IMAP server. Server error: '.
  516.             htmlspecialchars($imapline));
  517. }
  518.  
  519. echo $IND 'IMAP server ready (<tt><small>'.
  520.     htmlspecialchars(trim($imapline))."</small></tt>)<br />\n";
  521.  
  522. /** Check capabilities */
  523. fputs($stream"A001 CAPABILITY\r\n");
  524. $capline '';
  525. while ($line=fgets($stream1024)){
  526.     if (preg_match("/A001.*/",$line)) {
  527.         break;
  528.     else {
  529.         $capline.=$line;
  530.     }
  531. }
  532.  
  533. /* don't display capabilities before STARTTLS */
  534. if ($use_imap_tls===&& stristr($capline'STARTTLS'=== false{
  535.     do_err('Your server doesn\'t support STARTTLS.');
  536. elseif($use_imap_tls===2{
  537.     /* try starting starttls */
  538.     fwrite($stream,"A002 STARTTLS\r\n");
  539.     $starttls_line=fgets($stream1024);
  540.     if (preg_match("/^A002 OK.*/i",$starttls_line)) {
  541.         $imap_starttls_err 'IMAP STARTTLS failed. Server replied: '
  542.             .htmlspecialchars($starttls_line);
  543.         do_err($imap_starttls_err);
  544.     elseif (stream_socket_enable_crypto($stream,true,STREAM_CRYPTO_METHOD_TLS_CLIENT)) {
  545.         do_err('Failed to enable encryption on IMAP connection.');
  546.     else {
  547.         echo $IND "IMAP STARTTLS extension looks OK.<br />\n";
  548.     }
  549.  
  550.     // get new capability line
  551.         fwrite($stream,"A003 CAPABILITY\r\n");
  552.     $capline='';
  553.     while ($line=fgets($stream1024)){
  554.         if (preg_match("/A003.*/",$line)) {
  555.             break;
  556.         else {
  557.             $capline.=$line;
  558.         }
  559.     }
  560. }
  561.  
  562. echo $IND 'Capabilities: <tt>'.htmlspecialchars($capline)."</tt><br />\n";
  563.  
  564. if($imap_auth_mech == 'login' && stristr($capline'LOGINDISABLED'!== FALSE{
  565.     do_err('Your server doesn\'t allow plaintext logins. '.
  566.             'Try enabling another authentication mechanism like CRAM-MD5, DIGEST-MD5 or TLS-encryption '.
  567.             'in the SquirrelMail configuration.'FALSE);
  568. }
  569.  
  570. if (stristr($capline'XMAGICTRASH'!== false{
  571.     $magic_trash 'It looks like IMAP_MOVE_EXPUNGE_TO_TRASH option is turned on '
  572.         .'in your Courier IMAP configuration. Courier does not provide tools that '
  573.         .'allow to detect folder used for Trash or commands are not documented. '
  574.         .'SquirrelMail can\'t detect special trash folder. SquirrelMail manages '
  575.         .'all message deletion or move operations internally and '
  576.         .'IMAP_MOVE_EXPUNGE_TO_TRASH option can cause errors in message and '
  577.         .'folder management operations. Please turn off IMAP_MOVE_EXPUNGE_TO_TRASH '
  578.         .'option in Courier imapd configuration.';
  579.     do_err($magic_trash,false);
  580. }
  581.  
  582. /* add warning about IMAP delivery */
  583. if (stristr($capline'XCOURIEROUTBOX'!== false{
  584.     $courier_outbox 'OUTBOX setting is enabled in your Courier imapd '
  585.         .'configuration. SquirrelMail uses standard SMTP protocol or sendmail '
  586.         .'binary to send emails. Courier IMAP delivery method is not supported'
  587.         .' and can create duplicate email messages.';
  588.     do_err($courier_outbox,false);
  589. }
  590.  
  591. /** OK, close connection */
  592. fputs($stream"A004 LOGOUT\r\n");
  593. fclose($stream);
  594.  
  595. echo "Checking internationalization (i18n) settings...<br />\n";
  596. echo "$IND gettext - ";
  597. if (function_exists('gettext')) {
  598.     echo 'Gettext functions are available.'
  599.         .' On some systems you must have appropriate system locales compiled.'
  600.         ."<br />\n";
  601.  
  602.     /* optional setlocale() tests. Should work only on glibc systems. */
  603.     if (sqgetGlobalVar('testlocales',$testlocales,SQ_GET)) {
  604.         include_once(SM_PATH 'include/languages.php');
  605.         echo $IND $IND 'Testing translations:<br>';
  606.         foreach ($languages as $lang_code => $lang_data{
  607.             /* don't test aliases */
  608.             if (isset($lang_data['NAME'])) {
  609.                 /* locale can be $lang_code or $lang_data['LOCALE'] */
  610.                 if (isset($lang_data['LOCALE'])) {
  611.                     $setlocale $lang_data['LOCALE'];
  612.                 else {
  613.                     $setlocale $lang_code;
  614.                 }
  615.                 /* prepare information about tested locales */
  616.                 if (is_array($setlocale)) {
  617.                     $display_locale implode(', ',$setlocale);
  618.                     $locale_count count($setlocale);
  619.                 else {
  620.                     $display_locale $setlocale;
  621.                     $locale_count 1;
  622.                 }
  623.                 $tested_locales_msg 'Tested '.htmlspecialchars($display_locale).' '
  624.                     .($locale_count>'locales':'locale')'.';
  625.  
  626.                 echo $IND $IND .$IND $lang_data['NAME'].' (' .$lang_code') - ';
  627.                 $retlocale sq_setlocale(LC_ALL,$setlocale);
  628.                 if (is_bool($retlocale)) {
  629.                     echo '<font color="red">unsupported</font>. ';
  630.                     echo $tested_locales_msg;
  631.                 else {
  632.                     echo 'supported. '
  633.                         .$tested_locales_msg
  634.                         .' setlocale() returned "'.htmlspecialchars($retlocale).'"';
  635.                 }
  636.                 echo "<br />\n";
  637.             }
  638.         }
  639.         echo $IND $IND '<a href="configtest.php">Don\'t test translations</a>';
  640.     else {
  641.         echo $IND $IND '<a href="configtest.php?testlocales=1">Test translations</a>. '
  642.             .'This test is not accurate and might work only on some systems.'
  643.             ."\n";
  644.     }
  645.     echo "<br />\n";
  646.     /* end of translation tests */
  647. else {
  648.     echo 'Gettext functions are unavailable.'
  649.         .' SquirrelMail will use slower internal gettext functions.'
  650.         ."<br />\n";
  651. }
  652. echo "$IND mbstring - ";
  653. if (function_exists('mb_detect_encoding')) {
  654.     echo "Mbstring functions are available.<br />\n";
  655. else {
  656.     echo 'Mbstring functions are unavailable.'
  657.         ." Japanese translation won't work.<br />\n";
  658. }
  659. echo "$IND recode - ";
  660. if (function_exists('recode')) {
  661.     echo "Recode functions are available.<br />\n";
  662. elseif (isset($use_php_recode&& $use_php_recode{
  663.     echo "Recode functions are unavailable.<br />\n";
  664.     do_err('Your configuration requires recode support, but recode support is missing.');
  665. else {
  666.     echo "Recode functions are unavailable.<br />\n";
  667. }
  668. echo "$IND iconv - ";
  669. if (function_exists('iconv')) {
  670.     echo "Iconv functions are available.<br />\n";
  671. elseif (isset($use_php_iconv&& $use_php_iconv{
  672.     echo "Iconv functions are unavailable.<br />\n";
  673.     do_err('Your configuration requires iconv support, but iconv support is missing.');
  674. else {
  675.     echo "Iconv functions are unavailable.<br />\n";
  676. }
  677. // same test as in include/init.php + date_default_timezone_set check
  678. echo "$IND timezone - ";
  679. if ( (!ini_get('safe_mode')) || function_exists('date_default_timezone_set'||
  680.         !strcmp(ini_get('safe_mode_allowed_env_vars'),''||
  681.         preg_match('/^([\w_]+,)*TZ/'ini_get('safe_mode_allowed_env_vars')) ) {
  682.     echo "Webmail users can change their time zone settings. \n";
  683. else {
  684.     echo "Webmail users can't change their time zone settings. \n";
  685. }
  686. if (isset($_ENV['TZ'])) {
  687.     echo 'Default time zone is '.htmlspecialchars($_ENV['TZ']);
  688. else {
  689.     echo 'Current time zone is '.date('T');
  690. }
  691. echo ".<br />\n";
  692.  
  693. // Pear DB tests
  694. echo "Checking database functions...<br />\n";
  695. if($addrbook_dsn || $prefs_dsn || $addrbook_global_dsn{
  696.     @include_once('DB.php');
  697.     if (class_exists('DB')) {
  698.         echo "$IND PHP Pear DB support is present.<br />\n";
  699.         $db_functions=array(
  700.                 'dbase' => 'dbase_open',
  701.                 'fbsql' => 'fbsql_connect',
  702.                 'interbase' => 'ibase_connect',
  703.                 'informix' => 'ifx_connect',
  704.                 'msql' => 'msql_connect',
  705.                 'mssql' => 'mssql_connect',
  706.                 'mysql' => 'mysql_connect',
  707.                 'mysqli' => 'mysqli_connect',
  708.                 'oci8' => 'ocilogon',
  709.                 'odbc' => 'odbc_connect',
  710.                 'pgsql' => 'pg_connect',
  711.                 'sqlite' => 'sqlite_open',
  712.                 'sybase' => 'sybase_connect'
  713.                 );
  714.  
  715.         $dsns array();
  716.         if($prefs_dsn{
  717.             $dsns['preferences'$prefs_dsn;
  718.         }
  719.         if($addrbook_dsn{
  720.             $dsns['addressbook'$addrbook_dsn;
  721.         }
  722.         if($addrbook_global_dsn{
  723.             $dsns['global addressbook'$addrbook_global_dsn;
  724.         }
  725.  
  726.         foreach($dsns as $type => $dsn{
  727.             $aDsn explode(':'$dsn);
  728.             $dbtype array_shift($aDsn);
  729.             if(isset($db_functions[$dbtype]&& function_exists($db_functions[$dbtype])) {
  730.                 echo "$IND$dbtype database support present.<br />\n";
  731.  
  732.                 // now, test this interface:
  733.  
  734.                 
  735.                 $dbh DB::connect($dsntrue);
  736.                 if (DB::isError($dbh)) {
  737.                     do_err('Database error: 'htmlspecialchars(DB::errorMessage($dbh)) .
  738.                             ' in ' .$type .' DSN.');
  739.                 }
  740.                 $dbh->disconnect();
  741.                 echo "$IND$type database connect successful.<br />\n";
  742.  
  743.             else {
  744.                 do_err($dbtype.' database support not present!');
  745.             }
  746.         }
  747.     else {
  748.         $db_error='Required PHP PEAR DB support is not available.'
  749.             .' Is PEAR installed and is the include path set correctly to find <tt>DB.php</tt>?'
  750.             .' The include path is now:<tt>' ini_get('include_path''</tt>.';
  751.         do_err($db_error);
  752.     }
  753. else {
  754.     echo $IND."not using database functionality.<br />\n";
  755. }
  756.  
  757. // LDAP DB tests
  758. echo "Checking LDAP functions...<br />\n";
  759. ifempty($ldap_server) ) {
  760.     echo $IND."not using LDAP functionality.<br />\n";
  761. else {
  762.     if !function_exists('ldap_connect') ) {
  763.         do_err('Required LDAP support is not available.');
  764.     else {
  765.         echo "$IND LDAP support present.<br />\n";
  766.         foreach $ldap_server as $param {
  767.  
  768.             $linkid @ldap_connect($param['host'](empty($param['port']389 $param['port']) );
  769.  
  770.             if $linkid {
  771.                 echo "$IND LDAP connect to ".$param['host']." successful: ".$linkid."<br />\n";
  772.  
  773.                 if !empty($param['protocol']&&
  774.                         !ldap_set_option($linkidLDAP_OPT_PROTOCOL_VERSION$param['protocol']) ) {
  775.                     do_err('Unable to set LDAP protocol');
  776.                 }
  777.  
  778.                 if empty($param['binddn']) ) {
  779.                     $bind @ldap_bind($linkid);
  780.                 else {
  781.                     $bind @ldap_bind($param['binddn']$param['bindpw']);
  782.                 }
  783.  
  784.                 if $bind {
  785.                     echo "$IND LDAP Bind Successful <br />";
  786.                 else {
  787.                     do_err('Unable to Bind to LDAP Server');
  788.                 }
  789.  
  790.                 @ldap_close($linkid);
  791.             else {
  792.                 do_err('Connection to LDAP failed');
  793.             }
  794.         }
  795.     }
  796. }
  797.  
  798. echo '<hr width="75%" align="center">';
  799. echo '<h2 align="center">Summary</h2>';
  800. $footer '<hr width="75%" align="center">';
  801. if ($warnings{
  802.     echo '<p>No fatal errors were found, but there was at least 1 warning.  Please check the flagged issue(s) carefully, as correcting them may prevent erratic, undefined, or incorrect behavior (or flat out breakage).</p>';
  803.     echo $footer;
  804. else {
  805.     print <<< EOF
  806. <p>Congratulationsyour SquirrelMail setup looks fine to me!</p>
  807.  
  808. <p><a href="login.php">Login now</a></p>
  809.  
  810. </body>
  811. </html>
  812. EOF;
  813.     echo $footer;
  814. }
  815. ?>

Documentation generated on Sat, 07 Oct 2006 16:09:49 +0300 by phpDocumentor 1.3.0RC6