72 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			PHP
		
	
	
	
			
		
		
	
	
			72 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			PHP
		
	
	
	
| <?php
 | |
| 
 | |
| namespace BookStack\Http;
 | |
| 
 | |
| use Illuminate\Http\Request;
 | |
| use Illuminate\Http\Response;
 | |
| use Symfony\Component\HttpFoundation\StreamedResponse;
 | |
| 
 | |
| class DownloadResponseFactory
 | |
| {
 | |
|     public function __construct(
 | |
|         protected Request $request
 | |
|     ) {
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a response that directly forces a download in the browser.
 | |
|      */
 | |
|     public function directly(string $content, string $fileName): Response
 | |
|     {
 | |
|         return response()->make($content, 200, $this->getHeaders($fileName, strlen($content)));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a response that forces a download, from a given stream of content.
 | |
|      */
 | |
|     public function streamedDirectly($stream, string $fileName, int $fileSize): StreamedResponse
 | |
|     {
 | |
|         $rangeStream = new RangeSupportedStream($stream, $fileSize, $this->request);
 | |
|         $headers = array_merge($this->getHeaders($fileName, $fileSize), $rangeStream->getResponseHeaders());
 | |
|         return response()->stream(
 | |
|             fn() => $rangeStream->outputAndClose(),
 | |
|             $rangeStream->getResponseStatus(),
 | |
|             $headers,
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a file download response that provides the file with a content-type
 | |
|      * correct for the file, in a way so the browser can show the content in browser,
 | |
|      * for a given content stream.
 | |
|      */
 | |
|     public function streamedInline($stream, string $fileName, int $fileSize): StreamedResponse
 | |
|     {
 | |
|         $rangeStream = new RangeSupportedStream($stream, $fileSize, $this->request);
 | |
|         $mime = $rangeStream->sniffMime();
 | |
|         $headers = array_merge($this->getHeaders($fileName, $fileSize, $mime), $rangeStream->getResponseHeaders());
 | |
| 
 | |
|         return response()->stream(
 | |
|             fn() => $rangeStream->outputAndClose(),
 | |
|             $rangeStream->getResponseStatus(),
 | |
|             $headers,
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the common headers to provide for a download response.
 | |
|      */
 | |
|     protected function getHeaders(string $fileName, int $fileSize, string $mime = 'application/octet-stream'): array
 | |
|     {
 | |
|         $disposition = ($mime === 'application/octet-stream') ? 'attachment' : 'inline';
 | |
|         $downloadName = str_replace('"', '', $fileName);
 | |
| 
 | |
|         return [
 | |
|             'Content-Type'           => $mime,
 | |
|             'Content-Length'         => $fileSize,
 | |
|             'Content-Disposition'    => "{$disposition}; filename=\"{$downloadName}\"",
 | |
|             'X-Content-Type-Options' => 'nosniff',
 | |
|         ];
 | |
|     }
 | |
| }
 |