namespace BookStack\Http;
-use BookStack\Util\WebSafeMimeSniffer;
use Illuminate\Http\Request;
use Illuminate\Http\Response;
use Symfony\Component\HttpFoundation\StreamedResponse;
class DownloadResponseFactory
{
- protected Request $request;
-
- public function __construct(Request $request)
- {
- $this->request = $request;
+ public function __construct(
+ protected Request $request
+ ) {
}
/**
*/
public function directly(string $content, string $fileName): Response
{
- return response()->make($content, 200, $this->getHeaders($fileName));
+ 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): StreamedResponse
+ public function streamedDirectly($stream, string $fileName, int $fileSize): StreamedResponse
{
- return response()->stream(function () use ($stream) {
-
- // End & flush the output buffer, if we're in one, otherwise we still use memory.
- // Output buffer may or may not exist depending on PHP `output_buffering` setting.
- // Ignore in testing since output buffers are used to gather a response.
- if (!empty(ob_get_status()) && !app()->runningUnitTests()) {
- ob_end_clean();
- }
-
- fpassthru($stream);
- fclose($stream);
- }, 200, $this->getHeaders($fileName));
+ $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,
+ );
}
/**
* 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): StreamedResponse
+ public function streamedInline($stream, string $fileName, int $fileSize): StreamedResponse
{
- $sniffContent = fread($stream, 2000);
- $mime = (new WebSafeMimeSniffer())->sniff($sniffContent);
+ $rangeStream = new RangeSupportedStream($stream, $fileSize, $this->request);
+ $mime = $rangeStream->sniffMime();
+ $headers = array_merge($this->getHeaders($fileName, $fileSize, $mime), $rangeStream->getResponseHeaders());
- return response()->stream(function () use ($sniffContent, $stream) {
- echo $sniffContent;
- fpassthru($stream);
- fclose($stream);
- }, 200, $this->getHeaders($fileName, $mime));
+ return response()->stream(
+ fn() => $rangeStream->outputAndClose(),
+ $rangeStream->getResponseStatus(),
+ $headers,
+ );
}
/**
* Get the common headers to provide for a download response.
*/
- protected function getHeaders(string $fileName, string $mime = 'application/octet-stream'): array
+ 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',
];