middleware(function ($request, $next) { if (! Auth::user()->verified) { $this->sendVerificationEmail(); } return $next($request); })->only(['index', 'profile']); } public function user() { return json('', 0, auth()->user()->makeHidden(['password', 'ip', 'remember_token'])->toArray()); } public function index() { $user = Auth::user(); return view('user.index')->with([ 'statistics' => [ 'players' => $this->calculatePercentageUsed($user->players->count(), option('score_per_player')), 'storage' => $this->calculatePercentageUsed($this->getStorageUsed($user), option('score_per_storage')), ], 'announcement' => app('parsedown')->text(option_localized('announcement')), 'extra' => ['unverified' => option('require_verification') && ! $user->verified], ]); } public function scoreInfo() { $user = Auth::user(); return json('', 0, [ 'user' => [ 'score' => $user->score, 'lastSignAt' => $user->last_sign_at, ], 'stats' => [ 'players' => $this->calculatePercentageUsed($user->players->count(), option('score_per_player')), 'storage' => $this->calculatePercentageUsed($this->getStorageUsed($user), option('score_per_storage')), ], 'signAfterZero' => option('sign_after_zero'), 'signGapTime' => option('sign_gap_time'), ]); } /** * Calculate percentage of resources used by user. * * @param int $used * @param int $rate * @return array */ protected function calculatePercentageUsed($used, $rate) { $user = Auth::user(); // Initialize default value to avoid division by zero. $result['used'] = $used; $result['total'] = 'UNLIMITED'; $result['percentage'] = 0; if ($rate != 0) { $result['total'] = $used + floor($user->score / $rate); $result['percentage'] = $result['total'] ? $used / $result['total'] * 100 : 100; } return $result; } protected function getStorageUsed(User $user) { return Texture::where('uploader', $user->uid)->select('size')->sum('size'); } /** * Handle user signing. * * @return \Illuminate\Http\JsonResponse */ public function sign() { $user = Auth::user(); if ($this->getSignRemainingTime($user) <= 0) { $scoreLimits = explode(',', option('sign_score')); $acquiredScore = rand($scoreLimits[0], $scoreLimits[1]); $user->score += $acquiredScore; $user->last_sign_at = Carbon::now()->toDateTimeString(); $user->save(); $gap = option('sign_gap_time'); return json(trans('user.sign-success', ['score' => $acquiredScore]), 0, [ 'score' => $user->score, 'storage' => $this->calculatePercentageUsed($this->getStorageUsed($user), option('score_per_storage')), 'remaining_time' => $gap > 1 ? round($gap) : $gap, ]); } else { $remaining_time = $this->getUserSignRemainingTimeWithPrecision($user); return json(trans('user.cant-sign-until', [ 'time' => $remaining_time >= 1 ? $remaining_time : round($remaining_time * 60), 'unit' => $remaining_time >= 1 ? trans('user.time-unit-hour') : trans('user.time-unit-min'), ]), 1); } } protected function getUserSignRemainingTimeWithPrecision(User $user) { $hours = $this->getSignRemainingTime($user) / 3600; return $hours > 1 ? round($hours) : $hours; } protected function getSignRemainingTime(User $user) { $lastSignTime = Carbon::parse($user->last_sign_at); if (option('sign_after_zero')) { return Carbon::now()->diffInSeconds( $lastSignTime <= Carbon::today() ? $lastSignTime : Carbon::tomorrow(), false ); } return Carbon::now()->diffInSeconds($lastSignTime->addHours(option('sign_gap_time')), false); } public function sendVerificationEmail() { if (! option('require_verification')) { return json(trans('user.verification.disabled'), 1); } // Rate limit of 60s $remain = 60 + session('last_mail_time', 0) - time(); if ($remain > 0) { return json(trans('user.verification.frequent-mail'), 1); } $user = Auth::user(); if ($user->verified) { return json(trans('user.verification.verified'), 1); } $url = URL::signedRoute('auth.verify', ['uid' => $user->uid]); try { Mail::to($user->email)->send(new EmailVerification($url)); } catch (\Exception $e) { // Write the exception to log report($e); return json(trans('user.verification.failed', ['msg' => $e->getMessage()]), 2); } Session::put('last_mail_time', time()); return json(trans('user.verification.success'), 0); } public function profile() { $user = Auth::user(); return view('user.profile') ->with('extra', [ 'unverified' => option('require_verification') && ! $user->verified, 'admin' => $user->isAdmin(), ]); } public function handleProfile(Request $request, User $users) { $action = $request->input('action', ''); $user = Auth::user(); switch ($action) { case 'nickname': if (option('single_player', false)) { return json(trans('user.profile.nickname.single'), 1); } $this->validate($request, [ 'new_nickname' => 'required|no_special_chars|max:255', ]); $nickname = $request->input('new_nickname'); $user->nickname = $nickname; $user->save(); event(new UserProfileUpdated($action, $user)); return json(trans('user.profile.nickname.success', ['nickname' => $nickname]), 0); case 'password': $this->validate($request, [ 'current_password' => 'required|min:6|max:32', 'new_password' => 'required|min:8|max:32', ]); if (! $user->verifyPassword($request->input('current_password'))) { return json(trans('user.profile.password.wrong-password'), 1); } if ($user->changePassword($request->input('new_password'))) { event(new UserProfileUpdated($action, $user)); Auth::logout(); return json(trans('user.profile.password.success'), 0); } break; // @codeCoverageIgnore case 'email': $this->validate($request, [ 'new_email' => 'required|email', 'password' => 'required|min:6|max:32', ]); if ($users->where('email', $request->new_email)->count() > 0) { return json(trans('user.profile.email.existed'), 1); } if (! $user->verifyPassword($request->input('password'))) { return json(trans('user.profile.email.wrong-password'), 1); } $user->email = $request->input('new_email'); $user->verified = false; $user->save(); event(new UserProfileUpdated($action, $user)); Auth::logout(); return json(trans('user.profile.email.success'), 0); case 'delete': $this->validate($request, [ 'password' => 'required|min:6|max:32', ]); if ($user->isAdmin()) { return json(trans('user.profile.delete.admin'), 1); } if (! $user->verifyPassword($request->input('password'))) { return json(trans('user.profile.delete.wrong-password'), 1); } Auth::logout(); if ($user->delete()) { session()->flush(); return json(trans('user.profile.delete.success'), 0); } break; // @codeCoverageIgnore default: return json(trans('general.illegal-parameters'), 1); break; } } // @codeCoverageIgnore /** * Set user avatar. * * @param Request $request */ public function setAvatar(Request $request) { $this->validate($request, [ 'tid' => 'required|integer', ]); $tid = $request->input('tid'); $user = auth()->user(); if ($tid == 0) { $user->avatar = 0; $user->save(); return json(trans('user.profile.avatar.success'), 0); } $result = Texture::find($tid); if ($result) { if ($result->type == 'cape') { return json(trans('user.profile.avatar.wrong-type'), 1); } $user->avatar = $tid; $user->save(); return json(trans('user.profile.avatar.success'), 0); } else { return json(trans('skinlib.non-existent'), 1); } } public function readNotification($id) { $notification = auth() ->user() ->unreadNotifications ->first(function ($notification) use ($id) { return $notification->id === $id; }); $notification->markAsRead(); return [ 'title' => $notification->data['title'], 'content' => app('parsedown')->text($notification->data['content']), 'time' => $notification->created_at->toDateTimeString(), ]; } }