AuthController.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. <?php
  2. declare(strict_types=1);
  3. namespace App\Controllers;
  4. use App\Models\Config;
  5. use App\Models\InviteCode;
  6. use App\Models\LoginIp;
  7. use App\Models\User;
  8. use App\Services\Auth;
  9. use App\Services\Cache;
  10. use App\Services\Captcha;
  11. use App\Services\Mail;
  12. use App\Services\MFA;
  13. use App\Services\RateLimit;
  14. use App\Services\Reward;
  15. use App\Utils\Cookie;
  16. use App\Utils\Hash;
  17. use App\Utils\ResponseHelper;
  18. use App\Utils\Tools;
  19. use Exception;
  20. use Psr\Http\Client\ClientExceptionInterface;
  21. use Psr\Http\Message\ResponseInterface;
  22. use Ramsey\Uuid\Uuid;
  23. use RedisException;
  24. use Slim\Http\Response;
  25. use Slim\Http\ServerRequest;
  26. use function array_rand;
  27. use function date;
  28. use function explode;
  29. use function strlen;
  30. use function strtolower;
  31. use function time;
  32. use function trim;
  33. final class AuthController extends BaseController
  34. {
  35. /**
  36. * @throws Exception
  37. */
  38. public function login(ServerRequest $request, Response $response, array $args): ResponseInterface
  39. {
  40. $captcha = [];
  41. if (Config::obtain('enable_login_captcha')) {
  42. $captcha = Captcha::generate();
  43. }
  44. return $response->write($this->view()
  45. ->assign('base_url', $_ENV['baseUrl'])
  46. ->assign('captcha', $captcha)
  47. ->fetch('auth/login.tpl'));
  48. }
  49. public function loginHandle(ServerRequest $request, Response $response, array $args): Response|ResponseInterface
  50. {
  51. if (Config::obtain('enable_login_captcha') && ! Captcha::verify($request->getParams())) {
  52. return $response->withJson([
  53. 'ret' => 0,
  54. 'msg' => '系统无法接受你的验证结果,请刷新页面后重试。',
  55. ]);
  56. }
  57. $mfa_code = $this->antiXss->xss_clean($request->getParam('mfa_code'));
  58. $passwd = $request->getParam('passwd');
  59. $rememberMe = $request->getParam('remember_me') === 'true' ? 1 : 0;
  60. $email = strtolower(trim($this->antiXss->xss_clean($request->getParam('email'))));
  61. $redir = $this->antiXss->xss_clean(Cookie::get('redir')) ?? '/user';
  62. $user = (new User())->where('email', $email)->first();
  63. $loginIp = new LoginIp();
  64. if ($user === null) {
  65. $loginIp->collectLoginIP($_SERVER['REMOTE_ADDR'], 1);
  66. return $response->withJson([
  67. 'ret' => 0,
  68. 'msg' => '邮箱或者密码错误',
  69. ]);
  70. }
  71. if (! Hash::checkPassword($user->pass, $passwd)) {
  72. $loginIp->collectLoginIP($_SERVER['REMOTE_ADDR'], 1, $user->id);
  73. return $response->withJson([
  74. 'ret' => 0,
  75. 'msg' => '邮箱或者密码错误',
  76. ]);
  77. }
  78. if ($user->ga_enable && (strlen($mfa_code) !== 6 || ! MFA::verifyGa($user, $mfa_code))) {
  79. $loginIp->collectLoginIP($_SERVER['REMOTE_ADDR'], 1, $user->id);
  80. return $response->withJson([
  81. 'ret' => 0,
  82. 'msg' => '两步验证码错误',
  83. ]);
  84. }
  85. $time = 3600;
  86. if ($rememberMe) {
  87. $time = 86400 * ($_ENV['rememberMeDuration'] ?: 7);
  88. }
  89. Auth::login($user->id, $time);
  90. // 记录登录成功
  91. $loginIp->collectLoginIP($_SERVER['REMOTE_ADDR'], 0, $user->id);
  92. $user->last_login_time = time();
  93. $user->save();
  94. return $response->withHeader('HX-Redirect', $redir)->withJson([
  95. 'ret' => 1,
  96. 'msg' => '登录成功',
  97. ]);
  98. }
  99. /**
  100. * @throws Exception
  101. */
  102. public function register(ServerRequest $request, Response $response, $next): Response|ResponseInterface
  103. {
  104. $captcha = [];
  105. if (Config::obtain('enable_reg_captcha')) {
  106. $captcha = Captcha::generate();
  107. }
  108. $invite_code = $this->antiXss->xss_clean($request->getParam('code'));
  109. return $response->write(
  110. $this->view()
  111. ->assign('invite_code', $invite_code)
  112. ->assign('base_url', $_ENV['baseUrl'])
  113. ->assign('captcha', $captcha)
  114. ->fetch('auth/register.tpl')
  115. );
  116. }
  117. /**
  118. * @throws RedisException
  119. */
  120. public function sendVerify(ServerRequest $request, Response $response, $next): Response|ResponseInterface
  121. {
  122. if (Config::obtain('reg_email_verify')) {
  123. $email = strtolower(trim($this->antiXss->xss_clean($request->getParam('email'))));
  124. if ($email === '') {
  125. return ResponseHelper::error($response, '未填写邮箱');
  126. }
  127. // check email format
  128. $check_res = Tools::isEmailLegal($email);
  129. if ($check_res['ret'] === 0) {
  130. return $response->withJson($check_res);
  131. }
  132. if (! RateLimit::checkEmailIpLimit($request->getServerParam('REMOTE_ADDR')) ||
  133. ! RateLimit::checkEmailAddressLimit($email)
  134. ) {
  135. return ResponseHelper::error($response, '你的请求过于频繁,请稍后再试');
  136. }
  137. $user = (new User())->where('email', $email)->first();
  138. if ($user !== null) {
  139. return ResponseHelper::error($response, '此邮箱已经注册');
  140. }
  141. $email_code = Tools::genRandomChar(6);
  142. $redis = (new Cache())->initRedis();
  143. $redis->setex('email_verify:' . $email_code, Config::obtain('email_verify_code_ttl'), $email);
  144. try {
  145. Mail::send(
  146. $email,
  147. $_ENV['appName'] . '- 验证邮件',
  148. 'verify_code.tpl',
  149. [
  150. 'code' => $email_code,
  151. 'expire' => date('Y-m-d H:i:s', time() + Config::obtain('email_verify_code_ttl')),
  152. ]
  153. );
  154. } catch (Exception|ClientExceptionInterface $e) {
  155. return ResponseHelper::error($response, '邮件发送失败,请联系网站管理员。');
  156. }
  157. return ResponseHelper::success($response, '验证码发送成功,请查收邮件。');
  158. }
  159. return ResponseHelper::error($response, '站点未启用邮件验证');
  160. }
  161. /**
  162. * @throws Exception
  163. */
  164. public function registerHelper(
  165. Response $response,
  166. $name,
  167. $email,
  168. $passwd,
  169. $invite_code,
  170. $imtype,
  171. $imvalue,
  172. $money,
  173. $is_admin_reg
  174. ): ResponseInterface {
  175. $redir = $this->antiXss->xss_clean(Cookie::get('redir')) ?? '/user';
  176. $configs = Config::getClass('reg');
  177. // do reg user
  178. $user = new User();
  179. $user->user_name = $name;
  180. $user->email = $email;
  181. $user->remark = '';
  182. $user->pass = Hash::passwordHash($passwd);
  183. $user->passwd = Tools::genRandomChar(16);
  184. $user->uuid = Uuid::uuid4();
  185. $user->api_token = Uuid::uuid4();
  186. $user->port = Tools::getSsPort();
  187. $user->u = 0;
  188. $user->d = 0;
  189. $user->method = $configs['reg_method'];
  190. $user->forbidden_ip = Config::obtain('reg_forbidden_ip');
  191. $user->forbidden_port = Config::obtain('reg_forbidden_port');
  192. $user->im_type = $imtype;
  193. $user->im_value = $imvalue;
  194. $user->transfer_enable = Tools::toGB($configs['reg_traffic']);
  195. $user->auto_reset_day = Config::obtain('free_user_reset_day');
  196. $user->auto_reset_bandwidth = Config::obtain('free_user_reset_bandwidth');
  197. $user->daily_mail_enable = $configs['reg_daily_report'];
  198. if ($money > 0) {
  199. $user->money = $money;
  200. } else {
  201. $user->money = 0;
  202. }
  203. $user->ref_by = 0;
  204. if ($invite_code !== '') {
  205. $invite = (new InviteCode())->where('code', $invite_code)->first();
  206. if ($invite !== null) {
  207. $user->ref_by = $invite->user_id;
  208. }
  209. }
  210. $user->ga_token = MFA::generateGaToken();
  211. $user->ga_enable = 0;
  212. $user->class = $configs['reg_class'];
  213. $user->class_expire = date('Y-m-d H:i:s', time() + (int) $configs['reg_class_time'] * 86400);
  214. $user->node_iplimit = $configs['reg_ip_limit'];
  215. $user->node_speedlimit = $configs['reg_speed_limit'];
  216. $user->reg_date = date('Y-m-d H:i:s');
  217. $user->reg_ip = $_SERVER['REMOTE_ADDR'];
  218. $user->theme = $_ENV['theme'];
  219. $user->locale = $_ENV['locale'];
  220. $random_group = Config::obtain('random_group');
  221. if ($random_group === '') {
  222. $user->node_group = 0;
  223. } else {
  224. $user->node_group = $random_group[array_rand(explode(',', $random_group))];
  225. }
  226. if ($user->save() && ! $is_admin_reg) {
  227. if ($user->ref_by !== 0) {
  228. Reward::issueRegReward($user->id, $user->ref_by);
  229. }
  230. Auth::login($user->id, 3600);
  231. (new LoginIp())->collectLoginIP($_SERVER['REMOTE_ADDR'], 0, $user->id);
  232. return $response->withHeader('HX-Redirect', $redir)->withJson([
  233. 'ret' => 1,
  234. 'msg' => '注册成功!正在进入登录界面',
  235. ]);
  236. }
  237. return ResponseHelper::error($response, '未知错误');
  238. }
  239. /**
  240. * @throws RedisException
  241. * @throws Exception
  242. */
  243. public function registerHandle(ServerRequest $request, Response $response, array $args): Response|ResponseInterface
  244. {
  245. if (Config::obtain('reg_mode') === 'close') {
  246. return ResponseHelper::error($response, '未开放注册。');
  247. }
  248. if (Config::obtain('enable_reg_captcha') && ! Captcha::verify($request->getParams())) {
  249. return ResponseHelper::error($response, '系统无法接受你的验证结果,请刷新页面后重试。');
  250. }
  251. $tos = $request->getParam('tos') === 'true' ? 1 : 0;
  252. $email = strtolower(trim($this->antiXss->xss_clean($request->getParam('email'))));
  253. $name = $this->antiXss->xss_clean($request->getParam('name'));
  254. $passwd = $request->getParam('passwd');
  255. $repasswd = $request->getParam('repasswd');
  256. $invite_code = $this->antiXss->xss_clean(trim($request->getParam('invite_code')));
  257. // Check TOS agreement
  258. if (! $tos) {
  259. return ResponseHelper::error($response, '请同意服务条款');
  260. }
  261. // Check Invite Code
  262. if ($invite_code === '' && Config::obtain('reg_mode') === 'invite') {
  263. return ResponseHelper::error($response, '邀请码不能为空');
  264. }
  265. if ($invite_code !== '') {
  266. $invite = (new InviteCode())->where('code', $invite_code)->first();
  267. if ($invite === null) {
  268. return ResponseHelper::error($response, '邀请码无效');
  269. }
  270. $ref_user = (new User())->where('id', $invite->user_id)->first();
  271. if ($ref_user === null) {
  272. return ResponseHelper::error($response, '邀请码无效');
  273. }
  274. }
  275. $imtype = 0;
  276. $imvalue = '';
  277. // check email format
  278. $check_res = Tools::isEmailLegal($email);
  279. if ($check_res['ret'] === 0) {
  280. return $response->withJson($check_res);
  281. }
  282. // check email
  283. $user = (new User())->where('email', $email)->first();
  284. if ($user !== null) {
  285. return ResponseHelper::error($response, '邮箱已经被注册了');
  286. }
  287. // check pwd length
  288. if (strlen($passwd) < 8) {
  289. return ResponseHelper::error($response, '密码请大于8位');
  290. }
  291. // check pwd re
  292. if ($passwd !== $repasswd) {
  293. return ResponseHelper::error($response, '两次密码输入不符');
  294. }
  295. if (Config::obtain('reg_email_verify')) {
  296. $redis = (new Cache())->initRedis();
  297. $email_verify_code = trim($this->antiXss->xss_clean($request->getParam('emailcode')));
  298. $email_verify = $redis->get('email_verify:' . $email_verify_code);
  299. if (! $email_verify) {
  300. return ResponseHelper::error($response, '你的邮箱验证码不正确');
  301. }
  302. $redis->del('email_verify:' . $email_verify_code);
  303. }
  304. return $this->registerHelper($response, $name, $email, $passwd, $invite_code, $imtype, $imvalue, 0, 0);
  305. }
  306. public function logout(ServerRequest $request, Response $response, $next): Response
  307. {
  308. Auth::logout();
  309. return $response->withStatus(302)
  310. ->withHeader('Location', '/auth/login');
  311. }
  312. }