├── .DS_Store ├── .editorconfig ├── .env.example ├── .gitattributes ├── .gitignore ├── LICENSE ├── README.md ├── app ├── Console │ └── Kernel.php ├── Exceptions │ └── Handler.php ├── Http │ ├── Controllers │ │ ├── Auth │ │ │ ├── AuthenticatedSessionController.php │ │ │ ├── ConfirmablePasswordController.php │ │ │ ├── EmailVerificationNotificationController.php │ │ │ ├── EmailVerificationPromptController.php │ │ │ ├── NewPasswordController.php │ │ │ ├── PasswordController.php │ │ │ ├── PasswordResetLinkController.php │ │ │ ├── RegisteredUserController.php │ │ │ └── VerifyEmailController.php │ │ ├── Controller.php │ │ ├── MainController.php │ │ └── ProfileController.php │ ├── Kernel.php │ ├── Middleware │ │ ├── Authenticate.php │ │ ├── EncryptCookies.php │ │ ├── HandleInertiaRequests.php │ │ ├── PreventRequestsDuringMaintenance.php │ │ ├── RedirectIfAuthenticated.php │ │ ├── TrimStrings.php │ │ ├── TrustHosts.php │ │ ├── TrustProxies.php │ │ ├── ValidateSignature.php │ │ └── VerifyCsrfToken.php │ └── Requests │ │ ├── Auth │ │ └── LoginRequest.php │ │ └── ProfileUpdateRequest.php ├── Models │ └── User.php └── Providers │ ├── AppServiceProvider.php │ ├── AuthServiceProvider.php │ ├── BroadcastServiceProvider.php │ ├── EventServiceProvider.php │ └── RouteServiceProvider.php ├── artisan ├── bootstrap ├── app.php └── cache │ └── .gitignore ├── composer.json ├── composer.lock ├── config ├── app.php ├── auth.php ├── broadcasting.php ├── cache.php ├── cors.php ├── database.php ├── filesystems.php ├── hashing.php ├── logging.php ├── mail.php ├── queue.php ├── sanctum.php ├── services.php ├── session.php └── view.php ├── database ├── .gitignore ├── factories │ └── UserFactory.php ├── migrations │ ├── 2014_10_12_000000_create_users_table.php │ ├── 2014_10_12_100000_create_password_reset_tokens_table.php │ ├── 2019_08_19_000000_create_failed_jobs_table.php │ └── 2019_12_14_000001_create_personal_access_tokens_table.php └── seeders │ └── DatabaseSeeder.php ├── jsconfig.json ├── package-lock.json ├── package.json ├── phpunit.xml ├── postcss.config.js ├── public ├── .htaccess ├── favicon.ico ├── index.php └── robots.txt ├── resources ├── css │ └── app.css ├── js │ ├── Components │ │ ├── ApplicationLogo.jsx │ │ ├── Checkbox.jsx │ │ ├── DangerButton.jsx │ │ ├── Dropdown.jsx │ │ ├── InputError.jsx │ │ ├── InputLabel.jsx │ │ ├── Modal.jsx │ │ ├── NavLink.jsx │ │ ├── Navigation.jsx │ │ ├── PrimaryButton.jsx │ │ ├── ResponsiveNavLink.jsx │ │ ├── SecondaryButton.jsx │ │ ├── Spinner.jsx │ │ ├── Table.jsx │ │ └── TextInput.jsx │ ├── Layouts │ │ ├── AuthenticatedLayout.jsx │ │ └── GuestLayout.jsx │ ├── Pages │ │ ├── Auth │ │ │ ├── ConfirmPassword.jsx │ │ │ ├── ForgotPassword.jsx │ │ │ ├── Login.jsx │ │ │ ├── Register.jsx │ │ │ ├── ResetPassword.jsx │ │ │ └── VerifyEmail.jsx │ │ ├── Dashboard.jsx │ │ ├── Home.jsx │ │ ├── Inventory.jsx │ │ ├── List.jsx │ │ ├── Profile │ │ │ ├── Edit.jsx │ │ │ └── Partials │ │ │ │ ├── DeleteUserForm.jsx │ │ │ │ ├── UpdatePasswordForm.jsx │ │ │ │ └── UpdateProfileInformationForm.jsx │ │ ├── Reports.jsx │ │ └── Welcome.jsx │ ├── app.jsx │ ├── bootstrap.js │ └── services │ │ ├── ApiService.js │ │ ├── BatchApiService.js │ │ ├── index.js │ │ └── utils │ │ └── responseExtractors.js └── views │ ├── app.blade.php │ └── welcome.blade.php ├── routes ├── api.php ├── auth.php ├── channels.php ├── console.php └── web.php ├── storage ├── app │ ├── .gitignore │ └── public │ │ └── .gitignore ├── framework │ ├── .gitignore │ ├── cache │ │ ├── .gitignore │ │ └── data │ │ │ └── .gitignore │ ├── sessions │ │ └── .gitignore │ ├── testing │ │ └── .gitignore │ └── views │ │ └── .gitignore └── logs │ └── .gitignore ├── tailwind.config.js ├── tests ├── CreatesApplication.php ├── Feature │ ├── Auth │ │ ├── AuthenticationTest.php │ │ ├── EmailVerificationTest.php │ │ ├── PasswordConfirmationTest.php │ │ ├── PasswordResetTest.php │ │ ├── PasswordUpdateTest.php │ │ └── RegistrationTest.php │ ├── ExampleTest.php │ └── ProfileTest.php ├── TestCase.php └── Unit │ └── ExampleTest.php └── vite.config.js /.DS_Store: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bigcommerce/laravel-react-sample-app/564a2b9445c33966211f9345f064ff1b144efc68/.DS_Store -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [*] 4 | charset = utf-8 5 | end_of_line = lf 6 | indent_size = 4 7 | indent_style = space 8 | insert_final_newline = true 9 | trim_trailing_whitespace = true 10 | 11 | [*.md] 12 | trim_trailing_whitespace = false 13 | 14 | [*.{yml,yaml}] 15 | indent_size = 2 16 | 17 | [docker-compose.yml] 18 | indent_size = 4 19 | -------------------------------------------------------------------------------- /.env.example: -------------------------------------------------------------------------------- 1 | APP_NAME=Laravel 2 | APP_ENV=local 3 | APP_KEY= 4 | APP_DEBUG=true 5 | APP_URL=https://laravel-react-sample-app.test 6 | 7 | LOG_CHANNEL=stack 8 | LOG_DEPRECATIONS_CHANNEL=null 9 | LOG_LEVEL=debug 10 | 11 | DB_CONNECTION=mysql 12 | DB_HOST=127.0.0.1 13 | DB_PORT=3306 14 | DB_DATABASE=laravel 15 | DB_USERNAME=root 16 | DB_PASSWORD= 17 | 18 | BROADCAST_DRIVER=log 19 | CACHE_DRIVER=file 20 | FILESYSTEM_DISK=local 21 | QUEUE_CONNECTION=sync 22 | SESSION_DRIVER=file 23 | SESSION_LIFETIME=120 24 | 25 | MEMCACHED_HOST=127.0.0.1 26 | 27 | REDIS_HOST=127.0.0.1 28 | REDIS_PASSWORD=null 29 | REDIS_PORT=6379 30 | 31 | MAIL_MAILER=smtp 32 | MAIL_HOST=mailpit 33 | MAIL_PORT=1025 34 | MAIL_USERNAME=null 35 | MAIL_PASSWORD=null 36 | MAIL_ENCRYPTION=null 37 | MAIL_FROM_ADDRESS="hello@example.com" 38 | MAIL_FROM_NAME="${APP_NAME}" 39 | 40 | AWS_ACCESS_KEY_ID= 41 | AWS_SECRET_ACCESS_KEY= 42 | AWS_DEFAULT_REGION=us-east-1 43 | AWS_BUCKET= 44 | AWS_USE_PATH_STYLE_ENDPOINT=false 45 | 46 | PUSHER_APP_ID= 47 | PUSHER_APP_KEY= 48 | PUSHER_APP_SECRET= 49 | PUSHER_HOST= 50 | PUSHER_PORT=443 51 | PUSHER_SCHEME=https 52 | PUSHER_APP_CLUSTER=mt1 53 | 54 | VITE_PUSHER_APP_KEY="${PUSHER_APP_KEY}" 55 | VITE_PUSHER_HOST="${PUSHER_HOST}" 56 | VITE_PUSHER_PORT="${PUSHER_PORT}" 57 | VITE_PUSHER_SCHEME="${PUSHER_SCHEME}" 58 | VITE_PUSHER_APP_CLUSTER="${PUSHER_APP_CLUSTER}" 59 | 60 | # New env variables for BC app and a test API credentials for local dev 61 | # The Client ID and Secret can be found at https://devtools.bigcommerce.com/my/apps by selecting 'View Client ID' 62 | BC_APP_CLIENT_ID=xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx 63 | BC_APP_SECRET=xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx 64 | 65 | # These local credentials can be created by creating an API Account within your BigCommerce store (Settings > API Accounts) 66 | BC_LOCAL_CLIENT_ID=xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx 67 | BC_LOCAL_SECRET=xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx 68 | BC_LOCAL_ACCESS_TOKEN=xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx 69 | BC_LOCAL_STORE_HASH=stores/xxxxxxxxxxx 70 | -------------------------------------------------------------------------------- /.gitattributes: -------------------------------------------------------------------------------- 1 | * text=auto eol=lf 2 | 3 | *.blade.php diff=html 4 | *.css diff=css 5 | *.html diff=html 6 | *.md diff=markdown 7 | *.php diff=php 8 | 9 | /.github export-ignore 10 | CHANGELOG.md export-ignore 11 | .styleci.yml export-ignore 12 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | /.phpunit.cache 2 | /node_modules 3 | /public/build 4 | /public/hot 5 | /public/storage 6 | /storage/*.key 7 | /vendor 8 | .env 9 | .env.backup 10 | .env.production 11 | .phpunit.result.cache 12 | Homestead.json 13 | Homestead.yaml 14 | auth.json 15 | npm-debug.log 16 | yarn-error.log 17 | /.fleet 18 | /.idea 19 | /.vscode 20 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2019 BigCommerce 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Laravel React App for BigCommerce 2 | 3 | This is a basic BigCommerce app with two screens, a catalog summary view and list of orders that can be cancelled, built using Laravel and React. 4 | 5 | It's meant to fast track your ability to take a concept for an app to something usable within the BigCommerce control panel. A live store can install this app while it runs locally. 6 | 7 | A walkthrough going over the steps taken to produce this app, along with the steps required to create the app in BigCommerce, can be read [here](https://medium.com/p/711ceceb5006). 8 | 9 | ## Prerequisites & Installation 10 | 11 | Before jumping in, you'll want to make sure you have the system requirements met: 12 | - PHP ([Installation Guide](https://www.php.net/manual/en/install.php)) 13 | - Composer ([Installation Guide](https://getcomposer.org/doc/00-intro.md)) 14 | - Laravel ([Installation Guide](https://laravel.com/docs/10.x)) 15 | 16 | To ease PHP development and enable the app you develop to be easily shared, you’ll want to use either Valet or Homestead, depending on your OS: 17 | 18 | - Local SSL Cert (Recommend Valet or Homestead to ease set up) 19 | - Mac OS: Valet ([Installation Guide](https://laravel.com/docs/10.x/valet)) 20 | - Windows / Linux: Homestead ([Installation Guide](https://laravel.com/docs/10.x/homestead)) 21 | 22 | We’ll be using Valet for some of the steps below, but the functionality to host and share sites is similar across both Valet and Homestead. What’s more important in this tutorial is how to configure Laravel to use React and connect with BigCommerce. 23 | 24 | To install PHP dependancies: 25 | 26 | ```bash 27 | composer install 28 | ``` 29 | And JS dependancies: 30 | ```bash 31 | npm install 32 | ``` 33 | To test on a BigCommerce store, you can create a free trial on bigcommerce.com or request a free sandbox store by [signing up to be a tech partner](https://www.bigcommerce.com/partners/). 34 | 35 | ## Getting Laravel and React Running Together 36 | 37 | This is where we will create a baseline for future development: a simple application that loads at a specific URL in your browser and loads a React component instead of the default Laravel screen. 38 | 39 | Create a new Laravel codebase 40 | You can either use the Laravel command that creates the initial boilerplate for an app in the ~/Sites directory or use Composer: 41 | 42 | ```bash 43 | composer global require laravel/installer 44 | laravel new laravel-react-bigcommerce-app 45 | ``` 46 | or 47 | ```bash 48 | composer create-project laravel/laravel laravel-react-bigcommerce-app 49 | ``` 50 | 51 | Visit the app address to make sure it’s live locally: 52 | 53 | After the command above completes, we wiil need to set up the directory so Valet can serve the app securely. 54 | ```bash 55 | cd laravel-react-bigcommerce-app 56 | valet link 57 | valet secure 58 | ``` 59 | You should now be able to visit the following URL in your browser and see the default Laravel welcome screen: 60 | 61 | https://laravel-react-bigcommerce-app.test 62 | 63 | ## Set up React as the JS framework using Breeze and Inertia 64 | Larvel Breeze provides a minimal and simple starting point for building a Laravel application, with authentication features. It's powered by Blade and Tailwind but can be configured as an SPA using Inertia. In this example we will power our frontend with react. 65 | 66 | First let's install Breeze: 67 | ```bash 68 | composer require laravel/breeze --dev 69 | ``` 70 | Then we install the react scaffolding: 71 | ```bash 72 | php artisan breeze:install react 73 | ``` 74 | Note: this last command executed `npm install` so no need to run this command. 75 | 76 | Lastly, get the application running: 77 | ```bash 78 | npm run dev 79 | ``` 80 | a `Local` url should now be available in your console to visit your react application 81 | 82 | http://127.0.0.1:5173/ 83 | 84 | ## References 85 | 86 | https://laravel.com/docs/10.x/installation#your-first-laravel-project 87 | https://laravel.com/docs/10.x/starter-kits 88 | 89 | ## Contributing 90 | Pull requests are welcome. For major changes, please open an issue first to discuss what you would like to change. 91 | 92 | ## License 93 | MIT 94 | -------------------------------------------------------------------------------- /app/Console/Kernel.php: -------------------------------------------------------------------------------- 1 | command('inspire')->hourly(); 16 | } 17 | 18 | /** 19 | * Register the commands for the application. 20 | */ 21 | protected function commands(): void 22 | { 23 | $this->load(__DIR__.'/Commands'); 24 | 25 | require base_path('routes/console.php'); 26 | } 27 | } 28 | -------------------------------------------------------------------------------- /app/Exceptions/Handler.php: -------------------------------------------------------------------------------- 1 | , \Psr\Log\LogLevel::*> 14 | */ 15 | protected $levels = [ 16 | // 17 | ]; 18 | 19 | /** 20 | * A list of the exception types that are not reported. 21 | * 22 | * @var array> 23 | */ 24 | protected $dontReport = [ 25 | // 26 | ]; 27 | 28 | /** 29 | * A list of the inputs that are never flashed to the session on validation exceptions. 30 | * 31 | * @var array 32 | */ 33 | protected $dontFlash = [ 34 | 'current_password', 35 | 'password', 36 | 'password_confirmation', 37 | ]; 38 | 39 | /** 40 | * Register the exception handling callbacks for the application. 41 | */ 42 | public function register(): void 43 | { 44 | $this->reportable(function (Throwable $e) { 45 | // 46 | }); 47 | } 48 | } 49 | -------------------------------------------------------------------------------- /app/Http/Controllers/Auth/AuthenticatedSessionController.php: -------------------------------------------------------------------------------- 1 | Route::has('password.request'), 24 | 'status' => session('status'), 25 | ]); 26 | } 27 | 28 | /** 29 | * Handle an incoming authentication request. 30 | */ 31 | public function store(LoginRequest $request): RedirectResponse 32 | { 33 | $request->authenticate(); 34 | 35 | $request->session()->regenerate(); 36 | 37 | return redirect()->intended(RouteServiceProvider::HOME); 38 | } 39 | 40 | /** 41 | * Destroy an authenticated session. 42 | */ 43 | public function destroy(Request $request): RedirectResponse 44 | { 45 | Auth::guard('web')->logout(); 46 | 47 | $request->session()->invalidate(); 48 | 49 | $request->session()->regenerateToken(); 50 | 51 | return redirect('/'); 52 | } 53 | } 54 | -------------------------------------------------------------------------------- /app/Http/Controllers/Auth/ConfirmablePasswordController.php: -------------------------------------------------------------------------------- 1 | validate([ 30 | 'email' => $request->user()->email, 31 | 'password' => $request->password, 32 | ])) { 33 | throw ValidationException::withMessages([ 34 | 'password' => __('auth.password'), 35 | ]); 36 | } 37 | 38 | $request->session()->put('auth.password_confirmed_at', time()); 39 | 40 | return redirect()->intended(RouteServiceProvider::HOME); 41 | } 42 | } 43 | -------------------------------------------------------------------------------- /app/Http/Controllers/Auth/EmailVerificationNotificationController.php: -------------------------------------------------------------------------------- 1 | user()->hasVerifiedEmail()) { 18 | return redirect()->intended(RouteServiceProvider::HOME); 19 | } 20 | 21 | $request->user()->sendEmailVerificationNotification(); 22 | 23 | return back()->with('status', 'verification-link-sent'); 24 | } 25 | } 26 | -------------------------------------------------------------------------------- /app/Http/Controllers/Auth/EmailVerificationPromptController.php: -------------------------------------------------------------------------------- 1 | user()->hasVerifiedEmail() 20 | ? redirect()->intended(RouteServiceProvider::HOME) 21 | : Inertia::render('Auth/VerifyEmail', ['status' => session('status')]); 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /app/Http/Controllers/Auth/NewPasswordController.php: -------------------------------------------------------------------------------- 1 | $request->email, 26 | 'token' => $request->route('token'), 27 | ]); 28 | } 29 | 30 | /** 31 | * Handle an incoming new password request. 32 | * 33 | * @throws \Illuminate\Validation\ValidationException 34 | */ 35 | public function store(Request $request): RedirectResponse 36 | { 37 | $request->validate([ 38 | 'token' => 'required', 39 | 'email' => 'required|email', 40 | 'password' => ['required', 'confirmed', Rules\Password::defaults()], 41 | ]); 42 | 43 | // Here we will attempt to reset the user's password. If it is successful we 44 | // will update the password on an actual user model and persist it to the 45 | // database. Otherwise we will parse the error and return the response. 46 | $status = Password::reset( 47 | $request->only('email', 'password', 'password_confirmation', 'token'), 48 | function ($user) use ($request) { 49 | $user->forceFill([ 50 | 'password' => Hash::make($request->password), 51 | 'remember_token' => Str::random(60), 52 | ])->save(); 53 | 54 | event(new PasswordReset($user)); 55 | } 56 | ); 57 | 58 | // If the password was successfully reset, we will redirect the user back to 59 | // the application's home authenticated view. If there is an error we can 60 | // redirect them back to where they came from with their error message. 61 | if ($status == Password::PASSWORD_RESET) { 62 | return redirect()->route('login')->with('status', __($status)); 63 | } 64 | 65 | throw ValidationException::withMessages([ 66 | 'email' => [trans($status)], 67 | ]); 68 | } 69 | } 70 | -------------------------------------------------------------------------------- /app/Http/Controllers/Auth/PasswordController.php: -------------------------------------------------------------------------------- 1 | validate([ 19 | 'current_password' => ['required', 'current_password'], 20 | 'password' => ['required', Password::defaults(), 'confirmed'], 21 | ]); 22 | 23 | $request->user()->update([ 24 | 'password' => Hash::make($validated['password']), 25 | ]); 26 | 27 | return back(); 28 | } 29 | } 30 | -------------------------------------------------------------------------------- /app/Http/Controllers/Auth/PasswordResetLinkController.php: -------------------------------------------------------------------------------- 1 | session('status'), 22 | ]); 23 | } 24 | 25 | /** 26 | * Handle an incoming password reset link request. 27 | * 28 | * @throws \Illuminate\Validation\ValidationException 29 | */ 30 | public function store(Request $request): RedirectResponse 31 | { 32 | $request->validate([ 33 | 'email' => 'required|email', 34 | ]); 35 | 36 | // We will send the password reset link to this user. Once we have attempted 37 | // to send the link, we will examine the response then see the message we 38 | // need to show to the user. Finally, we'll send out a proper response. 39 | $status = Password::sendResetLink( 40 | $request->only('email') 41 | ); 42 | 43 | if ($status == Password::RESET_LINK_SENT) { 44 | return back()->with('status', __($status)); 45 | } 46 | 47 | throw ValidationException::withMessages([ 48 | 'email' => [trans($status)], 49 | ]); 50 | } 51 | } 52 | -------------------------------------------------------------------------------- /app/Http/Controllers/Auth/RegisteredUserController.php: -------------------------------------------------------------------------------- 1 | validate([ 35 | 'name' => 'required|string|max:255', 36 | 'email' => 'required|string|email|max:255|unique:'.User::class, 37 | 'password' => ['required', 'confirmed', Rules\Password::defaults()], 38 | ]); 39 | 40 | $user = User::create([ 41 | 'name' => $request->name, 42 | 'email' => $request->email, 43 | 'password' => Hash::make($request->password), 44 | ]); 45 | 46 | event(new Registered($user)); 47 | 48 | Auth::login($user); 49 | 50 | return redirect(RouteServiceProvider::HOME); 51 | } 52 | } 53 | -------------------------------------------------------------------------------- /app/Http/Controllers/Auth/VerifyEmailController.php: -------------------------------------------------------------------------------- 1 | user()->hasVerifiedEmail()) { 19 | return redirect()->intended(RouteServiceProvider::HOME.'?verified=1'); 20 | } 21 | 22 | if ($request->user()->markEmailAsVerified()) { 23 | event(new Verified($request->user())); 24 | } 25 | 26 | return redirect()->intended(RouteServiceProvider::HOME.'?verified=1'); 27 | } 28 | } 29 | -------------------------------------------------------------------------------- /app/Http/Controllers/Controller.php: -------------------------------------------------------------------------------- 1 | $request->user() instanceof MustVerifyEmail, 23 | 'status' => session('status'), 24 | ]); 25 | } 26 | 27 | /** 28 | * Update the user's profile information. 29 | */ 30 | public function update(ProfileUpdateRequest $request): RedirectResponse 31 | { 32 | $request->user()->fill($request->validated()); 33 | 34 | if ($request->user()->isDirty('email')) { 35 | $request->user()->email_verified_at = null; 36 | } 37 | 38 | $request->user()->save(); 39 | 40 | return Redirect::route('profile.edit'); 41 | } 42 | 43 | /** 44 | * Delete the user's account. 45 | */ 46 | public function destroy(Request $request): RedirectResponse 47 | { 48 | $request->validate([ 49 | 'password' => ['required', 'current-password'], 50 | ]); 51 | 52 | $user = $request->user(); 53 | 54 | Auth::logout(); 55 | 56 | $user->delete(); 57 | 58 | $request->session()->invalidate(); 59 | $request->session()->regenerateToken(); 60 | 61 | return Redirect::to('/'); 62 | } 63 | } 64 | -------------------------------------------------------------------------------- /app/Http/Kernel.php: -------------------------------------------------------------------------------- 1 | 15 | */ 16 | protected $middleware = [ 17 | // \App\Http\Middleware\TrustHosts::class, 18 | \App\Http\Middleware\TrustProxies::class, 19 | \Illuminate\Http\Middleware\HandleCors::class, 20 | \App\Http\Middleware\PreventRequestsDuringMaintenance::class, 21 | \Illuminate\Foundation\Http\Middleware\ValidatePostSize::class, 22 | \App\Http\Middleware\TrimStrings::class, 23 | \Illuminate\Foundation\Http\Middleware\ConvertEmptyStringsToNull::class, 24 | ]; 25 | 26 | /** 27 | * The application's route middleware groups. 28 | * 29 | * @var array> 30 | */ 31 | protected $middlewareGroups = [ 32 | 'web' => [ 33 | \App\Http\Middleware\EncryptCookies::class, 34 | \Illuminate\Cookie\Middleware\AddQueuedCookiesToResponse::class, 35 | \Illuminate\Session\Middleware\StartSession::class, 36 | \Illuminate\View\Middleware\ShareErrorsFromSession::class, 37 | \App\Http\Middleware\VerifyCsrfToken::class, 38 | \Illuminate\Routing\Middleware\SubstituteBindings::class, 39 | \App\Http\Middleware\HandleInertiaRequests::class, 40 | \Illuminate\Http\Middleware\AddLinkHeadersForPreloadedAssets::class, 41 | ], 42 | 43 | 'api' => [ 44 | // \Laravel\Sanctum\Http\Middleware\EnsureFrontendRequestsAreStateful::class, 45 | \Illuminate\Routing\Middleware\ThrottleRequests::class.':api', 46 | \Illuminate\Routing\Middleware\SubstituteBindings::class, 47 | ], 48 | ]; 49 | 50 | /** 51 | * The application's middleware aliases. 52 | * 53 | * Aliases may be used to conveniently assign middleware to routes and groups. 54 | * 55 | * @var array 56 | */ 57 | protected $middlewareAliases = [ 58 | 'auth' => \App\Http\Middleware\Authenticate::class, 59 | 'auth.basic' => \Illuminate\Auth\Middleware\AuthenticateWithBasicAuth::class, 60 | 'auth.session' => \Illuminate\Session\Middleware\AuthenticateSession::class, 61 | 'cache.headers' => \Illuminate\Http\Middleware\SetCacheHeaders::class, 62 | 'can' => \Illuminate\Auth\Middleware\Authorize::class, 63 | 'guest' => \App\Http\Middleware\RedirectIfAuthenticated::class, 64 | 'password.confirm' => \Illuminate\Auth\Middleware\RequirePassword::class, 65 | 'signed' => \App\Http\Middleware\ValidateSignature::class, 66 | 'throttle' => \Illuminate\Routing\Middleware\ThrottleRequests::class, 67 | 'verified' => \Illuminate\Auth\Middleware\EnsureEmailIsVerified::class, 68 | ]; 69 | } 70 | -------------------------------------------------------------------------------- /app/Http/Middleware/Authenticate.php: -------------------------------------------------------------------------------- 1 | expectsJson() ? null : route('login'); 16 | } 17 | } 18 | -------------------------------------------------------------------------------- /app/Http/Middleware/EncryptCookies.php: -------------------------------------------------------------------------------- 1 | 13 | */ 14 | protected $except = [ 15 | // 16 | ]; 17 | } 18 | -------------------------------------------------------------------------------- /app/Http/Middleware/HandleInertiaRequests.php: -------------------------------------------------------------------------------- 1 | 30 | */ 31 | public function share(Request $request): array 32 | { 33 | return array_merge(parent::share($request), [ 34 | 'auth' => [ 35 | 'user' => $request->user(), 36 | ], 37 | 'ziggy' => function () use ($request) { 38 | return array_merge((new Ziggy)->toArray(), [ 39 | 'location' => $request->url(), 40 | ]); 41 | }, 42 | ]); 43 | } 44 | } 45 | -------------------------------------------------------------------------------- /app/Http/Middleware/PreventRequestsDuringMaintenance.php: -------------------------------------------------------------------------------- 1 | 13 | */ 14 | protected $except = [ 15 | // 16 | ]; 17 | } 18 | -------------------------------------------------------------------------------- /app/Http/Middleware/RedirectIfAuthenticated.php: -------------------------------------------------------------------------------- 1 | check()) { 24 | return redirect(RouteServiceProvider::HOME); 25 | } 26 | } 27 | 28 | return $next($request); 29 | } 30 | } 31 | -------------------------------------------------------------------------------- /app/Http/Middleware/TrimStrings.php: -------------------------------------------------------------------------------- 1 | 13 | */ 14 | protected $except = [ 15 | 'current_password', 16 | 'password', 17 | 'password_confirmation', 18 | ]; 19 | } 20 | -------------------------------------------------------------------------------- /app/Http/Middleware/TrustHosts.php: -------------------------------------------------------------------------------- 1 | 13 | */ 14 | public function hosts(): array 15 | { 16 | return [ 17 | $this->allSubdomainsOfApplicationUrl(), 18 | ]; 19 | } 20 | } 21 | -------------------------------------------------------------------------------- /app/Http/Middleware/TrustProxies.php: -------------------------------------------------------------------------------- 1 | |string|null 14 | */ 15 | protected $proxies; 16 | 17 | /** 18 | * The headers that should be used to detect proxies. 19 | * 20 | * @var int 21 | */ 22 | protected $headers = 23 | Request::HEADER_X_FORWARDED_FOR | 24 | Request::HEADER_X_FORWARDED_HOST | 25 | Request::HEADER_X_FORWARDED_PORT | 26 | Request::HEADER_X_FORWARDED_PROTO | 27 | Request::HEADER_X_FORWARDED_AWS_ELB; 28 | } 29 | -------------------------------------------------------------------------------- /app/Http/Middleware/ValidateSignature.php: -------------------------------------------------------------------------------- 1 | 13 | */ 14 | protected $except = [ 15 | // 'fbclid', 16 | // 'utm_campaign', 17 | // 'utm_content', 18 | // 'utm_medium', 19 | // 'utm_source', 20 | // 'utm_term', 21 | ]; 22 | } 23 | -------------------------------------------------------------------------------- /app/Http/Middleware/VerifyCsrfToken.php: -------------------------------------------------------------------------------- 1 | 13 | */ 14 | protected $except = [ 15 | // 16 | ]; 17 | } 18 | -------------------------------------------------------------------------------- /app/Http/Requests/Auth/LoginRequest.php: -------------------------------------------------------------------------------- 1 | 26 | */ 27 | public function rules(): array 28 | { 29 | return [ 30 | 'email' => ['required', 'string', 'email'], 31 | 'password' => ['required', 'string'], 32 | ]; 33 | } 34 | 35 | /** 36 | * Attempt to authenticate the request's credentials. 37 | * 38 | * @throws \Illuminate\Validation\ValidationException 39 | */ 40 | public function authenticate(): void 41 | { 42 | $this->ensureIsNotRateLimited(); 43 | 44 | if (! Auth::attempt($this->only('email', 'password'), $this->boolean('remember'))) { 45 | RateLimiter::hit($this->throttleKey()); 46 | 47 | throw ValidationException::withMessages([ 48 | 'email' => trans('auth.failed'), 49 | ]); 50 | } 51 | 52 | RateLimiter::clear($this->throttleKey()); 53 | } 54 | 55 | /** 56 | * Ensure the login request is not rate limited. 57 | * 58 | * @throws \Illuminate\Validation\ValidationException 59 | */ 60 | public function ensureIsNotRateLimited(): void 61 | { 62 | if (! RateLimiter::tooManyAttempts($this->throttleKey(), 5)) { 63 | return; 64 | } 65 | 66 | event(new Lockout($this)); 67 | 68 | $seconds = RateLimiter::availableIn($this->throttleKey()); 69 | 70 | throw ValidationException::withMessages([ 71 | 'email' => trans('auth.throttle', [ 72 | 'seconds' => $seconds, 73 | 'minutes' => ceil($seconds / 60), 74 | ]), 75 | ]); 76 | } 77 | 78 | /** 79 | * Get the rate limiting throttle key for the request. 80 | */ 81 | public function throttleKey(): string 82 | { 83 | return Str::transliterate(Str::lower($this->input('email')).'|'.$this->ip()); 84 | } 85 | } 86 | -------------------------------------------------------------------------------- /app/Http/Requests/ProfileUpdateRequest.php: -------------------------------------------------------------------------------- 1 | 15 | */ 16 | public function rules(): array 17 | { 18 | return [ 19 | 'name' => ['string', 'max:255'], 20 | 'email' => ['email', 'max:255', Rule::unique(User::class)->ignore($this->user()->id)], 21 | ]; 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /app/Models/User.php: -------------------------------------------------------------------------------- 1 | 19 | */ 20 | protected $fillable = [ 21 | 'name', 22 | 'email', 23 | 'password', 24 | ]; 25 | 26 | /** 27 | * The attributes that should be hidden for serialization. 28 | * 29 | * @var array 30 | */ 31 | protected $hidden = [ 32 | 'password', 33 | 'remember_token', 34 | ]; 35 | 36 | /** 37 | * The attributes that should be cast. 38 | * 39 | * @var array 40 | */ 41 | protected $casts = [ 42 | 'email_verified_at' => 'datetime', 43 | ]; 44 | } 45 | -------------------------------------------------------------------------------- /app/Providers/AppServiceProvider.php: -------------------------------------------------------------------------------- 1 | 14 | */ 15 | protected $policies = [ 16 | // 'App\Models\Model' => 'App\Policies\ModelPolicy', 17 | ]; 18 | 19 | /** 20 | * Register any authentication / authorization services. 21 | */ 22 | public function boot(): void 23 | { 24 | // 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /app/Providers/BroadcastServiceProvider.php: -------------------------------------------------------------------------------- 1 | > 16 | */ 17 | protected $listen = [ 18 | Registered::class => [ 19 | SendEmailVerificationNotification::class, 20 | ], 21 | ]; 22 | 23 | /** 24 | * Register any events for your application. 25 | */ 26 | public function boot(): void 27 | { 28 | // 29 | } 30 | 31 | /** 32 | * Determine if events and listeners should be automatically discovered. 33 | */ 34 | public function shouldDiscoverEvents(): bool 35 | { 36 | return false; 37 | } 38 | } 39 | -------------------------------------------------------------------------------- /app/Providers/RouteServiceProvider.php: -------------------------------------------------------------------------------- 1 | configureRateLimiting(); 28 | 29 | $this->routes(function () { 30 | Route::middleware('api') 31 | ->prefix('api') 32 | ->group(base_path('routes/api.php')); 33 | 34 | Route::middleware('web') 35 | ->group(base_path('routes/web.php')); 36 | }); 37 | } 38 | 39 | /** 40 | * Configure the rate limiters for the application. 41 | */ 42 | protected function configureRateLimiting(): void 43 | { 44 | RateLimiter::for('api', function (Request $request) { 45 | return Limit::perMinute(60)->by($request->user()?->id ?: $request->ip()); 46 | }); 47 | } 48 | } 49 | -------------------------------------------------------------------------------- /artisan: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env php 2 | make(Illuminate\Contracts\Console\Kernel::class); 34 | 35 | $status = $kernel->handle( 36 | $input = new Symfony\Component\Console\Input\ArgvInput, 37 | new Symfony\Component\Console\Output\ConsoleOutput 38 | ); 39 | 40 | /* 41 | |-------------------------------------------------------------------------- 42 | | Shutdown The Application 43 | |-------------------------------------------------------------------------- 44 | | 45 | | Once Artisan has finished running, we will fire off the shutdown events 46 | | so that any final work may be done by the application before we shut 47 | | down the process. This is the last thing to happen to the request. 48 | | 49 | */ 50 | 51 | $kernel->terminate($input, $status); 52 | 53 | exit($status); 54 | -------------------------------------------------------------------------------- /bootstrap/app.php: -------------------------------------------------------------------------------- 1 | singleton( 30 | Illuminate\Contracts\Http\Kernel::class, 31 | App\Http\Kernel::class 32 | ); 33 | 34 | $app->singleton( 35 | Illuminate\Contracts\Console\Kernel::class, 36 | App\Console\Kernel::class 37 | ); 38 | 39 | $app->singleton( 40 | Illuminate\Contracts\Debug\ExceptionHandler::class, 41 | App\Exceptions\Handler::class 42 | ); 43 | 44 | /* 45 | |-------------------------------------------------------------------------- 46 | | Return The Application 47 | |-------------------------------------------------------------------------- 48 | | 49 | | This script returns the application instance. The instance is given to 50 | | the calling script so we can separate the building of the instances 51 | | from the actual running of the application and sending responses. 52 | | 53 | */ 54 | 55 | return $app; 56 | -------------------------------------------------------------------------------- /bootstrap/cache/.gitignore: -------------------------------------------------------------------------------- 1 | * 2 | !.gitignore 3 | -------------------------------------------------------------------------------- /composer.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "laravel/laravel", 3 | "type": "project", 4 | "description": "The Laravel Framework.", 5 | "keywords": ["framework", "laravel"], 6 | "license": "MIT", 7 | "require": { 8 | "php": "^8.1", 9 | "guzzlehttp/guzzle": "^7.5", 10 | "inertiajs/inertia-laravel": "^0.6.3", 11 | "laravel/framework": "^10.0", 12 | "laravel/sanctum": "^3.2", 13 | "laravel/tinker": "^2.8", 14 | "tightenco/ziggy": "^1.0" 15 | }, 16 | "require-dev": { 17 | "fakerphp/faker": "^1.9.1", 18 | "laravel/breeze": "^1.19", 19 | "laravel/pint": "^1.0", 20 | "laravel/sail": "^1.18", 21 | "mockery/mockery": "^1.4.4", 22 | "nunomaduro/collision": "^7.0", 23 | "phpunit/phpunit": "^10.0", 24 | "spatie/laravel-ignition": "^2.0" 25 | }, 26 | "autoload": { 27 | "psr-4": { 28 | "App\\": "app/", 29 | "Database\\Factories\\": "database/factories/", 30 | "Database\\Seeders\\": "database/seeders/" 31 | } 32 | }, 33 | "autoload-dev": { 34 | "psr-4": { 35 | "Tests\\": "tests/" 36 | } 37 | }, 38 | "scripts": { 39 | "post-autoload-dump": [ 40 | "Illuminate\\Foundation\\ComposerScripts::postAutoloadDump", 41 | "@php artisan package:discover --ansi" 42 | ], 43 | "post-update-cmd": [ 44 | "@php artisan vendor:publish --tag=laravel-assets --ansi --force" 45 | ], 46 | "post-root-package-install": [ 47 | "@php -r \"file_exists('.env') || copy('.env.example', '.env');\"" 48 | ], 49 | "post-create-project-cmd": [ 50 | "@php artisan key:generate --ansi" 51 | ] 52 | }, 53 | "extra": { 54 | "laravel": { 55 | "dont-discover": [] 56 | } 57 | }, 58 | "config": { 59 | "optimize-autoloader": true, 60 | "preferred-install": "dist", 61 | "sort-packages": true, 62 | "allow-plugins": { 63 | "pestphp/pest-plugin": true, 64 | "php-http/discovery": true 65 | } 66 | }, 67 | "minimum-stability": "stable", 68 | "prefer-stable": true 69 | } 70 | -------------------------------------------------------------------------------- /config/auth.php: -------------------------------------------------------------------------------- 1 | [ 17 | 'guard' => 'web', 18 | 'passwords' => 'users', 19 | ], 20 | 21 | /* 22 | |-------------------------------------------------------------------------- 23 | | Authentication Guards 24 | |-------------------------------------------------------------------------- 25 | | 26 | | Next, you may define every authentication guard for your application. 27 | | Of course, a great default configuration has been defined for you 28 | | here which uses session storage and the Eloquent user provider. 29 | | 30 | | All authentication drivers have a user provider. This defines how the 31 | | users are actually retrieved out of your database or other storage 32 | | mechanisms used by this application to persist your user's data. 33 | | 34 | | Supported: "session" 35 | | 36 | */ 37 | 38 | 'guards' => [ 39 | 'web' => [ 40 | 'driver' => 'session', 41 | 'provider' => 'users', 42 | ], 43 | ], 44 | 45 | /* 46 | |-------------------------------------------------------------------------- 47 | | User Providers 48 | |-------------------------------------------------------------------------- 49 | | 50 | | All authentication drivers have a user provider. This defines how the 51 | | users are actually retrieved out of your database or other storage 52 | | mechanisms used by this application to persist your user's data. 53 | | 54 | | If you have multiple user tables or models you may configure multiple 55 | | sources which represent each model / table. These sources may then 56 | | be assigned to any extra authentication guards you have defined. 57 | | 58 | | Supported: "database", "eloquent" 59 | | 60 | */ 61 | 62 | 'providers' => [ 63 | 'users' => [ 64 | 'driver' => 'eloquent', 65 | 'model' => App\Models\User::class, 66 | ], 67 | 68 | // 'users' => [ 69 | // 'driver' => 'database', 70 | // 'table' => 'users', 71 | // ], 72 | ], 73 | 74 | /* 75 | |-------------------------------------------------------------------------- 76 | | Resetting Passwords 77 | |-------------------------------------------------------------------------- 78 | | 79 | | You may specify multiple password reset configurations if you have more 80 | | than one user table or model in the application and you want to have 81 | | separate password reset settings based on the specific user types. 82 | | 83 | | The expire time is the number of minutes that each reset token will be 84 | | considered valid. This security feature keeps tokens short-lived so 85 | | they have less time to be guessed. You may change this as needed. 86 | | 87 | | The throttle setting is the number of seconds a user must wait before 88 | | generating more password reset tokens. This prevents the user from 89 | | quickly generating a very large amount of password reset tokens. 90 | | 91 | */ 92 | 93 | 'passwords' => [ 94 | 'users' => [ 95 | 'provider' => 'users', 96 | 'table' => 'password_reset_tokens', 97 | 'expire' => 60, 98 | 'throttle' => 60, 99 | ], 100 | ], 101 | 102 | /* 103 | |-------------------------------------------------------------------------- 104 | | Password Confirmation Timeout 105 | |-------------------------------------------------------------------------- 106 | | 107 | | Here you may define the amount of seconds before a password confirmation 108 | | times out and the user is prompted to re-enter their password via the 109 | | confirmation screen. By default, the timeout lasts for three hours. 110 | | 111 | */ 112 | 113 | 'password_timeout' => 10800, 114 | 115 | ]; 116 | -------------------------------------------------------------------------------- /config/broadcasting.php: -------------------------------------------------------------------------------- 1 | env('BROADCAST_DRIVER', 'null'), 19 | 20 | /* 21 | |-------------------------------------------------------------------------- 22 | | Broadcast Connections 23 | |-------------------------------------------------------------------------- 24 | | 25 | | Here you may define all of the broadcast connections that will be used 26 | | to broadcast events to other systems or over websockets. Samples of 27 | | each available type of connection are provided inside this array. 28 | | 29 | */ 30 | 31 | 'connections' => [ 32 | 33 | 'pusher' => [ 34 | 'driver' => 'pusher', 35 | 'key' => env('PUSHER_APP_KEY'), 36 | 'secret' => env('PUSHER_APP_SECRET'), 37 | 'app_id' => env('PUSHER_APP_ID'), 38 | 'options' => [ 39 | 'host' => env('PUSHER_HOST') ?: 'api-'.env('PUSHER_APP_CLUSTER', 'mt1').'.pusher.com', 40 | 'port' => env('PUSHER_PORT', 443), 41 | 'scheme' => env('PUSHER_SCHEME', 'https'), 42 | 'encrypted' => true, 43 | 'useTLS' => env('PUSHER_SCHEME', 'https') === 'https', 44 | ], 45 | 'client_options' => [ 46 | // Guzzle client options: https://docs.guzzlephp.org/en/stable/request-options.html 47 | ], 48 | ], 49 | 50 | 'ably' => [ 51 | 'driver' => 'ably', 52 | 'key' => env('ABLY_KEY'), 53 | ], 54 | 55 | 'redis' => [ 56 | 'driver' => 'redis', 57 | 'connection' => 'default', 58 | ], 59 | 60 | 'log' => [ 61 | 'driver' => 'log', 62 | ], 63 | 64 | 'null' => [ 65 | 'driver' => 'null', 66 | ], 67 | 68 | ], 69 | 70 | ]; 71 | -------------------------------------------------------------------------------- /config/cache.php: -------------------------------------------------------------------------------- 1 | env('CACHE_DRIVER', 'file'), 19 | 20 | /* 21 | |-------------------------------------------------------------------------- 22 | | Cache Stores 23 | |-------------------------------------------------------------------------- 24 | | 25 | | Here you may define all of the cache "stores" for your application as 26 | | well as their drivers. You may even define multiple stores for the 27 | | same cache driver to group types of items stored in your caches. 28 | | 29 | | Supported drivers: "apc", "array", "database", "file", 30 | | "memcached", "redis", "dynamodb", "octane", "null" 31 | | 32 | */ 33 | 34 | 'stores' => [ 35 | 36 | 'apc' => [ 37 | 'driver' => 'apc', 38 | ], 39 | 40 | 'array' => [ 41 | 'driver' => 'array', 42 | 'serialize' => false, 43 | ], 44 | 45 | 'database' => [ 46 | 'driver' => 'database', 47 | 'table' => 'cache', 48 | 'connection' => null, 49 | 'lock_connection' => null, 50 | ], 51 | 52 | 'file' => [ 53 | 'driver' => 'file', 54 | 'path' => storage_path('framework/cache/data'), 55 | ], 56 | 57 | 'memcached' => [ 58 | 'driver' => 'memcached', 59 | 'persistent_id' => env('MEMCACHED_PERSISTENT_ID'), 60 | 'sasl' => [ 61 | env('MEMCACHED_USERNAME'), 62 | env('MEMCACHED_PASSWORD'), 63 | ], 64 | 'options' => [ 65 | // Memcached::OPT_CONNECT_TIMEOUT => 2000, 66 | ], 67 | 'servers' => [ 68 | [ 69 | 'host' => env('MEMCACHED_HOST', '127.0.0.1'), 70 | 'port' => env('MEMCACHED_PORT', 11211), 71 | 'weight' => 100, 72 | ], 73 | ], 74 | ], 75 | 76 | 'redis' => [ 77 | 'driver' => 'redis', 78 | 'connection' => 'cache', 79 | 'lock_connection' => 'default', 80 | ], 81 | 82 | 'dynamodb' => [ 83 | 'driver' => 'dynamodb', 84 | 'key' => env('AWS_ACCESS_KEY_ID'), 85 | 'secret' => env('AWS_SECRET_ACCESS_KEY'), 86 | 'region' => env('AWS_DEFAULT_REGION', 'us-east-1'), 87 | 'table' => env('DYNAMODB_CACHE_TABLE', 'cache'), 88 | 'endpoint' => env('DYNAMODB_ENDPOINT'), 89 | ], 90 | 91 | 'octane' => [ 92 | 'driver' => 'octane', 93 | ], 94 | 95 | ], 96 | 97 | /* 98 | |-------------------------------------------------------------------------- 99 | | Cache Key Prefix 100 | |-------------------------------------------------------------------------- 101 | | 102 | | When utilizing the APC, database, memcached, Redis, or DynamoDB cache 103 | | stores there might be other applications using the same cache. For 104 | | that reason, you may prefix every cache key to avoid collisions. 105 | | 106 | */ 107 | 108 | 'prefix' => env('CACHE_PREFIX', Str::slug(env('APP_NAME', 'laravel'), '_').'_cache_'), 109 | 110 | ]; 111 | -------------------------------------------------------------------------------- /config/cors.php: -------------------------------------------------------------------------------- 1 | ['api/*', 'sanctum/csrf-cookie'], 19 | 20 | 'allowed_methods' => ['*'], 21 | 22 | 'allowed_origins' => ['*'], 23 | 24 | 'allowed_origins_patterns' => [], 25 | 26 | 'allowed_headers' => ['*'], 27 | 28 | 'exposed_headers' => [], 29 | 30 | 'max_age' => 0, 31 | 32 | 'supports_credentials' => false, 33 | 34 | ]; 35 | -------------------------------------------------------------------------------- /config/database.php: -------------------------------------------------------------------------------- 1 | env('DB_CONNECTION', 'mysql'), 19 | 20 | /* 21 | |-------------------------------------------------------------------------- 22 | | Database Connections 23 | |-------------------------------------------------------------------------- 24 | | 25 | | Here are each of the database connections setup for your application. 26 | | Of course, examples of configuring each database platform that is 27 | | supported by Laravel is shown below to make development simple. 28 | | 29 | | 30 | | All database work in Laravel is done through the PHP PDO facilities 31 | | so make sure you have the driver for your particular database of 32 | | choice installed on your machine before you begin development. 33 | | 34 | */ 35 | 36 | 'connections' => [ 37 | 38 | 'sqlite' => [ 39 | 'driver' => 'sqlite', 40 | 'url' => env('DATABASE_URL'), 41 | 'database' => env('DB_DATABASE', database_path('database.sqlite')), 42 | 'prefix' => '', 43 | 'foreign_key_constraints' => env('DB_FOREIGN_KEYS', true), 44 | ], 45 | 46 | 'mysql' => [ 47 | 'driver' => 'mysql', 48 | 'url' => env('DATABASE_URL'), 49 | 'host' => env('DB_HOST', '127.0.0.1'), 50 | 'port' => env('DB_PORT', '3306'), 51 | 'database' => env('DB_DATABASE', 'forge'), 52 | 'username' => env('DB_USERNAME', 'forge'), 53 | 'password' => env('DB_PASSWORD', ''), 54 | 'unix_socket' => env('DB_SOCKET', ''), 55 | 'charset' => 'utf8mb4', 56 | 'collation' => 'utf8mb4_unicode_ci', 57 | 'prefix' => '', 58 | 'prefix_indexes' => true, 59 | 'strict' => true, 60 | 'engine' => null, 61 | 'options' => extension_loaded('pdo_mysql') ? array_filter([ 62 | PDO::MYSQL_ATTR_SSL_CA => env('MYSQL_ATTR_SSL_CA'), 63 | ]) : [], 64 | ], 65 | 66 | 'pgsql' => [ 67 | 'driver' => 'pgsql', 68 | 'url' => env('DATABASE_URL'), 69 | 'host' => env('DB_HOST', '127.0.0.1'), 70 | 'port' => env('DB_PORT', '5432'), 71 | 'database' => env('DB_DATABASE', 'forge'), 72 | 'username' => env('DB_USERNAME', 'forge'), 73 | 'password' => env('DB_PASSWORD', ''), 74 | 'charset' => 'utf8', 75 | 'prefix' => '', 76 | 'prefix_indexes' => true, 77 | 'search_path' => 'public', 78 | 'sslmode' => 'prefer', 79 | ], 80 | 81 | 'sqlsrv' => [ 82 | 'driver' => 'sqlsrv', 83 | 'url' => env('DATABASE_URL'), 84 | 'host' => env('DB_HOST', 'localhost'), 85 | 'port' => env('DB_PORT', '1433'), 86 | 'database' => env('DB_DATABASE', 'forge'), 87 | 'username' => env('DB_USERNAME', 'forge'), 88 | 'password' => env('DB_PASSWORD', ''), 89 | 'charset' => 'utf8', 90 | 'prefix' => '', 91 | 'prefix_indexes' => true, 92 | // 'encrypt' => env('DB_ENCRYPT', 'yes'), 93 | // 'trust_server_certificate' => env('DB_TRUST_SERVER_CERTIFICATE', 'false'), 94 | ], 95 | 96 | ], 97 | 98 | /* 99 | |-------------------------------------------------------------------------- 100 | | Migration Repository Table 101 | |-------------------------------------------------------------------------- 102 | | 103 | | This table keeps track of all the migrations that have already run for 104 | | your application. Using this information, we can determine which of 105 | | the migrations on disk haven't actually been run in the database. 106 | | 107 | */ 108 | 109 | 'migrations' => 'migrations', 110 | 111 | /* 112 | |-------------------------------------------------------------------------- 113 | | Redis Databases 114 | |-------------------------------------------------------------------------- 115 | | 116 | | Redis is an open source, fast, and advanced key-value store that also 117 | | provides a richer body of commands than a typical key-value system 118 | | such as APC or Memcached. Laravel makes it easy to dig right in. 119 | | 120 | */ 121 | 122 | 'redis' => [ 123 | 124 | 'client' => env('REDIS_CLIENT', 'phpredis'), 125 | 126 | 'options' => [ 127 | 'cluster' => env('REDIS_CLUSTER', 'redis'), 128 | 'prefix' => env('REDIS_PREFIX', Str::slug(env('APP_NAME', 'laravel'), '_').'_database_'), 129 | ], 130 | 131 | 'default' => [ 132 | 'url' => env('REDIS_URL'), 133 | 'host' => env('REDIS_HOST', '127.0.0.1'), 134 | 'username' => env('REDIS_USERNAME'), 135 | 'password' => env('REDIS_PASSWORD'), 136 | 'port' => env('REDIS_PORT', '6379'), 137 | 'database' => env('REDIS_DB', '0'), 138 | ], 139 | 140 | 'cache' => [ 141 | 'url' => env('REDIS_URL'), 142 | 'host' => env('REDIS_HOST', '127.0.0.1'), 143 | 'username' => env('REDIS_USERNAME'), 144 | 'password' => env('REDIS_PASSWORD'), 145 | 'port' => env('REDIS_PORT', '6379'), 146 | 'database' => env('REDIS_CACHE_DB', '1'), 147 | ], 148 | 149 | ], 150 | 151 | ]; 152 | -------------------------------------------------------------------------------- /config/filesystems.php: -------------------------------------------------------------------------------- 1 | env('FILESYSTEM_DISK', 'local'), 17 | 18 | /* 19 | |-------------------------------------------------------------------------- 20 | | Filesystem Disks 21 | |-------------------------------------------------------------------------- 22 | | 23 | | Here you may configure as many filesystem "disks" as you wish, and you 24 | | may even configure multiple disks of the same driver. Defaults have 25 | | been set up for each driver as an example of the required values. 26 | | 27 | | Supported Drivers: "local", "ftp", "sftp", "s3" 28 | | 29 | */ 30 | 31 | 'disks' => [ 32 | 33 | 'local' => [ 34 | 'driver' => 'local', 35 | 'root' => storage_path('app'), 36 | 'throw' => false, 37 | ], 38 | 39 | 'public' => [ 40 | 'driver' => 'local', 41 | 'root' => storage_path('app/public'), 42 | 'url' => env('APP_URL').'/storage', 43 | 'visibility' => 'public', 44 | 'throw' => false, 45 | ], 46 | 47 | 's3' => [ 48 | 'driver' => 's3', 49 | 'key' => env('AWS_ACCESS_KEY_ID'), 50 | 'secret' => env('AWS_SECRET_ACCESS_KEY'), 51 | 'region' => env('AWS_DEFAULT_REGION'), 52 | 'bucket' => env('AWS_BUCKET'), 53 | 'url' => env('AWS_URL'), 54 | 'endpoint' => env('AWS_ENDPOINT'), 55 | 'use_path_style_endpoint' => env('AWS_USE_PATH_STYLE_ENDPOINT', false), 56 | 'throw' => false, 57 | ], 58 | 59 | ], 60 | 61 | /* 62 | |-------------------------------------------------------------------------- 63 | | Symbolic Links 64 | |-------------------------------------------------------------------------- 65 | | 66 | | Here you may configure the symbolic links that will be created when the 67 | | `storage:link` Artisan command is executed. The array keys should be 68 | | the locations of the links and the values should be their targets. 69 | | 70 | */ 71 | 72 | 'links' => [ 73 | public_path('storage') => storage_path('app/public'), 74 | ], 75 | 76 | ]; 77 | -------------------------------------------------------------------------------- /config/hashing.php: -------------------------------------------------------------------------------- 1 | 'bcrypt', 19 | 20 | /* 21 | |-------------------------------------------------------------------------- 22 | | Bcrypt Options 23 | |-------------------------------------------------------------------------- 24 | | 25 | | Here you may specify the configuration options that should be used when 26 | | passwords are hashed using the Bcrypt algorithm. This will allow you 27 | | to control the amount of time it takes to hash the given password. 28 | | 29 | */ 30 | 31 | 'bcrypt' => [ 32 | 'rounds' => env('BCRYPT_ROUNDS', 10), 33 | ], 34 | 35 | /* 36 | |-------------------------------------------------------------------------- 37 | | Argon Options 38 | |-------------------------------------------------------------------------- 39 | | 40 | | Here you may specify the configuration options that should be used when 41 | | passwords are hashed using the Argon algorithm. These will allow you 42 | | to control the amount of time it takes to hash the given password. 43 | | 44 | */ 45 | 46 | 'argon' => [ 47 | 'memory' => 65536, 48 | 'threads' => 1, 49 | 'time' => 4, 50 | ], 51 | 52 | ]; 53 | -------------------------------------------------------------------------------- /config/logging.php: -------------------------------------------------------------------------------- 1 | env('LOG_CHANNEL', 'stack'), 21 | 22 | /* 23 | |-------------------------------------------------------------------------- 24 | | Deprecations Log Channel 25 | |-------------------------------------------------------------------------- 26 | | 27 | | This option controls the log channel that should be used to log warnings 28 | | regarding deprecated PHP and library features. This allows you to get 29 | | your application ready for upcoming major versions of dependencies. 30 | | 31 | */ 32 | 33 | 'deprecations' => [ 34 | 'channel' => env('LOG_DEPRECATIONS_CHANNEL', 'null'), 35 | 'trace' => false, 36 | ], 37 | 38 | /* 39 | |-------------------------------------------------------------------------- 40 | | Log Channels 41 | |-------------------------------------------------------------------------- 42 | | 43 | | Here you may configure the log channels for your application. Out of 44 | | the box, Laravel uses the Monolog PHP logging library. This gives 45 | | you a variety of powerful log handlers / formatters to utilize. 46 | | 47 | | Available Drivers: "single", "daily", "slack", "syslog", 48 | | "errorlog", "monolog", 49 | | "custom", "stack" 50 | | 51 | */ 52 | 53 | 'channels' => [ 54 | 'stack' => [ 55 | 'driver' => 'stack', 56 | 'channels' => ['single'], 57 | 'ignore_exceptions' => false, 58 | ], 59 | 60 | 'single' => [ 61 | 'driver' => 'single', 62 | 'path' => storage_path('logs/laravel.log'), 63 | 'level' => env('LOG_LEVEL', 'debug'), 64 | ], 65 | 66 | 'daily' => [ 67 | 'driver' => 'daily', 68 | 'path' => storage_path('logs/laravel.log'), 69 | 'level' => env('LOG_LEVEL', 'debug'), 70 | 'days' => 14, 71 | ], 72 | 73 | 'slack' => [ 74 | 'driver' => 'slack', 75 | 'url' => env('LOG_SLACK_WEBHOOK_URL'), 76 | 'username' => 'Laravel Log', 77 | 'emoji' => ':boom:', 78 | 'level' => env('LOG_LEVEL', 'critical'), 79 | ], 80 | 81 | 'papertrail' => [ 82 | 'driver' => 'monolog', 83 | 'level' => env('LOG_LEVEL', 'debug'), 84 | 'handler' => env('LOG_PAPERTRAIL_HANDLER', SyslogUdpHandler::class), 85 | 'handler_with' => [ 86 | 'host' => env('PAPERTRAIL_URL'), 87 | 'port' => env('PAPERTRAIL_PORT'), 88 | 'connectionString' => 'tls://'.env('PAPERTRAIL_URL').':'.env('PAPERTRAIL_PORT'), 89 | ], 90 | ], 91 | 92 | 'stderr' => [ 93 | 'driver' => 'monolog', 94 | 'level' => env('LOG_LEVEL', 'debug'), 95 | 'handler' => StreamHandler::class, 96 | 'formatter' => env('LOG_STDERR_FORMATTER'), 97 | 'with' => [ 98 | 'stream' => 'php://stderr', 99 | ], 100 | ], 101 | 102 | 'syslog' => [ 103 | 'driver' => 'syslog', 104 | 'level' => env('LOG_LEVEL', 'debug'), 105 | ], 106 | 107 | 'errorlog' => [ 108 | 'driver' => 'errorlog', 109 | 'level' => env('LOG_LEVEL', 'debug'), 110 | ], 111 | 112 | 'null' => [ 113 | 'driver' => 'monolog', 114 | 'handler' => NullHandler::class, 115 | ], 116 | 117 | 'emergency' => [ 118 | 'path' => storage_path('logs/laravel.log'), 119 | ], 120 | ], 121 | 122 | ]; 123 | -------------------------------------------------------------------------------- /config/mail.php: -------------------------------------------------------------------------------- 1 | env('MAIL_MAILER', 'smtp'), 17 | 18 | /* 19 | |-------------------------------------------------------------------------- 20 | | Mailer Configurations 21 | |-------------------------------------------------------------------------- 22 | | 23 | | Here you may configure all of the mailers used by your application plus 24 | | their respective settings. Several examples have been configured for 25 | | you and you are free to add your own as your application requires. 26 | | 27 | | Laravel supports a variety of mail "transport" drivers to be used while 28 | | sending an e-mail. You will specify which one you are using for your 29 | | mailers below. You are free to add additional mailers as required. 30 | | 31 | | Supported: "smtp", "sendmail", "mailgun", "ses", "ses-v2", 32 | | "postmark", "log", "array", "failover" 33 | | 34 | */ 35 | 36 | 'mailers' => [ 37 | 'smtp' => [ 38 | 'transport' => 'smtp', 39 | 'host' => env('MAIL_HOST', 'smtp.mailgun.org'), 40 | 'port' => env('MAIL_PORT', 587), 41 | 'encryption' => env('MAIL_ENCRYPTION', 'tls'), 42 | 'username' => env('MAIL_USERNAME'), 43 | 'password' => env('MAIL_PASSWORD'), 44 | 'timeout' => null, 45 | 'local_domain' => env('MAIL_EHLO_DOMAIN'), 46 | ], 47 | 48 | 'ses' => [ 49 | 'transport' => 'ses', 50 | ], 51 | 52 | 'mailgun' => [ 53 | 'transport' => 'mailgun', 54 | // 'client' => [ 55 | // 'timeout' => 5, 56 | // ], 57 | ], 58 | 59 | 'postmark' => [ 60 | 'transport' => 'postmark', 61 | // 'client' => [ 62 | // 'timeout' => 5, 63 | // ], 64 | ], 65 | 66 | 'sendmail' => [ 67 | 'transport' => 'sendmail', 68 | 'path' => env('MAIL_SENDMAIL_PATH', '/usr/sbin/sendmail -bs -i'), 69 | ], 70 | 71 | 'log' => [ 72 | 'transport' => 'log', 73 | 'channel' => env('MAIL_LOG_CHANNEL'), 74 | ], 75 | 76 | 'array' => [ 77 | 'transport' => 'array', 78 | ], 79 | 80 | 'failover' => [ 81 | 'transport' => 'failover', 82 | 'mailers' => [ 83 | 'smtp', 84 | 'log', 85 | ], 86 | ], 87 | ], 88 | 89 | /* 90 | |-------------------------------------------------------------------------- 91 | | Global "From" Address 92 | |-------------------------------------------------------------------------- 93 | | 94 | | You may wish for all e-mails sent by your application to be sent from 95 | | the same address. Here, you may specify a name and address that is 96 | | used globally for all e-mails that are sent by your application. 97 | | 98 | */ 99 | 100 | 'from' => [ 101 | 'address' => env('MAIL_FROM_ADDRESS', 'hello@example.com'), 102 | 'name' => env('MAIL_FROM_NAME', 'Example'), 103 | ], 104 | 105 | /* 106 | |-------------------------------------------------------------------------- 107 | | Markdown Mail Settings 108 | |-------------------------------------------------------------------------- 109 | | 110 | | If you are using Markdown based email rendering, you may configure your 111 | | theme and component paths here, allowing you to customize the design 112 | | of the emails. Or, you may simply stick with the Laravel defaults! 113 | | 114 | */ 115 | 116 | 'markdown' => [ 117 | 'theme' => 'default', 118 | 119 | 'paths' => [ 120 | resource_path('views/vendor/mail'), 121 | ], 122 | ], 123 | 124 | ]; 125 | -------------------------------------------------------------------------------- /config/queue.php: -------------------------------------------------------------------------------- 1 | env('QUEUE_CONNECTION', 'sync'), 17 | 18 | /* 19 | |-------------------------------------------------------------------------- 20 | | Queue Connections 21 | |-------------------------------------------------------------------------- 22 | | 23 | | Here you may configure the connection information for each server that 24 | | is used by your application. A default configuration has been added 25 | | for each back-end shipped with Laravel. You are free to add more. 26 | | 27 | | Drivers: "sync", "database", "beanstalkd", "sqs", "redis", "null" 28 | | 29 | */ 30 | 31 | 'connections' => [ 32 | 33 | 'sync' => [ 34 | 'driver' => 'sync', 35 | ], 36 | 37 | 'database' => [ 38 | 'driver' => 'database', 39 | 'table' => 'jobs', 40 | 'queue' => 'default', 41 | 'retry_after' => 90, 42 | 'after_commit' => false, 43 | ], 44 | 45 | 'beanstalkd' => [ 46 | 'driver' => 'beanstalkd', 47 | 'host' => 'localhost', 48 | 'queue' => 'default', 49 | 'retry_after' => 90, 50 | 'block_for' => 0, 51 | 'after_commit' => false, 52 | ], 53 | 54 | 'sqs' => [ 55 | 'driver' => 'sqs', 56 | 'key' => env('AWS_ACCESS_KEY_ID'), 57 | 'secret' => env('AWS_SECRET_ACCESS_KEY'), 58 | 'prefix' => env('SQS_PREFIX', 'https://sqs.us-east-1.amazonaws.com/your-account-id'), 59 | 'queue' => env('SQS_QUEUE', 'default'), 60 | 'suffix' => env('SQS_SUFFIX'), 61 | 'region' => env('AWS_DEFAULT_REGION', 'us-east-1'), 62 | 'after_commit' => false, 63 | ], 64 | 65 | 'redis' => [ 66 | 'driver' => 'redis', 67 | 'connection' => 'default', 68 | 'queue' => env('REDIS_QUEUE', 'default'), 69 | 'retry_after' => 90, 70 | 'block_for' => null, 71 | 'after_commit' => false, 72 | ], 73 | 74 | ], 75 | 76 | /* 77 | |-------------------------------------------------------------------------- 78 | | Failed Queue Jobs 79 | |-------------------------------------------------------------------------- 80 | | 81 | | These options configure the behavior of failed queue job logging so you 82 | | can control which database and table are used to store the jobs that 83 | | have failed. You may change them to any database / table you wish. 84 | | 85 | */ 86 | 87 | 'failed' => [ 88 | 'driver' => env('QUEUE_FAILED_DRIVER', 'database-uuids'), 89 | 'database' => env('DB_CONNECTION', 'mysql'), 90 | 'table' => 'failed_jobs', 91 | ], 92 | 93 | ]; 94 | -------------------------------------------------------------------------------- /config/sanctum.php: -------------------------------------------------------------------------------- 1 | explode(',', env('SANCTUM_STATEFUL_DOMAINS', sprintf( 19 | '%s%s', 20 | 'localhost,localhost:3000,127.0.0.1,127.0.0.1:8000,::1', 21 | Sanctum::currentApplicationUrlWithPort() 22 | ))), 23 | 24 | /* 25 | |-------------------------------------------------------------------------- 26 | | Sanctum Guards 27 | |-------------------------------------------------------------------------- 28 | | 29 | | This array contains the authentication guards that will be checked when 30 | | Sanctum is trying to authenticate a request. If none of these guards 31 | | are able to authenticate the request, Sanctum will use the bearer 32 | | token that's present on an incoming request for authentication. 33 | | 34 | */ 35 | 36 | 'guard' => ['web'], 37 | 38 | /* 39 | |-------------------------------------------------------------------------- 40 | | Expiration Minutes 41 | |-------------------------------------------------------------------------- 42 | | 43 | | This value controls the number of minutes until an issued token will be 44 | | considered expired. If this value is null, personal access tokens do 45 | | not expire. This won't tweak the lifetime of first-party sessions. 46 | | 47 | */ 48 | 49 | 'expiration' => null, 50 | 51 | /* 52 | |-------------------------------------------------------------------------- 53 | | Sanctum Middleware 54 | |-------------------------------------------------------------------------- 55 | | 56 | | When authenticating your first-party SPA with Sanctum you may need to 57 | | customize some of the middleware Sanctum uses while processing the 58 | | request. You may change the middleware listed below as required. 59 | | 60 | */ 61 | 62 | 'middleware' => [ 63 | 'verify_csrf_token' => App\Http\Middleware\VerifyCsrfToken::class, 64 | 'encrypt_cookies' => App\Http\Middleware\EncryptCookies::class, 65 | ], 66 | 67 | ]; 68 | -------------------------------------------------------------------------------- /config/services.php: -------------------------------------------------------------------------------- 1 | [ 18 | 'domain' => env('MAILGUN_DOMAIN'), 19 | 'secret' => env('MAILGUN_SECRET'), 20 | 'endpoint' => env('MAILGUN_ENDPOINT', 'api.mailgun.net'), 21 | 'scheme' => 'https', 22 | ], 23 | 24 | 'postmark' => [ 25 | 'token' => env('POSTMARK_TOKEN'), 26 | ], 27 | 28 | 'ses' => [ 29 | 'key' => env('AWS_ACCESS_KEY_ID'), 30 | 'secret' => env('AWS_SECRET_ACCESS_KEY'), 31 | 'region' => env('AWS_DEFAULT_REGION', 'us-east-1'), 32 | ], 33 | 34 | ]; 35 | -------------------------------------------------------------------------------- /config/session.php: -------------------------------------------------------------------------------- 1 | env('SESSION_DRIVER', 'file'), 22 | 23 | /* 24 | |-------------------------------------------------------------------------- 25 | | Session Lifetime 26 | |-------------------------------------------------------------------------- 27 | | 28 | | Here you may specify the number of minutes that you wish the session 29 | | to be allowed to remain idle before it expires. If you want them 30 | | to immediately expire on the browser closing, set that option. 31 | | 32 | */ 33 | 34 | 'lifetime' => env('SESSION_LIFETIME', 120), 35 | 36 | 'expire_on_close' => false, 37 | 38 | /* 39 | |-------------------------------------------------------------------------- 40 | | Session Encryption 41 | |-------------------------------------------------------------------------- 42 | | 43 | | This option allows you to easily specify that all of your session data 44 | | should be encrypted before it is stored. All encryption will be run 45 | | automatically by Laravel and you can use the Session like normal. 46 | | 47 | */ 48 | 49 | 'encrypt' => false, 50 | 51 | /* 52 | |-------------------------------------------------------------------------- 53 | | Session File Location 54 | |-------------------------------------------------------------------------- 55 | | 56 | | When using the native session driver, we need a location where session 57 | | files may be stored. A default has been set for you but a different 58 | | location may be specified. This is only needed for file sessions. 59 | | 60 | */ 61 | 62 | 'files' => storage_path('framework/sessions'), 63 | 64 | /* 65 | |-------------------------------------------------------------------------- 66 | | Session Database Connection 67 | |-------------------------------------------------------------------------- 68 | | 69 | | When using the "database" or "redis" session drivers, you may specify a 70 | | connection that should be used to manage these sessions. This should 71 | | correspond to a connection in your database configuration options. 72 | | 73 | */ 74 | 75 | 'connection' => env('SESSION_CONNECTION'), 76 | 77 | /* 78 | |-------------------------------------------------------------------------- 79 | | Session Database Table 80 | |-------------------------------------------------------------------------- 81 | | 82 | | When using the "database" session driver, you may specify the table we 83 | | should use to manage the sessions. Of course, a sensible default is 84 | | provided for you; however, you are free to change this as needed. 85 | | 86 | */ 87 | 88 | 'table' => 'sessions', 89 | 90 | /* 91 | |-------------------------------------------------------------------------- 92 | | Session Cache Store 93 | |-------------------------------------------------------------------------- 94 | | 95 | | While using one of the framework's cache driven session backends you may 96 | | list a cache store that should be used for these sessions. This value 97 | | must match with one of the application's configured cache "stores". 98 | | 99 | | Affects: "apc", "dynamodb", "memcached", "redis" 100 | | 101 | */ 102 | 103 | 'store' => env('SESSION_STORE'), 104 | 105 | /* 106 | |-------------------------------------------------------------------------- 107 | | Session Sweeping Lottery 108 | |-------------------------------------------------------------------------- 109 | | 110 | | Some session drivers must manually sweep their storage location to get 111 | | rid of old sessions from storage. Here are the chances that it will 112 | | happen on a given request. By default, the odds are 2 out of 100. 113 | | 114 | */ 115 | 116 | 'lottery' => [2, 100], 117 | 118 | /* 119 | |-------------------------------------------------------------------------- 120 | | Session Cookie Name 121 | |-------------------------------------------------------------------------- 122 | | 123 | | Here you may change the name of the cookie used to identify a session 124 | | instance by ID. The name specified here will get used every time a 125 | | new session cookie is created by the framework for every driver. 126 | | 127 | */ 128 | 129 | 'cookie' => env( 130 | 'SESSION_COOKIE', 131 | Str::slug(env('APP_NAME', 'laravel'), '_').'_session' 132 | ), 133 | 134 | /* 135 | |-------------------------------------------------------------------------- 136 | | Session Cookie Path 137 | |-------------------------------------------------------------------------- 138 | | 139 | | The session cookie path determines the path for which the cookie will 140 | | be regarded as available. Typically, this will be the root path of 141 | | your application but you are free to change this when necessary. 142 | | 143 | */ 144 | 145 | 'path' => '/', 146 | 147 | /* 148 | |-------------------------------------------------------------------------- 149 | | Session Cookie Domain 150 | |-------------------------------------------------------------------------- 151 | | 152 | | Here you may change the domain of the cookie used to identify a session 153 | | in your application. This will determine which domains the cookie is 154 | | available to in your application. A sensible default has been set. 155 | | 156 | */ 157 | 158 | 'domain' => env('SESSION_DOMAIN'), 159 | 160 | /* 161 | |-------------------------------------------------------------------------- 162 | | HTTPS Only Cookies 163 | |-------------------------------------------------------------------------- 164 | | 165 | | By setting this option to true, session cookies will only be sent back 166 | | to the server if the browser has a HTTPS connection. This will keep 167 | | the cookie from being sent to you when it can't be done securely. 168 | | 169 | */ 170 | 171 | 'secure' => env('SESSION_SECURE_COOKIE', true), 172 | 173 | /* 174 | |-------------------------------------------------------------------------- 175 | | HTTP Access Only 176 | |-------------------------------------------------------------------------- 177 | | 178 | | Setting this value to true will prevent JavaScript from accessing the 179 | | value of the cookie and the cookie will only be accessible through 180 | | the HTTP protocol. You are free to modify this option if needed. 181 | | 182 | */ 183 | 184 | 'http_only' => true, 185 | 186 | /* 187 | |-------------------------------------------------------------------------- 188 | | Same-Site Cookies 189 | |-------------------------------------------------------------------------- 190 | | 191 | | This option determines how your cookies behave when cross-site requests 192 | | take place, and can be used to mitigate CSRF attacks. By default, we 193 | | will set this value to "lax" since this is a secure default value. 194 | | 195 | | Supported: "lax", "strict", "none", null 196 | | 197 | */ 198 | 199 | 'same_site' => 'none', 200 | 201 | ]; 202 | -------------------------------------------------------------------------------- /config/view.php: -------------------------------------------------------------------------------- 1 | [ 17 | resource_path('views'), 18 | ], 19 | 20 | /* 21 | |-------------------------------------------------------------------------- 22 | | Compiled View Path 23 | |-------------------------------------------------------------------------- 24 | | 25 | | This option determines where all the compiled Blade templates will be 26 | | stored for your application. Typically, this is within the storage 27 | | directory. However, as usual, you are free to change this value. 28 | | 29 | */ 30 | 31 | 'compiled' => env( 32 | 'VIEW_COMPILED_PATH', 33 | realpath(storage_path('framework/views')) 34 | ), 35 | 36 | ]; 37 | -------------------------------------------------------------------------------- /database/.gitignore: -------------------------------------------------------------------------------- 1 | *.sqlite* 2 | -------------------------------------------------------------------------------- /database/factories/UserFactory.php: -------------------------------------------------------------------------------- 1 | 10 | */ 11 | class UserFactory extends Factory 12 | { 13 | /** 14 | * Define the model's default state. 15 | * 16 | * @return array 17 | */ 18 | public function definition(): array 19 | { 20 | return [ 21 | 'name' => fake()->name(), 22 | 'email' => fake()->unique()->safeEmail(), 23 | 'email_verified_at' => now(), 24 | 'password' => '$2y$10$92IXUNpkjO0rOQ5byMi.Ye4oKoEa3Ro9llC/.og/at2.uheWG/igi', // password 25 | 'remember_token' => Str::random(10), 26 | ]; 27 | } 28 | 29 | /** 30 | * Indicate that the model's email address should be unverified. 31 | */ 32 | public function unverified(): static 33 | { 34 | return $this->state(fn (array $attributes) => [ 35 | 'email_verified_at' => null, 36 | ]); 37 | } 38 | } 39 | -------------------------------------------------------------------------------- /database/migrations/2014_10_12_000000_create_users_table.php: -------------------------------------------------------------------------------- 1 | id(); 16 | $table->string('name'); 17 | $table->string('email')->unique(); 18 | $table->timestamp('email_verified_at')->nullable(); 19 | $table->string('password'); 20 | $table->rememberToken(); 21 | $table->timestamps(); 22 | }); 23 | } 24 | 25 | /** 26 | * Reverse the migrations. 27 | */ 28 | public function down(): void 29 | { 30 | Schema::dropIfExists('users'); 31 | } 32 | }; 33 | -------------------------------------------------------------------------------- /database/migrations/2014_10_12_100000_create_password_reset_tokens_table.php: -------------------------------------------------------------------------------- 1 | string('email')->primary(); 16 | $table->string('token'); 17 | $table->timestamp('created_at')->nullable(); 18 | }); 19 | } 20 | 21 | /** 22 | * Reverse the migrations. 23 | */ 24 | public function down(): void 25 | { 26 | Schema::dropIfExists('password_reset_tokens'); 27 | } 28 | }; 29 | -------------------------------------------------------------------------------- /database/migrations/2019_08_19_000000_create_failed_jobs_table.php: -------------------------------------------------------------------------------- 1 | id(); 16 | $table->string('uuid')->unique(); 17 | $table->text('connection'); 18 | $table->text('queue'); 19 | $table->longText('payload'); 20 | $table->longText('exception'); 21 | $table->timestamp('failed_at')->useCurrent(); 22 | }); 23 | } 24 | 25 | /** 26 | * Reverse the migrations. 27 | */ 28 | public function down(): void 29 | { 30 | Schema::dropIfExists('failed_jobs'); 31 | } 32 | }; 33 | -------------------------------------------------------------------------------- /database/migrations/2019_12_14_000001_create_personal_access_tokens_table.php: -------------------------------------------------------------------------------- 1 | id(); 16 | $table->morphs('tokenable'); 17 | $table->string('name'); 18 | $table->string('token', 64)->unique(); 19 | $table->text('abilities')->nullable(); 20 | $table->timestamp('last_used_at')->nullable(); 21 | $table->timestamp('expires_at')->nullable(); 22 | $table->timestamps(); 23 | }); 24 | } 25 | 26 | /** 27 | * Reverse the migrations. 28 | */ 29 | public function down(): void 30 | { 31 | Schema::dropIfExists('personal_access_tokens'); 32 | } 33 | }; 34 | -------------------------------------------------------------------------------- /database/seeders/DatabaseSeeder.php: -------------------------------------------------------------------------------- 1 | create(); 16 | 17 | // \App\Models\User::factory()->create([ 18 | // 'name' => 'Test User', 19 | // 'email' => 'test@example.com', 20 | // ]); 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /jsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "baseUrl": ".", 4 | "paths": { 5 | "@/*": ["resources/js/*"] 6 | } 7 | }, 8 | "exclude": ["node_modules", "public"] 9 | } 10 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "private": true, 3 | "scripts": { 4 | "dev": "vite", 5 | "build": "vite build" 6 | }, 7 | "devDependencies": { 8 | "@headlessui/react": "^1.4.2", 9 | "@inertiajs/react": "^1.0.0", 10 | "@tailwindcss/forms": "^0.5.3", 11 | "@vitejs/plugin-react": "^3.0.0", 12 | "autoprefixer": "^10.4.12", 13 | "axios": "^1.1.2", 14 | "laravel-vite-plugin": "^0.7.2", 15 | "postcss": "^8.4.18", 16 | "react": "^18.2.0", 17 | "react-dom": "^18.2.0", 18 | "tailwindcss": "^3.2.1", 19 | "vite": "^4.0.0" 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /phpunit.xml: -------------------------------------------------------------------------------- 1 | 2 | 7 | 8 | 9 | ./tests/Unit 10 | 11 | 12 | ./tests/Feature 13 | 14 | 15 | 16 | 17 | ./app 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | -------------------------------------------------------------------------------- /postcss.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | plugins: { 3 | tailwindcss: {}, 4 | autoprefixer: {}, 5 | }, 6 | }; 7 | -------------------------------------------------------------------------------- /public/.htaccess: -------------------------------------------------------------------------------- 1 | 2 | 3 | Options -MultiViews -Indexes 4 | 5 | 6 | RewriteEngine On 7 | 8 | # Handle Authorization Header 9 | RewriteCond %{HTTP:Authorization} . 10 | RewriteRule .* - [E=HTTP_AUTHORIZATION:%{HTTP:Authorization}] 11 | 12 | # Redirect Trailing Slashes If Not A Folder... 13 | RewriteCond %{REQUEST_FILENAME} !-d 14 | RewriteCond %{REQUEST_URI} (.+)/$ 15 | RewriteRule ^ %1 [L,R=301] 16 | 17 | # Send Requests To Front Controller... 18 | RewriteCond %{REQUEST_FILENAME} !-d 19 | RewriteCond %{REQUEST_FILENAME} !-f 20 | RewriteRule ^ index.php [L] 21 | 22 | -------------------------------------------------------------------------------- /public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/bigcommerce/laravel-react-sample-app/564a2b9445c33966211f9345f064ff1b144efc68/public/favicon.ico -------------------------------------------------------------------------------- /public/index.php: -------------------------------------------------------------------------------- 1 | make(Kernel::class); 53 | 54 | $response = $kernel->handle( 55 | $request = Request::capture() 56 | )->send(); 57 | 58 | $kernel->terminate($request, $response); 59 | -------------------------------------------------------------------------------- /public/robots.txt: -------------------------------------------------------------------------------- 1 | User-agent: * 2 | Disallow: 3 | -------------------------------------------------------------------------------- /resources/css/app.css: -------------------------------------------------------------------------------- 1 | @tailwind base; 2 | @tailwind components; 3 | @tailwind utilities; 4 | -------------------------------------------------------------------------------- /resources/js/Components/ApplicationLogo.jsx: -------------------------------------------------------------------------------- 1 | export default function ApplicationLogo(props) { 2 | return ( 3 | 4 | 5 | 6 | ); 7 | } 8 | -------------------------------------------------------------------------------- /resources/js/Components/Checkbox.jsx: -------------------------------------------------------------------------------- 1 | export default function Checkbox({ className = '', ...props }) { 2 | return ( 3 | 11 | ); 12 | } 13 | -------------------------------------------------------------------------------- /resources/js/Components/DangerButton.jsx: -------------------------------------------------------------------------------- 1 | export default function DangerButton({ className = '', disabled, children, ...props }) { 2 | return ( 3 | 14 | ); 15 | } 16 | -------------------------------------------------------------------------------- /resources/js/Components/Dropdown.jsx: -------------------------------------------------------------------------------- 1 | import { useState, createContext, useContext, Fragment } from 'react'; 2 | import { Link } from '@inertiajs/react'; 3 | import { Transition } from '@headlessui/react'; 4 | 5 | const DropDownContext = createContext(); 6 | 7 | const Dropdown = ({ children }) => { 8 | const [open, setOpen] = useState(false); 9 | 10 | const toggleOpen = () => { 11 | setOpen((previousState) => !previousState); 12 | }; 13 | 14 | return ( 15 | 16 |
{children}
17 |
18 | ); 19 | }; 20 | 21 | const Trigger = ({ children }) => { 22 | const { open, setOpen, toggleOpen } = useContext(DropDownContext); 23 | 24 | return ( 25 | <> 26 |
{children}
27 | 28 | {open &&
setOpen(false)}>
} 29 | 30 | ); 31 | }; 32 | 33 | const Content = ({ align = 'right', width = '48', contentClasses = 'py-1 bg-white', children }) => { 34 | const { open, setOpen } = useContext(DropDownContext); 35 | 36 | let alignmentClasses = 'origin-top'; 37 | 38 | if (align === 'left') { 39 | alignmentClasses = 'origin-top-left left-0'; 40 | } else if (align === 'right') { 41 | alignmentClasses = 'origin-top-right right-0'; 42 | } 43 | 44 | let widthClasses = ''; 45 | 46 | if (width === '48') { 47 | widthClasses = 'w-48'; 48 | } 49 | 50 | return ( 51 | <> 52 | 62 |
setOpen(false)} 65 | > 66 |
{children}
67 |
68 |
69 | 70 | ); 71 | }; 72 | 73 | const DropdownLink = ({ className = '', children, ...props }) => { 74 | return ( 75 | 82 | {children} 83 | 84 | ); 85 | }; 86 | 87 | Dropdown.Trigger = Trigger; 88 | Dropdown.Content = Content; 89 | Dropdown.Link = DropdownLink; 90 | 91 | export default Dropdown; 92 | -------------------------------------------------------------------------------- /resources/js/Components/InputError.jsx: -------------------------------------------------------------------------------- 1 | export default function InputError({ message, className = '', ...props }) { 2 | return message ? ( 3 |

4 | {message} 5 |

6 | ) : null; 7 | } 8 | -------------------------------------------------------------------------------- /resources/js/Components/InputLabel.jsx: -------------------------------------------------------------------------------- 1 | export default function InputLabel({ value, className = '', children, ...props }) { 2 | return ( 3 | 6 | ); 7 | } 8 | -------------------------------------------------------------------------------- /resources/js/Components/Modal.jsx: -------------------------------------------------------------------------------- 1 | import { Fragment } from 'react'; 2 | import { Dialog, Transition } from '@headlessui/react'; 3 | 4 | export default function Modal({ children, show = false, maxWidth = '2xl', closeable = true, onClose = () => {} }) { 5 | const close = () => { 6 | if (closeable) { 7 | onClose(); 8 | } 9 | }; 10 | 11 | const maxWidthClass = { 12 | sm: 'sm:max-w-sm', 13 | md: 'sm:max-w-md', 14 | lg: 'sm:max-w-lg', 15 | xl: 'sm:max-w-xl', 16 | '2xl': 'sm:max-w-2xl', 17 | }[maxWidth]; 18 | 19 | return ( 20 | 21 | 27 | 36 |
37 | 38 | 39 | 48 | 51 | {children} 52 | 53 | 54 |
55 |
56 | ); 57 | } 58 | -------------------------------------------------------------------------------- /resources/js/Components/NavLink.jsx: -------------------------------------------------------------------------------- 1 | import { Link } from '@inertiajs/react'; 2 | 3 | export default function NavLink({ active = false, className = '', children, ...props }) { 4 | return ( 5 | 15 | {children} 16 | 17 | ); 18 | } 19 | -------------------------------------------------------------------------------- /resources/js/Components/Navigation.jsx: -------------------------------------------------------------------------------- 1 | import { Link } from '@inertiajs/react'; 2 | 3 | export default function Navigation() { 4 | return ( 5 | 17 | ); 18 | }; 19 | -------------------------------------------------------------------------------- /resources/js/Components/PrimaryButton.jsx: -------------------------------------------------------------------------------- 1 | export default function PrimaryButton({ className = '', disabled, children, ...props }) { 2 | return ( 3 | 14 | ); 15 | } 16 | -------------------------------------------------------------------------------- /resources/js/Components/ResponsiveNavLink.jsx: -------------------------------------------------------------------------------- 1 | import { Link } from '@inertiajs/react'; 2 | 3 | export default function ResponsiveNavLink({ active = false, className = '', children, ...props }) { 4 | return ( 5 | 13 | {children} 14 | 15 | ); 16 | } 17 | -------------------------------------------------------------------------------- /resources/js/Components/SecondaryButton.jsx: -------------------------------------------------------------------------------- 1 | export default function SecondaryButton({ type = 'button', className = '', disabled, children, ...props }) { 2 | return ( 3 | 15 | ); 16 | } 17 | -------------------------------------------------------------------------------- /resources/js/Components/Spinner.jsx: -------------------------------------------------------------------------------- 1 | export default function Spinner() { 2 | return ( 3 |
4 |
5 | 6 | Loading... 7 |
8 |
9 | ); 10 | } 11 | -------------------------------------------------------------------------------- /resources/js/Components/Table.jsx: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | 3 | export default class Table extends React.Component { 4 | constructor(props) { 5 | super(props); 6 | } 7 | 8 | getTableRow(data, index) { 9 | return ( 10 | 11 | {this.props.tableHeaders.map(function (header, index) { 12 | let value = data; 13 | if (header.index) { 14 | value = data[header.index]; 15 | } 16 | 17 | if (header.callback) { 18 | value = header.callback(value); 19 | } 20 | 21 | return {value}; 22 | })} 23 | 24 | ); 25 | } 26 | 27 | render() { 28 | return ( 29 | 30 | 31 | {this.props.tableHeaders.map(function (header, index) { 32 | return ; 33 | })} 34 | 35 | 36 | {this.props.tableData.map(this.getTableRow.bind(this))} 37 | 38 |
{header.label}
39 | ); 40 | } 41 | } -------------------------------------------------------------------------------- /resources/js/Components/TextInput.jsx: -------------------------------------------------------------------------------- 1 | import { forwardRef, useEffect, useRef } from 'react'; 2 | 3 | export default forwardRef(function TextInput({ type = 'text', className = '', isFocused = false, ...props }, ref) { 4 | const input = ref ? ref : useRef(); 5 | 6 | useEffect(() => { 7 | if (isFocused) { 8 | input.current.focus(); 9 | } 10 | }, []); 11 | 12 | return ( 13 |
14 | 23 |
24 | ); 25 | }); 26 | -------------------------------------------------------------------------------- /resources/js/Layouts/AuthenticatedLayout.jsx: -------------------------------------------------------------------------------- 1 | import { useState } from 'react'; 2 | import ApplicationLogo from '@/Components/ApplicationLogo'; 3 | import Dropdown from '@/Components/Dropdown'; 4 | import NavLink from '@/Components/NavLink'; 5 | import ResponsiveNavLink from '@/Components/ResponsiveNavLink'; 6 | import { Link } from '@inertiajs/react'; 7 | 8 | export default function Authenticated({ auth, header, children }) { 9 | const [showingNavigationDropdown, setShowingNavigationDropdown] = useState(false); 10 | 11 | return ( 12 |
13 | 117 | 118 | {header && ( 119 |
120 |
{header}
121 |
122 | )} 123 | 124 |
{children}
125 |
126 | ); 127 | } 128 | -------------------------------------------------------------------------------- /resources/js/Layouts/GuestLayout.jsx: -------------------------------------------------------------------------------- 1 | import ApplicationLogo from '@/Components/ApplicationLogo'; 2 | import { Link } from '@inertiajs/react'; 3 | 4 | export default function Guest({ children }) { 5 | return ( 6 |
7 |
8 | 9 | 10 | 11 |
12 | 13 |
14 | {children} 15 |
16 |
17 | ); 18 | } 19 | -------------------------------------------------------------------------------- /resources/js/Pages/Auth/ConfirmPassword.jsx: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react'; 2 | import GuestLayout from '@/Layouts/GuestLayout'; 3 | import InputError from '@/Components/InputError'; 4 | import InputLabel from '@/Components/InputLabel'; 5 | import PrimaryButton from '@/Components/PrimaryButton'; 6 | import TextInput from '@/Components/TextInput'; 7 | import { Head, useForm } from '@inertiajs/react'; 8 | 9 | export default function ConfirmPassword() { 10 | const { data, setData, post, processing, errors, reset } = useForm({ 11 | password: '', 12 | }); 13 | 14 | useEffect(() => { 15 | return () => { 16 | reset('password'); 17 | }; 18 | }, []); 19 | 20 | const handleOnChange = (event) => { 21 | setData(event.target.name, event.target.value); 22 | }; 23 | 24 | const submit = (e) => { 25 | e.preventDefault(); 26 | 27 | post(route('password.confirm')); 28 | }; 29 | 30 | return ( 31 | 32 | 33 | 34 |
35 | This is a secure area of the application. Please confirm your password before continuing. 36 |
37 | 38 |
39 |
40 | 41 | 42 | 51 | 52 | 53 |
54 | 55 |
56 | 57 | Confirm 58 | 59 |
60 |
61 |
62 | ); 63 | } 64 | -------------------------------------------------------------------------------- /resources/js/Pages/Auth/ForgotPassword.jsx: -------------------------------------------------------------------------------- 1 | import GuestLayout from '@/Layouts/GuestLayout'; 2 | import InputError from '@/Components/InputError'; 3 | import PrimaryButton from '@/Components/PrimaryButton'; 4 | import TextInput from '@/Components/TextInput'; 5 | import { Head, useForm } from '@inertiajs/react'; 6 | 7 | export default function ForgotPassword({ status }) { 8 | const { data, setData, post, processing, errors } = useForm({ 9 | email: '', 10 | }); 11 | 12 | const onHandleChange = (event) => { 13 | setData(event.target.name, event.target.value); 14 | }; 15 | 16 | const submit = (e) => { 17 | e.preventDefault(); 18 | 19 | post(route('password.email')); 20 | }; 21 | 22 | return ( 23 | 24 | 25 | 26 |
27 | Forgot your password? No problem. Just let us know your email address and we will email you a password 28 | reset link that will allow you to choose a new one. 29 |
30 | 31 | {status &&
{status}
} 32 | 33 |
34 | 43 | 44 | 45 | 46 |
47 | 48 | Email Password Reset Link 49 | 50 |
51 | 52 |
53 | ); 54 | } 55 | -------------------------------------------------------------------------------- /resources/js/Pages/Auth/Login.jsx: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react'; 2 | import Checkbox from '@/Components/Checkbox'; 3 | import GuestLayout from '@/Layouts/GuestLayout'; 4 | import InputError from '@/Components/InputError'; 5 | import InputLabel from '@/Components/InputLabel'; 6 | import PrimaryButton from '@/Components/PrimaryButton'; 7 | import TextInput from '@/Components/TextInput'; 8 | import { Head, Link, useForm } from '@inertiajs/react'; 9 | 10 | export default function Login({ status, canResetPassword }) { 11 | const { data, setData, post, processing, errors, reset } = useForm({ 12 | email: '', 13 | password: '', 14 | remember: '', 15 | }); 16 | 17 | useEffect(() => { 18 | return () => { 19 | reset('password'); 20 | }; 21 | }, []); 22 | 23 | const handleOnChange = (event) => { 24 | setData(event.target.name, event.target.type === 'checkbox' ? event.target.checked : event.target.value); 25 | }; 26 | 27 | const submit = (e) => { 28 | e.preventDefault(); 29 | 30 | post(route('login')); 31 | }; 32 | 33 | return ( 34 | 35 | 36 | 37 | {status &&
{status}
} 38 | 39 |
40 |
41 | 42 | 43 | 53 | 54 | 55 |
56 | 57 |
58 | 59 | 60 | 69 | 70 | 71 |
72 | 73 |
74 | 78 |
79 | 80 |
81 | {canResetPassword && ( 82 | 86 | Forgot your password? 87 | 88 | )} 89 | 90 | 91 | Log in 92 | 93 |
94 |
95 |
96 | ); 97 | } 98 | -------------------------------------------------------------------------------- /resources/js/Pages/Auth/Register.jsx: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react'; 2 | import GuestLayout from '@/Layouts/GuestLayout'; 3 | import InputError from '@/Components/InputError'; 4 | import InputLabel from '@/Components/InputLabel'; 5 | import PrimaryButton from '@/Components/PrimaryButton'; 6 | import TextInput from '@/Components/TextInput'; 7 | import { Head, Link, useForm } from '@inertiajs/react'; 8 | 9 | export default function Register() { 10 | const { data, setData, post, processing, errors, reset } = useForm({ 11 | name: '', 12 | email: '', 13 | password: '', 14 | password_confirmation: '', 15 | }); 16 | 17 | useEffect(() => { 18 | return () => { 19 | reset('password', 'password_confirmation'); 20 | }; 21 | }, []); 22 | 23 | const handleOnChange = (event) => { 24 | setData(event.target.name, event.target.type === 'checkbox' ? event.target.checked : event.target.value); 25 | }; 26 | 27 | const submit = (e) => { 28 | e.preventDefault(); 29 | 30 | post(route('register')); 31 | }; 32 | 33 | return ( 34 | 35 | 36 | 37 |
38 |
39 | 40 | 41 | 51 | 52 | 53 |
54 | 55 |
56 | 57 | 58 | 68 | 69 | 70 |
71 | 72 |
73 | 74 | 75 | 85 | 86 | 87 |
88 | 89 |
90 | 91 | 92 | 102 | 103 | 104 |
105 | 106 |
107 | 111 | Already registered? 112 | 113 | 114 | 115 | Register 116 | 117 |
118 |
119 |
120 | ); 121 | } 122 | -------------------------------------------------------------------------------- /resources/js/Pages/Auth/ResetPassword.jsx: -------------------------------------------------------------------------------- 1 | import { useEffect } from 'react'; 2 | import GuestLayout from '@/Layouts/GuestLayout'; 3 | import InputError from '@/Components/InputError'; 4 | import InputLabel from '@/Components/InputLabel'; 5 | import PrimaryButton from '@/Components/PrimaryButton'; 6 | import TextInput from '@/Components/TextInput'; 7 | import { Head, useForm } from '@inertiajs/react'; 8 | 9 | export default function ResetPassword({ token, email }) { 10 | const { data, setData, post, processing, errors, reset } = useForm({ 11 | token: token, 12 | email: email, 13 | password: '', 14 | password_confirmation: '', 15 | }); 16 | 17 | useEffect(() => { 18 | return () => { 19 | reset('password', 'password_confirmation'); 20 | }; 21 | }, []); 22 | 23 | const onHandleChange = (event) => { 24 | setData(event.target.name, event.target.value); 25 | }; 26 | 27 | const submit = (e) => { 28 | e.preventDefault(); 29 | 30 | post(route('password.store')); 31 | }; 32 | 33 | return ( 34 | 35 | 36 | 37 |
38 |
39 | 40 | 41 | 50 | 51 | 52 |
53 | 54 |
55 | 56 | 57 | 67 | 68 | 69 |
70 | 71 |
72 | 73 | 74 | 82 | 83 | 84 |
85 | 86 |
87 | 88 | Reset Password 89 | 90 |
91 |
92 |
93 | ); 94 | } 95 | -------------------------------------------------------------------------------- /resources/js/Pages/Auth/VerifyEmail.jsx: -------------------------------------------------------------------------------- 1 | import GuestLayout from '@/Layouts/GuestLayout'; 2 | import PrimaryButton from '@/Components/PrimaryButton'; 3 | import { Head, Link, useForm } from '@inertiajs/react'; 4 | 5 | export default function VerifyEmail({ status }) { 6 | const { post, processing } = useForm({}); 7 | 8 | const submit = (e) => { 9 | e.preventDefault(); 10 | 11 | post(route('verification.send')); 12 | }; 13 | 14 | return ( 15 | 16 | 17 | 18 |
19 | Thanks for signing up! Before getting started, could you verify your email address by clicking on the 20 | link we just emailed to you? If you didn't receive the email, we will gladly send you another. 21 |
22 | 23 | {status === 'verification-link-sent' && ( 24 |
25 | A new verification link has been sent to the email address you provided during registration. 26 |
27 | )} 28 | 29 |
30 |
31 | Resend Verification Email 32 | 33 | 39 | Log Out 40 | 41 |
42 |
43 |
44 | ); 45 | } 46 | -------------------------------------------------------------------------------- /resources/js/Pages/Dashboard.jsx: -------------------------------------------------------------------------------- 1 | import AuthenticatedLayout from '@/Layouts/AuthenticatedLayout'; 2 | import { Head } from '@inertiajs/react'; 3 | 4 | export default function Dashboard(props) { 5 | return ( 6 | Dashboard} 10 | > 11 | 12 | 13 |
14 |
15 |
16 |
You're logged in!
17 |
18 |
19 |
20 |
21 | ); 22 | } 23 | -------------------------------------------------------------------------------- /resources/js/Pages/Home.jsx: -------------------------------------------------------------------------------- 1 | import Navigation from '@/Components/Navigation'; 2 | import Spinner from '@/Components/Spinner'; 3 | import { ApiService } from '@/Services'; 4 | 5 | import { Head } from '@inertiajs/react'; 6 | import React from 'react'; 7 | 8 | export default class Home extends React.Component { 9 | constructor(props) { 10 | super(props); 11 | 12 | this.state = { 13 | isCatalogSummaryLoading: true, 14 | isStoreInfoLoading: true, 15 | catalogSummary: {}, 16 | storeInfo: {}, 17 | }; 18 | } 19 | 20 | componentDidMount() { 21 | ApiService.getResourceEntry('v2/store').then(this.handleStoreInfoResponse.bind(this)); 22 | ApiService.getResourceEntry('v3/catalog/summary').then(this.handleCatalogSummaryResponse.bind(this)); 23 | } 24 | 25 | handleStoreInfoResponse(response) { 26 | this.setState({ 27 | isStoreInfoLoading: false, 28 | storeInfo: response.data, 29 | }); 30 | } 31 | 32 | handleCatalogSummaryResponse(response) { 33 | this.setState({ 34 | isCatalogSummaryLoading: false, 35 | catalogSummary: response.data.data, 36 | }); 37 | } 38 | 39 | render() { 40 | const fieldsInSummary = [ 41 | { 42 | label: "Variant Count", 43 | index: "variant_count", 44 | format: "number", 45 | }, 46 | { 47 | label: "Inventory Count", 48 | index: "variant_count", 49 | format: "number", 50 | }, 51 | { 52 | label: "Inventory Value", 53 | index: "inventory_value", 54 | format: "currency", 55 | }, 56 | ]; 57 | return ( 58 | <> 59 | 60 | 61 |
62 |
63 |
64 |

This is the Home Page.

65 |
66 | {fieldsInSummary.map(function (summaryItem, index) { 67 | return
68 |

69 | {summaryItem.label} 70 |

71 | { 72 | this.state.isStoreInfoLoading 73 | ? 74 | 75 | : 76 | 77 | { 78 | summaryItem.format === 'currency' 79 | ? 80 | new Intl.NumberFormat(undefined, { style: 'currency', currency: this.state.storeInfo.currency }).format(this.state.catalogSummary[summaryItem.index]) 81 | : 82 | this.state.catalogSummary[summaryItem.index] 83 | 84 | } 85 | 86 | } 87 |
; 88 | }.bind(this))} 89 |
90 |
91 |
92 |

This is a Side Bar.

93 | { 94 | this.state.isStoreInfoLoading 95 | ? 96 | 97 | : 98 |
99 | { 100 | this.state.storeInfo.logo.url 101 | ? 102 | 103 | : 104 |
{this.state.storeInfo.name}
105 | } 106 | 107 |
    108 |
  • 109 |

    Domain:

    110 |

    {this.state.storeInfo.domain}

    111 |
  • 112 |
  • 113 |

    Secure URL:

    114 |

    {this.state.storeInfo.secure_url}

    115 |
  • 116 |
117 | 118 |
119 | } 120 |
121 |
122 |
123 | 124 | ); 125 | } 126 | } 127 | -------------------------------------------------------------------------------- /resources/js/Pages/List.jsx: -------------------------------------------------------------------------------- 1 | import Navigation from '@/Components/Navigation'; 2 | import Spinner from '@/Components/Spinner'; 3 | import Table from '@/Components/Table'; 4 | import { ApiService } from '@/Services'; 5 | 6 | import { Head } from '@inertiajs/react'; 7 | import React from 'react'; 8 | 9 | export default class List extends React.Component { 10 | constructor(props) { 11 | super(props); 12 | 13 | this.state = { 14 | isOrdersLoading: true, 15 | orders: { 16 | data: [], 17 | pagination: {}, 18 | }, 19 | tableHeaders: 20 | [ 21 | { 22 | label: "Order ID", 23 | index: "id", 24 | callback: function (orderId) { 25 | return orderId; 26 | }, 27 | }, 28 | { 29 | label: "Billing Name", 30 | index: "billing_address", 31 | callback: function (billingAddress) { 32 | return `${billingAddress.first_name} ${billingAddress.last_name}`; 33 | }, 34 | }, 35 | { 36 | label: "Order Total", 37 | index: "total_inc_tax", 38 | callback: function (orderTotal) { 39 | return orderTotal; 40 | }, 41 | }, 42 | { 43 | label: "Order Status", 44 | callback: function (data) { 45 | let badgeClass = 'badge badge-'; 46 | if (data.status_id === 5) { 47 | badgeClass += 'danger'; 48 | } else if (data.status_id === 2 || data.status_id === 10) { 49 | badgeClass += 'success'; 50 | } else { 51 | badgeClass += 'light'; 52 | } 53 | 54 | return ( 55 | {data.status} 56 | ); 57 | }, 58 | }, 59 | { 60 | label: "Actions", 61 | callback: function (data) { 62 | if (data.status_id !== 5) { 63 | return ( 64 | 65 | ); 66 | } 67 | }.bind(this), 68 | }, 69 | ], 70 | }; 71 | } 72 | 73 | componentWillMount() { 74 | this.loadOrders(); 75 | } 76 | 77 | loadOrders() { 78 | ApiService.getOrders({ 79 | limit: 5 80 | }).then(this.handleOrdersResponse.bind(this)); 81 | } 82 | 83 | handleOrdersResponse(response) { 84 | this.setState({ 85 | isOrdersLoading: false, 86 | orders: { 87 | data: response.data 88 | } 89 | }); 90 | } 91 | 92 | cancelOrder(orderId) { 93 | const newOrderData = { status_id: 5 }; 94 | 95 | this.setState({ 96 | isOrdersLoading: true, 97 | }); 98 | 99 | ApiService.updateOrder(orderId, newOrderData) 100 | .then(this.loadOrders.bind(this)); 101 | } 102 | 103 | hasOrders() { 104 | return (this.state.orders.data.length > 0); 105 | } 106 | 107 | render() { 108 | return ( 109 | <> 110 | 111 | 112 |
113 |
114 |

List of Orders

115 | { 116 | this.state.isOrdersLoading 117 | ? 118 | 119 | : 120 | this.hasOrders() 121 | ? 122 |
123 | 124 | 125 | : 126 |
127 |
No orders exist yet!
128 |
129 | } 130 | 131 | 132 | 133 | ); 134 | } 135 | } 136 | -------------------------------------------------------------------------------- /resources/js/Pages/Profile/Edit.jsx: -------------------------------------------------------------------------------- 1 | import AuthenticatedLayout from '@/Layouts/AuthenticatedLayout'; 2 | import DeleteUserForm from './Partials/DeleteUserForm'; 3 | import UpdatePasswordForm from './Partials/UpdatePasswordForm'; 4 | import UpdateProfileInformationForm from './Partials/UpdateProfileInformationForm'; 5 | import { Head } from '@inertiajs/react'; 6 | 7 | export default function Edit({ auth, mustVerifyEmail, status }) { 8 | return ( 9 | Profile} 12 | > 13 | 14 | 15 |
16 |
17 |
18 | 23 |
24 | 25 |
26 | 27 |
28 | 29 |
30 | 31 |
32 |
33 |
34 |
35 | ); 36 | } 37 | -------------------------------------------------------------------------------- /resources/js/Pages/Profile/Partials/DeleteUserForm.jsx: -------------------------------------------------------------------------------- 1 | import { useRef, useState } from 'react'; 2 | import DangerButton from '@/Components/DangerButton'; 3 | import InputError from '@/Components/InputError'; 4 | import InputLabel from '@/Components/InputLabel'; 5 | import Modal from '@/Components/Modal'; 6 | import SecondaryButton from '@/Components/SecondaryButton'; 7 | import TextInput from '@/Components/TextInput'; 8 | import { useForm } from '@inertiajs/react'; 9 | 10 | export default function DeleteUserForm({ className }) { 11 | const [confirmingUserDeletion, setConfirmingUserDeletion] = useState(false); 12 | const passwordInput = useRef(); 13 | 14 | const { 15 | data, 16 | setData, 17 | delete: destroy, 18 | processing, 19 | reset, 20 | errors, 21 | } = useForm({ 22 | password: '', 23 | }); 24 | 25 | const confirmUserDeletion = () => { 26 | setConfirmingUserDeletion(true); 27 | }; 28 | 29 | const deleteUser = (e) => { 30 | e.preventDefault(); 31 | 32 | destroy(route('profile.destroy'), { 33 | preserveScroll: true, 34 | onSuccess: () => closeModal(), 35 | onError: () => passwordInput.current.focus(), 36 | onFinish: () => reset(), 37 | }); 38 | }; 39 | 40 | const closeModal = () => { 41 | setConfirmingUserDeletion(false); 42 | 43 | reset(); 44 | }; 45 | 46 | return ( 47 |
48 |
49 |

Delete Account

50 | 51 |

52 | Once your account is deleted, all of its resources and data will be permanently deleted. Before 53 | deleting your account, please download any data or information that you wish to retain. 54 |

55 |
56 | 57 | Delete Account 58 | 59 | 60 |
61 |

62 | Are you sure you want to delete your account? 63 |

64 | 65 |

66 | Once your account is deleted, all of its resources and data will be permanently deleted. Please 67 | enter your password to confirm you would like to permanently delete your account. 68 |

69 | 70 |
71 | 72 | 73 | setData('password', e.target.value)} 80 | className="mt-1 block w-3/4" 81 | isFocused 82 | placeholder="Password" 83 | /> 84 | 85 | 86 |
87 | 88 |
89 | Cancel 90 | 91 | 92 | Delete Account 93 | 94 |
95 | 96 |
97 |
98 | ); 99 | } 100 | -------------------------------------------------------------------------------- /resources/js/Pages/Profile/Partials/UpdatePasswordForm.jsx: -------------------------------------------------------------------------------- 1 | import { useRef } from 'react'; 2 | import InputError from '@/Components/InputError'; 3 | import InputLabel from '@/Components/InputLabel'; 4 | import PrimaryButton from '@/Components/PrimaryButton'; 5 | import TextInput from '@/Components/TextInput'; 6 | import { useForm } from '@inertiajs/react'; 7 | import { Transition } from '@headlessui/react'; 8 | 9 | export default function UpdatePasswordForm({ className }) { 10 | const passwordInput = useRef(); 11 | const currentPasswordInput = useRef(); 12 | 13 | const { data, setData, errors, put, reset, processing, recentlySuccessful } = useForm({ 14 | current_password: '', 15 | password: '', 16 | password_confirmation: '', 17 | }); 18 | 19 | const updatePassword = (e) => { 20 | e.preventDefault(); 21 | 22 | put(route('password.update'), { 23 | preserveScroll: true, 24 | onSuccess: () => reset(), 25 | onError: () => { 26 | if (errors.password) { 27 | reset('password', 'password_confirmation'); 28 | passwordInput.current.focus(); 29 | } 30 | 31 | if (errors.current_password) { 32 | reset('current_password'); 33 | currentPasswordInput.current.focus(); 34 | } 35 | }, 36 | }); 37 | }; 38 | 39 | return ( 40 |
41 |
42 |

Update Password

43 | 44 |

45 | Ensure your account is using a long, random password to stay secure. 46 |

47 |
48 | 49 |
50 |
51 | 52 | 53 | setData('current_password', e.target.value)} 58 | type="password" 59 | className="mt-1 block w-full" 60 | autoComplete="current-password" 61 | /> 62 | 63 | 64 |
65 | 66 |
67 | 68 | 69 | setData('password', e.target.value)} 74 | type="password" 75 | className="mt-1 block w-full" 76 | autoComplete="new-password" 77 | /> 78 | 79 | 80 |
81 | 82 |
83 | 84 | 85 | setData('password_confirmation', e.target.value)} 89 | type="password" 90 | className="mt-1 block w-full" 91 | autoComplete="new-password" 92 | /> 93 | 94 | 95 |
96 | 97 |
98 | Save 99 | 100 | 106 |

Saved.

107 |
108 |
109 | 110 |
111 | ); 112 | } 113 | -------------------------------------------------------------------------------- /resources/js/Pages/Profile/Partials/UpdateProfileInformationForm.jsx: -------------------------------------------------------------------------------- 1 | import InputError from '@/Components/InputError'; 2 | import InputLabel from '@/Components/InputLabel'; 3 | import PrimaryButton from '@/Components/PrimaryButton'; 4 | import TextInput from '@/Components/TextInput'; 5 | import { Link, useForm, usePage } from '@inertiajs/react'; 6 | import { Transition } from '@headlessui/react'; 7 | 8 | export default function UpdateProfileInformation({ mustVerifyEmail, status, className }) { 9 | const user = usePage().props.auth.user; 10 | 11 | const { data, setData, patch, errors, processing, recentlySuccessful } = useForm({ 12 | name: user.name, 13 | email: user.email, 14 | }); 15 | 16 | const submit = (e) => { 17 | e.preventDefault(); 18 | 19 | patch(route('profile.update')); 20 | }; 21 | 22 | return ( 23 |
24 |
25 |

Profile Information

26 | 27 |

28 | Update your account's profile information and email address. 29 |

30 |
31 | 32 |
33 |
34 | 35 | 36 | setData('name', e.target.value)} 41 | required 42 | isFocused 43 | autoComplete="name" 44 | /> 45 | 46 | 47 |
48 | 49 |
50 | 51 | 52 | setData('email', e.target.value)} 58 | required 59 | autoComplete="username" 60 | /> 61 | 62 | 63 |
64 | 65 | {mustVerifyEmail && user.email_verified_at === null && ( 66 |
67 |

68 | Your email address is unverified. 69 | 75 | Click here to re-send the verification email. 76 | 77 |

78 | 79 | {status === 'verification-link-sent' && ( 80 |
81 | A new verification link has been sent to your email address. 82 |
83 | )} 84 |
85 | )} 86 | 87 |
88 | Save 89 | 90 | 96 |

Saved.

97 |
98 |
99 | 100 |
101 | ); 102 | } 103 | -------------------------------------------------------------------------------- /resources/js/Pages/Reports.jsx: -------------------------------------------------------------------------------- 1 | import Navigation from '@/Components/Navigation'; 2 | import Spinner from '@/Components/Spinner'; 3 | import { BatchApiService } from '@/Services'; 4 | import { Head } from '@inertiajs/react'; 5 | import React from 'react'; 6 | 7 | export default class Reports extends React.Component { 8 | constructor(props) { 9 | super(props); 10 | 11 | this.state = { 12 | isExporting: false, 13 | lastExportTime: null, 14 | }; 15 | } 16 | 17 | exportInventoryCSV = async () => { 18 | this.setState({ isExporting: true }); 19 | 20 | const response = await BatchApiService.exportInventory(); 21 | const csvData = this.convertToCSV(response.data.data); 22 | this.downloadCSV(csvData, `inventory-export-${new Date().toISOString()}.csv`); 23 | this.setState({ 24 | lastExportTime: new Date().toLocaleString(), 25 | isExporting: false 26 | }); 27 | }; 28 | 29 | convertToCSV(items) { 30 | // Define CSV headers 31 | const headers = [ 32 | 'Product ID', 33 | 'Variant ID', 34 | 'SKU', 35 | 'Available to Sell', 36 | 'Total On Hand', 37 | 'Warning Level', 38 | 'Safety Stock', 39 | 'Is In Stock', 40 | 'Bin Picking Number' 41 | ]; 42 | 43 | // Convert items to CSV rows 44 | const rows = items.map(item => [ 45 | item.identity.product_id, 46 | item.identity.variant_id, 47 | item.identity.sku, 48 | item.available_to_sell, 49 | item.total_inventory_onhand, 50 | item.settings.warning_level, 51 | item.settings.safety_stock, 52 | item.settings.is_in_stock ? 'Yes' : 'No', 53 | item.settings.bin_picking_number 54 | ]); 55 | 56 | // Combine headers and rows 57 | return [ 58 | headers.join(','), 59 | ...rows.map(row => row.join(',')) 60 | ].join('\n'); 61 | } 62 | 63 | downloadCSV(csvData, filename) { 64 | const blob = new Blob([csvData], { type: 'text/csv;charset=utf-8;' }); 65 | const link = document.createElement('a'); 66 | if (link.download !== undefined) { 67 | const url = URL.createObjectURL(blob); 68 | link.setAttribute('href', url); 69 | link.setAttribute('download', filename); 70 | link.style.visibility = 'hidden'; 71 | document.body.appendChild(link); 72 | link.click(); 73 | document.body.removeChild(link); 74 | } 75 | } 76 | 77 | render() { 78 | return ( 79 | <> 80 | 81 | 82 |
83 |
84 |

Inventory Export Tool

85 | 86 |
87 |

88 | Export a complete CSV file containing all inventory items across all pages. 89 | This operation may take a few moments to complete. 90 |

91 | 92 | 115 |
116 | 117 | {this.state.lastExportTime && ( 118 |
119 | ✓ Last export completed: {this.state.lastExportTime} 120 |
121 | )} 122 | 123 |
124 |
125 | 126 | ); 127 | } 128 | } -------------------------------------------------------------------------------- /resources/js/app.jsx: -------------------------------------------------------------------------------- 1 | import './bootstrap'; 2 | import '../css/app.css'; 3 | 4 | import { createRoot } from 'react-dom/client'; 5 | import { createInertiaApp } from '@inertiajs/react'; 6 | import { resolvePageComponent } from 'laravel-vite-plugin/inertia-helpers'; 7 | 8 | const appName = window.document.getElementsByTagName('title')[0]?.innerText || 'Laravel'; 9 | 10 | createInertiaApp({ 11 | title: (title) => `${title} - ${appName}`, 12 | resolve: (name) => resolvePageComponent(`./Pages/${name}.jsx`, import.meta.glob('./Pages/**/*.jsx')), 13 | setup({ el, App, props }) { 14 | const root = createRoot(el); 15 | 16 | root.render(); 17 | }, 18 | progress: { 19 | color: '#4B5563', 20 | }, 21 | }); 22 | -------------------------------------------------------------------------------- /resources/js/bootstrap.js: -------------------------------------------------------------------------------- 1 | /** 2 | * We'll load the axios HTTP library which allows us to easily issue requests 3 | * to our Laravel back-end. This library automatically handles sending the 4 | * CSRF token as a header based on the value of the "XSRF" token cookie. 5 | */ 6 | 7 | import axios from 'axios'; 8 | window.axios = axios; 9 | 10 | window.axios.defaults.headers.common['X-Requested-With'] = 'XMLHttpRequest'; 11 | 12 | /** 13 | * Echo exposes an expressive API for subscribing to channels and listening 14 | * for events that are broadcast by Laravel. Echo and event broadcasting 15 | * allows your team to easily build robust real-time web applications. 16 | */ 17 | 18 | // import Echo from 'laravel-echo'; 19 | 20 | // import Pusher from 'pusher-js'; 21 | // window.Pusher = Pusher; 22 | 23 | // window.Echo = new Echo({ 24 | // broadcaster: 'pusher', 25 | // key: import.meta.env.VITE_PUSHER_APP_KEY, 26 | // cluster: import.meta.env.VITE_PUSHER_APP_CLUSTER ?? 'mt1', 27 | // wsHost: import.meta.env.VITE_PUSHER_HOST ? import.meta.env.VITE_PUSHER_HOST : `ws-${import.meta.env.VITE_PUSHER_APP_CLUSTER}.pusher.com`, 28 | // wsPort: import.meta.env.VITE_PUSHER_PORT ?? 80, 29 | // wssPort: import.meta.env.VITE_PUSHER_PORT ?? 443, 30 | // forceTLS: (import.meta.env.VITE_PUSHER_SCHEME ?? 'https') === 'https', 31 | // enabledTransports: ['ws', 'wss'], 32 | // }); 33 | -------------------------------------------------------------------------------- /resources/js/services/ApiService.js: -------------------------------------------------------------------------------- 1 | import { extractRateLimitData } from './utils/responseExtractors'; 2 | 3 | export const ApiService = { 4 | 5 | _extractRateLimitData: extractRateLimitData, 6 | 7 | async _throttle(response) { 8 | const rateLimit = this._extractRateLimitData(response); 9 | if (rateLimit.requestsLeft <= 3) { // Buffer threshold 10 | const waitTime = rateLimit.windowMs / rateLimit.requestsQuota; 11 | await new Promise(resolve => setTimeout(resolve, waitTime)); 12 | } 13 | }, 14 | 15 | async getOrders(params) { 16 | params = Object.assign({ 17 | page: 1, 18 | limit: 10, 19 | }, params); 20 | 21 | const response = await axios({ 22 | method: 'get', 23 | url: '/bc-api/v2/orders', 24 | params, 25 | }); 26 | 27 | await this._throttle(response); 28 | 29 | return response; 30 | }, 31 | 32 | async updateOrder(orderId, data) { 33 | const response = await axios({ 34 | method: 'put', 35 | url: `/bc-api/v2/orders/${orderId}`, 36 | data, 37 | }); 38 | 39 | await this._throttle(response); 40 | return response; 41 | }, 42 | 43 | async deleteOrder(orderId) { 44 | const response = await axios({ 45 | method: 'delete', 46 | url: `/bc-api/v2/orders/${orderId}`, 47 | }); 48 | 49 | await this._throttle(response); 50 | return response; 51 | }, 52 | 53 | async getResourceCollection(resource, params) { 54 | params = Object.assign({ 55 | page: 1, 56 | limit: 10, 57 | }, params); 58 | 59 | const response = await axios({ 60 | method: 'get', 61 | url: `/bc-api/${resource}`, 62 | params, 63 | }); 64 | 65 | await this._throttle(response); 66 | 67 | return response; 68 | }, 69 | 70 | async getResourceEntry(resource, params) { 71 | const response = await axios({ 72 | method: 'get', 73 | url: `/bc-api/${resource}`, 74 | params, 75 | }); 76 | 77 | await this._throttle(response); 78 | return response; 79 | }, 80 | 81 | async updateResourceEntry(resource, data) { 82 | const response = await axios({ 83 | method: 'put', 84 | url: `/bc-api/${resource}`, 85 | data, 86 | }); 87 | 88 | await this._throttle(response); 89 | return response; 90 | }, 91 | 92 | async deleteResourceEntry(resource) { 93 | const response = await axios({ 94 | method: 'delete', 95 | url: `/bc-api/${resource}`, 96 | }); 97 | 98 | await this._throttle(response); 99 | return response; 100 | }, 101 | }; -------------------------------------------------------------------------------- /resources/js/services/BatchApiService.js: -------------------------------------------------------------------------------- 1 | import { extractPaginationData, extractRateLimitData } from './utils/responseExtractors'; 2 | 3 | export const BatchApiService = { 4 | _extractPaginationData: extractPaginationData, 5 | _extractRateLimitData: extractRateLimitData, 6 | 7 | 8 | // Helper throttle function 9 | async _throttle(response) { 10 | const rateLimit = this._extractRateLimitData(response); 11 | if (rateLimit.requestsLeft <= 3) { // Buffer threshold 12 | const waitTime = rateLimit.windowMs / rateLimit.requestsQuota; 13 | await new Promise(resolve => setTimeout(resolve, waitTime)); 14 | } 15 | }, 16 | 17 | async fetchAllPages(resource, params = {}) { 18 | params = Object.assign({ 19 | page: 1, 20 | limit: 250, 21 | }, params); 22 | 23 | let allData = []; 24 | let hasNextPage = true; 25 | let lastResponse = null; 26 | 27 | while (hasNextPage) { 28 | 29 | const response = await axios({ 30 | method: 'get', 31 | url: `/bc-api/${resource}`, 32 | params, 33 | }); 34 | 35 | await this._throttle(response); 36 | 37 | lastResponse = response; 38 | 39 | if (response?.data?.data) { 40 | allData = allData.concat(response.data.data); 41 | } 42 | 43 | const pagination = response.data?.meta?.pagination; 44 | hasNextPage = pagination?.current_page < pagination?.total_pages; 45 | 46 | if (hasNextPage) { 47 | params.page++; 48 | } 49 | } 50 | 51 | lastResponse.data = { 52 | data: allData, 53 | meta: lastResponse.data.meta 54 | }; 55 | return lastResponse; 56 | }, 57 | 58 | async exportInventory(params = {}) { 59 | return this.fetchAllPages('v3/inventory/locations/1/items', params); 60 | }, 61 | 62 | async generateInventoryReport() { 63 | const response = await this.fetchAllPages('v3/inventory/locations/1/items'); 64 | 65 | return { 66 | data: response.data.data, 67 | stats: this._calculateInventoryStats(response.data.data) 68 | }; 69 | }, 70 | 71 | _calculateInventoryStats(items) { 72 | return { 73 | totalItems: items.length, 74 | totalStock: items.reduce((sum, item) => sum + item.total_inventory_onhand, 0), 75 | lowStockItems: items.filter(item => item.available_to_sell <= item.settings.warning_level).length, 76 | outOfStockItems: items.filter(item => !item.settings.is_in_stock).length 77 | }; 78 | } 79 | }; -------------------------------------------------------------------------------- /resources/js/services/index.js: -------------------------------------------------------------------------------- 1 | import { ApiService } from './ApiService'; 2 | import { BatchApiService } from './BatchApiService'; 3 | 4 | export { 5 | ApiService, 6 | BatchApiService, 7 | }; -------------------------------------------------------------------------------- /resources/js/services/utils/responseExtractors.js: -------------------------------------------------------------------------------- 1 | export const extractPaginationData = (response) => { 2 | return { 3 | total: parseInt(response.data?.meta?.pagination?.total || null), 4 | totalPages: parseInt(response.data?.meta?.pagination?.total_pages || null), 5 | currentPage: parseInt(response.data?.meta?.pagination?.current_page || null), 6 | perPage: parseInt(response.data?.meta?.pagination?.per_page || null), 7 | count: parseInt(response.data?.meta?.pagination?.count || null), 8 | previousLink: response.data?.meta?.pagination?.links?.previous || null, 9 | currentLink: response.data?.meta?.pagination?.links?.current || null, 10 | nextLink: response.data?.meta?.pagination?.links?.next || null, 11 | }; 12 | }; 13 | 14 | export const extractRateLimitData = (response) => { 15 | return { 16 | resetMs: parseInt(response.headers['x-rate-limit-time-reset-ms']), 17 | windowMs: parseInt(response.headers['x-rate-limit-time-window-ms']), 18 | requestsLeft: parseInt(response.headers['x-rate-limit-requests-left']), 19 | requestsQuota: parseInt(response.headers['x-rate-limit-requests-quota']), 20 | }; 21 | }; -------------------------------------------------------------------------------- /resources/views/app.blade.php: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | {{ config('app.name', 'Laravel') }} 8 | 9 | 10 | 11 | 12 | 13 | 14 | @routes 15 | @viteReactRefresh 16 | @vite(['resources/js/app.jsx', "resources/js/Pages/{$page['component']}.jsx"]) 17 | @inertiaHead 18 | 19 | 20 | @inertia 21 | 22 | 23 | -------------------------------------------------------------------------------- /routes/api.php: -------------------------------------------------------------------------------- 1 | get('/user', function (Request $request) { 18 | return $request->user(); 19 | }); 20 | -------------------------------------------------------------------------------- /routes/auth.php: -------------------------------------------------------------------------------- 1 | group(function () { 15 | Route::get('register', [RegisteredUserController::class, 'create']) 16 | ->name('register'); 17 | 18 | Route::post('register', [RegisteredUserController::class, 'store']); 19 | 20 | Route::get('login', [AuthenticatedSessionController::class, 'create']) 21 | ->name('login'); 22 | 23 | Route::post('login', [AuthenticatedSessionController::class, 'store']); 24 | 25 | Route::get('forgot-password', [PasswordResetLinkController::class, 'create']) 26 | ->name('password.request'); 27 | 28 | Route::post('forgot-password', [PasswordResetLinkController::class, 'store']) 29 | ->name('password.email'); 30 | 31 | Route::get('reset-password/{token}', [NewPasswordController::class, 'create']) 32 | ->name('password.reset'); 33 | 34 | Route::post('reset-password', [NewPasswordController::class, 'store']) 35 | ->name('password.store'); 36 | }); 37 | 38 | Route::middleware('auth')->group(function () { 39 | Route::get('verify-email', EmailVerificationPromptController::class) 40 | ->name('verification.notice'); 41 | 42 | Route::get('verify-email/{id}/{hash}', VerifyEmailController::class) 43 | ->middleware(['signed', 'throttle:6,1']) 44 | ->name('verification.verify'); 45 | 46 | Route::post('email/verification-notification', [EmailVerificationNotificationController::class, 'store']) 47 | ->middleware('throttle:6,1') 48 | ->name('verification.send'); 49 | 50 | Route::get('confirm-password', [ConfirmablePasswordController::class, 'show']) 51 | ->name('password.confirm'); 52 | 53 | Route::post('confirm-password', [ConfirmablePasswordController::class, 'store']); 54 | 55 | Route::put('password', [PasswordController::class, 'update'])->name('password.update'); 56 | 57 | Route::post('logout', [AuthenticatedSessionController::class, 'destroy']) 58 | ->name('logout'); 59 | }); 60 | -------------------------------------------------------------------------------- /routes/channels.php: -------------------------------------------------------------------------------- 1 | id === (int) $id; 18 | }); 19 | -------------------------------------------------------------------------------- /routes/console.php: -------------------------------------------------------------------------------- 1 | comment(Inspiring::quote()); 19 | })->purpose('Display an inspiring quote'); 20 | -------------------------------------------------------------------------------- /routes/web.php: -------------------------------------------------------------------------------- 1 | name('inventory'); 32 | 33 | Route::get('/reports', function () { 34 | return Inertia::render('Reports'); 35 | })->name('reports'); 36 | 37 | Route::group(['prefix' => 'auth'], function () { 38 | Route::get('install', [MainController::class, 'install']); 39 | 40 | Route::get('load', [MainController::class, 'load']); 41 | 42 | Route::get('uninstall', function () { 43 | echo 'uninstall'; 44 | return app()->version(); 45 | }); 46 | 47 | Route::get('remove-user', function () { 48 | echo 'remove-user'; 49 | return app()->version(); 50 | }); 51 | }); 52 | 53 | Route::any('/bc-api/{endpoint}', [MainController::class, 'proxyBigCommerceAPIRequest']) 54 | ->where('endpoint', 'v2/.*|v3/.*'); 55 | 56 | require __DIR__.'/auth.php'; 57 | -------------------------------------------------------------------------------- /storage/app/.gitignore: -------------------------------------------------------------------------------- 1 | * 2 | !public/ 3 | !.gitignore 4 | -------------------------------------------------------------------------------- /storage/app/public/.gitignore: -------------------------------------------------------------------------------- 1 | * 2 | !.gitignore 3 | -------------------------------------------------------------------------------- /storage/framework/.gitignore: -------------------------------------------------------------------------------- 1 | compiled.php 2 | config.php 3 | down 4 | events.scanned.php 5 | maintenance.php 6 | routes.php 7 | routes.scanned.php 8 | schedule-* 9 | services.json 10 | -------------------------------------------------------------------------------- /storage/framework/cache/.gitignore: -------------------------------------------------------------------------------- 1 | * 2 | !data/ 3 | !.gitignore 4 | -------------------------------------------------------------------------------- /storage/framework/cache/data/.gitignore: -------------------------------------------------------------------------------- 1 | * 2 | !.gitignore 3 | -------------------------------------------------------------------------------- /storage/framework/sessions/.gitignore: -------------------------------------------------------------------------------- 1 | * 2 | !.gitignore 3 | -------------------------------------------------------------------------------- /storage/framework/testing/.gitignore: -------------------------------------------------------------------------------- 1 | * 2 | !.gitignore 3 | -------------------------------------------------------------------------------- /storage/framework/views/.gitignore: -------------------------------------------------------------------------------- 1 | * 2 | !.gitignore 3 | -------------------------------------------------------------------------------- /storage/logs/.gitignore: -------------------------------------------------------------------------------- 1 | * 2 | !.gitignore 3 | -------------------------------------------------------------------------------- /tailwind.config.js: -------------------------------------------------------------------------------- 1 | const defaultTheme = require('tailwindcss/defaultTheme'); 2 | 3 | /** @type {import('tailwindcss').Config} */ 4 | module.exports = { 5 | content: [ 6 | './vendor/laravel/framework/src/Illuminate/Pagination/resources/views/*.blade.php', 7 | './storage/framework/views/*.php', 8 | './resources/views/**/*.blade.php', 9 | './resources/js/**/*.jsx', 10 | ], 11 | 12 | theme: { 13 | extend: { 14 | fontFamily: { 15 | sans: ['Figtree', ...defaultTheme.fontFamily.sans], 16 | }, 17 | }, 18 | }, 19 | 20 | plugins: [require('@tailwindcss/forms')], 21 | }; 22 | -------------------------------------------------------------------------------- /tests/CreatesApplication.php: -------------------------------------------------------------------------------- 1 | make(Kernel::class)->bootstrap(); 18 | 19 | return $app; 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /tests/Feature/Auth/AuthenticationTest.php: -------------------------------------------------------------------------------- 1 | get('/login'); 17 | 18 | $response->assertStatus(200); 19 | } 20 | 21 | public function test_users_can_authenticate_using_the_login_screen(): void 22 | { 23 | $user = User::factory()->create(); 24 | 25 | $response = $this->post('/login', [ 26 | 'email' => $user->email, 27 | 'password' => 'password', 28 | ]); 29 | 30 | $this->assertAuthenticated(); 31 | $response->assertRedirect(RouteServiceProvider::HOME); 32 | } 33 | 34 | public function test_users_can_not_authenticate_with_invalid_password(): void 35 | { 36 | $user = User::factory()->create(); 37 | 38 | $this->post('/login', [ 39 | 'email' => $user->email, 40 | 'password' => 'wrong-password', 41 | ]); 42 | 43 | $this->assertGuest(); 44 | } 45 | } 46 | -------------------------------------------------------------------------------- /tests/Feature/Auth/EmailVerificationTest.php: -------------------------------------------------------------------------------- 1 | create([ 20 | 'email_verified_at' => null, 21 | ]); 22 | 23 | $response = $this->actingAs($user)->get('/verify-email'); 24 | 25 | $response->assertStatus(200); 26 | } 27 | 28 | public function test_email_can_be_verified(): void 29 | { 30 | $user = User::factory()->create([ 31 | 'email_verified_at' => null, 32 | ]); 33 | 34 | Event::fake(); 35 | 36 | $verificationUrl = URL::temporarySignedRoute( 37 | 'verification.verify', 38 | now()->addMinutes(60), 39 | ['id' => $user->id, 'hash' => sha1($user->email)] 40 | ); 41 | 42 | $response = $this->actingAs($user)->get($verificationUrl); 43 | 44 | Event::assertDispatched(Verified::class); 45 | $this->assertTrue($user->fresh()->hasVerifiedEmail()); 46 | $response->assertRedirect(RouteServiceProvider::HOME.'?verified=1'); 47 | } 48 | 49 | public function test_email_is_not_verified_with_invalid_hash(): void 50 | { 51 | $user = User::factory()->create([ 52 | 'email_verified_at' => null, 53 | ]); 54 | 55 | $verificationUrl = URL::temporarySignedRoute( 56 | 'verification.verify', 57 | now()->addMinutes(60), 58 | ['id' => $user->id, 'hash' => sha1('wrong-email')] 59 | ); 60 | 61 | $this->actingAs($user)->get($verificationUrl); 62 | 63 | $this->assertFalse($user->fresh()->hasVerifiedEmail()); 64 | } 65 | } 66 | -------------------------------------------------------------------------------- /tests/Feature/Auth/PasswordConfirmationTest.php: -------------------------------------------------------------------------------- 1 | create(); 16 | 17 | $response = $this->actingAs($user)->get('/confirm-password'); 18 | 19 | $response->assertStatus(200); 20 | } 21 | 22 | public function test_password_can_be_confirmed(): void 23 | { 24 | $user = User::factory()->create(); 25 | 26 | $response = $this->actingAs($user)->post('/confirm-password', [ 27 | 'password' => 'password', 28 | ]); 29 | 30 | $response->assertRedirect(); 31 | $response->assertSessionHasNoErrors(); 32 | } 33 | 34 | public function test_password_is_not_confirmed_with_invalid_password(): void 35 | { 36 | $user = User::factory()->create(); 37 | 38 | $response = $this->actingAs($user)->post('/confirm-password', [ 39 | 'password' => 'wrong-password', 40 | ]); 41 | 42 | $response->assertSessionHasErrors(); 43 | } 44 | } 45 | -------------------------------------------------------------------------------- /tests/Feature/Auth/PasswordResetTest.php: -------------------------------------------------------------------------------- 1 | get('/forgot-password'); 18 | 19 | $response->assertStatus(200); 20 | } 21 | 22 | public function test_reset_password_link_can_be_requested(): void 23 | { 24 | Notification::fake(); 25 | 26 | $user = User::factory()->create(); 27 | 28 | $this->post('/forgot-password', ['email' => $user->email]); 29 | 30 | Notification::assertSentTo($user, ResetPassword::class); 31 | } 32 | 33 | public function test_reset_password_screen_can_be_rendered(): void 34 | { 35 | Notification::fake(); 36 | 37 | $user = User::factory()->create(); 38 | 39 | $this->post('/forgot-password', ['email' => $user->email]); 40 | 41 | Notification::assertSentTo($user, ResetPassword::class, function ($notification) { 42 | $response = $this->get('/reset-password/'.$notification->token); 43 | 44 | $response->assertStatus(200); 45 | 46 | return true; 47 | }); 48 | } 49 | 50 | public function test_password_can_be_reset_with_valid_token(): void 51 | { 52 | Notification::fake(); 53 | 54 | $user = User::factory()->create(); 55 | 56 | $this->post('/forgot-password', ['email' => $user->email]); 57 | 58 | Notification::assertSentTo($user, ResetPassword::class, function ($notification) use ($user) { 59 | $response = $this->post('/reset-password', [ 60 | 'token' => $notification->token, 61 | 'email' => $user->email, 62 | 'password' => 'password', 63 | 'password_confirmation' => 'password', 64 | ]); 65 | 66 | $response->assertSessionHasNoErrors(); 67 | 68 | return true; 69 | }); 70 | } 71 | } 72 | -------------------------------------------------------------------------------- /tests/Feature/Auth/PasswordUpdateTest.php: -------------------------------------------------------------------------------- 1 | create(); 17 | 18 | $response = $this 19 | ->actingAs($user) 20 | ->from('/profile') 21 | ->put('/password', [ 22 | 'current_password' => 'password', 23 | 'password' => 'new-password', 24 | 'password_confirmation' => 'new-password', 25 | ]); 26 | 27 | $response 28 | ->assertSessionHasNoErrors() 29 | ->assertRedirect('/profile'); 30 | 31 | $this->assertTrue(Hash::check('new-password', $user->refresh()->password)); 32 | } 33 | 34 | public function test_correct_password_must_be_provided_to_update_password(): void 35 | { 36 | $user = User::factory()->create(); 37 | 38 | $response = $this 39 | ->actingAs($user) 40 | ->from('/profile') 41 | ->put('/password', [ 42 | 'current_password' => 'wrong-password', 43 | 'password' => 'new-password', 44 | 'password_confirmation' => 'new-password', 45 | ]); 46 | 47 | $response 48 | ->assertSessionHasErrors('current_password') 49 | ->assertRedirect('/profile'); 50 | } 51 | } 52 | -------------------------------------------------------------------------------- /tests/Feature/Auth/RegistrationTest.php: -------------------------------------------------------------------------------- 1 | get('/register'); 16 | 17 | $response->assertStatus(200); 18 | } 19 | 20 | public function test_new_users_can_register(): void 21 | { 22 | $response = $this->post('/register', [ 23 | 'name' => 'Test User', 24 | 'email' => 'test@example.com', 25 | 'password' => 'password', 26 | 'password_confirmation' => 'password', 27 | ]); 28 | 29 | $this->assertAuthenticated(); 30 | $response->assertRedirect(RouteServiceProvider::HOME); 31 | } 32 | } 33 | -------------------------------------------------------------------------------- /tests/Feature/ExampleTest.php: -------------------------------------------------------------------------------- 1 | get('/'); 16 | 17 | $response->assertStatus(200); 18 | } 19 | } 20 | -------------------------------------------------------------------------------- /tests/Feature/ProfileTest.php: -------------------------------------------------------------------------------- 1 | create(); 16 | 17 | $response = $this 18 | ->actingAs($user) 19 | ->get('/profile'); 20 | 21 | $response->assertOk(); 22 | } 23 | 24 | public function test_profile_information_can_be_updated(): void 25 | { 26 | $user = User::factory()->create(); 27 | 28 | $response = $this 29 | ->actingAs($user) 30 | ->patch('/profile', [ 31 | 'name' => 'Test User', 32 | 'email' => 'test@example.com', 33 | ]); 34 | 35 | $response 36 | ->assertSessionHasNoErrors() 37 | ->assertRedirect('/profile'); 38 | 39 | $user->refresh(); 40 | 41 | $this->assertSame('Test User', $user->name); 42 | $this->assertSame('test@example.com', $user->email); 43 | $this->assertNull($user->email_verified_at); 44 | } 45 | 46 | public function test_email_verification_status_is_unchanged_when_the_email_address_is_unchanged(): void 47 | { 48 | $user = User::factory()->create(); 49 | 50 | $response = $this 51 | ->actingAs($user) 52 | ->patch('/profile', [ 53 | 'name' => 'Test User', 54 | 'email' => $user->email, 55 | ]); 56 | 57 | $response 58 | ->assertSessionHasNoErrors() 59 | ->assertRedirect('/profile'); 60 | 61 | $this->assertNotNull($user->refresh()->email_verified_at); 62 | } 63 | 64 | public function test_user_can_delete_their_account(): void 65 | { 66 | $user = User::factory()->create(); 67 | 68 | $response = $this 69 | ->actingAs($user) 70 | ->delete('/profile', [ 71 | 'password' => 'password', 72 | ]); 73 | 74 | $response 75 | ->assertSessionHasNoErrors() 76 | ->assertRedirect('/'); 77 | 78 | $this->assertGuest(); 79 | $this->assertNull($user->fresh()); 80 | } 81 | 82 | public function test_correct_password_must_be_provided_to_delete_account(): void 83 | { 84 | $user = User::factory()->create(); 85 | 86 | $response = $this 87 | ->actingAs($user) 88 | ->from('/profile') 89 | ->delete('/profile', [ 90 | 'password' => 'wrong-password', 91 | ]); 92 | 93 | $response 94 | ->assertSessionHasErrors('password') 95 | ->assertRedirect('/profile'); 96 | 97 | $this->assertNotNull($user->fresh()); 98 | } 99 | } 100 | -------------------------------------------------------------------------------- /tests/TestCase.php: -------------------------------------------------------------------------------- 1 | assertTrue(true); 15 | } 16 | } 17 | -------------------------------------------------------------------------------- /vite.config.js: -------------------------------------------------------------------------------- 1 | import { defineConfig } from 'vite'; 2 | import laravel from 'laravel-vite-plugin'; 3 | import react from '@vitejs/plugin-react'; 4 | 5 | export default defineConfig({ 6 | plugins: [ 7 | laravel({ 8 | input: 'resources/js/app.jsx', 9 | refresh: true, 10 | }), 11 | react(), 12 | ], 13 | }); 14 | --------------------------------------------------------------------------------