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