configureRateLimiting(); // Disable trimming string values when requesting file information — it isn't helpful // and messes up the ability to actually open a directory that ends with a space. TrimStrings::skipWhen(function (Request $request) { return preg_match(self::FILE_PATH_REGEX, $request->getPathInfo()) === 1; }); // This is needed to make use of the "resolveRouteBinding" functionality in the // model. Without it you'll never trigger that logic flow thus resulting in a 404 // error because we request databases with a HashID, and not with a normal ID. Route::model('database', Database::class); $this->routes(function () { Route::middleware('web')->group(function () { Route::middleware(['auth.session', RequireTwoFactorAuthentication::class]) ->group(base_path('routes/base.php')); Route::middleware(['auth.session', RequireTwoFactorAuthentication::class, AdminAuthenticate::class]) ->prefix('/admin') ->group(base_path('routes/admin.php')); Route::middleware('guest')->prefix('/auth')->group(base_path('routes/auth.php')); }); Route::middleware(['api', RequireTwoFactorAuthentication::class])->group(function () { Route::middleware(['application-api', 'throttle:api.application']) ->prefix('/api/application') ->scopeBindings() ->group(base_path('routes/api-application.php')); Route::middleware(['client-api', 'throttle:api.client']) ->prefix('/api/client') ->scopeBindings() ->group(base_path('routes/api-client.php')); }); Route::middleware('daemon') ->prefix('/api/remote') ->scopeBindings() ->group(base_path('routes/api-remote.php')); $this->duskBoot(); }); } /** * Configure the rate limiters for the application. */ protected function configureRateLimiting() { // Authentication rate limiting. For login and checkpoint endpoints we'll apply // a limit of 10 requests per minute, for the forgot password endpoint apply a // limit of two per minute for the requester so that there is less ability to // trigger email spam. RateLimiter::for('authentication', function (Request $request) { if ($request->route()->named('auth.post.forgot-password')) { return Limit::perMinute(2)->by($request->ip()); } return Limit::perMinute(10); }); // Configure the throttles for both the application and client APIs below. // This is configurable per-instance in "config/http.php". By default this // limiter will be tied to the specific request user, and falls back to the // request IP if there is no request user present for the key. // // This means that an authenticated API user cannot use IP switching to get // around the limits. RateLimiter::for('api.client', function (Request $request) { $key = optional($request->user())->uuid ?: $request->ip(); return Limit::perMinutes( config('http.rate_limit.client_period'), config('http.rate_limit.client') )->by($key); }); RateLimiter::for('api.application', function (Request $request) { $key = optional($request->user())->uuid ?: $request->ip(); return Limit::perMinutes( config('http.rate_limit.application_period'), config('http.rate_limit.application') )->by($key); }); } // Laravel Dusk Browser Testing Route Helpers for the Daemon private function duskBoot() { // Make sure we're only running in the Dusk testing environment if (!app()->environment('dusk')) { return; } Route::middleware('cors')->group(function () { // Simulate Node Ping Route::get('/api/system', fn () => [ 'version' => '1.7.0', 'kernel_version' => '5.4.0-126-generic', 'architecture' => 'amd64', 'os' => 'linux', 'cpu_count' => 2, ]); // Simulate Successful Server Creation Route::post('/api/servers', fn () => []); }); } }