group(function () { Route::get('/dashboard', function () { return view('dashboard'); })->name('dashboard'); }); // R2 정적 이미지 프록시 (인증 불필요, 캐시 1일) Route::get('/images/{path}', function (string $path) { $r2Path = 'images/'.$path; if (! Storage::disk('r2')->exists($r2Path)) { abort(404); } $mime = Storage::disk('r2')->mimeType($r2Path); $stream = Storage::disk('r2')->readStream($r2Path); return response()->stream(function () use ($stream) { fpassthru($stream); if (is_resource($stream)) { fclose($stream); } }, 200, [ 'Content-Type' => $mime, 'Cache-Control' => 'public, max-age=86400', ]); })->where('path', '.*'); // R2 테넌트 파일 프록시 (문서 템플릿 이미지 등, 인증 불필요, 캐시 1일) Route::get('/storage/tenants/{path}', function (string $path) { if (! Storage::disk('r2')->exists($path)) { abort(404); } $mime = Storage::disk('r2')->mimeType($path); $stream = Storage::disk('r2')->readStream($path); return response()->stream(function () use ($stream) { fpassthru($stream); if (is_resource($stream)) { fclose($stream); } }, 200, [ 'Content-Type' => $mime, 'Cache-Control' => 'public, max-age=86400', ]); })->where('path', '.*'); // Swagger 설정 Route::get('/docs/api-docs.json', function () { return response()->file(storage_path('api-docs/api-docs.json')); }); Route::middleware('swagger.auth')->group(function () { Route::get('/api/documentation', function () { return response()->file(public_path('swagger-ui/index.html')); }); });