staffmanager/app/Http/Controllers/UserController.php

338 lines
11 KiB
PHP
Raw Normal View History

<?php
2020-10-10 16:30:26 +00:00
/*
* Copyright © 2020 Miguel Nogueira
*
* This file is part of Raspberry Staff Manager.
*
* Raspberry Staff Manager is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Raspberry Staff Manager is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Raspberry Staff Manager. If not, see <https://www.gnu.org/licenses/>.
*/
namespace App\Http\Controllers;
2020-10-10 16:30:26 +00:00
use App\Ban;
use App\Http\Requests\Add2FASecretRequest;
use App\Http\Requests\ChangeEmailRequest;
use App\Http\Requests\ChangePasswordRequest;
use App\Http\Requests\DeleteUserRequest;
2020-10-10 16:30:26 +00:00
use App\Http\Requests\FlushSessionsRequest;
use App\Http\Requests\Remove2FASecretRequest;
use App\Http\Requests\SearchPlayerRequest;
use App\Http\Requests\UpdateUserRequest;
2020-10-10 16:30:26 +00:00
use App\Notifications\ChangedPassword;
use App\Notifications\EmailChanged;
use App\Traits\ReceivesAccountTokens;
use App\User;
2020-10-10 16:30:26 +00:00
use Google2FA;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Hash;
use Illuminate\Support\Facades\Log;
use Spatie\Permission\Models\Role;
class UserController extends Controller
{
use ReceivesAccountTokens;
public function showStaffMembers()
{
$this->authorize('viewStaff', User::class);
$staffRoles = [
'reviewer',
'hiringManager',
2020-10-10 16:30:26 +00:00
'admin',
]; // TODO: Un-hardcode this, move to config/roles.php
$users = User::with('roles')->get();
$staffMembers = collect([]);
2020-10-10 16:30:26 +00:00
foreach ($users as $user) {
if (empty($user->roles)) {
Log::debug($user->role->name);
Log::debug('Staff list: User without role detected; Ignoring');
continue;
}
2020-10-10 16:30:26 +00:00
foreach ($user->roles as $role) {
if (in_array($role->name, $staffRoles)) {
$staffMembers->push($user);
continue 2; // Skip directly to the next user instead of comparing more roles for the current user
}
}
}
return view('dashboard.administration.staff-members')
->with([
2020-10-10 16:30:26 +00:00
'users' => $staffMembers,
]);
}
public function showPlayers()
{
$this->authorize('viewPlayers', User::class);
$users = User::with('roles')->get();
$players = collect([]);
2020-10-10 16:30:26 +00:00
foreach ($users as $user) {
// TODO: Might be problematic if we don't check if the role is user
2020-10-10 16:30:26 +00:00
if (count($user->roles) == 1) {
$players->push($user);
}
}
return view('dashboard.administration.players')
->with([
'users' => $players,
2020-10-10 16:30:26 +00:00
'bannedUserCount' => Ban::all()->count(),
]);
}
public function showPlayersLike(SearchPlayerRequest $request)
{
$this->authorize('viewPlayers', User::class);
$searchTerm = $request->searchTerm;
$matchingUsers = User::query()
->where('name', 'LIKE', "%{$searchTerm}%")
->orWhere('email', 'LIKE', "%{$searchTerm}%")
->get();
2020-10-10 16:30:26 +00:00
if (! $matchingUsers->isEmpty()) {
$request->session()->flash('success', 'There were '.$matchingUsers->count().' user(s) matching your search.');
return view('dashboard.administration.players')
->with([
'users' => $matchingUsers,
2020-10-10 16:30:26 +00:00
'bannedUserCount' => Ban::all()->count(),
]);
2020-10-10 16:30:26 +00:00
} else {
$request->session()->flash('error', 'Your search term did not return any results.');
2020-10-10 16:30:26 +00:00
return redirect(route('registeredPlayerList'));
}
}
2020-07-17 21:44:10 +00:00
public function showAccount(Request $request)
{
2020-07-17 21:44:10 +00:00
$QRCode = null;
2020-10-10 16:30:26 +00:00
if (! $request->user()->has2FA()) {
if ($request->session()->has('twofaAttemptFailed')) {
2020-07-17 21:44:10 +00:00
$twoFactorSecret = $request->session()->get('current2FA');
2020-10-10 16:30:26 +00:00
} else {
2020-07-17 21:44:10 +00:00
$twoFactorSecret = Google2FA::generateSecretKey(32, '');
$request->session()->put('current2FA', $twoFactorSecret);
}
$QRCode = Google2FA::getQRCodeInline(
config('app.name'),
$request->user()->email,
$twoFactorSecret
);
}
return view('dashboard.user.profile.useraccount')
2020-07-17 21:44:10 +00:00
->with('ip', request()->ip())
->with('twofaQRCode', $QRCode);
}
public function flushSessions(FlushSessionsRequest $request)
{
// TODO: Move all log calls to a listener, which binds to an event fired by each significant event, such as this one
// This will allow for other actions to be performed on certain events (like login failed event)
Auth::logoutOtherDevices($request->currentPasswordFlush);
2020-10-10 16:30:26 +00:00
Log::notice('User '.Auth::user()->name.' has logged out other devices in their account',
[
'originIPAddress' => $request->ip(),
'userID' => Auth::user()->id,
2020-10-10 16:30:26 +00:00
'timestamp' => now(),
]);
$request->session()->flash('success', 'Successfully logged out other devices. Remember to change your password if you think you\'ve been compromised.');
2020-10-10 16:30:26 +00:00
return redirect()->back();
}
public function changePassword(ChangePasswordRequest $request)
{
$user = User::find(Auth::user()->id);
2020-10-10 16:30:26 +00:00
if (! is_null($user)) {
$user->password = Hash::make($request->newPassword);
$user->save();
2020-10-10 16:30:26 +00:00
Log::info('User '.$user->name.' has changed their password', [
'originIPAddress' => $request->ip(),
'userID' => $user->id,
2020-10-10 16:30:26 +00:00
'timestamp' => now(),
]);
$user->notify(new ChangedPassword());
Auth::logout();
2020-10-10 16:30:26 +00:00
return redirect()->back();
}
}
public function changeEmail(ChangeEmailRequest $request)
{
$user = User::find(Auth::user()->id);
2020-10-10 16:30:26 +00:00
if (! is_null($user)) {
$user->email = $request->newEmail;
$user->save();
2020-10-10 16:30:26 +00:00
Log::notice('User '.$user->name.' has just changed their contact email address', [
'originIPAddress' => $request->ip(),
'userID' => $user->id,
2020-10-10 16:30:26 +00:00
'timestamp' => now(),
]);
$user->notify(new EmailChanged());
$request->session()->flash('success', 'Your email address has been changed!');
2020-10-10 16:30:26 +00:00
} else {
$request->session()->flash('error', 'There has been an error whilst trying to update your account. Please contact administrators.');
}
return redirect()->back();
}
public function delete(DeleteUserRequest $request, User $user)
{
$this->authorize('delete', $user);
2020-10-10 16:30:26 +00:00
if ($request->confirmPrompt == 'DELETE ACCOUNT') {
$user->forceDelete();
2020-10-10 16:30:26 +00:00
$request->session()->flash('success', 'User deleted successfully. PII has been erased.');
} else {
$request->session()->flash('error', 'Wrong confirmation text! Try again.');
}
return redirect()->route('registeredPlayerList');
}
public function update(UpdateUserRequest $request, User $user)
{
2020-10-10 16:30:26 +00:00
$this->authorize('adminEdit', $user);
2020-10-10 16:30:26 +00:00
// Mass update would not be possible here without extra code, making route model binding useless
$user->email = $request->email;
$user->name = $request->name;
$user->uuid = $request->uuid;
2020-10-10 16:30:26 +00:00
$existingRoles = Role::all()
->pluck('name')
->all();
2020-10-10 16:30:26 +00:00
$roleDiff = array_diff($existingRoles, $request->roles);
2020-10-10 16:30:26 +00:00
// Adds roles that were selected. Removes roles that aren't selected if the user has them.
foreach ($roleDiff as $deselectedRole) {
if ($user->hasRole($deselectedRole) && $deselectedRole !== 'user') {
$user->removeRole($deselectedRole);
}
}
2020-10-10 16:30:26 +00:00
foreach ($request->roles as $role) {
if (! $user->hasRole($role)) {
$user->assignRole($role);
}
}
2020-10-10 16:30:26 +00:00
$user->save();
$request->session()->flash('success', 'User updated successfully!');
2020-10-10 16:30:26 +00:00
return redirect()->back();
}
2020-07-17 21:44:10 +00:00
public function add2FASecret(Add2FASecretRequest $request)
{
$currentSecret = $request->session()->get('current2FA');
$isValid = Google2FA::verifyKey($currentSecret, $request->otp);
2020-10-10 16:30:26 +00:00
if ($isValid) {
$request->user()->twofa_secret = $currentSecret;
$request->user()->save();
2020-07-17 21:44:10 +00:00
2020-10-10 16:30:26 +00:00
Log::warning('SECURITY: User activated two-factor authentication', [
'initiator' => $request->user()->email,
'ip' => $request->ip(),
]);
2020-07-17 21:44:10 +00:00
2020-10-10 16:30:26 +00:00
Google2FA::login();
2020-07-17 21:44:10 +00:00
2020-10-10 16:30:26 +00:00
Log::warning('SECURITY: Started two factor session automatically', [
'initiator' => $request->user()->email,
'ip' => $request->ip(),
]);
2020-07-17 21:44:10 +00:00
2020-10-10 16:30:26 +00:00
$request->session()->forget('current2FA');
2020-07-17 21:44:10 +00:00
2020-10-10 16:30:26 +00:00
if ($request->session()->has('twofaAttemptFailed')) {
$request->session()->forget('twofaAttemptFailed');
}
2020-07-17 21:44:10 +00:00
2020-10-10 16:30:26 +00:00
$request->session()->flash('success', '2FA succesfully enabled! You\'ll now be prompted for an OTP each time you log in.');
} else {
$request->session()->flash('error', 'Incorrect code. Please reopen the 2FA settings panel and try again.');
$request->session()->put('twofaAttemptFailed', true);
2020-07-17 21:44:10 +00:00
}
return redirect()->back();
}
public function remove2FASecret(Remove2FASecretRequest $request)
{
Log::warning('SECURITY: Disabling two factor authentication (user initiated)', [
2020-10-10 16:30:26 +00:00
'initiator' => $request->user()->email,
'ip' => $request->ip(),
2020-07-17 21:44:10 +00:00
]);
$request->user()->twofa_secret = null;
$request->user()->save();
$request->session()->flash('success', 'Two-factor authentication disabled.');
2020-10-10 16:30:26 +00:00
2020-07-17 21:44:10 +00:00
return redirect()->back();
}
public function terminate(Request $request, User $user)
{
$this->authorize('terminate', User::class);
2020-07-17 21:44:10 +00:00
// TODO: move logic to policy
2020-10-10 16:30:26 +00:00
if (! $user->isStaffMember() || $user->is(Auth::user())) {
$request->session()->flash('error', 'You cannot terminate this user.');
2020-10-10 16:30:26 +00:00
return redirect()->back();
}
2020-10-10 16:30:26 +00:00
foreach ($user->roles as $role) {
if ($role->name == 'user') {
continue;
}
2020-10-10 16:30:26 +00:00
$user->removeRole($role->name);
}
2020-10-10 16:30:26 +00:00
Log::info('User '.$user->name.' has just been demoted.');
$request->session()->flash('success', 'User terminated successfully.');
//TODO: Dispatch event
return redirect()->back();
}
}