AuthController.php 13 KB

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