0
0
Fork 0
mirror of https://github.com/BookStackApp/BookStack.git synced 2025-04-21 03:30:38 +00:00

Range requests: Extracted stream output handling to new class

This commit is contained in:
Dan Brown 2024-01-07 14:03:13 +00:00
parent b191d8f99f
commit b4d9029dc3
No known key found for this signature in database
GPG key ID: 46D9F943C24A2EF9
4 changed files with 80 additions and 28 deletions

View file

@ -9,11 +9,9 @@ use Symfony\Component\HttpFoundation\StreamedResponse;
class DownloadResponseFactory class DownloadResponseFactory
{ {
protected Request $request; public function __construct(
protected Request $request
public function __construct(Request $request) ) {
{
$this->request = $request;
} }
/** /**
@ -27,19 +25,11 @@ class DownloadResponseFactory
/** /**
* Create a response that forces a download, from a given stream of 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) { $rangeStream = new RangeSupportedStream($stream, $fileSize, $this->request->headers);
return response()->stream(function () use ($rangeStream) {
// End & flush the output buffer, if we're in one, otherwise we still use memory. $rangeStream->outputAndClose();
// 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)); }, 200, $this->getHeaders($fileName));
} }
@ -48,15 +38,13 @@ class DownloadResponseFactory
* correct for the file, in a way so the browser can show the content in browser, * correct for the file, in a way so the browser can show the content in browser,
* for a given content stream. * 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); $rangeStream = new RangeSupportedStream($stream, $fileSize, $this->request->headers);
$mime = (new WebSafeMimeSniffer())->sniff($sniffContent); $mime = $rangeStream->sniffMime();
return response()->stream(function () use ($sniffContent, $stream) { return response()->stream(function () use ($rangeStream) {
echo $sniffContent; $rangeStream->outputAndClose();
fpassthru($stream);
fclose($stream);
}, 200, $this->getHeaders($fileName, $mime)); }, 200, $this->getHeaders($fileName, $mime));
} }

View file

@ -0,0 +1,57 @@
<?php
namespace BookStack\Http;
use BookStack\Util\WebSafeMimeSniffer;
use Symfony\Component\HttpFoundation\HeaderBag;
class RangeSupportedStream
{
protected string $sniffContent;
public function __construct(
protected $stream,
protected int $fileSize,
protected HeaderBag $requestHeaders,
) {
}
/**
* Sniff a mime type from the stream.
*/
public function sniffMime(): string
{
$offset = min(2000, $this->fileSize);
$this->sniffContent = fread($this->stream, $offset);
return (new WebSafeMimeSniffer())->sniff($this->sniffContent);
}
/**
* Output the current stream to stdout before closing out the stream.
*/
public function outputAndClose(): void
{
// 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();
}
$outStream = fopen('php://output', 'w');
$offset = 0;
if (!empty($this->sniffContent)) {
fwrite($outStream, $this->sniffContent);
$offset = strlen($this->sniffContent);
}
$toWrite = $this->fileSize - $offset;
stream_copy_to_stream($this->stream, $outStream, $toWrite);
fpassthru($this->stream);
fclose($this->stream);
fclose($outStream);
}
}

View file

@ -66,8 +66,6 @@ class AttachmentService
/** /**
* Stream an attachment from storage. * Stream an attachment from storage.
* *
* @throws FileNotFoundException
*
* @return resource|null * @return resource|null
*/ */
public function streamAttachmentFromStorage(Attachment $attachment) public function streamAttachmentFromStorage(Attachment $attachment)
@ -75,6 +73,14 @@ class AttachmentService
return $this->getStorageDisk()->readStream($this->adjustPathForStorageDisk($attachment->path)); return $this->getStorageDisk()->readStream($this->adjustPathForStorageDisk($attachment->path));
} }
/**
* Read the file size of an attachment from storage, in bytes.
*/
public function getAttachmentFileSize(Attachment $attachment): int
{
return $this->getStorageDisk()->size($this->adjustPathForStorageDisk($attachment->path));
}
/** /**
* Store a new attachment upon user upload. * Store a new attachment upon user upload.
* *

View file

@ -226,12 +226,13 @@ class AttachmentController extends Controller
$fileName = $attachment->getFileName(); $fileName = $attachment->getFileName();
$attachmentStream = $this->attachmentService->streamAttachmentFromStorage($attachment); $attachmentStream = $this->attachmentService->streamAttachmentFromStorage($attachment);
$attachmentSize = $this->attachmentService->getAttachmentFileSize($attachment);
if ($request->get('open') === 'true') { if ($request->get('open') === 'true') {
return $this->download()->streamedInline($attachmentStream, $fileName); return $this->download()->streamedInline($attachmentStream, $fileName, $attachmentSize);
} }
return $this->download()->streamedDirectly($attachmentStream, $fileName); return $this->download()->streamedDirectly($attachmentStream, $fileName, $attachmentSize);
} }
/** /**