auth.php 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. <?php
  2. const USERNAME_REGEX = '^.{1,1024}$';
  3. const PASSWORD_REGEX = '^(?=.*[\p{Ll}])(?=.*[\p{Lu}])(?=.*[\p{N}]).{8,1024}|.{10,1024}$';
  4. const PLACEHOLDER_USERNAME = 'lain';
  5. const PLACEHOLDER_PASSWORD = '••••••••••••••••••••••••';
  6. // Password storage security
  7. const ALGO_PASSWORD = PASSWORD_ARGON2ID;
  8. const OPTIONS_PASSWORD = [
  9. 'memory_cost' => 65536,
  10. 'time_cost' => 4,
  11. 'threads' => 64,
  12. ];
  13. function checkUsernameFormat($username) {
  14. if (preg_match('/' . USERNAME_REGEX . '/Du', $username) !== 1)
  15. output(403, 'Username malformed.');
  16. }
  17. function checkPasswordFormat($password) {
  18. if (preg_match('/' . PASSWORD_REGEX . '/Du', $password) !== 1)
  19. output(403, 'Password malformed.');
  20. }
  21. function hashUsername($username) {
  22. return base64_encode(sodium_crypto_pwhash(32, $username, hex2bin(query('select', 'params', ['name' => 'username_salt'], 'value')[0]), 2**10, 2**14, SODIUM_CRYPTO_PWHASH_ALG_ARGON2ID13));
  23. }
  24. function hashPassword($password) {
  25. return password_hash($password, ALGO_PASSWORD, OPTIONS_PASSWORD);
  26. }
  27. function usernameExists($username) {
  28. return isset(query('select', 'users', ['username' => $username], 'id')[0]);
  29. }
  30. function checkPassword($id, $password) {
  31. return password_verify($password, query('select', 'users', ['id' => $id], 'password')[0]);
  32. }
  33. function outdatedPasswordHash($id) {
  34. return password_needs_rehash(query('select', 'users', ['id' => $id], 'password')[0], ALGO_PASSWORD, OPTIONS_PASSWORD);
  35. }
  36. function changePassword($id, $password) {
  37. $db = new PDO('sqlite:' . DB_PATH);
  38. $stmt = $db->prepare('UPDATE users SET password = :password WHERE id = :id');
  39. $stmt->bindValue(':id', $id);
  40. $stmt->bindValue(':password', hashPassword($password));
  41. $stmt->execute();
  42. }
  43. function rateLimit() {
  44. if (PAGE_METADATA['tokens_account_cost'] ?? 0 > 0)
  45. rateLimitAccount(PAGE_METADATA['tokens_account_cost']);
  46. if (PAGE_METADATA['tokens_instance_cost'] ?? 0 > 0)
  47. rateLimitInstance(PAGE_METADATA['tokens_instance_cost']);
  48. }
  49. function rateLimitAccount($requestedTokens) {
  50. // Get
  51. $userData = query('select', 'users', ['id' => $_SESSION['id']]);
  52. $tokens = $userData[0]['bucket_tokens'];
  53. $bucketLastUpdate = $userData[0]['bucket_last_update'];
  54. // Compute
  55. $tokens = min(86400, $tokens + (time() - $bucketLastUpdate));
  56. if ($requestedTokens > $tokens)
  57. output(453, 'Limite d\'actions par compte atteinte. Réessayez plus tard.');
  58. $tokens -= $requestedTokens;
  59. // Update
  60. $db = new PDO('sqlite:' . DB_PATH);
  61. $stmt = $db->prepare('UPDATE users SET bucket_tokens = :bucket_tokens, bucket_last_update = :bucket_last_update WHERE id = :id');
  62. $stmt->bindValue(':id', $_SESSION['id']);
  63. $stmt->bindValue(':bucket_tokens', $tokens);
  64. $stmt->bindValue(':bucket_last_update', time());
  65. $stmt->execute();
  66. }
  67. function rateLimitInstance($requestedTokens) {
  68. // Get
  69. $tokens = query('select', 'params', ['name' => 'instance_bucket_tokens'], 'value')[0];
  70. $bucketLastUpdate = query('select', 'params', ['name' => 'instance_bucket_last_update'], 'value')[0];
  71. // Compute
  72. $tokens = min(86400, $tokens + (time() - $bucketLastUpdate));
  73. if ($requestedTokens > $tokens)
  74. output(453, 'Limite d\'actions globale atteinte. Réessayez plus tard.');
  75. $tokens -= $requestedTokens;
  76. // Update
  77. $db = new PDO('sqlite:' . DB_PATH);
  78. $stmt = $db->prepare("UPDATE params SET value = :bucket_tokens WHERE name = 'instance_bucket_tokens';");
  79. $stmt->bindValue(':bucket_tokens', $tokens);
  80. $stmt->execute();
  81. $stmt = $db->prepare("UPDATE params SET value = :bucket_last_update WHERE name = 'instance_bucket_last_update';");
  82. $stmt->bindValue(':bucket_last_update', time());
  83. $stmt->execute();
  84. }