1
0
mirror of https://github.com/flarum/core.git synced 2025-07-21 16:51:34 +02:00

Consolidate avatar uploading, allow avatarUrl to be used when updating user

This commit is contained in:
Toby Zerner
2017-10-07 17:39:27 +10:30
parent 87bf84ef6e
commit 2dbcfe02d8
7 changed files with 143 additions and 121 deletions

55
src/Core/AvatarUploader.php Executable file
View File

@@ -0,0 +1,55 @@
<?php
/*
* This file is part of Flarum.
*
* (c) Toby Zerner <toby.zerner@gmail.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
namespace Flarum\Core;
use Intervention\Image\Image;
use League\Flysystem\FilesystemInterface;
use Illuminate\Support\Str;
class AvatarUploader
{
protected $uploadDir;
public function __construct(FilesystemInterface $uploadDir)
{
$this->uploadDir = $uploadDir;
}
public function upload(User $user, Image $image)
{
if (extension_loaded('exif')) {
$image->orientate();
}
$encodedImage = $image->fit(100, 100)->encode('png');
$avatarPath = Str::quickRandom().'.png';
$this->remove($user);
$user->changeAvatarPath($avatarPath);
$this->uploadDir->put($avatarPath, $encodedImage);
}
public function remove(User $user)
{
$avatarPath = $user->avatar_path;
$user->afterSave(function () use ($avatarPath) {
if ($this->uploadDir->has($avatarPath)) {
$this->uploadDir->delete($avatarPath);
}
});
$user->changeAvatarPath(null);
}
}

View File

@@ -12,6 +12,7 @@
namespace Flarum\Core\Command; namespace Flarum\Core\Command;
use Flarum\Core\Access\AssertPermissionTrait; use Flarum\Core\Access\AssertPermissionTrait;
use Flarum\Core\AvatarUploader;
use Flarum\Core\Exception\PermissionDeniedException; use Flarum\Core\Exception\PermissionDeniedException;
use Flarum\Core\Repository\UserRepository; use Flarum\Core\Repository\UserRepository;
use Flarum\Core\Support\DispatchEventsTrait; use Flarum\Core\Support\DispatchEventsTrait;
@@ -30,20 +31,20 @@ class DeleteAvatarHandler
protected $users; protected $users;
/** /**
* @var FilesystemInterface * @var AvatarUploader
*/ */
protected $uploadDir; protected $uploader;
/** /**
* @param Dispatcher $events * @param Dispatcher $events
* @param UserRepository $users * @param UserRepository $users
* @param FilesystemInterface $uploadDir * @param AvatarUploader $uploader
*/ */
public function __construct(Dispatcher $events, UserRepository $users, FilesystemInterface $uploadDir) public function __construct(Dispatcher $events, UserRepository $users, AvatarUploader $uploader)
{ {
$this->events = $events; $this->events = $events;
$this->users = $users; $this->users = $users;
$this->uploadDir = $uploadDir; $this->uploader = $uploader;
} }
/** /**
@@ -61,8 +62,7 @@ class DeleteAvatarHandler
$this->assertCan($actor, 'edit', $user); $this->assertCan($actor, 'edit', $user);
} }
$avatarPath = $user->avatar_path; $this->uploader->remove($user);
$user->changeAvatarPath(null);
$this->events->fire( $this->events->fire(
new AvatarWillBeDeleted($user, $actor) new AvatarWillBeDeleted($user, $actor)
@@ -70,10 +70,6 @@ class DeleteAvatarHandler
$user->save(); $user->save();
if ($this->uploadDir->has($avatarPath)) {
$this->uploadDir->delete($avatarPath);
}
$this->dispatchEventsFor($user, $actor); $this->dispatchEventsFor($user, $actor);
return $user; return $user;

View File

@@ -11,7 +11,9 @@
namespace Flarum\Core\Command; namespace Flarum\Core\Command;
use Exception;
use Flarum\Core\Access\AssertPermissionTrait; use Flarum\Core\Access\AssertPermissionTrait;
use Flarum\Core\AvatarUploader;
use Flarum\Core\Repository\UserRepository; use Flarum\Core\Repository\UserRepository;
use Flarum\Core\Support\DispatchEventsTrait; use Flarum\Core\Support\DispatchEventsTrait;
use Flarum\Core\User; use Flarum\Core\User;
@@ -19,6 +21,9 @@ use Flarum\Core\Validator\UserValidator;
use Flarum\Event\UserGroupsWereChanged; use Flarum\Event\UserGroupsWereChanged;
use Flarum\Event\UserWillBeSaved; use Flarum\Event\UserWillBeSaved;
use Illuminate\Contracts\Events\Dispatcher; use Illuminate\Contracts\Events\Dispatcher;
use Illuminate\Contracts\Validation\Factory;
use Illuminate\Contracts\Validation\ValidationException;
use Intervention\Image\ImageManager;
class EditUserHandler class EditUserHandler
{ {
@@ -35,16 +40,30 @@ class EditUserHandler
*/ */
protected $validator; protected $validator;
/**
* @var AvatarUploader
*/
protected $avatarUploader;
/**
* @var Factory
*/
private $validatorFactory;
/** /**
* @param Dispatcher $events * @param Dispatcher $events
* @param UserRepository $users * @param UserRepository $users
* @param UserValidator $validator * @param UserValidator $validator
* @param AvatarUploader $avatarUploader
* @param Factory $validatorFactory
*/ */
public function __construct(Dispatcher $events, UserRepository $users, UserValidator $validator) public function __construct(Dispatcher $events, UserRepository $users, UserValidator $validator, AvatarUploader $avatarUploader, Factory $validatorFactory)
{ {
$this->events = $events; $this->events = $events;
$this->users = $users; $this->users = $users;
$this->validator = $validator; $this->validator = $validator;
$this->avatarUploader = $avatarUploader;
$this->validatorFactory = $validatorFactory;
} }
/** /**
@@ -135,6 +154,22 @@ class EditUserHandler
}); });
} }
if ($avatarUrl = array_get($attributes, 'avatarUrl')) {
$validation = $this->validatorFactory->make(compact('avatarUrl'), ['avatarUrl' => 'url']);
if ($validation->fails()) {
throw new ValidationException($validation);
}
try {
$image = (new ImageManager)->make($avatarUrl);
$this->avatarUploader->upload($user, $image);
} catch (Exception $e) {
//
}
}
$this->events->fire( $this->events->fire(
new UserWillBeSaved($user, $actor, $data) new UserWillBeSaved($user, $actor, $data)
); );

View File

@@ -14,22 +14,17 @@ namespace Flarum\Core\Command;
use Exception; use Exception;
use Flarum\Core\Access\AssertPermissionTrait; use Flarum\Core\Access\AssertPermissionTrait;
use Flarum\Core\AuthToken; use Flarum\Core\AuthToken;
use Flarum\Core\AvatarUploader;
use Flarum\Core\Exception\PermissionDeniedException; use Flarum\Core\Exception\PermissionDeniedException;
use Flarum\Core\Support\DispatchEventsTrait; use Flarum\Core\Support\DispatchEventsTrait;
use Flarum\Core\User; use Flarum\Core\User;
use Flarum\Core\Validator\UserValidator; use Flarum\Core\Validator\UserValidator;
use Flarum\Event\UserWillBeSaved; use Flarum\Event\UserWillBeSaved;
use Flarum\Foundation\Application;
use Flarum\Settings\SettingsRepositoryInterface; use Flarum\Settings\SettingsRepositoryInterface;
use Illuminate\Contracts\Events\Dispatcher; use Illuminate\Contracts\Events\Dispatcher;
use Illuminate\Contracts\Validation\Factory; use Illuminate\Contracts\Validation\Factory;
use Illuminate\Contracts\Validation\ValidationException; use Illuminate\Contracts\Validation\ValidationException;
use Illuminate\Support\Str;
use Intervention\Image\ImageManager; use Intervention\Image\ImageManager;
use League\Flysystem\Adapter\Local;
use League\Flysystem\Filesystem;
use League\Flysystem\FilesystemInterface;
use League\Flysystem\MountManager;
class RegisterUserHandler class RegisterUserHandler
{ {
@@ -47,14 +42,9 @@ class RegisterUserHandler
protected $validator; protected $validator;
/** /**
* @var Application * @var AvatarUploader
*/ */
protected $app; protected $avatarUploader;
/**
* @var FilesystemInterface
*/
protected $uploadDir;
/** /**
* @var Factory * @var Factory
@@ -65,17 +55,15 @@ class RegisterUserHandler
* @param Dispatcher $events * @param Dispatcher $events
* @param SettingsRepositoryInterface $settings * @param SettingsRepositoryInterface $settings
* @param UserValidator $validator * @param UserValidator $validator
* @param Application $app * @param AvatarUploader $avatarUploader
* @param FilesystemInterface $uploadDir
* @param Factory $validatorFactory * @param Factory $validatorFactory
*/ */
public function __construct(Dispatcher $events, SettingsRepositoryInterface $settings, UserValidator $validator, Application $app, FilesystemInterface $uploadDir, Factory $validatorFactory) public function __construct(Dispatcher $events, SettingsRepositoryInterface $settings, UserValidator $validator, AvatarUploader $avatarUploader, Factory $validatorFactory)
{ {
$this->events = $events; $this->events = $events;
$this->settings = $settings; $this->settings = $settings;
$this->validator = $validator; $this->validator = $validator;
$this->app = $app; $this->avatarUploader = $avatarUploader;
$this->uploadDir = $uploadDir;
$this->validatorFactory = $validatorFactory; $this->validatorFactory = $validatorFactory;
} }
@@ -130,12 +118,6 @@ class RegisterUserHandler
$user->activate(); $user->activate();
} }
$this->events->fire(
new UserWillBeSaved($user, $actor, $data)
);
$this->validator->assertValid(array_merge($user->getAttributes(), compact('password')));
if ($avatarUrl = array_get($data, 'attributes.avatarUrl')) { if ($avatarUrl = array_get($data, 'attributes.avatarUrl')) {
$validation = $this->validatorFactory->make(compact('avatarUrl'), ['avatarUrl' => 'url']); $validation = $this->validatorFactory->make(compact('avatarUrl'), ['avatarUrl' => 'url']);
@@ -144,12 +126,20 @@ class RegisterUserHandler
} }
try { try {
$this->saveAvatarFromUrl($user, $avatarUrl); $image = (new ImageManager)->make($avatarUrl);
$this->avatarUploader->upload($user, $avatarUrl);
} catch (Exception $e) { } catch (Exception $e) {
// //
} }
} }
$this->events->fire(
new UserWillBeSaved($user, $actor, $data)
);
$this->validator->assertValid(array_merge($user->getAttributes(), compact('password')));
$user->save(); $user->save();
if (isset($token)) { if (isset($token)) {
@@ -160,23 +150,4 @@ class RegisterUserHandler
return $user; return $user;
} }
private function saveAvatarFromUrl(User $user, $url)
{
$tmpFile = tempnam($this->app->storagePath().'/tmp', 'avatar');
$manager = new ImageManager;
$manager->make($url)->fit(100, 100)->save($tmpFile);
$mount = new MountManager([
'source' => new Filesystem(new Local(pathinfo($tmpFile, PATHINFO_DIRNAME))),
'target' => $this->uploadDir,
]);
$uploadName = Str::lower(Str::quickRandom()).'.png';
$user->changeAvatarPath($uploadName);
$mount->move('source://'.pathinfo($tmpFile, PATHINFO_BASENAME), "target://$uploadName");
}
} }

View File

@@ -13,18 +13,14 @@ namespace Flarum\Core\Command;
use Exception; use Exception;
use Flarum\Core\Access\AssertPermissionTrait; use Flarum\Core\Access\AssertPermissionTrait;
use Flarum\Core\AvatarUploader;
use Flarum\Core\Repository\UserRepository; use Flarum\Core\Repository\UserRepository;
use Flarum\Core\Support\DispatchEventsTrait; use Flarum\Core\Support\DispatchEventsTrait;
use Flarum\Core\Validator\AvatarValidator; use Flarum\Core\Validator\AvatarValidator;
use Flarum\Event\AvatarWillBeSaved; use Flarum\Event\AvatarWillBeSaved;
use Flarum\Foundation\Application; use Flarum\Foundation\Application;
use Illuminate\Contracts\Events\Dispatcher; use Illuminate\Contracts\Events\Dispatcher;
use Illuminate\Support\Str;
use Intervention\Image\ImageManager; use Intervention\Image\ImageManager;
use League\Flysystem\Adapter\Local;
use League\Flysystem\Filesystem;
use League\Flysystem\FilesystemInterface;
use League\Flysystem\MountManager;
use Symfony\Component\HttpFoundation\File\UploadedFile; use Symfony\Component\HttpFoundation\File\UploadedFile;
class UploadAvatarHandler class UploadAvatarHandler
@@ -37,16 +33,16 @@ class UploadAvatarHandler
*/ */
protected $users; protected $users;
/**
* @var FilesystemInterface
*/
protected $uploadDir;
/** /**
* @var Application * @var Application
*/ */
protected $app; protected $app;
/**
* @var AvatarUploader
*/
protected $uploader;
/** /**
* @var AvatarValidator * @var AvatarValidator
*/ */
@@ -55,16 +51,16 @@ class UploadAvatarHandler
/** /**
* @param Dispatcher $events * @param Dispatcher $events
* @param UserRepository $users * @param UserRepository $users
* @param FilesystemInterface $uploadDir
* @param Application $app * @param Application $app
* @param AvatarUploader $uploader
* @param AvatarValidator $validator * @param AvatarValidator $validator
*/ */
public function __construct(Dispatcher $events, UserRepository $users, FilesystemInterface $uploadDir, Application $app, AvatarValidator $validator) public function __construct(Dispatcher $events, UserRepository $users, Application $app, AvatarUploader $uploader, AvatarValidator $validator)
{ {
$this->events = $events; $this->events = $events;
$this->users = $users; $this->users = $users;
$this->uploadDir = $uploadDir;
$this->app = $app; $this->app = $app;
$this->uploader = $uploader;
$this->validator = $validator; $this->validator = $validator;
} }
@@ -83,60 +79,36 @@ class UploadAvatarHandler
$this->assertCan($actor, 'edit', $user); $this->assertCan($actor, 'edit', $user);
} }
$file = $command->file;
$tmpFile = tempnam($this->app->storagePath().'/tmp', 'avatar'); $tmpFile = tempnam($this->app->storagePath().'/tmp', 'avatar');
$command->file->moveTo($tmpFile); $file->moveTo($tmpFile);
try { try {
$file = new UploadedFile( $file = new UploadedFile(
$tmpFile, $tmpFile,
$command->file->getClientFilename(), $file->getClientFilename(),
$command->file->getClientMediaType(), $file->getClientMediaType(),
$command->file->getSize(), $file->getSize(),
$command->file->getError(), $file->getError(),
true true
); );
$this->validator->assertValid(['avatar' => $file]); $this->validator->assertValid(['avatar' => $file]);
$manager = new ImageManager; $image = (new ImageManager)->make($tmpFile);
// Explicitly tell Intervention to encode the image as PNG (instead of having to guess from the extension)
// Read exif data to orientate avatar only if EXIF extension is enabled
if (extension_loaded('exif')) {
$encodedImage = $manager->make($tmpFile)->orientate()->fit(100, 100)->encode('png', 100);
} else {
$encodedImage = $manager->make($tmpFile)->fit(100, 100)->encode('png', 100);
}
file_put_contents($tmpFile, $encodedImage);
$this->events->fire( $this->events->fire(
new AvatarWillBeSaved($user, $actor, $tmpFile) new AvatarWillBeSaved($user, $actor, $image)
); );
$mount = new MountManager([ $this->uploader->upload($user, $image);
'source' => new Filesystem(new Local(pathinfo($tmpFile, PATHINFO_DIRNAME))),
'target' => $this->uploadDir,
]);
if ($user->avatar_path && $mount->has($file = "target://$user->avatar_path")) {
$mount->delete($file);
}
$uploadName = Str::lower(Str::quickRandom()).'.png';
$user->changeAvatarPath($uploadName);
$mount->move('source://'.pathinfo($tmpFile, PATHINFO_BASENAME), "target://$uploadName");
$user->save(); $user->save();
} finally {
$this->dispatchEventsFor($user, $actor); @unlink($tmpFile);
}
return $user; return $user;
} catch (Exception $e) {
@unlink($tmpFile);
throw $e;
}
} }
} }

View File

@@ -49,15 +49,7 @@ class CoreServiceProvider extends AbstractServiceProvider
return $app->make('Illuminate\Contracts\Filesystem\Factory')->disk('flarum-avatars')->getDriver(); return $app->make('Illuminate\Contracts\Filesystem\Factory')->disk('flarum-avatars')->getDriver();
}; };
$this->app->when('Flarum\Core\Command\UploadAvatarHandler') $this->app->when('Flarum\Core\AvatarUploader')
->needs('League\Flysystem\FilesystemInterface')
->give($avatarsFilesystem);
$this->app->when('Flarum\Core\Command\DeleteAvatarHandler')
->needs('League\Flysystem\FilesystemInterface')
->give($avatarsFilesystem);
$this->app->when('Flarum\Core\Command\RegisterUserHandler')
->needs('League\Flysystem\FilesystemInterface') ->needs('League\Flysystem\FilesystemInterface')
->give($avatarsFilesystem); ->give($avatarsFilesystem);
} }

View File

@@ -12,6 +12,7 @@
namespace Flarum\Event; namespace Flarum\Event;
use Flarum\Core\User; use Flarum\Core\User;
use Intervention\Image\Image;
class AvatarWillBeSaved class AvatarWillBeSaved
{ {
@@ -30,21 +31,21 @@ class AvatarWillBeSaved
public $actor; public $actor;
/** /**
* The path to the avatar that will be saved. * The image that will be saved.
* *
* @var string * @var Image
*/ */
public $path; public $image;
/** /**
* @param User $user The user whose avatar will be saved. * @param User $user The user whose avatar will be saved.
* @param User $actor The user performing the action. * @param User $actor The user performing the action.
* @param string $path The path to the avatar that will be saved. * @param Image $image The image that will be saved.
*/ */
public function __construct(User $user, User $actor, $path) public function __construct(User $user, User $actor, Image $image)
{ {
$this->user = $user; $this->user = $user;
$this->actor = $actor; $this->actor = $actor;
$this->path = $path; $this->image = $image;
} }
} }