Skip to content

Commit

Permalink
Merge pull request #5379 from BookStackApp/better_cleanup
Browse files Browse the repository at this point in the history
Export limits and cleanup
  • Loading branch information
ssddanbrown authored Jan 4, 2025
2 parents ff6c5aa + 1ff2826 commit 6effc6d
Show file tree
Hide file tree
Showing 9 changed files with 130 additions and 10 deletions.
7 changes: 7 additions & 0 deletions app/App/Providers/RouteServiceProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -85,5 +85,12 @@ protected function configureRateLimiting(): void
RateLimiter::for('public', function (Request $request) {
return Limit::perMinute(10)->by($request->ip());
});

RateLimiter::for('exports', function (Request $request) {
$user = user();
$attempts = $user->isGuest() ? 4 : 10;
$key = $user->isGuest() ? $request->ip() : $user->id;
return Limit::perMinute($attempts)->by($key);
});
}
}
3 changes: 2 additions & 1 deletion app/Exports/Controllers/BookExportController.php
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ public function __construct(
protected ExportFormatter $exportFormatter,
) {
$this->middleware('can:content-export');
$this->middleware('throttle:exports');
}

/**
Expand Down Expand Up @@ -75,6 +76,6 @@ public function zip(string $bookSlug, ZipExportBuilder $builder)
$book = $this->queries->findVisibleBySlugOrFail($bookSlug);
$zip = $builder->buildForBook($book);

return $this->download()->streamedDirectly(fopen($zip, 'r'), $bookSlug . '.zip', filesize($zip));
return $this->download()->streamedFileDirectly($zip, $bookSlug . '.zip', filesize($zip), true);
}
}
3 changes: 2 additions & 1 deletion app/Exports/Controllers/ChapterExportController.php
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ public function __construct(
protected ExportFormatter $exportFormatter,
) {
$this->middleware('can:content-export');
$this->middleware('throttle:exports');
}

/**
Expand Down Expand Up @@ -81,6 +82,6 @@ public function zip(string $bookSlug, string $chapterSlug, ZipExportBuilder $bui
$chapter = $this->queries->findVisibleBySlugsOrFail($bookSlug, $chapterSlug);
$zip = $builder->buildForChapter($chapter);

return $this->download()->streamedDirectly(fopen($zip, 'r'), $chapterSlug . '.zip', filesize($zip));
return $this->download()->streamedFileDirectly($zip, $chapterSlug . '.zip', filesize($zip), true);
}
}
3 changes: 2 additions & 1 deletion app/Exports/Controllers/PageExportController.php
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ public function __construct(
protected ExportFormatter $exportFormatter,
) {
$this->middleware('can:content-export');
$this->middleware('throttle:exports');
}

/**
Expand Down Expand Up @@ -85,6 +86,6 @@ public function zip(string $bookSlug, string $pageSlug, ZipExportBuilder $builde
$page = $this->queries->findVisibleBySlugsOrFail($bookSlug, $pageSlug);
$zip = $builder->buildForPage($page);

return $this->download()->streamedDirectly(fopen($zip, 'r'), $pageSlug . '.zip', filesize($zip));
return $this->download()->streamedFileDirectly($zip, $pageSlug . '.zip', filesize($zip), true);
}
}
12 changes: 11 additions & 1 deletion app/Exports/PdfGenerator.php
Original file line number Diff line number Diff line change
Expand Up @@ -90,18 +90,28 @@ protected function renderUsingCommand(string $html): string
$process = Process::fromShellCommandline($command);
$process->setTimeout($timeout);

$cleanup = function () use ($inputHtml, $outputPdf) {
foreach ([$inputHtml, $outputPdf] as $file) {
if (file_exists($file)) {
unlink($file);
}
}
};

try {
$process->run();
} catch (ProcessTimedOutException $e) {
$cleanup();
throw new PdfExportException("PDF Export via command failed due to timeout at {$timeout} second(s)");
}

if (!$process->isSuccessful()) {
$cleanup();
throw new PdfExportException("PDF Export via command failed with exit code {$process->getExitCode()}, stdout: {$process->getOutput()}, stderr: {$process->getErrorOutput()}");
}

$pdfContents = file_get_contents($outputPdf);
unlink($outputPdf);
$cleanup();

if ($pdfContents === false) {
throw new PdfExportException("PDF Export via command failed, unable to read PDF output file");
Expand Down
25 changes: 21 additions & 4 deletions app/Exports/ZipExports/ZipExportBuilder.php
Original file line number Diff line number Diff line change
Expand Up @@ -84,10 +84,27 @@ protected function build(): string
$zip->addEmptyDir('files');

$toRemove = [];
$this->files->extractEach(function ($filePath, $fileRef) use ($zip, &$toRemove) {
$zip->addFile($filePath, "files/$fileRef");
$toRemove[] = $filePath;
});
$addedNames = [];

try {
$this->files->extractEach(function ($filePath, $fileRef) use ($zip, &$toRemove, &$addedNames) {
$entryName = "files/$fileRef";
$zip->addFile($filePath, $entryName);
$toRemove[] = $filePath;
$addedNames[] = $entryName;
});
} catch (\Exception $exception) {
// Cleanup the files we've processed so far and respond back with error
foreach ($toRemove as $file) {
unlink($file);
}
foreach ($addedNames as $name) {
$zip->deleteName($name);
}
$zip->close();
unlink($zipFile);
throw new ZipExportException("Failed to add files for ZIP export, received error: " . $exception->getMessage());
}

$zip->close();

Expand Down
28 changes: 27 additions & 1 deletion app/Http/DownloadResponseFactory.php
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
class DownloadResponseFactory
{
public function __construct(
protected Request $request
protected Request $request,
) {
}

Expand All @@ -35,6 +35,32 @@ public function streamedDirectly($stream, string $fileName, int $fileSize): Stre
);
}

/**
* Create a response that downloads the given file via a stream.
* Has the option to delete the provided file once the stream is closed.
*/
public function streamedFileDirectly(string $filePath, string $fileName, int $fileSize, bool $deleteAfter = false): StreamedResponse
{
$stream = fopen($filePath, 'r');

if ($deleteAfter) {
// Delete the given file if it still exists after the app terminates
$callback = function () use ($filePath) {
if (file_exists($filePath)) {
unlink($filePath);
}
};

// We watch both app terminate and php shutdown to cover both normal app termination
// as well as other potential scenarios (connection termination).
app()->terminating($callback);
register_shutdown_function($callback);
}

return $this->streamedDirectly($stream, $fileName, $fileSize);
}


/**
* 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,
Expand Down
18 changes: 17 additions & 1 deletion tests/Exports/PdfExportTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
use BookStack\Entities\Models\Page;
use BookStack\Exceptions\PdfExportException;
use BookStack\Exports\PdfGenerator;
use FilesystemIterator;
use Tests\TestCase;

class PdfExportTest extends TestCase
Expand Down Expand Up @@ -128,7 +129,7 @@ public function test_pdf_command_option_errors_if_command_returns_error_status()
}, PdfExportException::class);
}

public function test_pdf_command_timout_option_limits_export_time()
public function test_pdf_command_timeout_option_limits_export_time()
{
$page = $this->entities->page();
$command = 'php -r \'sleep(4);\'';
Expand All @@ -143,4 +144,19 @@ public function test_pdf_command_timout_option_limits_export_time()
}, PdfExportException::class,
"PDF Export via command failed due to timeout at 1 second(s)");
}

public function test_pdf_command_option_does_not_leave_temp_files()
{
$tempDir = sys_get_temp_dir();
$startTempFileCount = iterator_count((new FileSystemIterator($tempDir, FilesystemIterator::SKIP_DOTS)));

$page = $this->entities->page();
$command = 'cp {input_html_path} {output_pdf_path}';
config()->set('exports.pdf_command', $command);

$this->asEditor()->get($page->getUrl('/export/pdf'));

$afterTempFileCount = iterator_count((new FileSystemIterator($tempDir, FilesystemIterator::SKIP_DOTS)));
$this->assertEquals($startTempFileCount, $afterTempFileCount);
}
}
41 changes: 41 additions & 0 deletions tests/Exports/ZipExportTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
use BookStack\Entities\Tools\PageContent;
use BookStack\Uploads\Attachment;
use BookStack\Uploads\Image;
use FilesystemIterator;
use Illuminate\Support\Carbon;
use Illuminate\Testing\TestResponse;
use Tests\TestCase;
Expand Down Expand Up @@ -60,6 +61,24 @@ public function test_export_metadata()
$this->assertEquals($instanceId, $zipInstanceId);
}

public function test_export_leaves_no_temp_files()
{
$tempDir = sys_get_temp_dir();
$startTempFileCount = iterator_count((new FileSystemIterator($tempDir, FilesystemIterator::SKIP_DOTS)));

$page = $this->entities->pageWithinChapter();
$this->asEditor();
$pageResp = $this->get($page->getUrl("/export/zip"));
$pageResp->streamedContent();
$pageResp->assertOk();
$this->get($page->chapter->getUrl("/export/zip"))->assertOk();
$this->get($page->book->getUrl("/export/zip"))->assertOk();

$afterTempFileCount = iterator_count((new FileSystemIterator($tempDir, FilesystemIterator::SKIP_DOTS)));

$this->assertEquals($startTempFileCount, $afterTempFileCount);
}

public function test_page_export()
{
$page = $this->entities->page();
Expand Down Expand Up @@ -404,6 +423,28 @@ public function test_links_in_markdown_are_parsed()
$this->assertStringContainsString("[Link to chapter]([[bsexport:chapter:{$chapter->id}]])", $pageData['markdown']);
}

public function test_exports_rate_limited_low_for_guest_viewers()
{
$this->setSettings(['app-public' => 'true']);

$page = $this->entities->page();
for ($i = 0; $i < 4; $i++) {
$this->get($page->getUrl("/export/zip"))->assertOk();
}
$this->get($page->getUrl("/export/zip"))->assertStatus(429);
}

public function test_exports_rate_limited_higher_for_logged_in_viewers()
{
$this->asAdmin();

$page = $this->entities->page();
for ($i = 0; $i < 10; $i++) {
$this->get($page->getUrl("/export/zip"))->assertOk();
}
$this->get($page->getUrl("/export/zip"))->assertStatus(429);
}

protected function extractZipResponse(TestResponse $response): ZipResultData
{
$zipData = $response->streamedContent();
Expand Down

0 comments on commit 6effc6d

Please sign in to comment.