init.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608
  1. <?php
  2. /**
  3. * init.php -- initialisation file
  4. *
  5. * File should be loaded in every file in src/ or plugins that occupate an entire frame
  6. *
  7. * @copyright &copy; 2006 The SquirrelMail Project Team
  8. * @license http://opensource.org/licenses/gpl-license.php GNU Public License
  9. * @version $Id$
  10. * @package squirrelmail
  11. */
  12. /**
  13. * This is a development version so in order to track programmer mistakes we
  14. * set the error reporting to E_ALL
  15. */
  16. error_reporting(E_ALL);
  17. /**
  18. * If register_globals are on, unregister globals.
  19. * Second test covers boolean set as string (php_value register_globals off).
  20. */
  21. if ((bool) ini_get('register_globals') &&
  22. strtolower(ini_get('register_globals'))!='off') {
  23. /**
  24. * Remove all globals that are not reserved by PHP
  25. * 'value' and 'key' are used by foreach. Don't unset them inside foreach.
  26. */
  27. foreach ($GLOBALS as $key => $value) {
  28. switch($key) {
  29. case 'HTTP_POST_VARS':
  30. case '_POST':
  31. case 'HTTP_GET_VARS':
  32. case '_GET':
  33. case 'HTTP_COOKIE_VARS':
  34. case '_COOKIE':
  35. case 'HTTP_SERVER_VARS':
  36. case '_SERVER':
  37. case 'HTTP_ENV_VARS':
  38. case '_ENV':
  39. case 'HTTP_POST_FILES':
  40. case '_FILES':
  41. case '_REQUEST':
  42. case 'HTTP_SESSION_VARS':
  43. case '_SESSION':
  44. case 'GLOBALS':
  45. case 'key':
  46. case 'value':
  47. break;
  48. case 'sInitLocation':
  49. // FIXME: variable must be set only in src/login.php
  50. break;
  51. default:
  52. unset($GLOBALS[$key]);
  53. }
  54. }
  55. // Unset variables used in foreach
  56. unset($GLOBALS['key']);
  57. unset($GLOBALS['value']);
  58. }
  59. /**
  60. * [#1518885] session.use_cookies = off breaks SquirrelMail
  61. *
  62. * When session cookies are not used, all http redirects, meta refreshes,
  63. * src/download.php and javascript URLs are broken. Setting must be set
  64. * before session is started.
  65. */
  66. if (!(bool)ini_get('session.use_cookies') ||
  67. ini_get('session.use_cookies') == 'off') {
  68. ini_set('session.use_cookies','1');
  69. }
  70. /**
  71. * calculate SM_PATH and calculate the base_uri
  72. * assumptions made: init.php is only called from plugins or from the src dir.
  73. * files in the plugin directory may not be part of a subdirectory called "src"
  74. *
  75. */
  76. if (isset($_SERVER['SCRIPT_NAME'])) {
  77. $a = explode('/',$_SERVER['SCRIPT_NAME']);
  78. } elseif (isset($HTTP_SERVER_VARS['SCRIPT_NAME'])) {
  79. $a = explode('/',$HTTP_SERVER_VARS['SCRIPT_NAME']);
  80. } else {
  81. $error = 'Unable to detect script environment. '
  82. .'Please test your PHP settings and send PHP core config, $_SERVER '
  83. .'and $HTTP_SERVER_VARS to SquirrelMail developers.';
  84. die($error);
  85. }
  86. $sSM_PATH = '';
  87. for($i = count($a) -2;$i > -1; --$i) {
  88. $sSM_PATH .= '../';
  89. if ($a[$i] === 'src' || $a[$i] === 'plugins') {
  90. break;
  91. }
  92. }
  93. $base_uri = implode('/',array_slice($a,0,$i)). '/';
  94. define('SM_PATH',$sSM_PATH);
  95. define('SM_BASE_URI', $base_uri);
  96. /**
  97. * global var $bInit is used to check if initialisation took place.
  98. * At this moment it's a workarounf for the include of addrbook_search_html
  99. * inside compose.php. If we found a better way then remove this. Do only use
  100. * this var if you know for sure a page can be called stand alone and be included
  101. * in another file.
  102. */
  103. $bInit = true;
  104. /**
  105. * This theme as a failsafe if no themes were found, or if we error
  106. * out before anything could be initialised.
  107. */
  108. $color = array();
  109. $color[0] = '#DCDCDC'; /* light gray TitleBar */
  110. $color[1] = '#800000'; /* red */
  111. $color[2] = '#CC0000'; /* light red Warning/Error Messages */
  112. $color[3] = '#A0B8C8'; /* green-blue Left Bar Background */
  113. $color[4] = '#FFFFFF'; /* white Normal Background */
  114. $color[5] = '#FFFFCC'; /* light yellow Table Headers */
  115. $color[6] = '#000000'; /* black Text on left bar */
  116. $color[7] = '#0000CC'; /* blue Links */
  117. $color[8] = '#000000'; /* black Normal text */
  118. $color[9] = '#ABABAB'; /* mid-gray Darker version of #0 */
  119. $color[10] = '#666666'; /* dark gray Darker version of #9 */
  120. $color[11] = '#770000'; /* dark red Special Folders color */
  121. $color[12] = '#EDEDED';
  122. $color[13] = '#800000'; /* (dark red) Color for quoted text -- > 1 quote */
  123. $color[14] = '#ff0000'; /* (red) Color for quoted text -- >> 2 or more */
  124. $color[15] = '#002266'; /* (dark blue) Unselectable folders */
  125. $color[16] = '#ff9933'; /* (orange) Highlight color */
  126. require(SM_PATH . 'functions/global.php');
  127. require(SM_PATH . 'functions/arrays.php');
  128. /* load default configuration */
  129. require(SM_PATH . 'config/config_default.php');
  130. /* reset arrays in default configuration */
  131. $ldap_server = array();
  132. $plugins = array();
  133. $fontsets = array();
  134. $aTemplateSet = array();
  135. $aTemplateSet[0]['ID'] = 'default';
  136. $aTemplateSet[0]['NAME'] = 'Default';
  137. /* load site configuration */
  138. require(SM_PATH . 'config/config.php');
  139. /* load local configuration overrides */
  140. if (file_exists(SM_PATH . 'config/config_local.php')) {
  141. require(SM_PATH . 'config/config_local.php');
  142. }
  143. require(SM_PATH . 'functions/plugin.php');
  144. require(SM_PATH . 'include/constants.php');
  145. require(SM_PATH . 'include/languages.php');
  146. require(SM_PATH . 'class/template/Template.class.php');
  147. require(SM_PATH . 'class/error.class.php');
  148. /**
  149. * If magic_quotes_runtime is on, SquirrelMail breaks in new and creative ways.
  150. * Force magic_quotes_runtime off.
  151. * tassium@squirrelmail.org - I put it here in the hopes that all SM code includes this.
  152. * If there's a better place, please let me know.
  153. */
  154. ini_set('magic_quotes_runtime','0');
  155. /* if running with magic_quotes_gpc then strip the slashes
  156. from POST and GET global arrays */
  157. if (get_magic_quotes_gpc()) {
  158. sqstripslashes($_GET);
  159. sqstripslashes($_POST);
  160. }
  161. /* strip any tags added to the url from PHP_SELF.
  162. This fixes hand crafted url XXS expoits for any
  163. page that uses PHP_SELF as the FORM action */
  164. $_SERVER['PHP_SELF'] = strip_tags($_SERVER['PHP_SELF']);
  165. $PHP_SELF = php_self();
  166. /**
  167. * Initialize the session
  168. */
  169. /** set the name of the session cookie */
  170. if (!isset($session_name) || !$session_name) {
  171. $session_name = 'SQMSESSID';
  172. }
  173. /**
  174. * if session.auto_start is On then close the session
  175. */
  176. $sSessionAutostartName = session_name();
  177. if ((isset($sSessionAutostartName) || $sSessionAutostartName == '') &&
  178. $sSessionAutostartName !== $session_name) {
  179. $sCookiePath = ini_get('session.cookie_path');
  180. $sCookieDomain = ini_get('session.cookie_domain');
  181. // reset the cookie
  182. setcookie($sSessionAutostartName,'',time() - 604800,$sCookiePath,$sCookieDomain);
  183. @session_destroy();
  184. session_write_close();
  185. }
  186. /**
  187. * includes from classes stored in the session
  188. */
  189. require(SM_PATH . 'class/mime.class.php');
  190. ini_set('session.name' , $session_name);
  191. session_set_cookie_params (0, $base_uri);
  192. sqsession_is_active();
  193. /**
  194. * SquirrelMail version number -- DO NOT CHANGE
  195. */
  196. $version = '1.5.2 [CVS]';
  197. /**
  198. * SquirrelMail internal version number -- DO NOT CHANGE
  199. * $sm_internal_version = array (release, major, minor)
  200. */
  201. $SQM_INTERNAL_VERSION = array(1,5,2);
  202. /**
  203. * Include Compatibility plugin if available.
  204. */
  205. if (file_exists(SM_PATH . 'plugins/compatibility/functions.php'))
  206. include_once(SM_PATH . 'plugins/compatibility/functions.php');
  207. /**
  208. * MAIN PLUGIN LOADING CODE HERE
  209. * On init, we no longer need to load all plugin setup files.
  210. * Now, we load the statically generated hook registrations here
  211. * and let the hook calls include only the plugins needed.
  212. */
  213. $squirrelmail_plugin_hooks = array();
  214. if (file_exists(SM_PATH . 'config/plugin_hooks.php')) {
  215. require(SM_PATH . 'config/plugin_hooks.php');
  216. }
  217. /**
  218. * allow plugins to override main configuration; hook is placed
  219. * here to allow plugins to use session information to do their work
  220. */
  221. do_hook('config_override');
  222. /**
  223. * DISABLED.
  224. * Remove globalized session data in rg=on setups
  225. *
  226. * Code can be utilized when session is started, but data is not loaded.
  227. * We have already loaded configuration and other important vars. Can't
  228. * clean session globals here.
  229. if ((bool) @ini_get('register_globals') &&
  230. strtolower(ini_get('register_globals'))!='off') {
  231. foreach ($_SESSION as $key => $value) {
  232. unset($GLOBALS[$key]);
  233. }
  234. }
  235. */
  236. sqsession_register(SM_BASE_URI,'base_uri');
  237. /**
  238. * Retrieve the language cookie
  239. */
  240. if (! sqgetGlobalVar('squirrelmail_language',$squirrelmail_language,SQ_COOKIE)) {
  241. $squirrelmail_language = '';
  242. }
  243. /**
  244. * @var $sInitlocation From where do we include.
  245. */
  246. if (!isset($sInitLocation)) {
  247. $sInitLocation=NULL;
  248. }
  249. /**
  250. * Before 1.5.2 version hook was part of functions/constants.php.
  251. * After init layout changes, hook had to be moved because include/constants.php is
  252. * loaded before plugins are initialized.
  253. * @since 1.2.0
  254. */
  255. do_hook('loading_constants');
  256. switch ($sInitLocation) {
  257. case 'style':
  258. // need to get the right template set up
  259. //
  260. sqGetGlobalVar('templateid', $templateid, SQ_GET);
  261. // sanitize just in case...
  262. //
  263. $templateid = preg_replace('/(\.\.\/){1,}/', '', $templateid);
  264. // make sure given template actually is available
  265. //
  266. $found_templateset = false;
  267. for ($i = 0; $i < count($aTemplateSet); ++$i) {
  268. if ($aTemplateSet[$i]['ID'] == $templateid) {
  269. $found_templateset = true;
  270. break;
  271. }
  272. }
  273. // FIXME: do we need/want to check here for actual (physical) presence of template sets?
  274. // selected template not available, fall back to default template
  275. //
  276. if (!$found_templateset) {
  277. $sTemplateID = Template::get_default_template_set();
  278. } else {
  279. $sTemplateID = $templateid;
  280. }
  281. session_write_close();
  282. sqsetcookieflush();
  283. break;
  284. case 'redirect':
  285. /**
  286. * directory hashing functions are needed for all setups in case
  287. * plugins use own pref files.
  288. */
  289. require(SM_PATH . 'functions/prefs.php');
  290. require(SM_PATH . 'functions/auth.php');
  291. /* hook loads custom prefs backend plugins */
  292. $prefs_backend = do_hook_function('prefs_backend');
  293. if (isset($prefs_backend) && !empty($prefs_backend) && file_exists(SM_PATH . $prefs_backend)) {
  294. require(SM_PATH . $prefs_backend);
  295. } elseif (isset($prefs_dsn) && !empty($prefs_dsn)) {
  296. require(SM_PATH . 'functions/db_prefs.php');
  297. } else {
  298. require(SM_PATH . 'functions/file_prefs.php');
  299. }
  300. //nobreak;
  301. case 'login':
  302. require(SM_PATH . 'functions/display_messages.php' );
  303. require(SM_PATH . 'functions/page_header.php');
  304. require(SM_PATH . 'functions/html.php');
  305. // reset template file cache
  306. //
  307. $sTemplateID = Template::get_default_template_set();
  308. Template::cache_template_file_hierarchy(TRUE);
  309. /**
  310. * Make sure icon variables are setup for the login page.
  311. */
  312. $icon_theme = $icon_themes[$icon_theme_def]['PATH'];
  313. /*
  314. * NOTE: The $icon_theme_path var should contain the path to the icon
  315. * theme to use. If the admin has disabled icons, or the user has
  316. * set the icon theme to "None," no icons will be used.
  317. */
  318. $icon_theme_path = (!$use_icons || $icon_theme=='none') ? NULL : ($icon_theme == 'template' ? SM_PATH . Template::calculate_template_images_directory($sTemplateID) : $icon_theme);
  319. /**
  320. * cleanup old cookies with a cookie path the same as the standard php.ini
  321. * cookie path. All previous SquirrelMail version used the standard php.ini
  322. * cookie path for storing the session name. That behaviour changed.
  323. */
  324. if ($sCookiePath !== SM_BASE_URI) {
  325. /**
  326. * do not delete the standard sessions with session.name is i.e. PHPSESSID
  327. * because they probably belong to other php apps
  328. */
  329. if (ini_get('session.name') !== $sSessionAutostartName) {
  330. sqsetcookie(ini_get('session.name'),'',0,$sCookiePath);
  331. }
  332. }
  333. break;
  334. default:
  335. require(SM_PATH . 'functions/display_messages.php' );
  336. require(SM_PATH . 'functions/page_header.php');
  337. require(SM_PATH . 'functions/html.php');
  338. require(SM_PATH . 'functions/strings.php');
  339. /**
  340. * Check if we are logged in
  341. */
  342. require(SM_PATH . 'functions/auth.php');
  343. if ( !sqsession_is_registered('user_is_logged_in') ) {
  344. // First we store some information in the new session to prevent
  345. // information-loss.
  346. //
  347. $session_expired_post = $_POST;
  348. $session_expired_location = $PHP_SELF;
  349. if (!sqsession_is_registered('session_expired_post')) {
  350. sqsession_register($session_expired_post,'session_expired_post');
  351. }
  352. if (!sqsession_is_registered('session_expired_location')) {
  353. sqsession_register($session_expired_location,'session_expired_location');
  354. }
  355. // signout page will deal with users who aren't logged
  356. // in on its own; don't show error here
  357. //
  358. if (strpos($PHP_SELF, 'signout.php') !== FALSE) {
  359. return;
  360. }
  361. /**
  362. * Initialize the template object (logout_error uses it)
  363. */
  364. /*
  365. * $sTemplateID is not initialized when a user is not logged in, so we
  366. * will use the config file defaults here. If the neccesary variables
  367. * are net set, force a default value.
  368. */
  369. $sTemplateID = Template::get_default_template_set();
  370. $oTemplate = Template::construct_template($sTemplateID);
  371. set_up_language($squirrelmail_language, true);
  372. logout_error( _("You must be logged in to access this page.") );
  373. exit;
  374. }
  375. sqgetGlobalVar('username',$username,SQ_SESSION);
  376. sqgetGlobalVar('authz',$authz,SQ_SESSION);
  377. /**
  378. * Setting the prefs backend
  379. */
  380. sqgetGlobalVar('prefs_cache', $prefs_cache, SQ_SESSION );
  381. sqgetGlobalVar('prefs_are_cached', $prefs_are_cached, SQ_SESSION );
  382. if ( !sqsession_is_registered('prefs_are_cached') ||
  383. !isset( $prefs_cache) ||
  384. !is_array( $prefs_cache)) {
  385. $prefs_are_cached = false;
  386. $prefs_cache = false; //array();
  387. }
  388. /* see 'redirect' case */
  389. require(SM_PATH . 'functions/prefs.php');
  390. $prefs_backend = do_hook_function('prefs_backend');
  391. if (isset($prefs_backend) && !empty($prefs_backend) && file_exists(SM_PATH . $prefs_backend)) {
  392. require(SM_PATH . $prefs_backend);
  393. } elseif (isset($prefs_dsn) && !empty($prefs_dsn)) {
  394. require(SM_PATH . 'functions/db_prefs.php');
  395. } else {
  396. require(SM_PATH . 'functions/file_prefs.php');
  397. }
  398. /**
  399. * initializing user settings
  400. */
  401. require(SM_PATH . 'include/load_prefs.php');
  402. // i do not understand the frames language cookie story
  403. /**
  404. * We'll need this to later have a noframes version
  405. *
  406. * Check if the user has a language preference, but no cookie.
  407. * Send him a cookie with his language preference, if there is
  408. * such discrepancy.
  409. */
  410. $my_language = getPref($data_dir, $username, 'language');
  411. if ($my_language != $squirrelmail_language) {
  412. sqsetcookie('squirrelmail_language', $my_language, time()+2592000, $base_uri);
  413. }
  414. // /dont understand
  415. /**
  416. * Set up the language.
  417. */
  418. $err=set_up_language(getPref($data_dir, $username, 'language'));
  419. /* this is the last cookie we set so flush it. */
  420. sqsetcookieflush();
  421. // Japanese translation used without mbstring support
  422. if ($err==2) {
  423. $sError =
  424. "<p>You need to have PHP installed with the multibyte string function \n".
  425. "enabled (using configure option --enable-mbstring).</p>\n".
  426. "<p>System assumed that you accidently switched to Japanese translation \n".
  427. "and reverted your language preference to English.</p>\n".
  428. "<p>Please refresh this page in order to use webmail.</p>\n";
  429. error_box($sError);
  430. }
  431. $timeZone = getPref($data_dir, $username, 'timezone');
  432. /* Check to see if we are allowed to set the TZ environment variable.
  433. * We are able to do this if ...
  434. * safe_mode is disabled OR
  435. * safe_mode_allowed_env_vars is empty (you are allowed to set any) OR
  436. * safe_mode_allowed_env_vars contains TZ
  437. */
  438. $tzChangeAllowed = (!ini_get('safe_mode')) ||
  439. !strcmp(ini_get('safe_mode_allowed_env_vars'),'') ||
  440. preg_match('/^([\w_]+,)*TZ/', ini_get('safe_mode_allowed_env_vars'));
  441. if ( $timeZone != SMPREF_NONE && ($timeZone != "")
  442. && $tzChangeAllowed ) {
  443. // get time zone key, if strict or custom strict timezones are used
  444. if (isset($time_zone_type) &&
  445. ($time_zone_type == 1 || $time_zone_type == 3)) {
  446. /* load time zone functions */
  447. require(SM_PATH . 'include/timezones.php');
  448. $realTimeZone = sq_get_tz_key($timeZone);
  449. } else {
  450. $realTimeZone = $timeZone;
  451. }
  452. // set time zone
  453. if ($realTimeZone) {
  454. putenv("TZ=".$realTimeZone);
  455. }
  456. }
  457. /**
  458. * php 5.1.0 added time zone functions. Set time zone with them in order
  459. * to prevent E_STRICT notices and allow time zone modifications in safe_mode.
  460. */
  461. if (function_exists('date_default_timezone_set')) {
  462. if ($timeZone != SMPREF_NONE && $timeZone != "") {
  463. date_default_timezone_set($timeZone);
  464. } else {
  465. // interface runs on server's time zone. Remove php E_STRICT complains
  466. $default_timezone = @date_default_timezone_get();
  467. date_default_timezone_set($default_timezone);
  468. }
  469. }
  470. break;
  471. }
  472. /*
  473. * $sTemplateID is not initialized when a user is not logged in, so we
  474. * will use the config file defaults here. If the neccesary variables
  475. * are not set, force a default value.
  476. *
  477. * If the user is logged in, $sTemplateID will be set in load_prefs.php,
  478. * so we shouldn't change it here.
  479. */
  480. if (!isset($sTemplateID)) {
  481. $sTemplateID = Template::get_default_template_set();
  482. $icon_theme_path = !$use_icons ? NULL : Template::calculate_template_images_directory($sTemplateID);
  483. }
  484. // template object may have already been constructed in load_prefs.php
  485. //
  486. if (empty($oTemplate)) {
  487. $oTemplate = Template::construct_template($sTemplateID);
  488. }
  489. // We want some variables to always be available to the template
  490. $always_include = array('sTemplateID', 'icon_theme_path', 'javascript_on');
  491. foreach ($always_include as $var) {
  492. $oTemplate->assign($var, (isset($$var) ? $$var : NULL));
  493. }
  494. /**
  495. * Initialize our custom error handler object
  496. */
  497. $oErrorHandler = new ErrorHandler($oTemplate,'error_message.tpl');
  498. /**
  499. * Activate custom error handling
  500. */
  501. if (version_compare(PHP_VERSION, "4.3.0", ">=")) {
  502. $oldErrorHandler = set_error_handler(array($oErrorHandler, 'SquirrelMailErrorhandler'));
  503. } else {
  504. $oldErrorHandler = set_error_handler('SquirrelMailErrorhandler');
  505. }
  506. /**
  507. * Javascript support detection function
  508. * @param boolean $reset recheck javascript support if set to true.
  509. * @return integer SMPREF_JS_ON or SMPREF_JS_OFF ({@see include/constants.php})
  510. * @since 1.5.1
  511. */
  512. function checkForJavascript($reset = FALSE) {
  513. global $data_dir, $username, $javascript_on, $javascript_setting;
  514. if ( !$reset && sqGetGlobalVar('javascript_on', $javascript_on, SQ_SESSION) )
  515. return $javascript_on;
  516. if ( $reset || !isset($javascript_setting) )
  517. $javascript_setting = getPref($data_dir, $username, 'javascript_setting', SMPREF_JS_AUTODETECT);
  518. if ( !sqGetGlobalVar('new_js_autodetect_results', $js_autodetect_results) &&
  519. !sqGetGlobalVar('js_autodetect_results', $js_autodetect_results) )
  520. $js_autodetect_results = SMPREF_JS_OFF;
  521. if ( $javascript_setting == SMPREF_JS_AUTODETECT )
  522. $javascript_on = $js_autodetect_results;
  523. else
  524. $javascript_on = $javascript_setting;
  525. sqsession_register($javascript_on, 'javascript_on');
  526. return $javascript_on;
  527. }
  528. function sqm_baseuri() {
  529. global $base_uri;
  530. return $base_uri;
  531. }