auth.utils.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340
  1. <?php
  2. define("CONFIG", 1);
  3. define("SQL", 1);
  4. require_once($_SERVER['DOCUMENT_ROOT'] . "/cfg/config.php");
  5. require_once($_SERVER['DOCUMENT_ROOT'] . "/inc/sql.php");
  6. require_once($_SERVER['DOCUMENT_ROOT'] . "/inc/common.php");
  7. //ini_set('session.use_trans_sid', true);
  8. //ini_set('session.use_only_cookies', false);
  9. define('SESSION_TABLE', 'sessions');
  10. define('USER_SESSIONS_TABLE', 'user_sessions');
  11. //set default const values
  12. if (!defined('SESSION_LIFETIME') || SESSION_LIFETIME < 60) { define('SESSION_LIFETIME', 86400); }
  13. if (!defined("HTML_LANG")) { define("HTML_LANG","english"); }
  14. if (!defined("HTML_STYLE")) { define("HTML_STYLE","white"); }
  15. if (!defined("IPCAM_GROUP_ID")) { define("IPCAM_GROUP_ID","5"); }
  16. if (!defined("SNMP_timeout")) { define("SNMP_timeout","500000"); }
  17. if (!defined("SNMP_retry")) { define("SNMP_retry","1"); }
  18. // Инициализация сессий в БД
  19. function init_db_sessions($db) {
  20. // Настройка обработчиков сессий
  21. session_set_save_handler(
  22. 'sess_open',
  23. 'sess_close',
  24. 'sess_read',
  25. 'sess_write',
  26. 'sess_destroy',
  27. 'sess_gc'
  28. );
  29. register_shutdown_function('session_write_close');
  30. }
  31. // Обработчики сессий
  32. function sess_open($savePath, $sessionName) { return true; }
  33. function sess_close() { return true; }
  34. function sess_read($sessionId) {
  35. global $db_link;
  36. $sessionId = mysqli_real_escape_string($db_link, $sessionId);
  37. $result = mysqli_query($db_link, "SELECT data FROM ".SESSION_TABLE." WHERE id = '$sessionId'");
  38. if (!$result) {
  39. error_log("Session read failed: " . mysqli_error($db_link));
  40. return '';
  41. }
  42. return mysqli_num_rows($result) ? mysqli_fetch_assoc($result)['data'] : '';
  43. }
  44. function sess_write($sessionId, $data) {
  45. global $db_link;
  46. $sessionId = mysqli_real_escape_string($db_link, $sessionId);
  47. $data = mysqli_real_escape_string($db_link, $data);
  48. $time = time();
  49. $query = "INSERT INTO ".SESSION_TABLE." (id, data, last_accessed)
  50. VALUES ('$sessionId', '$data', $time)
  51. ON DUPLICATE KEY UPDATE data = '$data', last_accessed = $time";
  52. if (!mysqli_query($db_link, $query)) {
  53. error_log("Session write failed: " . mysqli_error($db_link));
  54. return false;
  55. }
  56. return true;
  57. }
  58. function sess_destroy($sessionId) {
  59. global $db_link;
  60. $sessionId = mysqli_real_escape_string($db_link, $sessionId);
  61. if (!mysqli_query($db_link, "DELETE FROM ".SESSION_TABLE." WHERE id = '$sessionId'")) {
  62. error_log("Session destroy failed: " . mysqli_error($db_link));
  63. return false;
  64. }
  65. return true;
  66. }
  67. function sess_gc($maxLifetime) {
  68. global $db_link;
  69. $old = time() - $maxLifetime;
  70. if (!mysqli_query($db_link, "DELETE FROM ".SESSION_TABLE." WHERE last_accessed < $old")) {
  71. error_log("Session GC failed: " . mysqli_error($db_link));
  72. return false;
  73. }
  74. return true;
  75. }
  76. function login($db) {
  77. $redirect_url = getSafeRedirectUrl(DEFAULT_PAGE);
  78. if ($redirect_url == DEFAULT_PAGE) {
  79. // 1. Сначала получаем путь из оригинального URL
  80. $current_path = parse_url($_SERVER['REQUEST_URI'], PHP_URL_PATH);
  81. $current_path = $current_path ? rtrim($current_path, '/') : '/';
  82. // 2. Подготавливаем пути для сравнения
  83. $login_path = rtrim(LOGIN_PAGE, '/');
  84. $logout_path = rtrim(LOGOUT_PAGE, '/');
  85. // 3. Сравниваем пути
  86. if ($current_path !== $login_path && $current_path !== $logout_path) {
  87. // 4. Кодируем только если нужно сохранить полный URL с параметрами
  88. $redirect_url = urlencode($_SERVER['REQUEST_URI']);
  89. }
  90. }
  91. // 1. Проверка активной сессии
  92. if (!empty($_SESSION['user_id']) && validate_session($db)) {
  93. // Дополнительная валидация сессии
  94. if ($_SESSION['ip'] === get_user_ip() &&
  95. $_SESSION['user_agent'] === ($_SERVER['HTTP_USER_AGENT'] ?? '')) {
  96. return true;
  97. }
  98. // Несоответствие параметров - разрушаем сессию
  99. logout($db,FALSE,$redirect_url);
  100. }
  101. // 2. Проверка API-авторизации (для API-запросов)
  102. if (strpos($_SERVER['REQUEST_URI'], '/api.php') === 0) {
  103. return IsSilentAuthenticated($db);
  104. }
  105. // 4. Проверка логина/пароля из POST-данных (обычная форма входа)
  106. if (!empty($_POST['login']) && !empty($_POST['password'])) {
  107. if (authenticate_by_credentials($db, $_POST['login'], $_POST['password'])) {
  108. LOG_INFO($db, "Logged in customer id: ".$_SESSION['user_id']." name: ".$_SESSION['login']." from ".$_SESSION['ip']." with acl: ".$_SESSION['acl']." url: ".$redirect_url);
  109. return true;
  110. }
  111. // Неудачная попытка входа
  112. sleep(1); // Защита от брутфорса
  113. }
  114. // 5. Если ни один метод не сработал - требовать авторизацию
  115. logout($db,FALSE,$redirect_url);
  116. exit;
  117. }
  118. function authenticate_by_credentials($db,$login,$password) {
  119. $login = mysqli_real_escape_string($db, trim($login));
  120. $query = "SELECT * FROM `Customers` WHERE Login='{$login}'";
  121. $user = get_record_sql($db, $query);
  122. if (empty($user)) {
  123. sleep(1);
  124. return false;
  125. }
  126. if (!password_verify($password, $user['password'])) {
  127. sleep(1);
  128. return false;
  129. }
  130. // Создание сессии
  131. session_regenerate_id(true);
  132. $_SESSION = [
  133. 'user_id' => $user['id'],
  134. 'login' => $user['Login'],
  135. 'acl' => $user['rights'],
  136. 'ip' => get_client_ip(),
  137. 'user_agent' => $_SERVER['HTTP_USER_AGENT'] ?? '',
  138. 'created' => time()
  139. ];
  140. // Запись сессии в БД
  141. $sessionId = mysqli_real_escape_string($db, session_id());
  142. $ip = mysqli_real_escape_string($db, $_SESSION['ip']);
  143. $userAgent = mysqli_real_escape_string($db, $_SESSION['user_agent']);
  144. $time = time();
  145. // Запись в БД
  146. $sessionId = mysqli_real_escape_string($db, session_id());
  147. $query = "INSERT INTO ".USER_SESSIONS_TABLE."
  148. (session_id, user_id, ip_address, user_agent, created_at, last_activity)
  149. VALUES (
  150. '$sessionId',
  151. {$user['id']},
  152. '$ip',
  153. '$userAgent',
  154. $time,
  155. $time
  156. )";
  157. if (!mysqli_query($db, $query)) {
  158. error_log("Session DB error: ".mysqli_error($db));
  159. return false;
  160. }
  161. return true;
  162. }
  163. function validate_session($db) {
  164. // Проверка IP и User-Agent
  165. if ($_SESSION['ip'] !== get_client_ip() ||
  166. $_SESSION['user_agent'] !== ($_SERVER['HTTP_USER_AGENT'] ?? '')) {
  167. logout($db);
  168. return false;
  169. }
  170. // Проверка активности сессии в БД
  171. $sessionId = mysqli_real_escape_string($db, session_id());
  172. $result = mysqli_query($db,
  173. "SELECT 1
  174. FROM ".USER_SESSIONS_TABLE."
  175. WHERE
  176. session_id = '$sessionId' AND
  177. user_id = {$_SESSION['user_id']} AND
  178. is_active = 1
  179. LIMIT 1");
  180. if (!$result || mysqli_num_rows($result) === 0) {
  181. logout($db);
  182. return false;
  183. }
  184. // Обновление времени активности
  185. mysqli_query($db,
  186. "UPDATE ".USER_SESSIONS_TABLE."
  187. SET last_activity = ".time()."
  188. WHERE session_id = '$sessionId'");
  189. return true;
  190. }
  191. function get_client_ip() {
  192. foreach (['HTTP_CLIENT_IP', 'HTTP_X_FORWARDED_FOR', 'REMOTE_ADDR'] as $key) {
  193. if (!empty($_SERVER[$key])) {
  194. $ip = trim(current(explode(',', $_SERVER[$key])));
  195. if (filter_var($ip, FILTER_VALIDATE_IP)) {
  196. return $ip;
  197. }
  198. }
  199. }
  200. return '127.0.0.1';
  201. }
  202. // Авторизация по API-ключу (без пароля)
  203. function IsSilentAuthenticated($db) {
  204. if (!empty($_SESSION['user_id'])) {
  205. return true;
  206. }
  207. $auth_ip = get_user_ip();
  208. $api_key = '';
  209. // Получаем ключ из GET или POST
  210. if (!empty($_GET['api_key'])) {
  211. $api_key = trim($_GET['api_key']);
  212. } elseif (!empty($_POST['api_key'])) {
  213. $api_key = trim($_POST['api_key']);
  214. }
  215. if (!empty($_GET['login'])) {
  216. $login = trim($_GET['login']);
  217. } elseif (!empty($_POST['login'])) {
  218. $login = trim($_POST['login']);
  219. }
  220. if (empty($login) || empty($api_key) || strlen($api_key) < 20) {
  221. return false;
  222. }
  223. // Экранирование и подготовка
  224. $login = mysqli_real_escape_string($db, $login);
  225. $api_key = mysqli_real_escape_string($db, $api_key);
  226. // Ищем пользователя с таким логином и API-ключом
  227. $query = "SELECT id, rights FROM Customers
  228. WHERE Login = '$login' AND api_key = '$api_key'
  229. LIMIT 1";
  230. $result = mysqli_query($db, $query);
  231. if (!$result || mysqli_num_rows($result) === 0) {
  232. error_log("API auth failed for: $login");
  233. return false;
  234. }
  235. $user = mysqli_fetch_assoc($result);
  236. // Создаем сессию
  237. $_SESSION = [
  238. 'user_id' => $user['id'],
  239. 'login' => $login,
  240. 'acl' => $user['rights'],
  241. 'ip' => $auth_ip,
  242. 'user_agent' => $_SERVER['HTTP_USER_AGENT'] ?? '',
  243. 'api_auth' => true // Метка API-аутентификации
  244. ];
  245. LOG_INFO($db, "Logged in to api customer id: ".$_SESSION['user_id']." name: ".$_SESSION['login']." from ".$_SESSION['ip']." with acl: ".$_SESSION['acl']);
  246. return true;
  247. }
  248. // Выход из системы (полная версия)
  249. function logout($db, $silent = FALSE, $redirect_url = DEFAULT_PAGE) {
  250. if (session_status() === PHP_SESSION_ACTIVE) {
  251. LOG_INFO($db, "Logout customer id: ".$_SESSION['user_id']." name: ".$_SESSION['login']." from ".$_SESSION['ip']." with acl: ".$_SESSION['acl']);
  252. // Деактивация сессии в БД
  253. $sessionId = mysqli_real_escape_string($db, session_id());
  254. mysqli_query($db,
  255. "UPDATE ".USER_SESSIONS_TABLE."
  256. SET is_active = 0
  257. WHERE session_id = '$sessionId'");
  258. // Очистка данных
  259. $_SESSION = [];
  260. session_destroy();
  261. if (!headers_sent()) {
  262. setcookie(session_name(), '', time() - SESSION_LIFETIME, '/');
  263. // Удаление авторизационной куки (если есть)
  264. if (isset($_COOKIE['Auth'])) {
  265. setcookie('Auth', '', time() - SESSION_LIFETIME, '/');
  266. }
  267. }
  268. }
  269. if (!$silent and !headers_sent()) {
  270. if ($redirect_url == DEFAULT_PAGE or empty($redirect_url) or $redirect_url=='/') {
  271. header('Location: '.LOGIN_PAGE);
  272. } else {
  273. header('Location: '.LOGIN_PAGE.'?redirect_url='.$redirect_url);
  274. }
  275. }
  276. }
  277. // Инициализация системы сессий
  278. init_db_sessions($db_link);
  279. // Инициализация сессии
  280. if (session_status() !== PHP_SESSION_ACTIVE) {
  281. // Старт сессии с безопасными настройками
  282. session_start([
  283. 'cookie_lifetime' => SESSION_LIFETIME,
  284. 'cookie_path' => '/', // ВАЖНО: кука должна быть доступна для всего сайта
  285. 'cookie_domain' => $_SERVER['HTTP_HOST'], // Текущий домен
  286. 'cookie_secure' => isset($_SERVER['HTTPS']), // true если HTTPS
  287. // 'cookie_httponly' => true,
  288. 'cookie_samesite' => 'Lax', // Или 'Strict', но 'Lax' лучше для переходов по ссылкам
  289. 'gc_maxlifetime' => SESSION_LIFETIME,
  290. ]);
  291. }