index.php 12.7 KB
Newer Older
1
2
3
4
5
<?php

/*
  This code is part of FusionDirectory (http://www.fusiondirectory.org/)
  Copyright (C) 2003-2010  Cajus Pollmeier
6
  Copyright (C) 2011-2013  FusionDirectory
7
8
9
10
11
12
13
14
15
16
17
18
19

  This program 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.

  This program is distributed in the hope that it will be useful,
  but WITHOUT ANY WARRANTY; without even the implied warranty of
  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  GNU General Public License for more details.

  You should have received a copy of the GNU General Public License
  along with this program; if not, write to the Free Software
20
  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA.
21
22
23
24
25
*/

/* Load required includes */
require_once ("../include/php_setup.inc");
require_once ("functions.inc");
26
require_once ("variables.inc");
27
28
29
30
require_once ("class_log.inc");
header("Content-type: text/html; charset=UTF-8");


31
/* Display the login page and exit() */
32
33
function displayLogin()
{
34
  global $smarty,$message,$config,$ssl,$error_collector,$error_collector_mailto,$lang;
35
36
37
  error_reporting(E_ALL | E_STRICT);
  /* Fill template with required values */
  $username = "";
38
39
  if (isset($_POST["username"])) {
    $username = trim(get_post("username"));
40
41
42
  }
  $smarty->assign ('date', gmdate("D, d M Y H:i:s"));
  $smarty->assign ('username', $username);
43
  $smarty->assign ('personal_img', "geticon.php?context=types&icon=user&size=48");
44
45
  $smarty->assign ('password_img', "geticon.php?context=status&icon=dialog-password&size=48");
  $smarty->assign ('directory_img', "geticon.php?context=places&icon=network-server&size=48");
46
  $smarty->append ('css_files',  get_template_path('login.css'));
47
48

  /* Some error to display? */
49
50
  if (!isset($message)) {
    $message = "";
51
52
53
54
  }
  $smarty->assign ("message", $message);

  /* Displasy SSL mode warning? */
55
  if (($ssl != "") && ($config->get_cfg_value('warnSSL') == 'TRUE')) {
56
57
58
59
60
    $smarty->assign ("ssl", _("Warning").": <a style=\"color:red;\" href=\"$ssl\">"._("Session is not encrypted!")."</a>");
  } else {
    $smarty->assign ("ssl", "");
  }

61
  if (!$config->check_session_lifetime()) {
62
    $smarty->assign ("lifetime", _("Warning").": ".
63
        _("The session lifetime configured in your fusiondirectory.conf will be overridden by php.ini settings."));
64
  } else {
65
66
67
68
    $smarty->assign ("lifetime", "");
  }

  /* Generate server list */
69
70
71
  $servers = array();
  if (isset($_POST['server'])) {
    $selected = get_post('server');
72
  } else {
73
    $selected = $config->data['MAIN']['DEFAULT'];
74
  }
75
76
  foreach ($config->data['LOCATIONS'] as $key => $ignored) {
    $servers[$key] = $key;
77
78
79
80
81
82
  }
  $smarty->assign ("server_options", $servers);
  $smarty->assign ("server_id", $selected);

  /* show login screen */
  $smarty->assign ("PHPSESSID", session_id());
83
  if (session::is_set('errors')) {
84
85
    $smarty->assign("errors", session::get('errors'));
  }
86
87
  if ($error_collector != "") {
    $smarty->assign("php_errors", preg_replace("/%BUGBODY%/", $error_collector_mailto, $error_collector)."</div>");
88
89
90
91
92
  } else {
    $smarty->assign("php_errors", "");
  }
  $smarty->assign("msg_dialogs", msg_dialog::get_dialogs());
  $smarty->assign("usePrototype", "false");
93
  $smarty->assign("date", date("l, dS F Y H:i:s O"));
94
95
  $smarty->assign("lang", preg_replace('/_.*$/', '', $lang));
  $smarty->assign("rtl", language_is_rtl($lang));
96

97
98
  $smarty->display (get_template_path('headers.tpl'));
  $smarty->assign("version", FD_VERSION);
99

100
101
102
103
104
105
106
107
108
109
110
  $smarty->display(get_template_path('login.tpl'));
  exit();
}

/*****************************************************************************
 *                               M   A   I   N                               *
 *****************************************************************************/

/* Set error handler to own one, initialize time calculation
   and start session. */
session::start();
111
session::set('errorsAlreadyPosted', array());
112

113
/* Destroy old session if exists.
114
   Else you will get your old session back, if you not logged out correctly. */
115
if (is_array(session::get_all()) && count(session::get_all())) {
116
117
118
119
  session::destroy();
  session::start();
}

120
$username = "";
121
122

/* Reset errors */
123
124
125
session::set('errors', "");
session::set('errorsAlreadyPosted', "");
session::set('LastError', "");
126
127

/* Check if we need to run setup */
128
if (!file_exists(CONFIG_DIR."/".CONFIG_FILE)) {
129
130
131
132
133
  header("location:setup.php");
  exit();
}

/* Reset errors */
134
session::set('errors', "");
135

136
/* Check if fusiondirectory.conf (.CONFIG_FILE) is accessible */
137
if (!is_readable(CONFIG_DIR."/".CONFIG_FILE)) {
138
  msg_dialog::display(_("Configuration error"), sprintf(_("FusionDirectory configuration %s/%s is not readable. Please run fusiondirectory-setup --check-config to fix this."), CONFIG_DIR, CONFIG_FILE), FATAL_ERROR_DIALOG);
139
140
141
142
  exit();
}

/* Parse configuration file */
143
$config = new config(CONFIG_DIR."/".CONFIG_FILE, $BASE_DIR);
144
145
146
147
if ($_SERVER["REQUEST_METHOD"] == "POST") {
  session::global_set('DEBUGLEVEL', 0);
} else {
  session::global_set('DEBUGLEVEL', $config->get_cfg_value('DEBUGLEVEL'));
148
149
150
151
  @DEBUG (DEBUG_CONFIG, __LINE__, __FUNCTION__, __FILE__, $config->data, "config");
}

/* Set template compile directory */
152
$smarty->compile_dir = $config->get_cfg_value("templateCompileDirectory", SPOOL_DIR);
153
154

/* Check for compile directory */
155
156
if (!(is_dir($smarty->compile_dir) && is_writable($smarty->compile_dir))) {
  msg_dialog::display(_("Smarty error"), sprintf(_("Directory '%s' specified as compile directory is not accessible!"),
157
        $smarty->compile_dir), FATAL_ERROR_DIALOG);
158
159
160
161
162
163
164
  exit();
}

/* Check for old files in compile directory */
clean_smarty_compile_dir($smarty->compile_dir);

/* Language setup */
165
$lang = get_browser_language();
166
167
168
putenv("LANGUAGE=");
putenv("LANG=$lang");
setlocale(LC_ALL, $lang);
169
$GLOBALS['t_language']            = $lang;
170
171
$GLOBALS['t_gettext_message_dir'] = $BASE_DIR.'/locale/';

172
173
/* Set the text domain as 'fusiondirectory' */
$domain = 'fusiondirectory';
174
175
176
177
bindtextdomain($domain, LOCALE_DIR);
textdomain($domain);
$smarty->assign ('nextfield', 'username');

178
if ($_SERVER["REQUEST_METHOD"] != "POST") {
179
180
181
182
  @DEBUG (DEBUG_TRACE, __LINE__, __FUNCTION__, __FILE__, $lang, "Setting language to");
}

/* Do we have htaccess authentification enabled? */
183
184
185
$htaccess_authenticated = FALSE;
if ($config->get_cfg_value("htaccessAuthentication") == "TRUE" ) {
  if (!isset($_SERVER['REMOTE_USER'])) {
186
187
188
189
    msg_dialog::display(_("Configuration error"), _("There is a problem with the authentication setup!"), FATAL_ERROR_DIALOG);
    exit;
  }

190
191
192
193
  $tmp      = process_htaccess($_SERVER['REMOTE_USER'], isset($_SERVER['KRB5CCNAME']));
  $username = $tmp['username'];
  $server   = $tmp['server'];
  if ($username == "") {
194
195
196
    msg_dialog::display(_("Error"), _("Cannot find a valid user for the current authentication setup!"), FATAL_ERROR_DIALOG);
    exit;
  }
197
  if ($server == "") {
Benoit Mortier's avatar
Benoit Mortier committed
198
    msg_dialog::display(_("Error"), _("User information is not unique across the configured LDAP trees!"), FATAL_ERROR_DIALOG);
199
200
201
    exit;
  }

202
203
204
205
206
207
208
209
210
211
  $htaccess_authenticated = TRUE;
}
if (!$htaccess_authenticated) {
  if (isset($_POST['server'])) {
    $server = get_post("server");
  } else {
    $server = $config->data['MAIN']['DEFAULT'];
  }
}
$config->set_current($server);
212
213
214
if ($_SERVER["REQUEST_METHOD"] == "POST") {
  session::global_set('DEBUGLEVEL', 0);
}
215
216

/* If SSL is forced, just forward to the SSL enabled site */
217
if (($config->get_cfg_value("forcessl") == "TRUE") && ($ssl != '')) {
218
219
  header ("Location: $ssl");
  exit;
220
221
}

222

223
/* Got a formular answer, validate and try to log in */
224
if (($_SERVER["REQUEST_METHOD"] == "POST" && isset($_POST['login'])) || $htaccess_authenticated) {
225
226

  /* Reset error messages */
227
228
  $message = "";

229
230

  /* Destroy old sessions, they cause a successfull login to relog again ...*/
231
  if (session::global_is_set('_LAST_PAGE_REQUEST')) {
232
    session::global_set('_LAST_PAGE_REQUEST', time());
233
234
235
236
  }

  /* Admin-logon and verify */
  $ldap = $config->get_ldap_link();
237
  if (is_null($ldap) || (is_int($ldap) && $ldap == 0)) {
238
    msg_dialog::display(_("LDAP error"), msgPool::ldaperror($ldap->get_error(), $this->dn, 0, get_class()), LDAP_ERROR);
239
240
241
242
    displayLogin();
    exit();
  }

243

244
  /* Check for schema file presence */
245
246
247
  if ($config->get_cfg_value("schemaCheck") == "TRUE") {
    $recursive  = ($config->get_cfg_value("ldapFollowReferrals") == "TRUE");
    $tls        = ($config->get_cfg_value("ldapTLS") == "TRUE");
248

249
    if (!count($ldap->get_objectclasses())) {
250
251
      msg_dialog::display(_("LDAP error"), _("Cannot detect information about the installed LDAP schema!"), ERROR_DIALOG);
      displayLogin();
252
      exit();
253
    } else {
254
      $cfg = array();
255
256
257
258
      $cfg['admin']       = $config->current['ADMINDN'];
      $cfg['password']    = $config->current['ADMINPASSWORD'];
      $cfg['connection']  = $config->current['SERVER'];
      $cfg['tls']         = $tls;
259
      $str = check_schema($cfg, $config->get_cfg_value("rfc2307bis") == "TRUE");
260
      $checkarr = array();
261
262
      foreach ($str as $tr) {
        if (isset($tr['IS_MUST_HAVE']) && !$tr['STATUS']) {
263
264
265
266
267
268
269
270
          msg_dialog::display(_("LDAP error"), _("Your LDAP setup contains old schema definitions:")."<br><br><i>".$tr['MSG']."</i>", ERROR_DIALOG);
          displayLogin();
          exit();
        }
      }
    }
  }

271

272
273
  /* Check for locking area */
  $ldap->cat($config->get_cfg_value("config"), array("dn"));
274
275
  $attrs = $ldap->fetch();
  if (!count ($attrs)) {
276
277
278
279
    $ldap->cd($config->current['BASE']);
    $ldap->create_missing_trees($config->get_cfg_value("config"));
  }

280

281
  /* Check for valid input */
282
283
284
285
286
287
  $ok = TRUE;
  if (!$htaccess_authenticated) {
    $username = trim(get_post("username"));
    if (!preg_match("/^[@A-Za-z0-9_.-]+$/", $username)) {
      $message = _("Please specify a valid username!");
      $ok = FALSE;
288
    } elseif (mb_strlen($_POST["password"], 'UTF-8') == 0) {
289
      $message = _("Please specify your password!");
290
      $smarty->assign ('nextfield', 'password');
291
      $ok = FALSE;
292
293
294
295
296
297
    }
  }

  if ($ok) {

    /* Login as user, initialize user ACL's */
298
299
300
    if ($htaccess_authenticated) {
      $ui = ldap_login_user_htaccess($username);
      if ($ui === NULL || !$ui) {
301
302
303
304
        msg_dialog::display(_("Authentication error"), _("Cannot retrieve user information for htaccess authentication!"), FATAL_ERROR_DIALOG);
        exit;
      }
    } else {
305
      $ui = ldap_login_user($username, $_POST["password"]);
306
    }
307
308
    if ($ui === NULL || !$ui) {
      $message = _("Please check the username/password combination.");
309
      $smarty->assign ('nextfield', 'password');
310
      session::global_set('config', $config);
311

312
313
314
      if (isset($_SERVER['REMOTE_ADDR'])) {
        $ip = $_SERVER['REMOTE_ADDR'];
        new log("security", "login", "", array(), "Authentication failed for user \"$username\" [from $ip]");
315
      } else {
316
        new log("security", "login", "", array(), "Authentication failed for user \"$username\"");
317
      }
318
319
320
321
322
    } else {
      /* Remove all locks of this user */
      del_user_locks($ui->dn);

      /* Save userinfo and plugin structure */
323
      session::global_set('ui', $ui);
324

325
      /* Let FusionDirectory trigger a new connection for each POST, save config to session. */
326
      session::global_set('config', $config);
327

328
329
330
      /* We need a fully loaded plist and config to test account expiration */
      $plist = load_plist();

331
      /* are we using accountexpiration */
332
      if ($config->get_cfg_value("handleExpiredAccounts") == "TRUE") {
333
        $expired = $ui->expired_status();
334

335
        if ($expired == POSIX_ACCOUNT_EXPIRED) {
336
          $message = _("Account locked. Please contact your system administrator!");
337
          $smarty->assign ('nextfield', 'password');
338
          new log("security", "login", "", array(), "Account for user \"$username\" has expired");
339
340
341
342
343
344
          displayLogin();
          exit();
        }
      }

      /* Not account expired or password forced change go to main page */
345
      new log("security", "login", "", array(), "User \"$username\" logged in successfully");
346
      session::global_set('connected', 1);
347
      $config->checkLdapConfig(); // check that newly installed plugins have their configuration in the LDAP
348
      header ("Location: main.php?global_check=1");
349
350
351
352
353
354
355
356
357
      exit;
    }
  }
}

/* Translation of cookie-warning. Whether to display it, is determined by JavaScript */
$smarty->assign ("cookies", "<b>"._("Warning").":<\/b> "._("Your browser has cookies disabled. Please enable cookies and reload this page before logging in!"));

/* Set focus to the error button if we've an error message */
358
359
360
361
362
$focus = "";
if (session::is_set('errors') && session::get('errors') != "") {
  $focus = '<script type="text/javascript">';
  $focus .= 'document.forms[0].error_accept.focus();';
  $focus .= '</script>';
363
364
365
366
367
368
369
370
}
$smarty->assign("focus", $focus);
displayLogin();

?>

</body>
</html>