AuthController.php 14 KB

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