Auth.php 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. <?php
  2. /**
  3. * This class handles all authentication process.
  4. */
  5. class FreshRSS_Auth {
  6. /**
  7. * Determines if user is connected.
  8. */
  9. private static $login_ok = false;
  10. /**
  11. * This method initializes authentication system.
  12. */
  13. public static function init() {
  14. self::$login_ok = Minz_Session::param('loginOk', false);
  15. $current_user = Minz_Session::param('currentUser', '');
  16. if ($current_user === '') {
  17. $current_user = Minz_Configuration::defaultUser();
  18. Minz_Session::_param('currentUser', $current_user);
  19. }
  20. if (self::$login_ok) {
  21. self::giveAccess();
  22. } elseif (self::accessControl()) {
  23. self::giveAccess();
  24. FreshRSS_UserDAO::touch($current_user);
  25. } else {
  26. // Be sure all accesses are removed!
  27. self::removeAccess();
  28. }
  29. }
  30. /**
  31. * This method checks if user is allowed to connect.
  32. *
  33. * Required session parameters are also set in this method (such as
  34. * currentUser).
  35. *
  36. * @return boolean true if user can be connected, false else.
  37. */
  38. private static function accessControl() {
  39. switch (Minz_Configuration::authType()) {
  40. case 'form':
  41. $credentials = FreshRSS_FormAuth::getCredentialsFromCookie();
  42. $current_user = '';
  43. if (isset($credentials[1])) {
  44. $current_user = trim($credentials[0]);
  45. Minz_Session::_param('currentUser', $current_user);
  46. Minz_Session::_param('passwordHash', trim($credentials[1]));
  47. }
  48. return $current_user != '';
  49. case 'http_auth':
  50. $current_user = httpAuthUser();
  51. $login_ok = $current_user != '';
  52. if ($login_ok) {
  53. Minz_Session::_param('currentUser', $current_user);
  54. }
  55. return $login_ok;
  56. case 'persona':
  57. $email = filter_var(Minz_Session::param('mail'), FILTER_VALIDATE_EMAIL);
  58. $persona_file = DATA_PATH . '/persona/' . $email . '.txt';
  59. if (($current_user = @file_get_contents($persona_file)) !== false) {
  60. $current_user = trim($current_user);
  61. Minz_Session::_param('currentUser', $current_user);
  62. Minz_Session::_param('mail', $email);
  63. return true;
  64. }
  65. return false;
  66. case 'none':
  67. return true;
  68. default:
  69. // TODO load extension
  70. return false;
  71. }
  72. }
  73. /**
  74. * Gives access to the current user.
  75. */
  76. public static function giveAccess() {
  77. $current_user = Minz_Session::param('currentUser');
  78. try {
  79. $conf = new FreshRSS_Configuration($current_user);
  80. } catch(Minz_Exception $e) {
  81. die($e->getMessage());
  82. }
  83. switch (Minz_Configuration::authType()) {
  84. case 'form':
  85. self::$login_ok = Minz_Session::param('passwordHash') === $conf->passwordHash;
  86. break;
  87. case 'http_auth':
  88. self::$login_ok = strcasecmp($current_user, httpAuthUser()) === 0;
  89. break;
  90. case 'persona':
  91. self::$login_ok = strcasecmp(Minz_Session::param('mail'), $conf->mail_login) === 0;
  92. break;
  93. case 'none':
  94. self::$login_ok = true;
  95. break;
  96. default:
  97. // TODO: extensions
  98. self::$login_ok = false;
  99. }
  100. Minz_Session::_param('loginOk', self::$login_ok);
  101. }
  102. /**
  103. * Returns if current user has access to the given scope.
  104. *
  105. * @param string $scope general (default) or admin
  106. * @return boolean true if user has corresponding access, false else.
  107. */
  108. public static function hasAccess($scope = 'general') {
  109. $ok = self::$login_ok;
  110. switch ($scope) {
  111. case 'general':
  112. break;
  113. case 'admin':
  114. $ok &= Minz_Session::param('currentUser') === Minz_Configuration::defaultUser();
  115. break;
  116. default:
  117. $ok = false;
  118. }
  119. return $ok;
  120. }
  121. /**
  122. * Removes all accesses for the current user.
  123. */
  124. public static function removeAccess() {
  125. Minz_Session::_param('loginOk');
  126. self::$login_ok = false;
  127. Minz_Session::_param('currentUser', Minz_Configuration::defaultUser());
  128. switch (Minz_Configuration::authType()) {
  129. case 'form':
  130. Minz_Session::_param('passwordHash');
  131. FreshRSS_FormAuth::deleteCookie();
  132. break;
  133. case 'persona':
  134. Minz_Session::_param('mail');
  135. break;
  136. case 'http_auth':
  137. case 'none':
  138. // Nothing to do...
  139. break;
  140. default:
  141. // TODO: extensions
  142. }
  143. }
  144. }
  145. class FreshRSS_FormAuth {
  146. public static function checkCredentials($username, $hash, $nonce, $challenge) {
  147. if (!ctype_alnum($username) ||
  148. !ctype_graph($challenge) ||
  149. !ctype_alnum($nonce)) {
  150. Minz_Log::debug('Invalid credential parameters:' .
  151. ' user=' . $username .
  152. ' challenge=' . $challenge .
  153. ' nonce=' . $nonce);
  154. return false;
  155. }
  156. if (!function_exists('password_verify')) {
  157. include_once(LIB_PATH . '/password_compat.php');
  158. }
  159. return password_verify($nonce . $hash, $challenge);
  160. }
  161. public static function getCredentialsFromCookie() {
  162. $token = Minz_Session::getLongTermCookie('FreshRSS_login');
  163. if (!ctype_alnum($token)) {
  164. return array();
  165. }
  166. $token_file = DATA_PATH . '/tokens/' . $token . '.txt';
  167. $mtime = @filemtime($token_file);
  168. if ($mtime + 2629744 < time()) {
  169. // Token has expired (> 1 month) or does not exist.
  170. // TODO: 1 month -> use a configuration instead
  171. @unlink($token_file);
  172. return array();
  173. }
  174. $credentials = @file_get_contents($token_file);
  175. return $credentials === false ? array() : explode("\t", $credentials, 2);
  176. }
  177. public static function makeCookie($username, $password_hash) {
  178. do {
  179. $token = sha1(Minz_Configuration::salt() . $username . uniqid(mt_rand(), true));
  180. $token_file = DATA_PATH . '/tokens/' . $token . '.txt';
  181. } while (file_exists($token_file));
  182. if (@file_put_contents($token_file, $username . "\t" . $password_hash) === false) {
  183. return false;
  184. }
  185. $expire = time() + 2629744; //1 month //TODO: Use a configuration instead
  186. Minz_Session::setLongTermCookie('FreshRSS_login', $token, $expire);
  187. return $token;
  188. }
  189. public static function deleteCookie() {
  190. $token = Minz_Session::getLongTermCookie('FreshRSS_login');
  191. Minz_Session::deleteLongTermCookie('FreshRSS_login');
  192. if (ctype_alnum($token)) {
  193. @unlink(DATA_PATH . '/tokens/' . $token . '.txt');
  194. }
  195. if (rand(0, 10) === 1) {
  196. self::purgeTokens();
  197. }
  198. }
  199. public static function purgeTokens() {
  200. $oldest = time() - 2629744; // 1 month // TODO: Use a configuration instead
  201. foreach (new DirectoryIterator(DATA_PATH . '/tokens/') as $file_info) {
  202. // $extension = $file_info->getExtension(); doesn't work in PHP < 5.3.7
  203. $extension = pathinfo($file_info->getFilename(), PATHINFO_EXTENSION);
  204. if ($extension === 'txt' && $file_info->getMTime() < $oldest) {
  205. @unlink($file_info->getPathname());
  206. }
  207. }
  208. }
  209. }