Refactored thumbs; fixed setting custom avatars
This commit is contained in:
parent
7e492e044c
commit
109aa1c39e
13 changed files with 159 additions and 22 deletions
1
data/.gitignore
vendored
1
data/.gitignore
vendored
|
@ -1 +1,2 @@
|
|||
local.ini
|
||||
thumbnails
|
||||
|
|
|
@ -61,7 +61,7 @@ final class UserAvatarController extends AbstractController
|
|||
if (!$this->fileService->exists($file))
|
||||
$file = $this->userService->getBlankAvatarSourcePath();
|
||||
|
||||
$sizedFile = $this->thumbnailService->generateFromFile($file, $size, $size);
|
||||
$sizedFile = $this->thumbnailService->getOrGenerate($file, $size, $size);
|
||||
$this->fileService->serve($sizedFile);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -59,6 +59,13 @@ class FileService
|
|||
exit;
|
||||
}
|
||||
|
||||
public function createFolders($target)
|
||||
{
|
||||
$finalTarget = $this->getFullPath($target);
|
||||
if (!file_exists($finalTarget))
|
||||
mkdir($finalTarget, 0777, true);
|
||||
}
|
||||
|
||||
public function exists($source)
|
||||
{
|
||||
$finalSource = $this->getFullPath($source);
|
||||
|
|
|
@ -3,5 +3,5 @@ namespace Szurubooru\Services\ThumbnailGenerators;
|
|||
|
||||
interface IThumbnailGenerator
|
||||
{
|
||||
public function generateFromFile($srcPath, $dstPath, $width, $height);
|
||||
public function generate($srcPath, $dstPath, $width, $height);
|
||||
}
|
||||
|
|
|
@ -10,7 +10,7 @@ class ImageGdThumbnailGenerator implements IThumbnailGenerator
|
|||
$this->config = $config;
|
||||
}
|
||||
|
||||
public function generateFromFile($srcPath, $dstPath, $width, $height)
|
||||
public function generate($srcPath, $dstPath, $width, $height)
|
||||
{
|
||||
if (!file_exists($srcPath))
|
||||
throw new \InvalidArgumentException($srcPath . ' does not exist');
|
||||
|
|
|
@ -10,7 +10,7 @@ class ImageImagickThumbnailGenerator implements IThumbnailGenerator
|
|||
$this->config = $config;
|
||||
}
|
||||
|
||||
public function generateFromFile($srcPath, $dstPath, $width, $height)
|
||||
public function generate($srcPath, $dstPath, $width, $height)
|
||||
{
|
||||
if (!file_exists($srcPath))
|
||||
throw new \InvalidArgumentException($srcPath . ' does not exist');
|
||||
|
|
|
@ -14,7 +14,7 @@ class ImageThumbnailGenerator implements IThumbnailGenerator
|
|||
$this->imageGdThumbnailGenerator = $imageGdThumbnailGenerator;
|
||||
}
|
||||
|
||||
public function generateFromFile($srcPath, $dstPath, $width, $height)
|
||||
public function generate($srcPath, $dstPath, $width, $height)
|
||||
{
|
||||
if (extension_loaded('imagick'))
|
||||
$strategy = $this->imageImagickThumbnailGenerator;
|
||||
|
@ -23,6 +23,6 @@ class ImageThumbnailGenerator implements IThumbnailGenerator
|
|||
else
|
||||
throw new \Exception('Both imagick and gd extensions are disabled');
|
||||
|
||||
return $strategy->generateFromFile($srcPath, $dstPath, $width, $height);
|
||||
return $strategy->generate($srcPath, $dstPath, $width, $height);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -14,17 +14,58 @@ class ThumbnailService
|
|||
$this->thumbnailGenerator = $thumbnailGenerator;
|
||||
}
|
||||
|
||||
public function generateFromFile($source, $width, $height)
|
||||
public function getOrGenerate($source, $width, $height)
|
||||
{
|
||||
$target = $source . '-thumb' . $width . 'x' . $height . '.jpg';
|
||||
$target = $this->getPath($source, $width, $height);
|
||||
|
||||
if (!$this->fileService->exists($target))
|
||||
{
|
||||
$fullSource = $this->fileService->getFullPath($source);
|
||||
$fullTarget = $this->fileService->getFullPath($target);
|
||||
$this->thumbnailGenerator->generateFromFile($fullSource, $fullTarget, $width, $height);
|
||||
}
|
||||
$this->generate($source, $width, $height);
|
||||
|
||||
return $target;
|
||||
}
|
||||
|
||||
public function deleteUsedThumbnails($source)
|
||||
{
|
||||
foreach ($this->getUsedThumbnailSizes() as $size)
|
||||
{
|
||||
list ($width, $height) = $size;
|
||||
$target = $this->getPath($source, $width, $height);
|
||||
if ($this->fileService->exists($target))
|
||||
$this->fileService->delete($target);
|
||||
}
|
||||
}
|
||||
|
||||
public function generate($source, $width, $height)
|
||||
{
|
||||
$target = $this->getPath($source, $width, $height);
|
||||
|
||||
$fullSource = $this->fileService->getFullPath($source);
|
||||
$fullTarget = $this->fileService->getFullPath($target);
|
||||
$this->fileService->createFolders(dirname($target));
|
||||
$this->thumbnailGenerator->generate($fullSource, $fullTarget, $width, $height);
|
||||
|
||||
return $target;
|
||||
}
|
||||
|
||||
public function getUsedThumbnailSizes()
|
||||
{
|
||||
foreach (glob($this->fileService->getFullPath('thumbnails') . DIRECTORY_SEPARATOR . '*x*') as $fn)
|
||||
{
|
||||
if (!is_dir($fn))
|
||||
continue;
|
||||
|
||||
preg_match('/(?P<width>\d+)x(?P<height>\d+)/', $fn, $matches);
|
||||
if ($matches)
|
||||
{
|
||||
$width = intval($matches['width']);
|
||||
$height = intval($matches['height']);
|
||||
yield [$width, $height];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private function getPath($source, $width, $height)
|
||||
{
|
||||
return 'thumbnails' . DIRECTORY_SEPARATOR . $width . 'x' . $height . DIRECTORY_SEPARATOR . $source;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -10,6 +10,7 @@ class UserService
|
|||
private $passwordService;
|
||||
private $emailService;
|
||||
private $fileService;
|
||||
private $thumbnailService;
|
||||
private $timeService;
|
||||
private $tokenService;
|
||||
|
||||
|
@ -21,6 +22,7 @@ class UserService
|
|||
\Szurubooru\Services\PasswordService $passwordService,
|
||||
\Szurubooru\Services\EmailService $emailService,
|
||||
\Szurubooru\Services\FileService $fileService,
|
||||
\Szurubooru\Services\ThumbnailService $thumbnailService,
|
||||
\Szurubooru\Services\TimeService $timeService,
|
||||
\Szurubooru\Services\TokenService $tokenService)
|
||||
{
|
||||
|
@ -31,6 +33,7 @@ class UserService
|
|||
$this->passwordService = $passwordService;
|
||||
$this->emailService = $emailService;
|
||||
$this->fileService = $fileService;
|
||||
$this->thumbnailService = $thumbnailService;
|
||||
$this->timeService = $timeService;
|
||||
$this->tokenService = $tokenService;
|
||||
}
|
||||
|
@ -106,7 +109,11 @@ class UserService
|
|||
{
|
||||
$user->avatarStyle = \Szurubooru\Helpers\EnumHelper::avatarStyleFromString($formData->avatarStyle);
|
||||
if ($formData->avatarContent)
|
||||
$this->fileService->saveFromBase64($formData->avatarContent, $this->getCustomAvatarSourcePath($user));
|
||||
{
|
||||
$target = $this->getCustomAvatarSourcePath($user);
|
||||
$this->fileService->saveFromBase64($formData->avatarContent, $target);
|
||||
$this->thumbnailService->deleteUsedThumbnails($target);
|
||||
}
|
||||
}
|
||||
|
||||
if ($formData->userName !== null and $formData->userName !== $user->name)
|
||||
|
@ -155,7 +162,10 @@ class UserService
|
|||
public function deleteUser(\Szurubooru\Entities\User $user)
|
||||
{
|
||||
$this->userDao->deleteById($user->id);
|
||||
$this->fileService->delete($this->getCustomAvatarSourcePath($user));
|
||||
|
||||
$avatarSource = $this->getCustomAvatarSourcePath($user);
|
||||
$this->fileService->delete($avatarSource);
|
||||
$this->thumbnailService->deleteUsedThumbnails($avatarSource);
|
||||
}
|
||||
|
||||
public function getCustomAvatarSourcePath(\Szurubooru\Entities\User $user)
|
||||
|
|
|
@ -13,9 +13,12 @@ abstract class AbstractTestCase extends \PHPUnit_Framework_TestCase
|
|||
return new ConfigMock();
|
||||
}
|
||||
|
||||
public function getTestDirectory()
|
||||
public function createTestDirectory()
|
||||
{
|
||||
return __DIR__ . DIRECTORY_SEPARATOR . 'files';
|
||||
$path = $this->getTestDirectoryPath();
|
||||
if (!file_exists($path))
|
||||
mkdir($path, 0777, true);
|
||||
return $path;
|
||||
}
|
||||
|
||||
protected function tearDown()
|
||||
|
@ -23,11 +26,31 @@ abstract class AbstractTestCase extends \PHPUnit_Framework_TestCase
|
|||
$this->cleanTestDirectory();
|
||||
}
|
||||
|
||||
private function getTestDirectoryPath()
|
||||
{
|
||||
return __DIR__ . DIRECTORY_SEPARATOR . 'files';
|
||||
}
|
||||
|
||||
private function cleanTestDirectory()
|
||||
{
|
||||
foreach (scandir($this->getTestDirectory()) as $fn)
|
||||
if ($fn{0} != '.')
|
||||
unlink($this->getTestDirectory() . DIRECTORY_SEPARATOR . $fn);
|
||||
if (!file_exists($this->getTestDirectoryPath()))
|
||||
return;
|
||||
|
||||
$dirIterator = new \RecursiveDirectoryIterator(
|
||||
$this->getTestDirectoryPath(),
|
||||
\RecursiveDirectoryIterator::SKIP_DOTS);
|
||||
|
||||
$files = new \RecursiveIteratorIterator(
|
||||
$dirIterator,
|
||||
\RecursiveIteratorIterator::CHILD_FIRST);
|
||||
|
||||
foreach ($files as $fileInfo)
|
||||
{
|
||||
if ($fileInfo->isDir())
|
||||
rmdir($fileInfo->getRealPath());
|
||||
else
|
||||
unlink($fileInfo->getRealPath());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -5,12 +5,13 @@ class FileServiceTest extends \Szurubooru\Tests\AbstractTestCase
|
|||
{
|
||||
public function testSaving()
|
||||
{
|
||||
$testDirectory = $this->createTestDirectory();
|
||||
$httpHelper = $this->mock( \Szurubooru\Helpers\HttpHelper::class);
|
||||
$fileService = new \Szurubooru\Services\FileService($this->getTestDirectory(), $httpHelper);
|
||||
$fileService = new \Szurubooru\Services\FileService($testDirectory, $httpHelper);
|
||||
$input = 'data:text/plain,YXdlc29tZSBkb2c=';
|
||||
$fileService->saveFromBase64($input, 'dog.txt');
|
||||
$expected = 'awesome dog';
|
||||
$actual = file_get_contents($this->getTestDirectory() . DIRECTORY_SEPARATOR . 'dog.txt');
|
||||
$actual = file_get_contents($testDirectory . DIRECTORY_SEPARATOR . 'dog.txt');
|
||||
$this->assertEquals($expected, $actual);
|
||||
}
|
||||
}
|
||||
|
|
51
tests/Services/ThumbnailServiceTest.php
Normal file
51
tests/Services/ThumbnailServiceTest.php
Normal file
|
@ -0,0 +1,51 @@
|
|||
<?php
|
||||
namespace Szurubooru\Tests\Services;
|
||||
|
||||
class ThumbnailServiceTest extends \Szurubooru\Tests\AbstractTestCase
|
||||
{
|
||||
public function testDeleteUsedThumbnails()
|
||||
{
|
||||
define('DS', DIRECTORY_SEPARATOR);
|
||||
|
||||
$tempDirectory = $this->createTestDirectory();
|
||||
mkdir($tempDirectory . DS . 'thumbnails');
|
||||
mkdir($tempDirectory . DS . 'thumbnails' . DS . '5x5');
|
||||
mkdir($tempDirectory . DS . 'thumbnails' . DS . '10x10');
|
||||
touch($tempDirectory . DS . 'thumbnails' . DS . '5x5' . DS . 'remove');
|
||||
touch($tempDirectory . DS . 'thumbnails' . DS . '5x5' . DS . 'keep');
|
||||
touch($tempDirectory . DS . 'thumbnails' . DS . '10x10' . DS . 'remove');
|
||||
|
||||
$httpHelperMock = $this->mock(\Szurubooru\Helpers\HttpHelper::class);
|
||||
$fileService = new \Szurubooru\Services\FileService($tempDirectory, $httpHelperMock);
|
||||
$thumbnailGeneratorMock = $this->mock(\Szurubooru\Services\ThumbnailGenerators\SmartThumbnailGenerator::class);
|
||||
|
||||
$thumbnailService = new \Szurubooru\Services\ThumbnailService($fileService, $thumbnailGeneratorMock);
|
||||
$thumbnailService->deleteUsedThumbnails('remove');
|
||||
|
||||
$this->assertFalse(file_exists($tempDirectory . DS . 'thumbnails' . DS . '5x5' . DS . 'remove'));
|
||||
$this->assertTrue(file_exists($tempDirectory . DS . 'thumbnails' . DS . '5x5' . DS . 'keep'));
|
||||
$this->assertFalse(file_exists($tempDirectory . DS . 'thumbnails' . DS . '10x10' . DS . 'remove'));
|
||||
}
|
||||
|
||||
public function testGetUsedThumbnailSizes()
|
||||
{
|
||||
$tempDirectory = $this->createTestDirectory();
|
||||
mkdir($tempDirectory . DIRECTORY_SEPARATOR . '5x5');
|
||||
mkdir($tempDirectory . DIRECTORY_SEPARATOR . '10x10');
|
||||
mkdir($tempDirectory . DIRECTORY_SEPARATOR . 'something unexpected');
|
||||
touch($tempDirectory . DIRECTORY_SEPARATOR . '15x15');
|
||||
|
||||
$fileServiceMock = $this->mock(\Szurubooru\Services\FileService::class);
|
||||
$fileServiceMock->expects($this->once())->method('getFullPath')->willReturn($tempDirectory);
|
||||
$thumbnailGeneratorMock = $this->mock(\Szurubooru\Services\ThumbnailGenerators\SmartThumbnailGenerator::class);
|
||||
|
||||
$thumbnailService = new \Szurubooru\Services\ThumbnailService($fileServiceMock, $thumbnailGeneratorMock);
|
||||
|
||||
$expected = [[5, 5], [10, 10]];
|
||||
$actual = iterator_to_array($thumbnailService->getUsedThumbnailSizes());
|
||||
|
||||
$this->assertEquals(count($expected), count($actual));
|
||||
foreach ($expected as $v)
|
||||
$this->assertContains($v, $actual);
|
||||
}
|
||||
}
|
|
@ -10,6 +10,7 @@ final class UserServiceTest extends \Szurubooru\Tests\AbstractTestCase
|
|||
private $passwordServiceMock;
|
||||
private $emailServiceMock;
|
||||
private $fileServiceMock;
|
||||
private $thumbnailServiceMock;
|
||||
private $timeServiceMock;
|
||||
private $tokenServiceMock;
|
||||
|
||||
|
@ -22,6 +23,7 @@ final class UserServiceTest extends \Szurubooru\Tests\AbstractTestCase
|
|||
$this->passwordServiceMock = $this->mock(\Szurubooru\Services\PasswordService::class);
|
||||
$this->emailServiceMock = $this->mock(\Szurubooru\Services\EmailService::class);
|
||||
$this->fileServiceMock = $this->mock(\Szurubooru\Services\FileService::class);
|
||||
$this->thumbnailServiceMock = $this->mock(\Szurubooru\Services\ThumbnailService::class);
|
||||
$this->timeServiceMock = $this->mock(\Szurubooru\Services\TimeService::class);
|
||||
$this->tokenServiceMock = $this->mock(\Szurubooru\Services\TokenService::class);
|
||||
}
|
||||
|
@ -179,6 +181,7 @@ final class UserServiceTest extends \Szurubooru\Tests\AbstractTestCase
|
|||
$this->passwordServiceMock,
|
||||
$this->emailServiceMock,
|
||||
$this->fileServiceMock,
|
||||
$this->thumbnailServiceMock,
|
||||
$this->timeServiceMock,
|
||||
$this->tokenServiceMock);
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue