mirror of
https://github.com/pacnpal/thrillwiki_laravel.git
synced 2025-12-23 19:31:10 -05:00
feat: Implement rides management with CRUD functionality
- Added rides index view with search and filter options. - Created rides show view to display ride details. - Implemented API routes for rides. - Developed authentication routes for user registration, login, and email verification. - Created tests for authentication, email verification, password reset, and user profile management. - Added feature tests for rides and operators, including creation, updating, deletion, and searching. - Implemented soft deletes and caching for rides and operators. - Enhanced manufacturer and operator model tests for various functionalities.
This commit is contained in:
95
app/Http/Controllers/Api/OperatorController.php
Normal file
95
app/Http/Controllers/Api/OperatorController.php
Normal file
@@ -0,0 +1,95 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Controllers\Api;
|
||||
|
||||
use App\Http\Controllers\Controller;
|
||||
use App\Models\Operator;
|
||||
use App\Http\Requests\OperatorRequest;
|
||||
use App\Http\Resources\OperatorResource;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\Http\JsonResponse;
|
||||
|
||||
class OperatorController extends Controller
|
||||
{
|
||||
/**
|
||||
* Display a listing of the resource.
|
||||
*/
|
||||
public function index(Request $request): JsonResponse
|
||||
{
|
||||
$query = Operator::query();
|
||||
|
||||
// Search functionality
|
||||
if ($request->filled('search')) {
|
||||
$search = $request->get('search');
|
||||
$query->where(function ($q) use ($search) {
|
||||
$q->where('name', 'ILIKE', "%{$search}%")
|
||||
->orWhere('description', 'ILIKE', "%{$search}%");
|
||||
});
|
||||
}
|
||||
|
||||
// Filter by status
|
||||
if ($request->filled('status')) {
|
||||
$query->where('is_active', $request->get('status') === 'active');
|
||||
}
|
||||
|
||||
$operators = $query->latest()->paginate(15);
|
||||
|
||||
return response()->json([
|
||||
'data' => OperatorResource::collection($operators),
|
||||
'meta' => [
|
||||
'current_page' => $operators->currentPage(),
|
||||
'last_page' => $operators->lastPage(),
|
||||
'per_page' => $operators->perPage(),
|
||||
'total' => $operators->total(),
|
||||
]
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Store a newly created resource in storage.
|
||||
*/
|
||||
public function store(OperatorRequest $request): JsonResponse
|
||||
{
|
||||
$operator = Operator::create($request->validated());
|
||||
|
||||
return response()->json([
|
||||
'message' => 'Operator created successfully',
|
||||
'data' => new OperatorResource($operator)
|
||||
], 201);
|
||||
}
|
||||
|
||||
/**
|
||||
* Display the specified resource.
|
||||
*/
|
||||
public function show(Operator $operator): JsonResponse
|
||||
{
|
||||
return response()->json([
|
||||
'data' => new OperatorResource($operator)
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Update the specified resource in storage.
|
||||
*/
|
||||
public function update(OperatorRequest $request, Operator $operator): JsonResponse
|
||||
{
|
||||
$operator->update($request->validated());
|
||||
|
||||
return response()->json([
|
||||
'message' => 'Operator updated successfully',
|
||||
'data' => new OperatorResource($operator)
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the specified resource from storage.
|
||||
*/
|
||||
public function destroy(Operator $operator): JsonResponse
|
||||
{
|
||||
$operator->delete();
|
||||
|
||||
return response()->json([
|
||||
'message' => 'Operator deleted successfully'
|
||||
]);
|
||||
}
|
||||
}
|
||||
95
app/Http/Controllers/Api/RideController.php
Normal file
95
app/Http/Controllers/Api/RideController.php
Normal file
@@ -0,0 +1,95 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Controllers\Api;
|
||||
|
||||
use App\Http\Controllers\Controller;
|
||||
use App\Models\Ride;
|
||||
use App\Http\Requests\RideRequest;
|
||||
use App\Http\Resources\RideResource;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\Http\JsonResponse;
|
||||
|
||||
class RideController extends Controller
|
||||
{
|
||||
/**
|
||||
* Display a listing of the resource.
|
||||
*/
|
||||
public function index(Request $request): JsonResponse
|
||||
{
|
||||
$query = Ride::query();
|
||||
|
||||
// Search functionality
|
||||
if ($request->filled('search')) {
|
||||
$search = $request->get('search');
|
||||
$query->where(function ($q) use ($search) {
|
||||
$q->where('name', 'ILIKE', "%{$search}%")
|
||||
->orWhere('description', 'ILIKE', "%{$search}%");
|
||||
});
|
||||
}
|
||||
|
||||
// Filter by status
|
||||
if ($request->filled('status')) {
|
||||
$query->where('is_active', $request->get('status') === 'active');
|
||||
}
|
||||
|
||||
$rides = $query->latest()->paginate(15);
|
||||
|
||||
return response()->json([
|
||||
'data' => RideResource::collection($rides),
|
||||
'meta' => [
|
||||
'current_page' => $rides->currentPage(),
|
||||
'last_page' => $rides->lastPage(),
|
||||
'per_page' => $rides->perPage(),
|
||||
'total' => $rides->total(),
|
||||
]
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Store a newly created resource in storage.
|
||||
*/
|
||||
public function store(RideRequest $request): JsonResponse
|
||||
{
|
||||
$ride = Ride::create($request->validated());
|
||||
|
||||
return response()->json([
|
||||
'message' => 'Ride created successfully',
|
||||
'data' => new RideResource($ride)
|
||||
], 201);
|
||||
}
|
||||
|
||||
/**
|
||||
* Display the specified resource.
|
||||
*/
|
||||
public function show(Ride $ride): JsonResponse
|
||||
{
|
||||
return response()->json([
|
||||
'data' => new RideResource($ride)
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Update the specified resource in storage.
|
||||
*/
|
||||
public function update(RideRequest $request, Ride $ride): JsonResponse
|
||||
{
|
||||
$ride->update($request->validated());
|
||||
|
||||
return response()->json([
|
||||
'message' => 'Ride updated successfully',
|
||||
'data' => new RideResource($ride)
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the specified resource from storage.
|
||||
*/
|
||||
public function destroy(Ride $ride): JsonResponse
|
||||
{
|
||||
$ride->delete();
|
||||
|
||||
return response()->json([
|
||||
'message' => 'Ride deleted successfully'
|
||||
]);
|
||||
}
|
||||
}
|
||||
27
app/Http/Controllers/Auth/VerifyEmailController.php
Normal file
27
app/Http/Controllers/Auth/VerifyEmailController.php
Normal file
@@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Controllers\Auth;
|
||||
|
||||
use App\Http\Controllers\Controller;
|
||||
use Illuminate\Auth\Events\Verified;
|
||||
use Illuminate\Foundation\Auth\EmailVerificationRequest;
|
||||
use Illuminate\Http\RedirectResponse;
|
||||
|
||||
class VerifyEmailController extends Controller
|
||||
{
|
||||
/**
|
||||
* Mark the authenticated user's email address as verified.
|
||||
*/
|
||||
public function __invoke(EmailVerificationRequest $request): RedirectResponse
|
||||
{
|
||||
if ($request->user()->hasVerifiedEmail()) {
|
||||
return redirect()->intended(route('dashboard', absolute: false).'?verified=1');
|
||||
}
|
||||
|
||||
if ($request->user()->markEmailAsVerified()) {
|
||||
event(new Verified($request->user()));
|
||||
}
|
||||
|
||||
return redirect()->intended(route('dashboard', absolute: false).'?verified=1');
|
||||
}
|
||||
}
|
||||
98
app/Http/Controllers/OperatorController.php
Normal file
98
app/Http/Controllers/OperatorController.php
Normal file
@@ -0,0 +1,98 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Controllers;
|
||||
|
||||
use App\Models\Operator;
|
||||
use App\Http\Requests\OperatorRequest;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\View\View;
|
||||
use Illuminate\Http\RedirectResponse;
|
||||
|
||||
class OperatorController extends Controller
|
||||
{
|
||||
/**
|
||||
* Display a listing of the resource.
|
||||
*/
|
||||
public function index(Request $request): View
|
||||
{
|
||||
$query = Operator::query();
|
||||
|
||||
// Search functionality
|
||||
if ($request->filled('search')) {
|
||||
$search = $request->get('search');
|
||||
$query->where(function ($q) use ($search) {
|
||||
$q->where('name', 'ILIKE', "%{$search}%")
|
||||
->orWhere('description', 'ILIKE', "%{$search}%");
|
||||
});
|
||||
}
|
||||
|
||||
// Filter by status
|
||||
if ($request->filled('status')) {
|
||||
$query->where('is_active', $request->get('status') === 'active');
|
||||
}
|
||||
|
||||
$operators = $query->latest()->paginate(15)->withQueryString();
|
||||
|
||||
return view('operators.index', compact('operators'));
|
||||
}
|
||||
|
||||
/**
|
||||
* Show the form for creating a new resource.
|
||||
*/
|
||||
public function create(): View
|
||||
{
|
||||
return view('operators.create');
|
||||
}
|
||||
|
||||
/**
|
||||
* Store a newly created resource in storage.
|
||||
*/
|
||||
public function store(OperatorRequest $request): RedirectResponse
|
||||
{
|
||||
$operator = Operator::create($request->validated());
|
||||
|
||||
return redirect()
|
||||
->route('operators.show', $operator)
|
||||
->with('success', 'Operator created successfully!');
|
||||
}
|
||||
|
||||
/**
|
||||
* Display the specified resource.
|
||||
*/
|
||||
public function show(Operator $operator): View
|
||||
{
|
||||
return view('operators.show', compact('operator'));
|
||||
}
|
||||
|
||||
/**
|
||||
* Show the form for editing the specified resource.
|
||||
*/
|
||||
public function edit(Operator $operator): View
|
||||
{
|
||||
return view('operators.edit', compact('operator'));
|
||||
}
|
||||
|
||||
/**
|
||||
* Update the specified resource in storage.
|
||||
*/
|
||||
public function update(OperatorRequest $request, Operator $operator): RedirectResponse
|
||||
{
|
||||
$operator->update($request->validated());
|
||||
|
||||
return redirect()
|
||||
->route('operators.show', $operator)
|
||||
->with('success', 'Operator updated successfully!');
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the specified resource from storage.
|
||||
*/
|
||||
public function destroy(Operator $operator): RedirectResponse
|
||||
{
|
||||
$operator->delete();
|
||||
|
||||
return redirect()
|
||||
->route('operators.index')
|
||||
->with('success', 'Operator deleted successfully!');
|
||||
}
|
||||
}
|
||||
57
app/Http/Controllers/ParkController.php
Normal file
57
app/Http/Controllers/ParkController.php
Normal file
@@ -0,0 +1,57 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Controllers;
|
||||
|
||||
use App\Models\Park;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\View\View;
|
||||
|
||||
class ParkController extends Controller
|
||||
{
|
||||
/**
|
||||
* Display a listing of parks.
|
||||
*/
|
||||
public function index(): View
|
||||
{
|
||||
return view('parks.index');
|
||||
}
|
||||
|
||||
/**
|
||||
* Show the form for creating a new park.
|
||||
*/
|
||||
public function create(): View
|
||||
{
|
||||
return view('parks.create');
|
||||
}
|
||||
|
||||
/**
|
||||
* Display the specified park.
|
||||
*/
|
||||
public function show(Park $park): View
|
||||
{
|
||||
// Load relationships for the park detail page
|
||||
$park->load([
|
||||
'operator',
|
||||
'location',
|
||||
'areas.rides' => function ($query) {
|
||||
$query->orderBy('position')->orderBy('name');
|
||||
},
|
||||
'areas' => function ($query) {
|
||||
$query->orderBy('position')->orderBy('name');
|
||||
},
|
||||
'photos' => function ($query) {
|
||||
$query->orderBy('is_featured', 'desc')->orderBy('created_at', 'desc');
|
||||
}
|
||||
]);
|
||||
|
||||
return view('parks.show', compact('park'));
|
||||
}
|
||||
|
||||
/**
|
||||
* Show the form for editing the specified park.
|
||||
*/
|
||||
public function edit(Park $park): View
|
||||
{
|
||||
return view('parks.edit', compact('park'));
|
||||
}
|
||||
}
|
||||
98
app/Http/Controllers/RideController.php
Normal file
98
app/Http/Controllers/RideController.php
Normal file
@@ -0,0 +1,98 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Controllers;
|
||||
|
||||
use App\Models\Ride;
|
||||
use App\Http\Requests\RideRequest;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\View\View;
|
||||
use Illuminate\Http\RedirectResponse;
|
||||
|
||||
class RideController extends Controller
|
||||
{
|
||||
/**
|
||||
* Display a listing of the resource.
|
||||
*/
|
||||
public function index(Request $request): View
|
||||
{
|
||||
$query = Ride::query();
|
||||
|
||||
// Search functionality
|
||||
if ($request->filled('search')) {
|
||||
$search = $request->get('search');
|
||||
$query->where(function ($q) use ($search) {
|
||||
$q->where('name', 'ILIKE', "%{$search}%")
|
||||
->orWhere('description', 'ILIKE', "%{$search}%");
|
||||
});
|
||||
}
|
||||
|
||||
// Filter by status
|
||||
if ($request->filled('status')) {
|
||||
$query->where('is_active', $request->get('status') === 'active');
|
||||
}
|
||||
|
||||
$rides = $query->latest()->paginate(15)->withQueryString();
|
||||
|
||||
return view('rides.index', compact('rides'));
|
||||
}
|
||||
|
||||
/**
|
||||
* Show the form for creating a new resource.
|
||||
*/
|
||||
public function create(): View
|
||||
{
|
||||
return view('rides.create');
|
||||
}
|
||||
|
||||
/**
|
||||
* Store a newly created resource in storage.
|
||||
*/
|
||||
public function store(RideRequest $request): RedirectResponse
|
||||
{
|
||||
$ride = Ride::create($request->validated());
|
||||
|
||||
return redirect()
|
||||
->route('rides.show', $ride)
|
||||
->with('success', 'Ride created successfully!');
|
||||
}
|
||||
|
||||
/**
|
||||
* Display the specified resource.
|
||||
*/
|
||||
public function show(Ride $ride): View
|
||||
{
|
||||
return view('rides.show', compact('ride'));
|
||||
}
|
||||
|
||||
/**
|
||||
* Show the form for editing the specified resource.
|
||||
*/
|
||||
public function edit(Ride $ride): View
|
||||
{
|
||||
return view('rides.edit', compact('ride'));
|
||||
}
|
||||
|
||||
/**
|
||||
* Update the specified resource in storage.
|
||||
*/
|
||||
public function update(RideRequest $request, Ride $ride): RedirectResponse
|
||||
{
|
||||
$ride->update($request->validated());
|
||||
|
||||
return redirect()
|
||||
->route('rides.show', $ride)
|
||||
->with('success', 'Ride updated successfully!');
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove the specified resource from storage.
|
||||
*/
|
||||
public function destroy(Ride $ride): RedirectResponse
|
||||
{
|
||||
$ride->delete();
|
||||
|
||||
return redirect()
|
||||
->route('rides.index')
|
||||
->with('success', 'Ride deleted successfully!');
|
||||
}
|
||||
}
|
||||
48
app/Http/Requests/OperatorRequest.php
Normal file
48
app/Http/Requests/OperatorRequest.php
Normal file
@@ -0,0 +1,48 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Requests;
|
||||
|
||||
use Illuminate\Foundation\Http\FormRequest;
|
||||
|
||||
class OperatorRequest extends FormRequest
|
||||
{
|
||||
/**
|
||||
* Determine if the user is authorized to make this request.
|
||||
*/
|
||||
public function authorize(): bool
|
||||
{
|
||||
return true; // Add authorization logic as needed
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the validation rules that apply to the request.
|
||||
*/
|
||||
public function rules(): array
|
||||
{
|
||||
$rules = [
|
||||
'name' => ['required', 'string', 'max:255'],
|
||||
'description' => ['nullable', 'string'],
|
||||
'is_active' => ['boolean'],
|
||||
];
|
||||
|
||||
// For updates, make name unique except for current record
|
||||
if ($this->route('operator')) {
|
||||
$rules['name'][] = 'unique:operators,name,' . $this->route('operator')->id;
|
||||
} else {
|
||||
$rules['name'][] = 'unique:operators,name';
|
||||
}
|
||||
|
||||
return $rules;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get custom messages for validator errors.
|
||||
*/
|
||||
public function messages(): array
|
||||
{
|
||||
return [
|
||||
'name.required' => 'The operator name is required.',
|
||||
'name.unique' => 'A operator with this name already exists.',
|
||||
];
|
||||
}
|
||||
}
|
||||
48
app/Http/Requests/RideRequest.php
Normal file
48
app/Http/Requests/RideRequest.php
Normal file
@@ -0,0 +1,48 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Requests;
|
||||
|
||||
use Illuminate\Foundation\Http\FormRequest;
|
||||
|
||||
class RideRequest extends FormRequest
|
||||
{
|
||||
/**
|
||||
* Determine if the user is authorized to make this request.
|
||||
*/
|
||||
public function authorize(): bool
|
||||
{
|
||||
return true; // Add authorization logic as needed
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the validation rules that apply to the request.
|
||||
*/
|
||||
public function rules(): array
|
||||
{
|
||||
$rules = [
|
||||
'name' => ['required', 'string', 'max:255'],
|
||||
'description' => ['nullable', 'string'],
|
||||
'is_active' => ['boolean'],
|
||||
];
|
||||
|
||||
// For updates, make name unique except for current record
|
||||
if ($this->route('ride')) {
|
||||
$rules['name'][] = 'unique:rides,name,' . $this->route('ride')->id;
|
||||
} else {
|
||||
$rules['name'][] = 'unique:rides,name';
|
||||
}
|
||||
|
||||
return $rules;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get custom messages for validator errors.
|
||||
*/
|
||||
public function messages(): array
|
||||
{
|
||||
return [
|
||||
'name.required' => 'The ride name is required.',
|
||||
'name.unique' => 'A ride with this name already exists.',
|
||||
];
|
||||
}
|
||||
}
|
||||
53
app/Http/Resources/ManufacturerResource.php
Normal file
53
app/Http/Resources/ManufacturerResource.php
Normal file
@@ -0,0 +1,53 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Resources;
|
||||
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\Http\Resources\Json\JsonResource;
|
||||
|
||||
/**
|
||||
* Manufacturer API Resource
|
||||
*
|
||||
* Transforms Manufacturer model data for API responses
|
||||
* Includes ThrillWiki optimization patterns
|
||||
*/
|
||||
class ManufacturerResource extends JsonResource
|
||||
{
|
||||
/**
|
||||
* Transform the resource into an array.
|
||||
*
|
||||
* @return array<string, mixed>
|
||||
*/
|
||||
public function toArray(Request $request): array
|
||||
{
|
||||
return [
|
||||
'id' => $this->id,
|
||||
'name' => $this->name,
|
||||
'slug' => $this->slug,
|
||||
'description' => $this->description,
|
||||
'is_active' => $this->is_active,
|
||||
'created_at' => $this->created_at?->toISOString(),
|
||||
'updated_at' => $this->updated_at?->toISOString(),
|
||||
|
||||
// Include relationships when loaded
|
||||
$this->mergeWhen($this->relationLoaded('relationships'), [
|
||||
// Add relationship data here
|
||||
]),
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* Get additional data that should be returned with the resource array.
|
||||
*
|
||||
* @return array<string, mixed>
|
||||
*/
|
||||
public function with(Request $request): array
|
||||
{
|
||||
return [
|
||||
'meta' => [
|
||||
'model' => 'Manufacturer',
|
||||
'generated_at' => now()->toISOString(),
|
||||
],
|
||||
];
|
||||
}
|
||||
}
|
||||
24
app/Http/Resources/OperatorResource.php
Normal file
24
app/Http/Resources/OperatorResource.php
Normal file
@@ -0,0 +1,24 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Resources;
|
||||
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\Http\Resources\Json\JsonResource;
|
||||
|
||||
class OperatorResource extends JsonResource
|
||||
{
|
||||
/**
|
||||
* Transform the resource into an array.
|
||||
*/
|
||||
public function toArray(Request $request): array
|
||||
{
|
||||
return [
|
||||
'id' => $this->id,
|
||||
'name' => $this->name,
|
||||
'description' => $this->description,
|
||||
'is_active' => $this->is_active,
|
||||
'created_at' => $this->created_at,
|
||||
'updated_at' => $this->updated_at,
|
||||
];
|
||||
}
|
||||
}
|
||||
24
app/Http/Resources/RideResource.php
Normal file
24
app/Http/Resources/RideResource.php
Normal file
@@ -0,0 +1,24 @@
|
||||
<?php
|
||||
|
||||
namespace App\Http\Resources;
|
||||
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\Http\Resources\Json\JsonResource;
|
||||
|
||||
class RideResource extends JsonResource
|
||||
{
|
||||
/**
|
||||
* Transform the resource into an array.
|
||||
*/
|
||||
public function toArray(Request $request): array
|
||||
{
|
||||
return [
|
||||
'id' => $this->id,
|
||||
'name' => $this->name,
|
||||
'description' => $this->description,
|
||||
'is_active' => $this->is_active,
|
||||
'created_at' => $this->created_at,
|
||||
'updated_at' => $this->updated_at,
|
||||
];
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user