This commit is contained in:
2025-12-23 19:26:23 +01:00
commit da7e984965
94 changed files with 26350 additions and 0 deletions

View File

@@ -0,0 +1,68 @@
<?php
namespace App\Http\Controllers\Auth;
use App\Http\Controllers\Controller;
use App\Models\User;
use Illuminate\Auth\Events\Registered;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Hash;
use Illuminate\Support\Str;
use Inertia\Inertia;
use Inertia\Response;
class CompleteProfileController extends Controller
{
/**
* Display the complete profile view.
*/
public function create(Request $request): Response|RedirectResponse
{
$socialiteUser = session('socialite_user');
if (! $socialiteUser) {
return redirect()->route('login');
}
return Inertia::render('Auth/CompleteProfile', [
'socialiteUser' => $socialiteUser,
]);
}
/**
* Handle the complete profile request.
*/
public function store(Request $request): RedirectResponse
{
$socialiteUser = session('socialite_user');
if (! $socialiteUser) {
return redirect()->route('login');
}
$request->validate([
'username' => ['required', 'string', 'max:255', 'alpha_dash', 'unique:'.User::class],
'first_name' => ['required', 'string', 'max:255'],
'last_name' => ['required', 'string', 'max:255'],
]);
$user = User::create([
'username' => $request->username,
'first_name' => $request->first_name,
'last_name' => $request->last_name,
'email' => $socialiteUser['email'],
'password' => Hash::make(Str::random(24)),
'email_verified_at' => now(),
]);
session()->forget('socialite_user');
event(new Registered($user));
Auth::login($user, remember: true);
return redirect()->intended(config('auth-ui.redirects.login', '/'));
}
}

View File

@@ -0,0 +1,49 @@
<?php
namespace App\Http\Controllers\Auth;
use App\Http\Controllers\Controller;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Password;
use Inertia\Inertia;
use Inertia\Response;
class ForgotPasswordController extends Controller
{
/**
* Display the password reset link request view.
*/
public function create(): Response
{
if (! config('auth-ui.features.password_reset')) {
abort(404);
}
return Inertia::render('Auth/ForgotPassword');
}
/**
* Handle an incoming password reset link request.
*/
public function store(Request $request): RedirectResponse
{
if (! config('auth-ui.features.password_reset')) {
abort(404);
}
$request->validate([
'email' => ['required', 'email'],
]);
$status = Password::sendResetLink(
$request->only('email')
);
if ($status === Password::RESET_LINK_SENT) {
return back()->with('status', __($status));
}
return back()->withErrors(['email' => __($status)]);
}
}

View File

@@ -0,0 +1,60 @@
<?php
namespace App\Http\Controllers\Auth;
use App\Http\Controllers\Controller;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
use Illuminate\Validation\ValidationException;
use Inertia\Inertia;
use Inertia\Response;
class LoginController extends Controller
{
/**
* Display the login view.
*/
public function create(): Response
{
return Inertia::render('Auth/Login');
}
/**
* Handle an incoming authentication request.
*/
public function store(Request $request): RedirectResponse
{
$credentials = $request->validate([
'email' => ['required', 'string', 'email'],
'password' => ['required', 'string'],
]);
$remember = config('auth-ui.features.remember_me')
? $request->boolean('remember')
: false;
if (! Auth::attempt($credentials, $remember)) {
throw ValidationException::withMessages([
'email' => __('auth.failed'),
]);
}
$request->session()->regenerate();
return redirect()->intended(config('auth-ui.redirects.login', '/'));
}
/**
* Destroy an authenticated session.
*/
public function destroy(Request $request): RedirectResponse
{
Auth::logout();
$request->session()->invalidate();
$request->session()->regenerateToken();
return redirect(config('auth-ui.redirects.logout', '/'));
}
}

View File

@@ -0,0 +1,61 @@
<?php
namespace App\Http\Controllers\Auth;
use App\Http\Controllers\Controller;
use App\Models\User;
use Illuminate\Auth\Events\Registered;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Hash;
use Illuminate\Validation\Rules;
use Inertia\Inertia;
use Inertia\Response;
class RegisterController extends Controller
{
/**
* Display the registration view.
*/
public function create(): Response
{
if (! config('auth-ui.features.registration')) {
abort(404);
}
return Inertia::render('Auth/Register');
}
/**
* Handle an incoming registration request.
*/
public function store(Request $request): RedirectResponse
{
if (! config('auth-ui.features.registration')) {
abort(404);
}
$request->validate([
'username' => ['required', 'string', 'max:255', 'alpha_dash', 'unique:'.User::class],
'first_name' => ['required', 'string', 'max:255'],
'last_name' => ['required', 'string', 'max:255'],
'email' => ['required', 'string', 'lowercase', 'email', 'max:255', 'unique:'.User::class],
'password' => ['required', 'confirmed', Rules\Password::defaults()],
]);
$user = User::create([
'username' => $request->username,
'first_name' => $request->first_name,
'last_name' => $request->last_name,
'email' => $request->email,
'password' => Hash::make($request->password),
]);
event(new Registered($user));
Auth::login($user);
return redirect(config('auth-ui.redirects.register', '/'));
}
}

View File

@@ -0,0 +1,66 @@
<?php
namespace App\Http\Controllers\Auth;
use App\Http\Controllers\Controller;
use Illuminate\Auth\Events\PasswordReset;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Hash;
use Illuminate\Support\Facades\Password;
use Illuminate\Support\Str;
use Illuminate\Validation\Rules;
use Inertia\Inertia;
use Inertia\Response;
class ResetPasswordController extends Controller
{
/**
* Display the password reset view.
*/
public function create(Request $request): Response
{
if (! config('auth-ui.features.password_reset')) {
abort(404);
}
return Inertia::render('Auth/ResetPassword', [
'email' => $request->email,
'token' => $request->route('token'),
]);
}
/**
* Handle an incoming new password request.
*/
public function store(Request $request): RedirectResponse
{
if (! config('auth-ui.features.password_reset')) {
abort(404);
}
$request->validate([
'token' => ['required'],
'email' => ['required', 'email'],
'password' => ['required', 'confirmed', Rules\Password::defaults()],
]);
$status = Password::reset(
$request->only('email', 'password', 'password_confirmation', 'token'),
function ($user) use ($request) {
$user->forceFill([
'password' => Hash::make($request->password),
'remember_token' => Str::random(60),
])->save();
event(new PasswordReset($user));
}
);
if ($status === Password::PASSWORD_RESET) {
return redirect()->route('login')->with('status', __($status));
}
return back()->withErrors(['email' => __($status)]);
}
}

View File

@@ -0,0 +1,109 @@
<?php
namespace App\Http\Controllers\Auth;
use App\Http\Controllers\Controller;
use App\Models\User;
use Illuminate\Http\RedirectResponse;
use Illuminate\Support\Facades\Auth;
use Illuminate\Support\Facades\Hash;
use Illuminate\Support\Str;
use Laravel\Socialite\Facades\Socialite;
class SocialiteController extends Controller
{
/**
* Suggest a username based on social profile.
*/
protected function suggestUsername($socialUser): string
{
return Str::slug(
$socialUser->getNickname() ?? explode('@', $socialUser->getEmail())[0],
'_'
);
}
/**
* Get enabled providers from config.
*/
protected function getEnabledProviders(): array
{
return collect(config('auth-ui.providers', []))
->filter(fn ($provider) => $provider['enabled'] ?? false)
->keys()
->all();
}
/**
* Redirect the user to the provider authentication page.
*/
public function redirect(string $provider): RedirectResponse
{
if (! in_array($provider, $this->getEnabledProviders())) {
abort(404, 'Provider not enabled');
}
return Socialite::driver($provider)->redirect();
}
/**
* Obtain the user information from provider.
*/
public function callback(string $provider): RedirectResponse
{
if (! in_array($provider, $this->getEnabledProviders())) {
abort(404, 'Provider not enabled');
}
try {
$socialUser = Socialite::driver($provider)->user();
} catch (\Exception $e) {
return redirect()->route('login')->withErrors([
'email' => 'Unable to authenticate with '.$provider.'. Please try again.',
]);
}
// Find existing user by provider ID or email
$user = User::where('email', $socialUser->getEmail())->first();
if (! $user) {
// Create new user if registration is enabled
if (! config('auth-ui.features.registration')) {
return redirect()->route('login')->withErrors([
'email' => 'Registration is disabled. Please contact an administrator.',
]);
}
$name = $socialUser->getName() ?? $socialUser->getNickname() ?? 'User';
$nameParts = explode(' ', $name, 2);
$suggestedUsername = $this->suggestUsername($socialUser);
// Check if username is already taken
if (User::where('username', $suggestedUsername)->exists()) {
// Store social data in session and redirect to complete profile
session()->put('socialite_user', [
'email' => $socialUser->getEmail(),
'first_name' => $nameParts[0],
'last_name' => $nameParts[1] ?? '',
'suggested_username' => $suggestedUsername,
'provider' => $provider,
]);
return redirect()->route('auth.complete-profile');
}
$user = User::create([
'username' => $suggestedUsername,
'first_name' => $nameParts[0],
'last_name' => $nameParts[1] ?? '',
'email' => $socialUser->getEmail(),
'password' => Hash::make(Str::random(24)),
'email_verified_at' => now(),
]);
}
Auth::login($user, remember: true);
return redirect()->intended(config('auth-ui.redirects.login', '/'));
}
}