mirror of
https://github.com/BookStackApp/BookStack.git
synced 2025-04-02 21:59:06 +08:00
Made display thumbnail generation use original data if smaller
Thumbnail generation would sometimes create a file larger than the original, if the original was already well optimized, therefore making the thumbnail counter-productive. This change compares the sizes of the original and the generated thumbnail, and uses the smaller of the two if the thumbnail does not change the aspect ratio of the image. Fixes #1751
This commit is contained in:
parent
a83a7f34f4
commit
32e7f0a2e6
@ -2,6 +2,8 @@
|
|||||||
|
|
||||||
use BookStack\Auth\Permissions\PermissionService;
|
use BookStack\Auth\Permissions\PermissionService;
|
||||||
use BookStack\Entities\Page;
|
use BookStack\Entities\Page;
|
||||||
|
use BookStack\Exceptions\ImageUploadException;
|
||||||
|
use Exception;
|
||||||
use Illuminate\Database\Eloquent\Builder;
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
use Symfony\Component\HttpFoundation\File\UploadedFile;
|
use Symfony\Component\HttpFoundation\File\UploadedFile;
|
||||||
|
|
||||||
@ -15,10 +17,6 @@ class ImageRepo
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* ImageRepo constructor.
|
* ImageRepo constructor.
|
||||||
* @param Image $image
|
|
||||||
* @param ImageService $imageService
|
|
||||||
* @param \BookStack\Auth\Permissions\PermissionService $permissionService
|
|
||||||
* @param \BookStack\Entities\Page $page
|
|
||||||
*/
|
*/
|
||||||
public function __construct(
|
public function __construct(
|
||||||
Image $image,
|
Image $image,
|
||||||
@ -35,10 +33,8 @@ class ImageRepo
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get an image with the given id.
|
* Get an image with the given id.
|
||||||
* @param $id
|
|
||||||
* @return Image
|
|
||||||
*/
|
*/
|
||||||
public function getById($id)
|
public function getById($id): Image
|
||||||
{
|
{
|
||||||
return $this->image->findOrFail($id);
|
return $this->image->findOrFail($id);
|
||||||
}
|
}
|
||||||
@ -46,13 +42,8 @@ class ImageRepo
|
|||||||
/**
|
/**
|
||||||
* Execute a paginated query, returning in a standard format.
|
* Execute a paginated query, returning in a standard format.
|
||||||
* Also runs the query through the restriction system.
|
* Also runs the query through the restriction system.
|
||||||
* @param $query
|
|
||||||
* @param int $page
|
|
||||||
* @param int $pageSize
|
|
||||||
* @param bool $filterOnPage
|
|
||||||
* @return array
|
|
||||||
*/
|
*/
|
||||||
private function returnPaginated($query, $page = 1, $pageSize = 24)
|
private function returnPaginated($query, $page = 1, $pageSize = 24): array
|
||||||
{
|
{
|
||||||
$images = $query->orderBy('created_at', 'desc')->skip($pageSize * ($page - 1))->take($pageSize + 1)->get();
|
$images = $query->orderBy('created_at', 'desc')->skip($pageSize * ($page - 1))->take($pageSize + 1)->get();
|
||||||
$hasMore = count($images) > $pageSize;
|
$hasMore = count($images) > $pageSize;
|
||||||
@ -71,13 +62,6 @@ class ImageRepo
|
|||||||
/**
|
/**
|
||||||
* Fetch a list of images in a paginated format, filtered by image type.
|
* Fetch a list of images in a paginated format, filtered by image type.
|
||||||
* Can be filtered by uploaded to and also by name.
|
* Can be filtered by uploaded to and also by name.
|
||||||
* @param string $type
|
|
||||||
* @param int $page
|
|
||||||
* @param int $pageSize
|
|
||||||
* @param int $uploadedTo
|
|
||||||
* @param string|null $search
|
|
||||||
* @param callable|null $whereClause
|
|
||||||
* @return array
|
|
||||||
*/
|
*/
|
||||||
public function getPaginatedByType(
|
public function getPaginatedByType(
|
||||||
string $type,
|
string $type,
|
||||||
@ -86,7 +70,8 @@ class ImageRepo
|
|||||||
int $uploadedTo = null,
|
int $uploadedTo = null,
|
||||||
string $search = null,
|
string $search = null,
|
||||||
callable $whereClause = null
|
callable $whereClause = null
|
||||||
) {
|
): array
|
||||||
|
{
|
||||||
$imageQuery = $this->image->newQuery()->where('type', '=', strtolower($type));
|
$imageQuery = $this->image->newQuery()->where('type', '=', strtolower($type));
|
||||||
|
|
||||||
if ($uploadedTo !== null) {
|
if ($uploadedTo !== null) {
|
||||||
@ -109,13 +94,6 @@ class ImageRepo
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get paginated gallery images within a specific page or book.
|
* Get paginated gallery images within a specific page or book.
|
||||||
* @param string $type
|
|
||||||
* @param string $filterType
|
|
||||||
* @param int $page
|
|
||||||
* @param int $pageSize
|
|
||||||
* @param int|null $uploadedTo
|
|
||||||
* @param string|null $search
|
|
||||||
* @return array
|
|
||||||
*/
|
*/
|
||||||
public function getEntityFiltered(
|
public function getEntityFiltered(
|
||||||
string $type,
|
string $type,
|
||||||
@ -124,7 +102,8 @@ class ImageRepo
|
|||||||
int $pageSize = 24,
|
int $pageSize = 24,
|
||||||
int $uploadedTo = null,
|
int $uploadedTo = null,
|
||||||
string $search = null
|
string $search = null
|
||||||
) {
|
): array
|
||||||
|
{
|
||||||
$contextPage = $this->page->findOrFail($uploadedTo);
|
$contextPage = $this->page->findOrFail($uploadedTo);
|
||||||
$parentFilter = null;
|
$parentFilter = null;
|
||||||
|
|
||||||
@ -144,16 +123,9 @@ class ImageRepo
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Save a new image into storage and return the new image.
|
* Save a new image into storage and return the new image.
|
||||||
* @param UploadedFile $uploadFile
|
* @throws ImageUploadException
|
||||||
* @param string $type
|
|
||||||
* @param int $uploadedTo
|
|
||||||
* @param int|null $resizeWidth
|
|
||||||
* @param int|null $resizeHeight
|
|
||||||
* @param bool $keepRatio
|
|
||||||
* @return Image
|
|
||||||
* @throws \BookStack\Exceptions\ImageUploadException
|
|
||||||
*/
|
*/
|
||||||
public function saveNew(UploadedFile $uploadFile, $type, $uploadedTo = 0, int $resizeWidth = null, int $resizeHeight = null, bool $keepRatio = true)
|
public function saveNew(UploadedFile $uploadFile, string $type, int $uploadedTo = 0, int $resizeWidth = null, int $resizeHeight = null, bool $keepRatio = true): Image
|
||||||
{
|
{
|
||||||
$image = $this->imageService->saveNewFromUpload($uploadFile, $type, $uploadedTo, $resizeWidth, $resizeHeight, $keepRatio);
|
$image = $this->imageService->saveNewFromUpload($uploadFile, $type, $uploadedTo, $resizeWidth, $resizeHeight, $keepRatio);
|
||||||
$this->loadThumbs($image);
|
$this->loadThumbs($image);
|
||||||
@ -161,29 +133,22 @@ class ImageRepo
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Save a drawing the the database;
|
* Save a drawing the the database.
|
||||||
* @param string $base64Uri
|
* @throws ImageUploadException
|
||||||
* @param int $uploadedTo
|
|
||||||
* @return Image
|
|
||||||
* @throws \BookStack\Exceptions\ImageUploadException
|
|
||||||
*/
|
*/
|
||||||
public function saveDrawing(string $base64Uri, int $uploadedTo)
|
public function saveDrawing(string $base64Uri, int $uploadedTo): Image
|
||||||
{
|
{
|
||||||
$name = 'Drawing-' . user()->getShortName(40) . '-' . strval(time()) . '.png';
|
$name = 'Drawing-' . user()->getShortName(40) . '-' . strval(time()) . '.png';
|
||||||
$image = $this->imageService->saveNewFromBase64Uri($base64Uri, $name, 'drawio', $uploadedTo);
|
return $this->imageService->saveNewFromBase64Uri($base64Uri, $name, 'drawio', $uploadedTo);
|
||||||
return $image;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update the details of an image via an array of properties.
|
* Update the details of an image via an array of properties.
|
||||||
* @param Image $image
|
* @throws ImageUploadException
|
||||||
* @param array $updateDetails
|
* @throws Exception
|
||||||
* @return Image
|
|
||||||
* @throws \BookStack\Exceptions\ImageUploadException
|
|
||||||
* @throws \Exception
|
|
||||||
*/
|
*/
|
||||||
public function updateImageDetails(Image $image, $updateDetails)
|
public function updateImageDetails(Image $image, $updateDetails): Image
|
||||||
{
|
{
|
||||||
$image->fill($updateDetails);
|
$image->fill($updateDetails);
|
||||||
$image->save();
|
$image->save();
|
||||||
@ -191,14 +156,11 @@ class ImageRepo
|
|||||||
return $image;
|
return $image;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Destroys an Image object along with its revisions, files and thumbnails.
|
* Destroys an Image object along with its revisions, files and thumbnails.
|
||||||
* @param Image $image
|
* @throws Exception
|
||||||
* @return bool
|
|
||||||
* @throws \Exception
|
|
||||||
*/
|
*/
|
||||||
public function destroyImage(Image $image = null)
|
public function destroyImage(Image $image = null): bool
|
||||||
{
|
{
|
||||||
if ($image) {
|
if ($image) {
|
||||||
$this->imageService->destroy($image);
|
$this->imageService->destroy($image);
|
||||||
@ -208,8 +170,7 @@ class ImageRepo
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Destroy all images of a certain type.
|
* Destroy all images of a certain type.
|
||||||
* @param string $imageType
|
* @throws Exception
|
||||||
* @throws \Exception
|
|
||||||
*/
|
*/
|
||||||
public function destroyByType(string $imageType)
|
public function destroyByType(string $imageType)
|
||||||
{
|
{
|
||||||
@ -222,9 +183,7 @@ class ImageRepo
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Load thumbnails onto an image object.
|
* Load thumbnails onto an image object.
|
||||||
* @param Image $image
|
* @throws Exception
|
||||||
* @throws \BookStack\Exceptions\ImageUploadException
|
|
||||||
* @throws \Exception
|
|
||||||
*/
|
*/
|
||||||
protected function loadThumbs(Image $image)
|
protected function loadThumbs(Image $image)
|
||||||
{
|
{
|
||||||
@ -238,42 +197,33 @@ class ImageRepo
|
|||||||
* Get the thumbnail for an image.
|
* Get the thumbnail for an image.
|
||||||
* If $keepRatio is true only the width will be used.
|
* If $keepRatio is true only the width will be used.
|
||||||
* Checks the cache then storage to avoid creating / accessing the filesystem on every check.
|
* Checks the cache then storage to avoid creating / accessing the filesystem on every check.
|
||||||
* @param Image $image
|
* @throws Exception
|
||||||
* @param int $width
|
|
||||||
* @param int $height
|
|
||||||
* @param bool $keepRatio
|
|
||||||
* @return string
|
|
||||||
* @throws \BookStack\Exceptions\ImageUploadException
|
|
||||||
* @throws \Exception
|
|
||||||
*/
|
*/
|
||||||
protected function getThumbnail(Image $image, $width = 220, $height = 220, $keepRatio = false)
|
protected function getThumbnail(Image $image, ?int $width = 220, ?int $height = 220, bool $keepRatio = false): ?string
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
return $this->imageService->getThumbnail($image, $width, $height, $keepRatio);
|
return $this->imageService->getThumbnail($image, $width, $height, $keepRatio);
|
||||||
} catch (\Exception $exception) {
|
} catch (Exception $exception) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the raw image data from an Image.
|
* Get the raw image data from an Image.
|
||||||
* @param Image $image
|
|
||||||
* @return null|string
|
|
||||||
*/
|
*/
|
||||||
public function getImageData(Image $image)
|
public function getImageData(Image $image): ?string
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
return $this->imageService->getImageData($image);
|
return $this->imageService->getImageData($image);
|
||||||
} catch (\Exception $exception) {
|
} catch (Exception $exception) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the validation rules for image files.
|
* Get the validation rules for image files.
|
||||||
* @return string
|
|
||||||
*/
|
*/
|
||||||
public function getImageValidationRules()
|
public function getImageValidationRules(): string
|
||||||
{
|
{
|
||||||
return 'image_extension|no_double_extension|mimes:jpeg,png,gif,bmp,webp,tiff';
|
return 'image_extension|no_double_extension|mimes:jpeg,png,gif,bmp,webp,tiff';
|
||||||
}
|
}
|
||||||
|
@ -254,7 +254,16 @@ class ImageService extends UploadService
|
|||||||
} else {
|
} else {
|
||||||
$thumb->fit($width, $height);
|
$thumb->fit($width, $height);
|
||||||
}
|
}
|
||||||
return (string)$thumb->encode();
|
|
||||||
|
$thumbData = (string)$thumb->encode();
|
||||||
|
|
||||||
|
// Use original image data if we're keeping the ratio
|
||||||
|
// and the resizing does not save any space.
|
||||||
|
if ($keepRatio && strlen($thumbData) > strlen($imageData)) {
|
||||||
|
return $imageData;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $thumbData;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -36,6 +36,30 @@ class ImageTest extends TestCase
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function test_image_display_thumbnail_generation_does_not_increase_image_size()
|
||||||
|
{
|
||||||
|
$page = Page::first();
|
||||||
|
$admin = $this->getAdmin();
|
||||||
|
$this->actingAs($admin);
|
||||||
|
|
||||||
|
$originalFile = $this->getTestImageFilePath('compressed.png');
|
||||||
|
$originalFileSize = filesize($originalFile);
|
||||||
|
$imgDetails = $this->uploadGalleryImage($page, 'compressed.png');
|
||||||
|
$relPath = $imgDetails['path'];
|
||||||
|
|
||||||
|
$this->assertTrue(file_exists(public_path($relPath)), 'Uploaded image found at path: '. public_path($relPath));
|
||||||
|
$displayImage = $imgDetails['response']->thumbs->display;
|
||||||
|
|
||||||
|
$displayImageRelPath = implode('/', array_slice(explode('/', $displayImage), 3));
|
||||||
|
$displayImagePath = public_path($displayImageRelPath);
|
||||||
|
$displayFileSize = filesize($displayImagePath);
|
||||||
|
|
||||||
|
$this->deleteImage($relPath);
|
||||||
|
$this->deleteImage($displayImageRelPath);
|
||||||
|
|
||||||
|
$this->assertEquals($originalFileSize, $displayFileSize, 'Display thumbnail generation should not increase image size');
|
||||||
|
}
|
||||||
|
|
||||||
public function test_image_edit()
|
public function test_image_edit()
|
||||||
{
|
{
|
||||||
$editor = $this->getEditor();
|
$editor = $this->getEditor();
|
||||||
|
@ -10,9 +10,13 @@ trait UsesImages
|
|||||||
* Get the path to our basic test image.
|
* Get the path to our basic test image.
|
||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
protected function getTestImageFilePath()
|
protected function getTestImageFilePath(?string $fileName = null)
|
||||||
{
|
{
|
||||||
return base_path('tests/test-data/test-image.png');
|
if (is_null($fileName)) {
|
||||||
|
$fileName = 'test-image.png';
|
||||||
|
}
|
||||||
|
|
||||||
|
return base_path('tests/test-data/' . $fileName);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -20,9 +24,9 @@ trait UsesImages
|
|||||||
* @param $fileName
|
* @param $fileName
|
||||||
* @return UploadedFile
|
* @return UploadedFile
|
||||||
*/
|
*/
|
||||||
protected function getTestImage($fileName)
|
protected function getTestImage($fileName, ?string $testDataFileName = null)
|
||||||
{
|
{
|
||||||
return new UploadedFile($this->getTestImageFilePath(), $fileName, 'image/png', 5238, null, true);
|
return new UploadedFile($this->getTestImageFilePath($testDataFileName), $fileName, 'image/png', 5238, null, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -52,9 +56,9 @@ trait UsesImages
|
|||||||
* @param string $contentType
|
* @param string $contentType
|
||||||
* @return \Illuminate\Foundation\Testing\TestResponse
|
* @return \Illuminate\Foundation\Testing\TestResponse
|
||||||
*/
|
*/
|
||||||
protected function uploadImage($name, $uploadedTo = 0, $contentType = 'image/png')
|
protected function uploadImage($name, $uploadedTo = 0, $contentType = 'image/png', ?string $testDataFileName = null)
|
||||||
{
|
{
|
||||||
$file = $this->getTestImage($name);
|
$file = $this->getTestImage($name, $testDataFileName);
|
||||||
return $this->withHeader('Content-Type', $contentType)
|
return $this->withHeader('Content-Type', $contentType)
|
||||||
->call('POST', '/images/gallery', ['uploaded_to' => $uploadedTo], [], ['file' => $file], []);
|
->call('POST', '/images/gallery', ['uploaded_to' => $uploadedTo], [], ['file' => $file], []);
|
||||||
}
|
}
|
||||||
@ -66,22 +70,23 @@ trait UsesImages
|
|||||||
* @param Page|null $page
|
* @param Page|null $page
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
protected function uploadGalleryImage(Page $page = null)
|
protected function uploadGalleryImage(Page $page = null, ?string $testDataFileName = null)
|
||||||
{
|
{
|
||||||
if ($page === null) {
|
if ($page === null) {
|
||||||
$page = Page::query()->first();
|
$page = Page::query()->first();
|
||||||
}
|
}
|
||||||
|
|
||||||
$imageName = 'first-image.png';
|
$imageName = $testDataFileName ?? 'first-image.png';
|
||||||
$relPath = $this->getTestImagePath('gallery', $imageName);
|
$relPath = $this->getTestImagePath('gallery', $imageName);
|
||||||
$this->deleteImage($relPath);
|
$this->deleteImage($relPath);
|
||||||
|
|
||||||
$upload = $this->uploadImage($imageName, $page->id);
|
$upload = $this->uploadImage($imageName, $page->id, 'image/png', $testDataFileName);
|
||||||
$upload->assertStatus(200);
|
$upload->assertStatus(200);
|
||||||
return [
|
return [
|
||||||
'name' => $imageName,
|
'name' => $imageName,
|
||||||
'path' => $relPath,
|
'path' => $relPath,
|
||||||
'page' => $page
|
'page' => $page,
|
||||||
|
'response' => json_decode($upload->getContent()),
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BIN
tests/test-data/compressed.png
Normal file
BIN
tests/test-data/compressed.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 732 B |
Loading…
x
Reference in New Issue
Block a user