blessing-skin-server/app/Http/Controllers/AuthController.php

340 lines
11 KiB
PHP

<?php
namespace App\Http\Controllers;
use Log;
use Mail;
use View;
use Cache;
use Cookie;
use Option;
use Session;
use App\Events;
use App\Models\User;
use App\Models\Player;
use Illuminate\Http\Request;
use App\Exceptions\PrettyPageException;
use App\Services\Repositories\UserRepository;
class AuthController extends Controller
{
public function login()
{
return view('auth.login');
}
public function handleLogin(Request $request, UserRepository $users)
{
$this->validate($request, [
'identification' => 'required',
'password' => 'required|min:6|max:32'
]);
$identification = $request->input('identification');
// Guess type of identification
$authType = (validate($identification, 'email')) ? "email" : "username";
event(new Events\UserTryToLogin($identification, $authType));
// Get user instance from repository.
// If the given identification is not registered yet,
// it will return a null value.
$user = $users->get($identification, $authType);
// Require CAPTCHA if user fails to login more than 3 times
$loginFailsCacheKey = sha1('login_fails_'.get_client_ip());
$loginFails = (int) Cache::get($loginFailsCacheKey, 0);
if ($loginFails > 3) {
if (strtolower($request->input('captcha')) != strtolower(session('phrase')))
return json(trans('auth.validation.captcha'), 1);
}
if (! $user) {
return json(trans('auth.validation.user'), 2);
} else {
if ($user->verifyPassword($request->input('password'))) {
Cache::forget($loginFailsCacheKey);
Session::put('uid' , $user->uid);
Session::put('token', $user->getToken());
// Time in minutes
$time = $request->input('keep') == true ? 10080 : 60;
event(new Events\UserLoggedIn($user));
session()->forget('last_requested_path');
return json(trans('auth.login.success'), 0, [
'token' => $user->getToken()
]) // Set cookies
->withCookie('uid', $user->uid, $time)
->withCookie('token', $user->getToken(), $time);
} else {
// Increase the counter
Cache::put($loginFailsCacheKey, ++$loginFails, 60);
return json(trans('auth.validation.password'), 1, [
'login_fails' => $loginFails
]);
}
}
}
public function logout(Request $request)
{
if (Session::has('uid') && Session::has('token')) {
// Flush sessions
Session::flush();
// Delete cookies
return json(trans('auth.logout.success'), 0)
->withCookie(Cookie::forget('uid'))
->withCookie(Cookie::forget('token'));
} else {
return json(trans('auth.logout.fail'), 1);
}
}
public function register()
{
if (option('user_can_register')) {
return view('auth.register');
} else {
throw new PrettyPageException(trans('auth.register.close'), 7);
}
}
public function handleRegister(Request $request, UserRepository $users)
{
if (! $this->checkCaptcha($request))
return json(trans('auth.validation.captcha'), 1);
if (! option('user_can_register'))
return json(trans('auth.register.close'), 7);
// Validate nickname or player name
$rule = option('register_with_player_name') ?
['player_name' => 'required|player_name|min:'.option('player_name_length_min').'|max:'.option('player_name_length_max')] :
['nickname' => 'required|no_special_chars|max:255'];
$this->validate($request, array_merge([
'email' => 'required|email',
'password' => 'required|min:8|max:32'
], $rule));
if (option('register_with_player_name')) {
event(new Events\CheckPlayerExists($request->get('player_name')));
if (Player::where('player_name', $request->get('player_name'))->first()) {
return json(trans('user.player.add.repeated'), 2);
}
}
// If amount of registered accounts of IP is more than allowed amounts,
// then reject the register.
if (User::where('ip', get_client_ip())->count() >= option('regs_per_ip')) {
return json(trans('auth.register.max', ['regs' => option('regs_per_ip')]), 7);
}
// Register a new user.
// If the email is already registered,
// it will return a false value.
$user = User::register(
$request->get('email'),
$request->get('password'), function($user) use ($request)
{
$user->ip = get_client_ip();
$user->score = option('user_initial_score');
$user->register_at = get_datetime_string();
$user->last_sign_at = get_datetime_string(time() - 86400);
$user->permission = User::NORMAL;
$user->nickname = $request->get(
option('register_with_player_name') ? 'player_name' : 'nickname'
);
});
if (! $user) {
return json(trans('auth.register.registered'), 5);
}
event(new Events\UserRegistered($user));
// Add player with chosen name
if (option('register_with_player_name')) {
$player = new Player;
$player->uid = $user->uid;
$player->player_name = $request->get('player_name');
$player->preference = 'default';
$player->last_modified = get_datetime_string();
$player->save();
event(new Events\PlayerWasAdded($player));
}
return json([
'errno' => 0,
'msg' => trans('auth.register.success'),
'token' => $user->getToken(),
]) // Set cookies
->withCookie('uid', $user->uid, 60)
->withCookie('token', $user->getToken(), 60);
}
public function forgot()
{
if (config('mail.driver') != "") {
return view('auth.forgot');
} else {
throw new PrettyPageException(trans('auth.forgot.disabled'), 8);
}
}
public function handleForgot(Request $request, UserRepository $users)
{
if (! $this->checkCaptcha($request))
return json(trans('auth.validation.captcha'), 1);
if (! config('mail.driver')) {
return json(trans('auth.forgot.disabled'), 1);
}
$rateLimit = 180;
$lastMailCacheKey = sha1('last_mail_'.get_client_ip());
$remain = $rateLimit + Cache::get($lastMailCacheKey, 0) - time();
// Rate limit
if ($remain > 0) {
return json([
'errno' => 2,
'msg' => trans('auth.forgot.frequent-mail'),
'remain' => $remain
]);
}
// Get user instance
$user = $users->get($request->input('email'), 'email');
if (! $user)
return json(trans('auth.forgot.unregistered'), 1);
$uid = $user->uid;
// Generate token for password resetting
$token = generate_random_token();
$url = Option::get('site_url')."/auth/reset?uid=$uid&token=$token";
try {
Mail::send('mails.password-reset', compact('url'), function ($m) use ($request) {
$site_name = option_localized('site_name');
$m->from(config('mail.username'), $site_name);
$m->to($request->input('email'))->subject(trans('auth.forgot.mail.title', ['sitename' => $site_name]));
});
} catch (\Exception $e) {
// Write the exception to log
report($e);
return json(trans('auth.forgot.failed', ['msg' => $e->getMessage()]), 2);
}
Cache::put("pwd_reset_token_$uid", $token, 60);
Cache::put($lastMailCacheKey, time(), 60);
return json(trans('auth.forgot.success'), 0);
}
public function reset(UserRepository $users, Request $request)
{
// Retrieve token from cache
$uid = $request->get('uid');
$token = Cache::get("pwd_reset_token_$uid");
// Get user instance from repository
$user = $users->get($uid);
if (! $user) {
return redirect('auth/forgot')->with('msg', trans('auth.reset.invalid'));
}
// No token exist or token mismatch (maybe expired)
if (is_null($token) || $token != $request->get('token')) {
return redirect('auth/forgot')->with('msg', trans('auth.reset.expired'));
}
return view('auth.reset')->with('user', $user);
}
public function handleReset(Request $request, UserRepository $users)
{
$this->validate($request, [
'uid' => 'required|integer',
'password' => 'required|min:8|max:32',
'token' => 'required',
]);
// Retrieve token from cache
$uid = $request->get('uid');
$token = Cache::get("pwd_reset_token_$uid");
// Get user instance from repository
$user = $users->get($uid);
if (! $user) {
return json(trans('auth.reset.invalid'), 1);
}
// No token exist or token mismatch (maybe expired)
if (is_null($token) || $token != $request->get('token')) {
return json(trans('auth.reset.expired'), 1);
}
$user->changePasswd($request->get('password'));
return json(trans('auth.reset.success'), 0);
}
public function verify(Request $request, UserRepository $users)
{
if (! option('require_verification')) {
throw new PrettyPageException(trans('user.verification.disabled'), 1);
}
// Get user instance from repository
$user = $users->get($request->get('uid'));
if (! $user || $user->verified) {
throw new PrettyPageException(trans('auth.verify.invalid'), 1);
}
if ($user->verification_token != $request->get('token')) {
throw new PrettyPageException(trans('auth.verify.expired'), 1);
}
$user->verified = true;
$user->save();
return view('auth.verify');
}
public function captcha()
{
$builder = new \Gregwar\Captcha\CaptchaBuilder;
$builder->build($width = 100, $height = 34);
Session::put('phrase', $builder->getPhrase());
ob_start();
$builder->output();
$captcha = ob_get_contents();
ob_end_clean();
return \Response::png($captcha);
}
protected function checkCaptcha($request)
{
return (strtolower($request->input('captcha')) == strtolower(session('phrase')));
}
}